您好,欢迎访问三七文档
当前位置:首页 > 电子/通信 > 综合/其它 > Visual_C++面向对象编程教程_第5章_文档与视图
版权所有复制必究文档与视图结构的工作原理文档的读写操作机制菜单编程工具栏编程状态栏编程文档与视图结构是MFC应用程序最基本的程序结构,适用于大多数Windows应用程序。文档和视图完成了程序的大部分功能,它们是MFC应用程序的核心。文档与视图结构是MFC的基石,掌握文档与视图结构对于利用MFC编程有着至关重要的意义。本章对文档与视图结构进行更深入的讨论。信息管理是计算机的一个主要应用,而信息是用数据表示的,因此数据的处理是一般软件都要完成的一项主要工作。采用传统的编程方法,数据处理是一项复杂的任务,并且每一个程序员都可能有不同的处理方法。为了统一和简化数据处理方法,Microsoft公司在MFC中提出了文档/视图结构的概念,其产品Word就是典型的文档/视图结构应用程序。5.1文档与视图结构标题栏主菜单工具栏客户区状态栏不同程序的相同菜单项和工具栏按钮表示相同的操作。5.1.1文档与视图结构概述Windows应用程序界面特点:分为数据的管理和显示文档用于管理和维护数据视图用来显示和编辑数据MFC通过其文档类和视图类提供了大量有关数据处理的方法。MFC文档/视图结构数据处理工作分工:文档的概念在MFC应用程序中的适用范围很广,一般说来,文档是能够被逻辑地组合的一系列数据,包括文本、图形、图象和表格数据。一个文档代表了用户存储或打开的一个文件单位。文档的主要作用是把对数据的处理从对用户界面的处理中分离出来,集中处理数据,同时提供了一个与其它类交互的接口。什么是文档?视图是文档在屏幕上的一个映像,它就像一个观景器,用户通过视图看到文档,也是通过视图来改变文档,视图充当了文档与用户之间的媒介物。应用程序通过视图向用户显示文档中的数据,并把用户的输入解释为对文档的操作。一个视图总是与一个文档对象相关联,用户通过与文档相关联的视图与文档进行交互。当用户打开一个文档时,应用程序就会创建一个与之相关联的视图。什么是视图?视图负责显示和编辑文档数据,但不负责存储。用户对数据的编辑需要依靠窗口上的鼠标与键盘操作才得以完成,这些消息都是由视图类接收后进行处理或通知文档类,如收到窗口刷新消息时调用视图类的成员函数OnDraw()显示文档内容。视图还可在打印机上输出。文档负责数据的读写操作,数据通常被保存在文档类的成员变量中,文档类通过一个称为序列化的成员函数将成员变量的数据保存到磁盘文件中。MFC应用程序为数据的序列化提供了默认支持。视图和文档的功能:文档、视图、框架窗口之间的关系一个视图是一个没有边框的窗口,它位于主框架窗口中的客户区。视图是文档对外显示的窗口,但它并不能完全独立,它必须依存在一个框架窗口内。一个视图只能拥有一个文档,但一个文档可以同时拥有多个视图。视图是文档在屏幕上的一个映像,它就像一个观景器文档/视图结构的优点:把数据处理类从用户界面处理类中分离出来,使得每一个类都能集中地执行一项工作。把Windows程序通常要做的工作分成若干定义好的类,这样有助于应用程序的模块化,程序也易于扩展,编程时只需修改所涉及的类。虽然文档/视图结构牵涉到许多类,其中的也关系比较复杂,但MFCAppWizard向导建立的MFC应用程序框架已经把程序的主要结构完成了,模块间的消息传递以及各函数的功能都已确定。MFC应用程序框架起到了穿针引线的作用,按照消息处理函数功能的不同,将不同消息的响应分别分布在文档类和视图类中。文档/视图结构并没有完全要求所有数据都属于文档类,视图类也可以有自己的数据。如果在视图类中不定义任何数据,在需要时都从文档类中获取,这样做会影响程序的效率。例如,在文本编辑程序中,往往在视图中缓存部分数据,这样可以避免对文档的频繁访问,提高运行效率。在视图类中定义数据包含多个类的MFC文档/视图结构应用程序要管理这些类中的数据,除了考虑在程序的哪一部分拥有数据和在哪一部分显示数据,一个主要的问题是文档数据更改后如何保持视图显示的同步,即文档与视图如何进行交互。在文档、视图和应用程序框架之间包含了一系列复杂的相互作用过程,文档与视图的交互是通过类的公有成员变量和成员函数实现的。5.1.2文档与视图之间的相互作用1.视图类的成员函数GetDocument()一个视图对象只有一个与之相关联的文档对象。在MFC应用程序中,视图对象通过调用成员函数函数GetDocument()得到当前文档。GetDocument()是视图类的成员函数,调用它可以返回与视图相关联的文档对象的指针,利用这个指针可以访问文档类及其派生类的公有成员。当利用MFCAppWizard向导创建一个SDI单文档应用程序Mysdi时,生成了视图类的一个派生类,并在派生类中定义了函数GetDocument()。文档和视图类常用的成员函数CMysdiDoc*CMysdiView::GetDocument(){ASSERT(m_pDocument-IsKindOf(RUNTIME_CLASS(CMysdiDoc)));return(CMysdiDoc*)m_pDocument;//m_pDocument是CArchive类的数据成员,//指向当前文档对象}GetDocument()的Debug版函数代码:一个文档对象可以有多个与之相关联的视图对象,当一个文档的数据通过某个视图被修改后,与它关联的每一个视图都必须反映出这些修改。因此,视图在需要时必须进行重绘,即当文档数据发生改变时,必须通知到所有相关联的视图对象,以便更新所显示的数据。更新与该文档有关的所有视图的方法是调用成员函数CDocument::UpdateAllViews()。2.CDocument类的成员函数UpdateAllViews()如果在文档派生类的成员函数中调用UpdateAllViews()函数,其第一个参数pSender设为NULL,表示所有与当前文档相关的视图都要重绘(参见例5-3)。如果在视图派生类的成员函数中通过当前文档指针调用UpdateAllViews()函数,其第一个参数pSender设为当前视图,如下形式:GetDocument()-UpdateAllViews(this)函数声明:voidUpdateAllViews(CView*pSender,LPARAMlHint=0L,CObject*pHint=NULL);当程序调用CDocument::UpdateAllViews()函数时,实际上是调用了所有相关视图的OnUpdate()函数,以更新相关的视图。需要时,可以直接在视图派生类的成员函数中调用该函数刷新当前视图。3.视图类的成员函数OnUpdate()voidCView::OnUpdate(CView*pSender,LPARAM/*lHint*/,CObject*/*pHint*/){ASSERT(pSender!=this);UNUSED(pSender);//unusedinreleasebuilds//invalidatetheentirepane,erasebackgroundtooInvalidate(TRUE);//使整个窗口矩形无效,通过调//用OnDraw()更新整个视图窗口}基类CView的成员函数在OnUpdate()中通过调用函数CWnd::Invalidate()刷新整个客户区,我们也可以在自己的CWnd派生类中直接调用函数Invalidate()。总结:刷新视图时默认的函数调用过程:CDocument::UpdateAllViews()→CView::OnUpdate()→CWnd::Invalidate()→OnPaint()→OnDraw()MFC基于文档/视图结构的应用程序分为单文档和多文档两种类型,一个多文档应用程序有一个主窗口,但在主窗口中可以同时打开多个子窗口,每一个子窗口对应一个不同的文档。利用MFCAppWizard[exe]向导可以很方便地建立一个多文档应用程序,只需在MFCAppWizard向导第1步选择Multipledocuments程序类型。SDI和MDI使用不同框架窗口。SDI的框架窗口是唯一的主框架窗口,窗口类是CMainFrame,由CFrameWnd派生而来。5.1.3多文档MDI的框架窗口分为主框架窗口和子框架窗口,区别于SDI,MDI的主框架窗口不包含视图,分别由每个子框架窗口包含一个视图。MDI的主框架窗口类不与某个打开的文档相关联,而只与子框架窗口相关联。MDI主框架窗口类CMainFrame由CMDIFrameWnd派生而来,而MDI子框架窗口类CChildFrame由CMDIChildWnd派生而来。在文档/视图结构中,数据以文档类对象的形式存在。文档对象通过视图对象显示,而视图对象又是主框架窗口的一个子窗口,并且涉及文档操作的菜单和工具栏等资源也是建立在主框架窗口上。这样,文档、视图、框架类和所涉及的资源形成了一种固定的联系,这种固定的联系就称为文档模板。也就是说,文档模板描述了相对应每一种类型文档的视图和窗口的风格类型。当打开某种类型的文件时,应用程序必须确定那一种文档模板用于解释这种文件。在初始化程序时,必须首先注册文档模板,以便程序利用这个模板来完成主框架窗口、视图、文档对象的创建和资源的装入。文档模板的概念:标准Windows应用程序界面窗口组成:客户区非客户区:5.2菜单设计窗口的边框标题栏菜单栏工具栏状态栏滚动条菜单、工具栏、状态栏是用户与应用程序进行交互的重要工具。菜单和工具栏为应用程序提供了传递用户命令的选择区域,而状态栏提供了提示信息的输出区域。5.2.1建立菜单资源使用MFCAppWizard向导创建文档/视图结构应用程序时,向导将自动生成Windows标准的菜单资源和命令处理函数。但这个默认生成的主框架菜单资源往往不能满足实际的需要,因此我们需要利用菜单资源编辑器对其进行修改和添加。例编写一个单文档应用程序DrawCoin,为程序添加一个“画硬币”主菜单,并在其中添加“增加硬币”和“减少硬币”两个菜单项。1.利用MFCAppWizard[exe]向导创建SDI应用程序。在项目工作区的ResourceView页面中选择Menu并展开它,双击下面的IDR_MAINFRAME项弹出菜单资源编辑器,显示应用程序向导所创建的菜单资源。2.为程序添加主菜单。双击菜单栏右边虚空白框,弹出属性对话框,在Caption框输入主菜单的标题“画硬币(&C)”,字符&用于在显示C时加上下划线,并表示其快捷键为Alt+C。3.在主菜单“画硬币(C)”下方双击带虚框的空白菜单项,弹出属性对话框。在ID栏输入ID_COIN_ADD。在Caption框输入菜单项的标题“增加硬币(&A)\tCtrl+A”。5.2.2添加菜单命令处理函数菜单实际上是一系列命令的列表,当一个菜单项被选中后,一个含有该菜单项ID标识的WM_COMMAND命令消息将发送到应用程序窗口,应用程序将该消息转换为一个函数(命令消息处理函数)调用。命令消息来自于用户界面对象,是由菜单项、工具栏按钮和快捷键等程序界面元素发送的WM_COMMAND消息。在MFC应用程序中,许多类都能接收菜单被选中而引发的消息。总的来说,从类CCmdTarget派生出来的类都可以加入应用程序的消息循环。应该将菜单命令映射到哪个类中,需要由该命令的功能决定。如果一个命令同视图的显示有关,就应该将其映射到视图类;如果同文档的读写有关,就映射到文档类中;如果命令完成通用功能,一般映射到窗口框架类。有时无法对功能进行准确分类,则可以将菜单命令映射到任意一个类,看看是否能够完成指定的功能。将菜单命令映射到哪个类?利用ClassWizard类向导添加菜单命令WM_COMMAND消息处理函数后,向导将自动添加一个如下格式消息映射:ON_COMMAND(MenuItemID,MemberFuntion)其中参数MenuItemID是菜单项的ID标识,参数MemberFuntion是处理该消息的成员
本文标题:Visual_C++面向对象编程教程_第5章_文档与视图
链接地址:https://www.777doc.com/doc-2854962 .html