多文档DOC/VIEW框架不创建新文档

转自:http://www.cppblog.com/totti1006/archive/2007/11/08/36115.html

在文档窗口创建的时候 ,它缺省总是会新建一个新文档 , 那么怎么让它不新建文档呢?就这个问题 , 我对文档视图窗口应用程序启动时的文档创建机制 , 稍稍的浅浅挖了一下 , 做了一个详细的分析 , 希望能够对初学者有所帮助 .

在App文件的InitInstance()函数中, 有如下几行代码:
CCommandLineInfo  cmdInfo;
ParseCommandLine(cmdInfo);

          if (!ProcessShellCommand(cmdInfo)) return FALSE;

这几行代码是程序启动时创建新文档的关键代码 .

 

1: 我们首先来看看让CCommandLineInfo类是个什么东西:( 部分源代码 )

//in afxwin.h

 class CCommandLineInfo : public CObject

{

    public:

    // Sets default values

   CCommandLineInfo();

   BOOL m_bShowSplash;

   BOOL m_bRunEmbedded;

   BOOL m_bRunAutomated;

 

   enum { FileNew, FileOpen, FilePrint, FilePrintTo, FileDDE, AppRegister,

   AppUnregister, FileNothing = -1 } m_nShellCommand;

 

 // not valid for FileNew

 CString m_strFileName;

   . . .

   ~CCommandLineInfo();

   . . .

 };

  这里要重点注意enum {FileNew, . . . , FileNothing = -1 }m_nShellCommand;

这里联合类型定义的m_nShellCommand 就是外壳程序执行的命令类型 , 如果m_nShellCommand设置为FileNew ,那么程序就会创建新文档 . 如果想在文档开始时不创建新文档 , 就必须将m_nShellCommand设置为FilleNothing .

下面我们再看看CCommandLineInfo的构造函数 .

//in appcore.cpp

 CCommandLineInfo::CCommandLineInfo()

 {

        m_bShowSplash   = TRUE;

        m_bRunEmbedded  = FALSE;

        m_bRunAutomated = FALSE;

        m_nShellCommand = FileNew;

 }

这里很明白的看出 , 构造函数中 , 缺省将 m_nShellCommand设置为 FileNew .

 

2:再来看看ParseCommandLine(cmdInfo); 函数 .

 

void CWinApp::ParseCommandLine(CCommandLineInfo& rCmdInfo)

{

    for (int i = 1; i < __argc; i++)

    {

        LPCTSTR pszParam = __targv[i];

        BOOL bFlag = FALSE;

        BOOL bLast = ((i + 1) == __argc);

        if (pszParam[0] == '-' || pszParam[0] == '/')

        {

            // remove flag specifier

            bFlag = TRUE;

            ++pszParam;

        }

        rCmdInfo.ParseParam(pszParam, bFlag, bLast);

    }

}

可以看出ParseCommandLine主要是对输入的命令行参数做一些分析 , 并调用ParseParam来进行处理 .继续分析 ParseParam函数 , 查看如下源代码:

void CCommandLineInfo::ParseParam(const TCHAR* pszParam,BOOL bFlag,BOOL bLast)

{

    if (bFlag)

    {

        USES_CONVERSION;

        ParseParamFlag(T2CA(pszParam));

    }

    else

        ParseParamNotFlag(pszParam);

 

    ParseLast(bLast);

}

其它的函数撇开不看 , 我们重点来分析一下ParseParamFlag()和ParseLast()函数 .

void CCommandLineInfo::ParseParamFlag(const char* pszParam)

{

    // OLE command switches are case insensitive, while

    // shell command switches are case sensitive

 

    if (lstrcmpA(pszParam, "pt") == 0)

        m_nShellCommand = FilePrintTo;

    else if (lstrcmpA(pszParam, "p") == 0)

        m_nShellCommand = FilePrint;

    else if (lstrcmpiA(pszParam, "Unregister") == 0 ||

             lstrcmpiA(pszParam, "Unregserver") == 0)

        m_nShellCommand = AppUnregister;

    else if (lstrcmpA(pszParam, "dde") == 0)

    {

        AfxOleSetUserCtrl(FALSE);

        m_nShellCommand = FileDDE;

    }

    else if (lstrcmpiA(pszParam, "Embedding") == 0)

    {

        AfxOleSetUserCtrl(FALSE);

        m_bRunEmbedded = TRUE;

        m_bShowSplash = FALSE;

    }

    else if (lstrcmpiA(pszParam, "Automation") == 0)

    {

        AfxOleSetUserCtrl(FALSE);

        m_bRunAutomated = TRUE;

        m_bShowSplash = FALSE;

    }

}

ParseParamFlag判断传过来的字符串 ,判断它的参数类型 , 并根据参数类型做不同的处理 .

void CCommandLineInfo::ParseLast(BOOL bLast)

{

    if (bLast)

    {

        if (m_nShellCommand == FileNew && !m_strFileName.IsEmpty())

            m_nShellCommand = FileOpen;

        m_bShowSplash = !m_bRunEmbedded && !m_bRunAutomated;

    }

}

ParseLast会判断是否是是FileNew打开新文档 , 如果是打开新文档 , 并且打开的文档名不为空的话, 就假定用户想打开这个文档 , 把命令设置为FileOpen .


最后 , 我们可以总结一下ParseCommandLine的作用 . ParseCommandLine的作用主要是分析命令行参数,如果没有命令行参数 ,ParseCommandLine()就假定用户想新建一个文档,于是设置一个FileNew命令,如果命令行参数中有一个文件名,ParseCommandLine()就假定用户想打开该文件,于是设置一个FileOpen命令。

 

3: 最后 , 我们来重点看看外壳命令解析的主角 : ProcessShellCommand ();(部分源代码)

BOOL CWinApp::ProcessShellCommand(CCommandLineInfo& rCmdInfo)

 {

      BOOL bResult = TRUE;

      switch (rCmdInfo.m_nShellCommand)

     {

          case CCommandLineInfo::FileNew:

                  if (!AfxGetApp()->OnCmdMsg(ID_FILE_NEW, 0, NULL, NULL))

                         OnFileNew();

                  if (m_pMainWnd == NULL)

                         bResult = FALSE;

                  break;

        case CCommandLineInfo::FileOpen:      . . .

        case CCommandLineInfo::FilePrintTo:    . . .

        case CCommandLineInfo::FilePrint:      . . .

        case CCommandLineInfo::FileDDE:       . . .

        case CCommandLineInfo::AppRegister:   . . .

        case CCommandLineInfo::AppUnregister: . . .

        . . .

      }

}

代码看到这里 , 一切都很明白了 . ProcessShellCommand分析m_nShellCommand ,并根据m_nShellCommand不同的类型值进行不同的处理 .

再来分析下面两行代码:


         CCommandLineInfo cmdInfo;

         ParseCommandLine(cmdInfo);

       if (!ProcessShellCommand(cmdInfo)) return FALSE;


  1: 当CCommandLineInfo cmdInfo进行定义时 , 首先调用构造函数 , 构造函数中m_nShellCommand被设置为FileNew 
 
2: 然后执行ParseCommandLine(cmdInfo);对命令进行分析 .

  3: 最后执行ProcessShellCommand (cmdInfo) , ProcessShellCommand ()判断m_nShellCommand为FileNew , 于是调用OnFileNew()创建了一个新的文档 .

   这也就是创建新文档的来龙去脉 .

 

最后, 我们看怎么样解决不想在应用程序启动时的创建新文档的问题:

直接在InitInstance()函数中用如下代码代替原来的几行即可:

CCommandLineInfo cmdInfo;
cmdInfo.m_nShellCommand = CCommandLineInfo::FileNothing;
ParseCommandLine(cmdInfo);

       if (!ProcessShellCommand(cmdInfo)) return FALSE;


单文档是不可以

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
内容介绍 项目结构: Controller层:使用Spring MVC来处理用户请求,负责将请求分发到相应的业务逻辑层,并将数据传递给视图层进行展示。Controller层通常包含控制器类,这些类通过注解如@Controller、@RequestMapping等标记,负责处理HTTP请求并返回响应。 Service层:Spring的核心部分,用于处理业务逻辑。Service层通过接口和实现类的方式,将业务逻辑与具体的实现细节分离。常见的注解有@Service和@Transactional,后者用于管理事务。 DAO层:使用MyBatis来实现数据持久化,DAO层与数据库直接交互,执行CRUD操作。MyBatis通过XML映射文件或注解的方式,将SQL语句与Java对象绑定,实现高效的数据访问。 Spring整合: Spring核心配置:包括Spring的IOC容器配置,管理Service和DAO层的Bean。配置文件通常包括applicationContext.xml或采用Java配置类。 事务管理:通过Spring的声明式事务管理,简化了事务的处理,确保数据一致性和完整性。 Spring MVC整合: 视图解析器:配置Spring MVC的视图解析器,将逻辑视图名解析为具体的JSP或其他类型的视图。 拦截器:通过配置Spring MVC的拦截器,处理请求的预处理和后处理,常用于权限验证、日志记录等功能。 MyBatis整合: 数据源配置:配置数据库连接池(如Druid或C3P0),确保应用可以高效地访问数据库。 SQL映射文件:使用MyBatis的XML文件或注解配置,将SQL语句与Java对象映射,支持复杂的查询、插入、更和删除操作。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值