分类: C/C++
2008-03-18 17:20:41
虽然MFC的功能很强大,但Windows Template Library(WTL)的出现,无疑说明了MFC有一定的弊病。和MFC相比,功能并没有MFC完善。比如MFC支持doc/view架构,而WTL并不支持。同时,WTL也没有Microsoft的官方支持。但是,WTL是基于模版(template)的,其应用程序最小只有24KB,同时不象MFC,依赖DLL。但是WTL也实现了CString、CRect、CSize、CPoint等常用的类,还CStaticT
2.准备工作
首先安装WTL AppWizard,现在最高版本应该是WTL7.0,直接运行setup脚本文件就可以了,这里给大家几个下载地址:
1)
2)
3)
这样当你启动VC6.0后,File/New时,在Project属性页就能看到添加了一项ATL/WTL AppWizard。你可以直接把WTL的库文件( 共16个.h文件)拷贝到vc的安装目录VC98/Include中,也可以放到你的工程文件夹中。
3.应用实例1---SDI中状态栏的应用
(1) File/New,如图:
(2) OK后,
SDI(Single Document Interface)应用程序通常只有一个主窗口(通常是一个框架窗口,Frame Window)。框架窗口包含菜单、工具栏、状态栏和称为视(View)的客户工作区。
Multip-SDI(Multiple Threads SDI),就像IE浏览器,使用"文件/新建/窗口"命令后,会出现另一个IE窗口。
MDI(Multiple Document Interface)应用程序有一个主框架窗口,但有多个子框架窗口。每个子窗口都有自己的视(View),和MFC的相似。
Dialog应用程序是基于对话框的。
(3) Next
(4) 在产生的文件中可以看到WTL确实不支持Doc/View.
WTL对单界面线程的封装:WTL使用一个_Module全局变量来保存全局数据,并通过它来引用应用程序级的代码。在WTL中,该变量是CAppModule的实例,想象MFC的theApp。
●下面对MyTestWTL.cpp文件中的函数作一些说明:
Ⅰ:
int WINAPI _tWinMain(HINSTANCE hInstance, HINSTANCE /*hPrevInstance*/, LPTSTR lpstrCmdLine, int nCmdShow) { HRESULT hRes = ::CoInitialize(NULL); // If you are running on NT 4.0 or higher you can use the following call instead to // make the EXE free threaded. This means that calls come in on a random RPC thread. // HRESULT hRes = ::CoInitializeEx(NULL, COINIT_MULTITHREADED); ATLASSERT(SUCCEEDED(hRes)); // this resolves ATL window thunking problem when Microsoft Layer for Unicode (MSLU) is used ::DefWindowProc(NULL, 0, 0, 0L); AtlInitCommonControls(ICC_COOL_CLASSES | ICC_BAR_CLASSES); // add flags to support other controls hRes = _Module.Init(NULL, hInstance); ATLASSERT(SUCCEEDED(hRes)); int nRet = Run(lpstrCmdLine, nCmdShow); //程序逻辑,调用全局函数Run() _Module.Term(); // 终止模块 ::CoUninitialize(); return nRet; }入口函数名为_tWinMain()。当使用UNICODE时,编译器会将它替换为wWinMain(),否则,为WinMain()。入口函数其实就是主线程(_Module)的起始点,这和SDK,MFC一个道理。一个_Module还维持一个消息循环Map。
int Run(LPTSTR /*lpstrCmdLine*/ = NULL, int nCmdShow = SW_SHOWDEFAULT) { CMessageLoop theLoop; _Module.AddMessageLoop(&theLoop); CMainFrame wndMain; if(wndMain.CreateEx() == NULL) { ATLTRACE(_T("Main window creation failed!\n")); return 0; } wndMain.ShowWindow(nCmdShow); int nRet = theLoop.Run(); _Module.RemoveMessageLoop(); return nRet; }该函数创建了一个CMessageLoop实例,该实例包含了这个线程的消息循环。这些消息循环都放在模块的全局消息循环中,通过线程的ID来索引。这样,该线程的其它代码就能访问得到。每一个应用程序维护一个消息循环队列Map,应用程序中的每个线程都通过"_Module.AddMessageLoop(&theLoop)",把该线程的消息循环加入到_Module的消息循环Map中。消息循环对象包含了消息过滤和空闲处理。每个线程都可以加入空闲处理代码和消息过滤。
● 添加代码:
Ⅰ:在stdafx.h中添加:
#includeⅡ:添加3个图标资源:IDR_DEFAULT, IDR_DATE, IDR_TIME#include #include //这三个是WTL 中实现常用控件类 #include #include //WTL 使用类像 CString等 #include //WTL 使用像MFC中DDX/DDV 机制
Ⅲ:在MainFrm.h中添加 CMultiPaneStatusBarCtrl m_Mstatus;
Ⅳ:在MainFrm.cpp中添加
在函数最前面声明两个数组:
static int arrPanes[] = { IDR_DEFAULT, IDR_DATE, IDR_TIME }; static int arrWidths[] = { 450,200, 600 }; //你自己修改一下值看看有什么不同在OnCreate函数中添加:
CreateSimpleStatusBar(); m_Mstatus.SubclassWindow(m_hWndStatusBar); m_Mstatus.SetPanes(arrPanes,sizeof(arrPanes)/sizeof(int), false); m_Mstatus.SetPaneWidth(IDR_DEFAULT,400); m_Mstatus.SetPaneWidth(IDR_DATE,200); m_Mstatus.SetPaneWidth(IDR_TIME,900); m_Mstatus.SetPaneIcon(IDR_DEFAULT,AtlLoadIconImage(IDR_DEFAULT, LR_DEFAULTCOLOR)); m_Mstatus.SetPaneIcon(IDR_DATE, AtlLoadIconImage(IDR_DATE, LR_DEFAULTCOLOR)); m_Mstatus.SetPaneIcon(IDR_TIME, AtlLoadIconImage(IDR_TIME, LR_DEFAULTCOLOR)); m_Mstatus.SetPaneText(IDR_DATE,"WTL is very wonderful!",0); m_Mstatus.SetPaneText(IDR_TIME,"",0);这些代码相信大家一看就会明白!
参考资料:
深入剖析WTL——Win32模型,作者:建新
WTL for MFC Programmers 作者:Michael Dunn
在此深表感谢