MFC深入浅出读书笔记第一部分

最近看侯捷的MFC深入浅出,简单总结一下。

第一章首先就是先了解一下windows程序设计的基础知识,包括win32程序开发基础,什么*.lib,*.h,*.cpp的,程序入口点WinMain函数,窗口注册,消息循环,消息映射等。还有控制台(console)应用程序的运行过程以及与MFC的区别,进程与线程的诞生(CreatProcess()、CreateThread()、_beginthread()等)与消亡(ExitProcess()、endthread()等)。

第二章讲解了一些C++的重要性质,有this指针的使用,虚函数与多态,静态成员(函数与变量),执行时识别(RTTI),动态生成(Dynamic Creation),异常处理(Exception Handling),模板(Template)等。

  虚函数的总结:
  1 、 父类指针指向子类对象时,该指针只能引用父类的方法。
  2 、子类指针指向父类对象时,问题多多,需要强制转换,不符合实际。
  3 、父类和子类有同名成员函数时,父类指针永远只能调用父类方法而无论其指向是父类还是子类,子类指针也类似。
  如果你预期衍生类别有可能重新定义某一个成员函数,那么你就在基础类别中把此函数设为virtual。
  四种不同对象的生存方式:(in stack ,in heap,global,local static)
 

 1  void Func()
 2   {
 3     CObject ob;//在堆栈(stack)中产生ob对象
 4   }
 5   void Func()
 6   {
 7     CObject *ob = new CObject();//在堆(heap)中产生ob对象
 8   }
 9   CObject ob;//全局对象
10   void Func()
11   {
12     static CObject ob;//局部静态对象,产生在固定的内存上(但是它既不是stack,也不是heap)
13   }

  其中静态全局对象的构造必须依靠startup码实现,startup码是更早于程序进入点(main or WinMain)执行的代码,有C++编译器提供。

  模板总结

  1、模板函数

   例子: 

  template <class T>
  T power(T base, int exponent);

  其中的class不是C++中的class,这里它只是指一个普通的数据类型,而T就是一种具体的数据类型,且只有在调用函数时才确定T的值。宏观上说T可以是任何类型的数据类型,前提是函数内的操作合法,编译器能识别出。

  2、类模板

类的声明比较简单,比一般的类声明多了一行    template <class T>  

 1  template <class T>  //声明
 2  class CTest
 3  {
 4     public :
 5      CTest(T t1, T t2, T t3);
 6      T Min();
 7      T Max();
 8      private:
 9      T a, b, c;
10  };

函数的定义如下,规则就是:每一个成员函数前都要加上template <class T>,而且类别名称应该使用CTest<T>。

1 template <class T>//这个必须加上
2   T CTest<T>::Min()
3   {
4       T minab = a < b ? a : b;
5       return minab < c ? minab : c;
6   }

类的使用如下,必须首先指定类型

1 CTest<int> obj1(2, 5, 4);    //int类型
2      cout << obj1.Min() << endl;
3      cout << obj1.Max() << endl;
4 
5 CTest<float> obj1(2.23, -3.35, 9.334);    //float类型
6      cout << obj1.Min() << endl;
7      cout << obj1.Max() << endl;

第三章介绍MFC的六大关键技术,包括MFC初始化、执行时类型识别(RTTI)、动态生成(Dynamic Creation)、永久保存(Persistence)、消息映射(Message Mapping)、命令绕行(Command Routing)。

  1、MFC初始化----寻找main函数

  C++规定,全局对象的构造将比main或WinMain函数更早。 

  首先是全局构造

    CObject构造函数 -> CCmdTarget -> CWinThread -> CWinApp -> theApp构造函数

  然后进入WinMain函数

    WinMain -> AfxWinMain -> AfxWinInit -> theApp.InitApplication -> theApp.InitInstance

  接着执行线程过程。

    theApp.Run()

  最后清理

    AfxWinTerm

  2、执行时类型识别(RTTI)-- -- 查询父子关系(true or false)

  CRuntimeClass的实现要通过几个复杂的宏DECLARE_DYNAMIX(Cxxx)和IMPLEMENT_DYNAMIC(Cxxx,Cxxxbase),在类内部声明这两个宏,就可以将该类加入到类型识别库。而CObject的函数IsKindOf()则可以指向任何派生至CObject的类,然后调用它就可以找到基类。

  3、动态生成(Dynamic Creation) -----  如何从文件中读取一个类并实现它

  动态生成也通过CRuntimeClass来实现,加入两个成员变量,并通过宏来实现。DECLARE_DYNCREATE 和IMPLEMENT_DYNCREATE 。

  4、永久保存(Persistence) -- -- 将对象写入到文件(Serialize)

  这需要用到宏DECLARE_SERIAL  / IMPLEMENT_SERIAL,以及CObject的虚函数 virtual void Serialize(CArchive& ar);

  5 、消息映射(Message Mapping)---- 消息如何获取

  宏DECLARE_MESSAGE_MAP / BEGIN_MESSAGE_MAP / END_MESSAGE_MAP

  在WinMain函数里加入一死循环

1   while (GetMessage (&msg, NULL, 0, 0))
2      {
3            TranslateMessage (&msg) ;
4            DispatchMessage (&msg) ;       
5      }

  6、命令绕行(Command Routing)---- 消息流动方向

   一般Window消息(WM_xxx),一定是从子类流向父类,不可能旁流。

   命令消息WM_COMMAND 的消息流向比较另类,可以横向流动

 

 

posted @ 2013-08-26 10:52  struggle_time  阅读(1578)  评论(10编辑  收藏  举报