让MFC支持命令行参数

//注意:在BOOL CXXXApp::InitInstance()里面

//如果在CXXXDlg dlg; 之前加入,这个过程就是程序框架出来之前。

//简介 LPTSTR WINAPI GetCommandLine(void);

//没有参数,返回值是一个LPTSTR类型的字符串指针,即命令行参数的字符串。

// LPWSTR *CommandLineToArgvW(

// LPCWSTR lpCmdLine,

// int *pNumArgs);


//定义int接收参数个数

int nArgs = 0;

//此项单独用来读取参数个数,用于一些需要先判断个数的程序

CommandLineToArgvW(GetCommandLineW(), &nArgs);

if (nArgs>=2)

{

//处理过程

}


//定义接受参数数据结构

LPWSTR *szArglist = NULL;

 //获取参数 以及参数个数。

szArglist = CommandLineToArgvW(GetCommandLineW(), &nArgs);  

if( NULL != szArglist)

 {

//szArglist就是保存参数的数组

  //nArgs是数组中参数的个数

//数组的第一个元素表示程序本身,也就是szArglist[0],

  //其他的元素依次是输入参数

}  

 //取得参数后,释放CommandLineToArgvW申请的空间

CString str; 

 for (int i=1;i

{

//处理szArglist[i]过程

}

//释放szArglist

LocalFree(szArglist); 

//如需要直接退出

//exit(0);


以下为参考资料
让MFC程序支持命令行参数

一般情况下,如果我们的VC工程是Console控制台类型或者是Win32类型的情况下,这两类的工程类型比较容易获得命令行参数,
即,通过其入口点函数

int main(int argc, char *argv[])
int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPTSTR lpCmdLine, int nCmdShow)

但是,如果我们的工程类型是MFC对话框或单文档类型的话,如何让我们的程序支持命令行参数呢?
系统没有为我们提供好,那我们只有自己获取了……

一般情况下,需要在MFC程序的主线程初始化函数中进行处理,主线程的初始化函数即:
BOOL CProjCleanerApp::InitInstance(),这里面我们假设工程的名字是ProjCleaner,CProjCleanerApp 即主线程类。
在 BOOL CProjCleanerApp::InitInstance() 函数中进行程序的初始化以及对话框界面的显示工作,其大致代码如下:

BOOL CProjCleanerApp::InitInstance()
{
//省略的其他代码……;
CWinApp::InitInstance();
//省略的其他代码……;

CProjCleanerDlg dlg; //新建主对话框类型对象;
m_pMainWnd = &dlg;
INT_PTR nResponse = dlg.DoModal(); //进行模态对话框的显示;
if (nResponse == IDOK)
{
// TODO: Place code here to handle when the dialog is
// dismissed with OK
}
else if (nResponse == IDCANCEL)
{
// TODO: Place code here to handle when the dialog is
// dismissed with Cancel
}
return FALSE;
}
我们要获取程序的命令行参数,一般是在主对话框界面显示之前,即 CProjCleanerDlg dlg; 之前获取。
获取的方法主要是调用 GetCommandLine 和 CommandLineToArgvW 函数。

GetCommandLine 函数的原型定义如下:

LPTSTR WINAPI GetCommandLine(void);
没有参数,返回值是一个LPTSTR类型的字符串指针,即命令行参数的字符串。

默认情况下,每个程序都会有一个命令行参数,即可执行程序本身的路径,第二个参数才是真正的附加命令行参数。
举个例子,调用GetCommandLine函数获取到的命令行参数的字符串大致如下:

"E:\SoftWare\ProjCleaner.exe" "-help"
其中,"E:\SoftWare\ProjCleaner.exe" 为可执行文件的路径,"-help" 为附加的参数。

调用完 GetCommandLine 函数获取到了命令行参数之后,
我们就要调用 CommandLineToArgvW 函数去分解得到的命令行参数字符串,

CommandLineToArgvW 函数的原型定义如下:

LPWSTR *CommandLineToArgvW(
LPCWSTR lpCmdLine,
int *pNumArgs
);

他有两个参数,第一个参数 lpCmdLine 即通过 GetCommandLine 函数获取到的命令行参数字符串指针,
第二个参数是一个 int 类型的指针,用这个参数返回命令行参数中的参数个数,
默认情况下,pNumArgs 会返回1,如果有附加的命令行参数,该值会大于1,比如:
"E:\SoftWare\ProjCleaner.exe" "-help"
则,pNumArgs 会返回2。

CommandLineToArgvW 函数的返回值是一个宽字节LPWSTR类型的字符串指针数组,
数组中的每一个字符串指针代表着一个命令行参数,根据上面的示例,
设返回值为:
LPWSTR *lpszArgv,则 lpszArgv[0] 就是 "E:\SoftWare\ProjCleaner.exe",lpszArgv[1] 就是 "-help",这样大家明白了吧?

通过 GetCommandLine、CommandLineToArgvW 两个函数就可以成功的获取并拆分命令行参数了。




MFC也可以像控制台那样获取命令行参数:
利用GetCommandLineW()函数获得命令行参数,
利用CommandLineToArgvW()函数解析命令行参数。

LPWSTR *szArglist = NULL;
int nArgs = 0;
szArglist = CommandLineToArgvW(GetCommandLineW(), &nArgs);
if( NULL != szArglist)
{
//szArglist就是保存参数的数组
//nArgs是数组中参数的个数
//数组的第一个元素表示进程的path,也就是szArglist[0],其他的元素依次是输入参数。
}
//取得参数后,释放CommandLineToArgvW申请的空间
LocalFree(szArglist);




在MFC程序中,可以用以下几种方法来获取命令行参数。
为方便说明,我们假设执行了命令:C:\test\app.exe -1 -2

方法一
::GetCommandLine();
将获取到 "C:\test\app.exe" -1 -2

方法二
for (int i=0;i<__argc;i++)
{
argv[i];
将依次得到C:\test\app.exe -1 -2
}

方法三
AfxGetApp()->m_lpCmdLine;
将获取到 -1 -2



获取命令行的方法:

1、GetCommandLine() 获取输入的所有信息,包括程序所在路径及参数

2、AfxGetApp()->m_lpCmdLine 只包含参数

一般情况下,获取到命令行后就可以针对命令行中的内容进行相应的处理了



CObject
└CCommandLineInfo
类CCommandLineInfo用于分析启动应用时的命令行参数。
MFC应用一般都会在它的应用对象中使用函数InitInstance创建这个类的一个本地实例。
然后把该对象传给CWinApp::ParseCommandLine,ParseCommandLine又重复调用ParseParam填充CCommandLineInfo对象。
最后,CCommandLineInfo对象被传给CWinApp::ProcessShellCommand来处理命令行参数和选项。

BOOL CExampleApp::InitInstance()
{
...
// 分析标准外壳命令、DDE、打开文件操作的命令行
CCommandLineInfo cmdInfo;
ParseCommandLine(cmdInfo);


// 调度在命令行中指定的命令。如果
// 用 /RegServer、/Register、/Unregserver 或 /Unregister 启动应用程序,则返回 FALSE。
if (!ProcessShellCommand(cmdInfo))
return FALSE;
...
}


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);
}
}




目标
让应用程序处理这里所见的命令行标志。
>XXX.exe /c /d
策略
一个MFC应用程序可以用CCommandLineInfo类的成员函数ParseParam()处理一些标准标
志。要添加我们自己的标志,而仍然能够支持另外一些标志,我们将从CCommandLineInfo派
生类,然后重载ParseParam()。
步骤
1.创建一个新的CCommandLineInfo类
1)用ClassWizard创建一个派生于CCommandLineInfo的新类。在新类中,为应用程序要
处理的每个新的标志添加一个Boolean或String成员变量。
classCWzdCommandLineInfo:publicCCommandLineInfo
{
public:
BOOLm_bAFlag;
BOOLm_bCFlag;
BOOLm_bDAFlag;
CStringm_sArg;
2)添加一个ParseParam()函数,以重载基类的ParseParam()函数。
//Operations
public:
voidParseParam(constTCHAR*pszParam,BOOLbFlag,BOOLbLast);
};
3)如下实现ParseParam():
voidCWzdCommandLineInfo::ParseParam(constTCHAR*pszParam,
BOOLbFlag,BOOLbLast)
{
CStringsArg(pszParam);
if(bFlag)
{
m_bAFlag=!sArg.CompareNoCase("a");
m_bCFlag=!sArg.CompareNoCase("c");
m_bDAFlag=!sArg.CompareNoCase("da");
}
//m_strFileNamegetsthefirstnonflagname
elseif(m_strFileName.IsEmpty())
{
m_sArg=sArg;
}


第二部分用户界面实例

CCommandLineInfo::ParseParam(pszParam,bFlag,bLast);
}
注意到变量pszParam包括命令行中的下一项。如果pszParam的后面是一个—(连字符)或
/(正斜杠)字符,则bFlag变量为TRUE,这些字符将被删除;如果pszParam是一行中最后一个
变量,则bLast为TRUE。确信最后调用基类的ParseParam(),否则标准标志不被处理。
4)有关命令行消息类的详细清单,参见本节的“清单—命令行消息类”。
2.把新的命令行消息类插到应用程序类中
1)在应用程序类中找到ParseCommandLine(),并用该新类替换CCommandLineInfo类。
//Parsecommandlineforstandardshellcommands,DDE,fileopen
CWzdCommandLineInfocmdInfo;
ParseCommandLine(cmdInfo);
2)现在,命令行选项不能作为cmdInfo变量的成员变量。
if(cmdInfo.m_bAFlag)
{
:::
}
3)要使这些选项在整个应用程序中可得,则把cmdInfo嵌入应用程序中,并访问它的成员
变量。
(CXXXApp*)AfxGetApp()->m_cmdInfo.m_bAFlag;
注意:CXXXApp类是你自己创建的应用程序类
说明
■标准MFC标志如下,真正处理这些标准命令行发生在ProcessShellCommand(cmdInfo)
中,它正好在应用程序类中ParseCommandLine()之后。
nothing使应用程序试图打开一个新文档
filename使应用程序试图以文档方式打开文件名
/pfilename使应用程序打开并打印给定的文件名到默认的打印机
/ptfilename与上面相同,但输入到指定的打印机
printerdriverport
/dde使应用程序开始运行,并等待DDE命令
/AutomationCOM标志
/Embedding
/Unregister
/Unregserver
■处理非标准标志(如名字)会有点复杂,我们认为出现的第一个非标准标志是文档文件名。
然而,一旦一个文件名被发现,可以根据目的攫取任何非标准标志,这就是说,除非
遇到/pt标志,在这种情况下,下面三个非标准标志变量用来初始化打印。为了简化起
见,也可通过不把/pt标志传递给基类中的ParseParam()来禁用/pt标志。
■当然,如果不需继续支持前面所示的标准MFC标志,则可以更加自由地行动。只要不
用调用基类的ParseParam(),可以使用任何标志或非标准标志选项。但是,不要因为能
用非标准标志,而轻易放弃这些标准标志提供的功能。

你可能感兴趣的:(让MFC支持命令行参数)