Win32 API编程:使用CreateProcess创建新进程

1. CreateProcess
说明:
WIN32API函数CreateProcess用来创建一个新的进程和它的主线程,这个新进程运行指定的可执行文件。

函数原型:
BOOL CreateProcess
(
    LPCTSTR lpApplicationName,       
    LPTSTR lpCommandLine,       
    LPSECURITY_ATTRIBUTES lpProcessAttributes。
    LPSECURITY_ATTRIBUTES lpThreadAttributes,       
    BOOL bInheritHandles,       
    DWORD dwCreationFlags,
    LPVOID lpEnvironment,       
    LPCTSTR lpCurrentDirectory,       
    LPSTARTUPINFO lpStartupInfo,       
    LPPROCESS_INFORMATION lpProcessInformation
);

       一个线程调用它来首先创建一个进程内核对象,进程内核对象是用来管理这个新的进程的,然后,系统为新进程创建虚拟地址空间,并将可执行文件(和DLL)的代码和数据加载到这个地址空间,然后系统为新进程的主线程创建一个线程内核对象.

1)先解释下lpApplicationName和lpCommandLine

       lpApplicationName和lpCommandLine分别指向新进程要使用的可执行文件的名称,以及要传给新进程的命令行字符串,lpCommandLine的类型为LPTSTR,这是因为在内部,CreateProcess实际上会修改我们传给他的命令行字符串,当然在它返回前,它会把这个字符串还原,所以这样的代码是错误的:

STARTUPINFO si = {sizeof(si)} ;  
PROCESS_INFORMATION pi ;  
CreateProcess(NULL,TEXT("NOTEPAD"),NULL,NULL,  
    FALSE,0,NULL,NULL,&si,&pi) ;  

解决这个BUG的方法是把常量字符串复制到一个临时缓冲区中,如下所示:

STARTUPINFO si = {sizeof(si)} ;  
    PROCESS_INFORMATION pi ;  
    TCHAR szCommandLine[] = TEXT("NOTEPAD") ;  
    CreateProcess(NULL,szCommandLine,NULL,NULL,  
        FALSE,0,NULL,NULL,&si,&pi) ;  

001:lpApplicationName为NULL,(99%都设为NULL)

    在这种情况下,可执行模块的名字必须处于 lpCommandLine 参数的最前面并由空格符与后面的字符分开,当CreateProcess解析lpCommandLine 字符串时,它会检查字符串中的第一个标记(token),并假记此标记为我们想运行的可执行文件的名称,如果可执行文件的名称没有扩展名,默认为.exe,并且如果文件名不包含一个完整的路径,CreateProcess还会按以下顺序来搜索可执行文件:

(1)主调进程.exe文件所在的目录

(2)主调进程的当前目录

(3)windows系统目录,即GetSystemDirectory返回的System32子文件夹

(4)windows目录

(5)PATH环境变量中列出的目录

002:lpApplicationName不为NULL

       在这种情况下,必须指定文件扩展名,系统不会像1那样自动假定扩展名了,并且如果文件名不包含一个完整的路径,CreateProcess只会在当前目录查找可执行文件,不会在其他任何目录查找了。

 

       一旦系统找到了可执行文件,就创建一个新进程,并将可执行文件的代码和数据映射到新进程的地址空间,然后启动例程,C/C++会将可执行文件名之后的第一个实参的地址传给(w)WinMain的pszCmdLine参数

003  通过CreateProcess()调用的参数说明

BOOL bRet = CreateProcess(
sCmd,
sParam,
NULL,
NULL,
FALSE,
0,
NULL,
NULL,
&si,
&pi);

(1)如果sCmd = "D:\test\abc.exe", sParam = "1 2 3", 则在abc.exe程序中通过GetCommandLine()得到字串 "1 2 3"

(2)如果sCmd = "D:\test\abc.exe",sParam = "D:\test\abc.exe 1 2 3", 则在abc.exe程序中通过GetCommandLine()得到字串 "D:\test\abc.exe 1 2 3"

(3)如果sCmd = "D:\test\abc.exe", sParam = "\"D:\test\abc.exe\" 1 2 3", 则在abc.exe程序中通过GetCommandLine()得到字串 "\"D:\test\abc.exe\" 1 2 3"

总结:在使用CreateProcess()调用程序时

(1).在子进程中通过GetCommandLine()得到的命令行结果跟调用时传入的sParam完全一致。

(2).方式(2)和(3)效果其实是一样的,对进程全路径文件名是否单独加引号,Windows都兼容。只是程序在使用这个参数时,要做脱引号处理。

(3).标准做法是采用方式(3),这样既可以兼容执行程序路径中包含的空格,也可以和ShellExecute()调用保持一致。

 

2)lpProcessAttributes和lpThreadAttributes

      为了创建一个新的进程,系统必须创建一个进程内核对象和新进程的主线程的线程内核对象,这两个参数就标识着这两个内核对象的安全描述符。

3)bInheritHandles就不用说了,设置父子进程的句柄描绘表的继承性

4)dwCreationFlags标识了影响新进程创建方式的标志

C R E AT E _ N O _ W I N D O W 标志用于告诉系统不要为应用程序创建任何控制台窗口。可以使用本标志运行一个没有用户界面的控制台应用程序

值:CREATE_DEFAULT_ERROR_MODE
含义:新的进程不继承调用进程的错误模式。CreateProcess函数赋予新进程当前的默认错误模式作为替代。应用程序可以调用SetErrorMode函数设置当前的默认错误模式。
这个标志对于那些运行在没有硬件错误环境下的多线程外壳程序是十分有用的。
对于CreateProcess函数,默认的行为是为新进程继承调用者的错误模式。设置这个标志以改变默认的处理方式。

值:CREATE_NEW_CONSOLE
含义:新的进程将使用一个新的控制台,而不是继承父进程的控制台。这个标志不能与DETACHED_PROCESS标志一起使用。

值:CREATE_NEW_PROCESS_GROUP
含义:新进程将使一个进程树的根进程。进程树种的全部进程都是根进程的子进程。新进程树的用户标识符与这个进程的标识符是相同的,由lpProcessInformation参数返回。进程树经常使用GenerateConsoleCtrlEvent函数允许发送CTRL+C或CTRL+BREAK信号到一组控制台进程。

值:CREATE_SEPARATE_WOW_VDM
含义:(只适用于Windows NT)这个标志只有当运行一个16位的Windows应用程序时才是有效的。如果被设置,新进程将会在一个私有的虚拟DOS机(VDM)中运行。另外,默认情况下所有的16位Windows应用程序都会在同一个共享的VDM中以线程的方式运行。单独运行一个16位程序的优点是一个应用程序的崩溃只会结束这一个VDM的运行;其他那些在不同VDM中运行的程序会继续正常的运行。同样的,在不同VDM中运行的16位Windows应用程序拥有不同的输入队列,这意味着如果一个程序暂时失去响应,在独立的VDM中的应用程序能够继续获得输入。

值:CREATE_SHARED_WOW_VDM
含义:(只适用于Windows NT)这个标志只有当运行一个16位的Windows应用程序时才是有效的。如果WIN.INI中的Windows段的DefaultSeparateVDM选项被设置为真,这个标识使得CreateProcess函数越过这个选项并在共享的虚拟DOS机中运行新进程。

值:CREATE_SUSPENDED
含义:新进程的主线程会在创建后被挂起,直到调用ResumeThread函数被调用时才运行,这样一来,父进程就可以修改子进程地址空间中的内存,更改子进程的主线程的优先级,或者在进程执行任何代码前,把它加入到一个作业中,父进程修改好子进程后,再调用ResumeThread来允许子进程执行代码。

值:CREATE_UNICODE_ENVIRONMENT
含义:如果被设置,由lpEnvironment参数指定的环境块使用Unicode字符,如果为空,环境块使用ANSI字符。

值:DEBUG_PROCESS
含义:如果这个标志被设置,调用进程将被当作一个调试程序,并且新进程会被当作被调试的进程。系统把被调试程序发生的所有调试事件通知给调试器。
如果你使用这个标志创建进程,只有调用进程(调用CreateProcess函数的进程)可以调用WaitForDebugEvent函数。

值:DEBUG_ONLY_THIS_PROCESS
含义:如果此标志没有被设置且调用进程正在被调试,新进程将成为调试调用进程的调试器的另一个调试对象。如果调用进程没有被调试,有关调试的行为就不会产生。

值:DETACHED_PROCESS
含义:对于控制台进程,新进程没有访问父进程控制台的权限。新进程可以通过AllocConsole函数自己创建一个新的控制台。这个标志不可以与CREATE_NEW_CONSOLE标志一起使用。

dwCreationFlags参数还用来控制新进程的优先类,优先类用来决定此进程的线程调度的优先级。如果下面的优先级类标志都没有被指定,那么默认的优先类是NORMAL_PRIORITY_CLASS,除非被创建的进程是IDLE_PRIORITY_CLASS。在这种情况下子进程的默认优先类是IDLE_PRIORITY_CLASS。
可以下面的标志中的一个:

优先级:HIGH_PRIORITY_CLASS       
含义:指示这个进程将执行时间临界的任务,所以它必须被立即运行以保证正确。这个优先级的程序优先于正常优先级或空闲优先级的程序。一个例子是Windows任务列表,为了保证当用户调用时可以立刻响应,放弃了对系统负荷的考虑。确保在使用高优先级时应该足够谨慎,因为一个高优先级的CPU关联应用程序可以占用几乎全部的CPU可用时间。

优先级:IDLE_PRIORITY_CLASS       
含义:指示这个进程的线程只有在系统空闲时才会运行并且可以被任何高优先级的任务打断。例如屏幕保护程序。空闲优先级会被子进程继承。

优先级:NORMAL_PRIORITY_CLASS       
含义:指示这个进程没有特殊的任务调度要求。

优先级:REALTIME_PRIORITY_CLASS       
含义:指示这个进程拥有可用的最高优先级。一个拥有实时优先级的进程的线程可以打断所有其他进程线程的执行,包括正在执行重要任务的系统进程。例如,一个执行时间稍长一点的实时进程可能导致磁盘缓存不足或鼠标反映迟钝。

 

5)lpEnvironment:

 

        指向一个新进程的环境块。如果此参数为空,新进程使用调用进程的环境。
  一个环境块存在于一个由以NULL结尾的字符串组成的块中,这个块也是以NULL结尾的。每个字符串都是name=value的形式。
  因为相等标志被当作分隔符,所以它不能被环境变量当作变量名。
  与其使用应用程序提供的环境块,不如直接把这个参数设为空,系统驱动器上的当前目录信息不会被自动传递给新创建的进程。对于这个情况的探讨和如何处理,请参见注释一节。
  环境块可以包含Unicode或ANSI字符。如果lpEnvironment指向的环境块包含Unicode字符,那么dwCreationFlags字段的CREATE_UNICODE_ENVIRONMENT标志将被设置。如果块包含ANSI字符,该标志将被清空。
  请注意一个ANSI环境块是由两个零字节结束的:一个是字符串的结尾,另一个用来结束这个快。一个Unicode环境块石油四个零字节结束的:两个代表字符串结束,另两个用来结束块。

 

6)lpCurrentDirectory

    指向一个以NULL结尾的字符串,这个字符串用来指定子进程的工作路径。这个字符串必须是一个包含驱动器名的绝对路径。如果这个参数为NULL,新进程将使用与调用进程相同的驱动器和目录。这个选项是一个需要启动启动应用程序并指定它们的驱动器和工作目录的外壳程序的主要条件。

7) lpStartupInfo

typedef struct _STARTUPINFO   
{   
DWORD cb;  
LPTSTR lpReserved;    
LPTSTR lpDesktop;   
LPTSTR lpTitle;    
DWORD dwX;    
DWORD dwY;    
DWORD dwXSize;  
DWORD dwYSize;    
DWORD dwXCountChars;    
DWORD dwYCountChars;    
DWORD dwFillAttribute;    
DWORD dwFlags;    
WORD wShowWindow;    
WORD cbReserved2;    
LPBYTE lpReserved2;    
HANDLE hStdInput;    
HANDLE hStdOutput;    
HANDLE hStdError;  
} STARTUPINFO, *LPSTARTUPINFO;  

指向一个用于决定新进程的主窗体如何显示的STARTUPINFO结构体。

      大多数应用程序都希望生成的应用程序只是使用默认值,最起码要全部初始化为0,再把cb成员设为此结构体的大小,如果没有清0,则新进程可能创建失败.

 

表4-6 STARTUPINFO 结构的成员

成员窗口,控制台还是两者兼有作用
cb两者兼有包含S TA RT U P I N F O 结构中的字节数。如果M i c r o s o f t 将来扩展该结构,它可用作版本控制手段。应用程序必须将c b 初始化为s i z e o f ( S TA RT U P I N F O )
lpReserved两者兼有保留。必须初始化为N U L L
lpDesktop两者兼有用于标识启动应用程序所在的桌面的名字。如果该桌面存在,新进程便与指定的桌面相关联。如果桌面不存在,便创建一个带有默认属性的桌面,并使用为新进程指定的名字。如果l p D e s k t o p 是N U L L (这是最常见的情况),那么该进程将与当前桌面相关联
lpTitle控制台用于设定控制台窗口的名称。如果l p Ti t l e 是N U L L ,则可执行文件的名字将用作窗口名
dwX
dwY
两者兼有用于设定应用程序窗口在屏幕上应该放置的位置的x 和y 坐标(以像素为单位)。只有当子进程用C W _ U S E D E FA U LT 作为C r e a t e Wi n d o w 的x 参数来创建它的第一个重叠窗口时,才使用这两个坐标。若是创建控制台窗口的应用程序,这些成员用于指明控制台窗口的左上角
dwXSize两者兼有用于设定应用程序窗口的宽度和长度(以像素为单位)只有dwYsize 当子进程将C W _ U S E D E FA U LT 用作C r e a t e Wi n d o w 的n Wi d t h参数来创建它的第一个重叠窗口时,才使用这些值。若是创建控制台窗口的应用程序,这些成员将用于指明控制台窗口的宽度
dwXCountChars
dwYCountChars
控制台用于设定子应用程序的控制台窗口的宽度和高度(以字符为单位)
dwFillAttribute控制台用于设定子应用程序的控制台窗口使用的文本和背景颜色
dwFlags两者兼有请参见下一段和表4 - 7 的说明
wShowWindow窗口用于设定如果子应用程序初次调用的S h o w Wi n d o w 将S W _ S H O W D E FA U LT 作为n C m d S h o w 参数传递时,该应用程序的第一个重叠窗口应该如何出现。本成员可以是通常用于Show Wi n d o w 函数的任何一个S W _ *标识符
cbReserved2两者兼有保留。必须被初始化为0
lpReserved2两者兼有保留。必须被初始化为N U L L
hStdInput
hStdOutput
hStdError
控制台用于设定供控制台输入和输出用的缓存的句柄。按照默认设置,h S t d I n p u t 用于标识键盘缓存,h S t d O u t p u t 和h S t d E r r o r用于标识控制台窗口的缓存

 

2. 实例 1

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

#include <windows.h>

#include <tchar.h>

#include <stdio.h>

 

int main(int argc, char* argv[])

{

    TCHAR szCommandLine[] = TEXT("NOTEPAD");//或者WCHAR

    //LPWSTR szCommandLine = TEXT("NOTEPAD");//错误

    //STARTUPINFO si = { sizeof(si) };

    STARTUPINFO si;

    PROCESS_INFORMATION pi;

    ZeroMemory(&si, sizeof(si));

    si.cb = sizeof(si);

    ZeroMemory(&pi, sizeof(pi));

 

    si.dwFlags = STARTF_USESHOWWINDOW;  // 指定wShowWindow成员有效

    si.wShowWindow = TRUE;          // 此成员设为TRUE的话则显示新建进程的主窗口,

                        // 为FALSE的话则不显示

    BOOL bRet = ::CreateProcess (

        NULL,           // 不在此指定可执行文件的文件名

        szCommandLine,      // 命令行参数

        NULL,           // 默认进程安全性

        NULL,           // 默认线程安全性

        FALSE,          // 指定当前进程内的句柄不可以被子进程继承

        CREATE_NEW_CONSOLE, // 为新进程创建一个新的控制台窗口

        NULL,           // 使用本进程的环境变量

        NULL,           // 使用本进程的驱动器和目录

        &si,

        &pi);

 

    if(bRet)

    {

        WaitForSingleObject(pi.hProcess, INFINITE);

        // 既然我们不使用两个句柄,最好是立刻将它们关闭

        ::CloseHandle (pi.hThread);

        ::CloseHandle (pi.hProcess);

 

        printf(" 新进程的进程ID号:%d \n", pi.dwProcessId);

        printf(" 新进程的主线程ID号:%d \n", pi.dwThreadId);

    }

    return 0;

}

在调用CreatePricess的前两个参数:pszApplicationName和pszCommandLine时可能崩溃的问题:

pszApplicationName和pszCommandLine参数分别指定新进程要使用的执行体文件的名称,以及要传给新进程的命令行字符串。先来谈谈pszCommandLine参数。
注意,pszCommandLine参数被原型化为一个PTSTR。这意味着CreateProces期望你传入的是一个非“常量字符串”的地址。在内部,CreateProcess实际上会修改你传给它的命令行字符串。但在CreateProcess返回之前,它会将这个字符串还原为原来的形式。
这是很重要的,因为如果命令行字符串包含在你的文件映像的只读部分,就会引起访问冲突(违例)。例如,以下代码就会导致冲突,因为Microsoft的C/C++编译器把"NOTEPAD"字符串放在只读内存中:

1

2

3

4

STARTUPINFO si = { sizeof(si) };

PROCESS_INFORMATION pi;

CreateProcess(NULL, TEXT("NOTEPAD"), NULL, NULL,

FALSE, 0, NULL, NULL, &si, &pi);

CreateProcess试图修改字符串时,会引起一个访问冲突(Microsoft C/C++编译器的早期版本把字符串放在可读/写内存中。所以对CreateProcess函数的调用不会引起访问冲突)。

解决这个问题的最佳方式是:在调用CreateProcess之前,把常量字符串复制到一个临时缓冲区,如下所示:
 

1

2

3

TCHAR szCommandLine[] = TEXT("NOTEPAD");

CreateProcess(NULL, szCommandLine, NULL, NULL,

FALSE, 0, NULL, NULL, &si, &pi);

涉及的字符关系:

char* 替换: LPSTR
const char* 替换: LPCSTR
WCHAR* 替换: LPWSTR
const WCHAR* 替换: LPCWSTR (C在W之前, 因为 const 在 WCHAR之前)
TCHAR* 替换: LPTSTR
const TCHAR* 替换: LPCTSTR

 

3. 实例2

原文:https://blog.csdn.net/yuyan987/article/details/78644922 

如果要执行c:\\program files\\internet explorer\\iexplore.exe并传入命令行"http://community.csdn.net/"(用IE打开http://community.csdn.net网址),一般有两种方法。


方法1:
#include "stdafx.h"
#include <windows.h>
#include <stdio.h>
 
void RunExe()
{
    STARTUPINFO stStartUpInfo;
    ::memset(&stStartUpInfo, 0 ,sizeof(stStartUpInfo));
    stStartUpInfo.cb = sizeof(stStartUpInfo);
 
    PROCESS_INFORMATION stProcessInfo;
    ::memset(&stProcessInfo, 0 ,sizeof(stProcessInfo));
 
    TCHAR szPath[] = _T("c:\\program files\\internet explorer\\iexplore.exe");
    TCHAR szCmd[] = _T(" http://community.csdn.net/");   //lpCommandLine的内容中开头需要一个空格,不然就和lpApplicationName连在一起去了
    try
    {
        bool bRet = ::CreateProcess(
            szPath,
            szCmd,
            NULL,
            NULL,
            false,
            CREATE_NEW_CONSOLE,
            NULL,
            NULL,
            &stStartUpInfo,
            &stProcessInfo);
 
        if (bRet)
        {
            //等待3s后关闭进程
            WaitForSingleObject(stProcessInfo.hProcess,3000L);
            ::CloseHandle(stProcessInfo.hProcess);
            ::CloseHandle(stProcessInfo.hThread);
            stProcessInfo.hProcess = NULL;
            stProcessInfo.hThread = NULL;
            stProcessInfo.dwProcessId = 0;
            stProcessInfo.dwThreadId = 0;
        }
        else
        {
            //如果创建进程失败,查看错误码
            DWORD dwErrCode = GetLastError();
            printf_s("ErrCode : %d\n",dwErrCode);
 
        }
    }
    catch( ... )
    {
    }
}
 
int main(int argc, char* argv[])   
{   
    RunExe();
 
    system("pause");
    return 0;   
}   

方法2:

 #include "stdafx.h"
#include <windows.h>
#include <stdio.h>
 
void RunExe()
{
    STARTUPINFO stStartUpInfo;
    ::memset(&stStartUpInfo, 0 ,sizeof(stStartUpInfo));
    stStartUpInfo.cb = sizeof(stStartUpInfo);
 
    PROCESS_INFORMATION stProcessInfo;
    ::memset(&stProcessInfo, 0 ,sizeof(stProcessInfo));
 
    
    TCHAR szCmd[] = _T("\"c:\\program files\\internet explorer\\iexplore.exe\" http://community.csdn.net/");
 
    try
    {
        bool bRet = ::CreateProcess(
            NULL,
            szCmd,
            NULL,
            NULL,
            false,
            CREATE_NEW_CONSOLE,
            NULL,
            NULL,
            &stStartUpInfo,
            &stProcessInfo);
 
        if (bRet)
        {
            //等待3s后关闭进程
            WaitForSingleObject(stProcessInfo.hProcess,3000L);
            ::CloseHandle(stProcessInfo.hProcess);
            ::CloseHandle(stProcessInfo.hThread);
            stProcessInfo.hProcess = NULL;
            stProcessInfo.hThread = NULL;
            stProcessInfo.dwProcessId = 0;
            stProcessInfo.dwThreadId = 0;
        }
        else
        {
            //如果创建进程失败,查看错误码
            DWORD dwErrCode = GetLastError();
            printf_s("ErrCode : %d\n",dwErrCode);
 
        }
    }
    catch( ... )
    {
    }
}
 
int main(int argc, char* argv[])   
{   
    RunExe();
 
    system("pause");
    return 0;   
}   


3. 关于退出

如果想创建一个新进程,并等待结果,可用以下类似代码:

 

PROCESS_INFORMATION pi;  
    DWORD dwExitCode;  
  
    //Spawn the child process.  
    BOOL fSuccess = CreateProcess(..., π);  
  
    if(fSuccess)  
    {  
        //Close the thread handle as soon as  
        //it is no longer needed!  
        CloseHandle(pi.hThread);  
  
        //Suspend our execution until  
        //the child has terminated.  
        WaitForSingleObject(pi.hProcess,INFINITE);  
  
        //The child process terminated;  
        //get its exit code.  
        GetExitCodeProcess(pi.hProcess,  
            &dwExitCode);  
  
        //Close the process handle as soon as  
        //it is no longer needed.  
        CloseHandle(pi.hProcess);  
    }  

只有当进程对象终止运行时,WaitForSingleObject才能得到通知,因此对WaitForSingleObject的调用会将父进程的线程挂起,直到子进程终止运行,当WaitForSingleObject返回时,通过GetExitCodeProcess,就可以得到子进程的退出码

 

注释:
     1. CreateProcess函数用来运行一个新程序。WinExec和LoadModule函数依旧可用,但是它们同样通过调用CreateProcess函数实现。

    2. 另外CreateProcess函数除了创建一个进程,还创建一个线程对象。这个线程将连同一个已初始化了的堆栈一起被创建,堆栈的大小由可执行文件的文件头中的描述决定。线程由文件头处开始执行。

    3.  新进程和新线程的句柄被以全局访问权限创建。对于这两个句柄中的任一个,如果没有安全描述符,那么这个句柄就可以在任何需要句柄类型作为参数的函数中被使用。当提供安全描述符时,在接下来的时候当句柄被使用时,总是会先进行访问权限的检查,如果访问权限检查拒绝访问,请求的进程将不能使用这个句柄访问这个进程。

    4. 这个进程会被分配给一个32位的进程标识符。直到进程中止这个标识符都是有效的。它可以被用来标识这个进程,或在OpenProcess函数中被指定以打开这个进程的句柄。进程中被初始化了的线程一样会被分配一个32位的线程标识符。这个标识符直到县城中止都是有效的且可以用来在系统中唯一标识这个线程。这些标识符在PROCESS_INFORMATION结构体中返回。

  5. 当在lpApplicationName或lpCommandLine参数中指定应用程序名时,应用程序名中是否包含扩展名都不会影响运行,只有一种情况例外:一个以.com为扩展名的MS-DOS程序或Windows程序必须包含.com扩展名。

   6. 调用进程可以通过WaitForInputIdle函数来等待新进程完成它的初始化并等待用户输入。这对于父进程和子进程之间的同步是极其有用的,因为CreateProcess函数不会等待新进程完成它的初始化工作。举例来说,在试图与新进程关联的窗口之前,进程应该先调用WaitForInputIdle。

  7. 首选的结束一个进程的方式是调用ExitProcess函数,因为这个函数通知这个进程的所有动态链接库(DLLs)程序已进入结束状态。其他的结束进程的方法不会通知关联的动态链接库。注意当一个进程调用ExitProcess时,这个进程的其他县城没有机会运行其他任何代码(包括关联动态链接库的终止代码)。

  8. ExitProcess, ExitThread, CreateThread, CreateRemoteThread,当一个进程启动时(调用了CreateProcess的结果)是在进程中序列化进行的。在一段地址空间中,同一时间内这些事件中只有一个可以发生。这意味着下面的限制将保留:
  *在进程启动和DLL初始化阶段,新的线程可以被创建,但是直到进程的DLL初始化完成前它们都不能开始运行。
  *在DLL初始化或卸下例程中进程中只能有一个线程。
  *直到所有的线程都完成DLL初始化或卸下后,ExitProcess函数才返回。

  9.  在进程中的所有线程都终止且进程所有的句柄和它们的线程被通过调用CloseHandle函数终止前,进程会留在系统中。进程和主线程的句柄都必须通过调用CloseHandle函数关闭。如果不再需要这些句柄,最好在创建进程后立刻关闭它们。

10.  当进程中最后一个线程终止时,下列的事件发生:
  *所有由进程打开的对象都会关闭。
  *进程的终止状态(由GetExitCodeProcess函数返回)从它的初始值STILL_ACTIVE变为最后一个结束的线程的结束状态。
  *主线程的线程对象被设置为标志状态,供其他等待这个对象的线程使用。
  *进程对象被设置为标志状态,供其他等待这个对象的线程使用。

假设当前在C盘上的目录是/MSVC/MFC且有一个环境变量叫做C:,它的值是C:/MSVC/MFC,就像前面lpEnvironment中提到过的那样,这样的系统驱动器上的目录信息在CreateProcess函数的lpEnvironment参数不为空时不会被自动传递到新进程里。一个应用程序必须手动地把当前目录信息传递到新的进程中。为了这样做,应用程序必须直接创建环境字符串,并把它们按字母顺序排列(因为Windows NT和Windows 95使用一种简略的环境变量),并把它们放进lpEnvironment中指定的环境块中。类似的,他们要找到环境块的开头,又要重复一次前面提到的环境块的排序。

一种获得驱动器X的当前目录变量的方法是调用GetFullPathName("x:",..)。这避免了一个应用程序必须去扫描环境块。如果返回的绝对路径是X:/,就不需要把这个值当作一个环境数据去传递了,因为根目录是驱动器X上的新进程的默认当前目录。


11.  由CreateProcess函数返回的句柄对于进程对象具有PROCESS_ALL_ACCESS的访问权限。

由lpcurrentDirectory参数指定的当前目录室子进程对象的当前目录。lpCommandLine参数指定的第二个项目是父进程的当前目录。

12.  对于Windows NT,当一个进程在指定了CREATE_NEW_PROCESS_GROUP的情况下被创建时,一个对于SetConsoleCtrlHandler(NULL,True)的调用被用在新的进程上,这意味着对新进程来说CTRL+C是无效的。这使得上层的外科程序可以自己处理CTRL+C信息并有选择的把这些信号传递给子进程。CTRL+BREAK依旧有效,并可被用来中断进程/进程树的执行。

 

注意:

若是创建的进程窗口想无窗口的话有以下两种方式

1. CreateProcess(
        szPath, 
        szTmp,
        NULL,
        NULL, 
        true, 
        CREATE_NO_WINDOW
        NULL, 
        NULL, 
        &stStartUpInfo,
        &stProcessInfo);  第5个参数为 CREATE_NO_WINDOW

2.  第8个参数设置一下

 stStartUpInfo.dwFlags=STARTF_USESHOWWINDOW;    
    stStartUpInfo.wShowWindow=SW_HIDE; 

  • 0
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
使用 C# 调用 Win32 API 创建进程并返回进程句柄的详细方法如下: 1. 首先需要设置一个 StartInfo 对象,用于存储启动进程的相关信息,如进程名称、命令行参数等。 ```csharp ProcessStartInfo startInfo = new ProcessStartInfo(); startInfo.FileName = "notepad.exe"; // 进程名称 startInfo.Arguments = "test.txt"; // 命令行参数 ``` 2. 创建一个 Process 对象,并将 StartInfo 对象赋值给它。 ```csharp Process process = new Process(); process.StartInfo = startInfo; ``` 3. 调用 Win32 APICreateProcess 方法,创建进程并返回进程句柄。 ```csharp bool success = CreateProcess( startInfo.FileName, startInfo.Arguments, IntPtr.Zero, IntPtr.Zero, false, 0, IntPtr.Zero, null, ref startupInfo, out PROCESS_INFORMATION processInfo); if (success) { process.Handle = processInfo.hProcess; process.Id = processInfo.dwProcessId; } ``` 其中,CreateProcess 方法的参数含义如下: - lpApplicationName:要执行的可执行文件的名称,如果指定了完整路径,则 lpCommandLine 参数可以为 null。 - lpCommandLine:要传递给可执行文件的命令行参数。 - lpProcessAttributes:进程句柄的安全属性,通常设置为 IntPtr.Zero。 - lpThreadAttributes:线程句柄的安全属性,通常设置为 IntPtr.Zero。 - bInheritHandles:指定是否从父进程继承句柄,通常设置为 false。 - dwCreationFlags:指定进程创建标志,通常设置为 0。 - lpEnvironment:进程的环境变量,通常设置为 IntPtr.Zero。 - lpCurrentDirectory:进程的当前工作目录,通常设置为 null。 - lpStartupInfo:启动信息,包括窗口大小、标题等。 - lpProcessInformation:输出参数,包括进程句柄和进程 ID。 4. 最后可以使用 Process 对象的方法来操作创建进程,如 WaitForExit() 等。 ```csharp process.WaitForExit(); ``` 完整的代码示例: ```csharp using System; using System.Diagnostics; using System.Runtime.InteropServices; namespace CreateProcessDemo { class Program { static void Main(string[] args) { // 设置启动信息 ProcessStartInfo startInfo = new ProcessStartInfo(); startInfo.FileName = "notepad.exe"; startInfo.Arguments = "test.txt"; // 创建进程并返回进程句柄 Process process = new Process(); process.StartInfo = startInfo; bool success = CreateProcess( startInfo.FileName, startInfo.Arguments, IntPtr.Zero, IntPtr.Zero, false, 0, IntPtr.Zero, null, ref startupInfo, out PROCESS_INFORMATION processInfo); if (success) { process.Handle = processInfo.hProcess; process.Id = processInfo.dwProcessId; // 等待进程结束 process.WaitForExit(); } } // 声明 CreateProcess 方法 [DllImport("kernel32.dll", SetLastError = true, CharSet = CharSet.Unicode)] static extern bool CreateProcess( string lpApplicationName, string lpCommandLine, IntPtr lpProcessAttributes, IntPtr lpThreadAttributes, bool bInheritHandles, uint dwCreationFlags, IntPtr lpEnvironment, string lpCurrentDirectory, ref STARTUPINFO lpStartupInfo, out PROCESS_INFORMATION lpProcessInformation); // 定义 STARTUPINFO 结构体 [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)] struct STARTUPINFO { public uint cb; public string lpReserved; public string lpDesktop; public string lpTitle; public uint dwX; public uint dwY; public uint dwXSize; public uint dwYSize; public uint dwXCountChars; public uint dwYCountChars; public uint dwFillAttribute; public uint dwFlags; public short wShowWindow; public short cbReserved2; public IntPtr lpReserved2; public IntPtr hStdInput; public IntPtr hStdOutput; public IntPtr hStdError; } // 定义 PROCESS_INFORMATION 结构体 [StructLayout(LayoutKind.Sequential)] struct PROCESS_INFORMATION { public IntPtr hProcess; public IntPtr hThread; public uint dwProcessId; public uint dwThreadId; } } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值