曲径通幽论坛

 找回密码
 立即注册
搜索
查看: 7379|回复: 0
打印 上一主题 下一主题

[进程,线程,模块] CreateProcess() -- 创建新的进程

[复制链接]

4918

主题

5880

帖子

3万

积分

GROAD

曲径通幽,安觅芳踪。

Rank: 6Rank: 6

积分
34397
跳转到指定楼层
楼主
发表于 2012-3-2 20:39:30 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
CreateProcess() 函数原型如下:
[C++] 纯文本查看 复制代码
BOOL WINAPI CreateProcess(
  __in_opt     LPCTSTR lpApplicationName,
  __inout_opt  LPTSTR lpCommandLine,
  __in_opt     LPSECURITY_ATTRIBUTES lpProcessAttributes,
  __in_opt     LPSECURITY_ATTRIBUTES lpThreadAttributes,
  __in         BOOL bInheritHandles,
  __in         DWORD dwCreationFlags,
  __in_opt     LPVOID lpEnvironment,
  __in_opt     LPCTSTR lpCurrentDirectory,
  __in         LPSTARTUPINFO lpStartupInfo,
  __out        LPPROCESS_INFORMATION lpProcessInformation
);

该函数用来创建一个新的进程。

第 1 个参数 lpApplicationName 是输入参数,指向启动进程的 exe 文件。

第 2 个参数 lpCommandLine 是输入参数,是启动进程的命令行中的参数。

当这两个参数都不为 NULL 时,第 1 个参数指定要启动的进程 exe 文件(不带参数),第 2 个参数指定启动进程所需参数。第 1 个参数也可以为 NULL,此时第 2 个参数就不能为 NULL,在 lpCommandLine 需要指定出要启动的程序名以及所接参数,彼此间以空格隔开,其中第 1 个参数即是程序名。

第 3 个参数 lpProcessAttributes 是输入参数,指向 SECURITY_ATTRIBUTES 结构变量,是进程的安全属性,可以为 NULL 则使用默认的安全属性。

第 4 个参数 lpThreadAttributes 是输入参数,同第 3 个参数一样,指向 SECURITY_ATTRIBUTES 结构变量。

第 5个参数 bInheritHandles 是输入参数,表示新进程是否从调用进程处继承了句柄。如果参数的值为 TRUE,调用进程中的每一个可继承的打开句柄都将被子进程继承。被继承的句柄与原进程拥有完全相同的值和访问权限;如果设为 FALSE,那么不继承。

第 6 个参数 dwCreationFlags 是输入参数,表示进程的创建标志以及优先级控制。如 :CREATE_NEW_CONSOLE 会使新建的控制台程序拥有一个新的控制台;DEBUG_PROCESS 调用进程将被当作一个调试程序,并且新进程会被当作被调试的进程。系统把被调试程序发生的所有调试事件通知给调试器。

第 7 个参数 lpEnvironment 是输入参数,指向新进程的环境变量块,如果设置为 NULL,那么使用父进程的环境变量。

第 8 个参数 lpCurrentDirectory 是输入参数,指定创建后新进程的当前目录,如果设置为 NULL,那么就在父进程所在的当前目录。

第 9 个参数 lpStartupInfo 是输入参数,指向一个 STARTUPINFO 结构,该结构里可以设定启动信息,可以设置为 NULL 。

第 10 个参数 lpProcessInformation 是输出参数,指向一个 PROCESS_INFORMATION 结构,返回被创建进程的信息。

测试代码
[C++] 纯文本查看 复制代码
#include "stdafx.h"
#include <locale.h>
#include <Windows.h>

int _tmain(int argc, _TCHAR* argv[])
{

    PROCESS_INFORMATION ProInfo;    //进程信息结构

    STARTUPINFO    StartInfo;
    ZeroMemory ( &StartInfo, sizeof(StartInfo));

        LPTSTR  szPrameter = TEXT("C:\\Users\\Administrator\\AppData\\Local\\Google\\Chrome\\Application\\chrome.exe [url]www.groad.net[/url]");
    TCHAR szCmdLine[2048] = {0};
    CopyMemory(szCmdLine, szPrameter, 2*_tcslen(szPrameter));

        ZeroMemory (&ProInfo, sizeof(ProInfo));

 if (!CreateProcess ( NULL,        // 执行的程序名
            szCmdLine,                    // 命令行指定
            NULL,                        // 进程安全属性,NULL 时使用默认安全属性
            NULL,                        // 线程安全属性,NULL 时使用默认安全属性
            FALSE,                        // 不继承句柄
            0,                            // 进程创建标志
            NULL,                        // 环境变量块,为 NULL 时使用父进程环境变量
            NULL,                    // 新进程目录
            &StartInfo,                        // 启动信息结构
            &ProInfo)                    // 进程信息结构
        ) {
            _tprintf (TEXT("CreateProcess failed : %d\n"), GetLastError());
            return (-1);
    }

    // 等待子进程结束
    WaitForSingleObject(ProInfo.hProcess, INFINITE);
                    
    CloseHandle ( ProInfo.hProcess );
    CloseHandle ( ProInfo.hThread );

    return 0;
}

上面程序启动谷歌 Chrome 浏览器并打开 www.groad.net 这个主页。在程序中,当第 1 个参数为 NULL 时,要启动的程序以及网址参数均指定在第 2 个参数中。注意,不能直接将参数 szPrameter 直接填写到第 2 个参数中,因为指定的命令行参数中含有空格,这样往往会造成参数解析错误,比如它会被解析成:
C:\\Users\\Administrator\\AppData\\Local\\Google\\Chrome\\Application\\chrome.exe www.groad.net.exe
也就是会认为 chrome.exe www.groad.net.exe 这就是一个程序,这是无法执行的。

而将命令行参数拷贝到数组中却是可行的,但如此一来却要多定义一个数组。如果是英文 Windows 环境,并无需构建多字符程序,那么可以将引号把路径括起来,这也不会出错,比如:
CreateProcess (NULL, "\"Path to exe\" -p1-p2 -p3", ...);
注意上面的路径使用反斜杠转义了括起路径的双引号。

但是,如果在第 1 个参数和第 2 个参数里分别指定程序名和参数,那么也无需添加一个数组来存储命令行参数,但第 2 个参数中指定的命令行参数注意前面要添加一个空格,否则可能无法正确解析参数,比如:
LPTSTR szPname = TEXT("D:\\Program Files (x86)\\Maxthon3\\Bin\\Maxthon.exe");
    LPTSTR szPrameter = TEXT("www.groad.net");     //网址前无空格,错误。正确的形式是添加一个空格
这里只是针对浏览器打开网址这种情况,并非所有的情形都会没问题,比如我测试时,IE 和 MaxThon 是需要在网址面前添加空格的,否则打开的是一个空白页。但对于 Chrome ,不管网址前加不加空格都只能打开空白页,得用上面添加的数组情况方能正常。这也许和浏览器程序本身有关。如果用的是记事打开一些文本,如果后面的文本路径参数不添加空格,那么打开空白,反之正常;而对于 UE ,那么即使后面不添加空格也照样打开,所以从这两个例子看来,仍然是和程序本身有关。但是,添加一个空格会保持较好的兼容性,如若还不行,可以考虑用一个数组来接纳所有的命令行参数。

另外,程序中使用了 WaitForSingleObject() 函数以等待子进程的退出,如当我们关闭了浏览器,那么上面的程序也随之结束,否则一直在那等待。
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|曲径通幽 ( 琼ICP备11001422号-1|公安备案:46900502000207 )

GMT+8, 2025-6-18 13:08 , Processed in 0.087377 second(s), 23 queries .

Powered by Discuz! X3.2

© 2001-2013 Comsenz Inc.

快速回复 返回顶部 返回列表