您的位置:首页 > 其它

获取和更改环境变量的值SetEnvironmentVariable()

2012-09-26 17:49 489 查看
#include <windows.h>
#include <stdio.h>
#include <tchar.h>
void DumpEnvVariables(PTSTR pEnvBlock[]);
void myDumpEnvVariables0(PTSTR pEnvBlock[]);
void myDumpEnvVariables1(PTSTR pEnvBlock[]);
void PrintEnvironmentVariable(PCTSTR pszVariableName);
int _tmain(int argc,TCHAR *argv[],TCHAR *envp[])
{
	DumpEnvVariables(envp);
	//myDumpEnvVariables0(envp);
	//myDumpEnvVariables1(envp);
	//输出指定环境变量和值
	PrintEnvironmentVariable(L"TEMP");
	int  i;
	scanf("%S",&i);//  接受键盘数据继续
	return 0;
}
void PrintEnvironmentVariable(PCTSTR pszVariableName) { 
	PTSTR pszValue = NULL; 
	// Get the size of the buffer that is required to store the value 
	DWORD dwResult = GetEnvironmentVariable(pszVariableName, pszValue, 0); 
	if (dwResult != 0) { 
		// Allocate the buffer to store the environment variable value 
		DWORD size = dwResult * sizeof(TCHAR); 
		pszValue = (PTSTR)malloc(size); 
		GetEnvironmentVariable(pszVariableName, pszValue, size); 
		_tprintf(TEXT("%s=%s\n"), pszVariableName, pszValue); 
		free(pszValue); 
	}
	else {
	 _tprintf(TEXT("'%s'=<unknown value>\n"), pszVariableName); 
	}
}
void DumpEnvVariables(PTSTR pEnvBlock[])
{
	int current =0;
	PTSTR *pElement=pEnvBlock;
	PTSTR pCurrent=NULL;     //初始化为NULL,避免成为野指针
	while(pElement!=NULL)
	{
		pCurrent=(PTSTR)(*pElement);
		if (pCurrent==NULL) //如果是最后一项,则将pElement赋值为NULL,使循环退出
		{
			pElement=NULL; //循环退出的条件
		}                  
		else
		{
			_tprintf(TEXT("[%u] %s\r\n"),current,pCurrent);
			current++;
			pElement++;
		}
	}
}
void myDumpEnvVariables0(PTSTR pEnvBlock[]) //自己改写的版本,避免了循环中的if判断
{
	int current=0;
	PTSTR *pElement=pEnvBlock;
	PTSTR pCurrent=NULL;
	pCurrent=*pElement;    //先赋值一次

	_tprintf(TEXT("\r\n getEnv by myDumpEnvVariables0\r\n"));
	while(pCurrent!=NULL)   //用pCurrent作为循环判断的变量
	{

		_tprintf(TEXT("[%u] %s\r\n"),current,pCurrent);
		pElement++;
		current++;
		pCurrent=*pElement;
	}
}
void myDumpEnvVariables1(PTSTR pEnvBlock[])//自己改写的版本,使程序更简洁,更易理解
{
	int current=0;
	PTSTR *pElement=pEnvBlock;

	_tprintf(TEXT("\r\n getEnv by myDumpEnvVariables1\r\n"));
	while(*pElement!=NULL)   //用pCurrent作为循环判断的变量
	{

		_tprintf(TEXT("[%u] %s\r\n"),current,*pElement);
		pElement++;
		current++;

	}

}


环境变量是进程中一组变量信息,环境变量分为系统环境变量、用户环境变量和进程环境变量。系统有全局的环境变量,在进程创建时,进程继承了系统的全局环境变量、当前登录用户的用户环境变量和父进程的环境变量。进程也可以有自己的环境变量。

设置和获取所在进程的环境变量使用API函数GetEnvironmentStrings、GetEnvironmentVariable和SetEnvironmentVariable:

1)GetEnvironmentStrings函数用于获取所有环境变量字符串:

LPTCH WINAPI GetEnvironmentStrings(void);

返回值:

成功时,返回指向保存环境变量的缓冲区;

失败时,返回值为NULL。

2)FreeEnvironmentStrings函数用来释放由GetEnvironmentStrings返回的内存块:

BOOL WINAPI FreeEnvironmentStrings(

__in LPTCH lpszEnvironmentBlock

);

返回值:

成功时,返回非零值;

失败时,返回零值,可调用GetLastError()查看进一步错误消息。

3)GetEnvironmentVariable函数用于获取指定的环境变量:

DWORD WINAPI GetEnvironmentVariable(

__in_opt LPCTSTR lpName, //环境变量名

__out_opt LPTSTR lpBuffer, //指向保存环境变量值的缓冲区

__in DWORD nSize //缓冲区大小(字符数)

);

返回值:

成功时,返回真实的环境变量值大小,不包括null结束符;

如果lpBuffer大小不足,则返回值是实际所需的字符数大小,lpBuffer内容未定义;

失败时,返回0;如果指定的环境变量找不到,GetLastError()返回ERROR_ENVVAR_NOT_FOUND。

4)SetEnvironmentVariable函数用于设置指定的环境变量:

BOOL WINAPI SetEnvironmentVariable(

__in LPCTSTR lpName, //环境变量名,当该值不存在且lpValue不为NULL时,将创建一个新的

__in_opt LPCTSTR lpValue //环境变量值

);

返回值:

成功时,返回非零值;

失败时,返回零值,调用GetLastError()查看具体的错误信息。

该函数对系统环境变量以及其他进程的环境变量不起作用!

实例一:调用GetEnvironmentStrings函数检索进程的环境变量内存块,并将其中内容打印到控制台:

#include<windows.h>

#include<tchar.h>

#include<stdio.h>

int _tmain()

{

LPTSTR lpszVariable;

LPTCH lpvEnv;

//获得环境变量内存块的指针

lpvEnv = GetEnvironmentStrings();

if(lpvEnv == NULL)

{

printf("GetEnvironmentStrins failed(%d)/n", GetLastError());

return 0;

}

//环境变量字符串是以NULL分隔的,内存块以NULL结尾

lpszVariable = (LPTSTR)lpvEnv;

while(*lpszVariable)

{

_tprintf(TEXT("%s/n"), lpszVariable);

lpszVariable += lstrlen(lpszVariable) + 1; //移动指针

}

FreeEnvironmentStrings(lpvEnv);

system("pause");

return 1;

}

实例二:默认情况下,子进程继承父进程环境变量内存块的一份拷贝;下面代码通过调用CreateProcess函数实现将一个环境变量块传递给子进程(asce.exe就是实例一编译链接得到的exe文件,因此,该代码的运行结果就是子进程打印从父进程继承而来的环境变量):

#include<Windows.h>

#include<tchar.h>

#include<stdio.h>

#include<strsafe.h>

#define BUFSIZE 4096

int _tmain()

{

TCHAR chNewEnv[BUFSIZE];

LPTSTR lpszCurrentVariable;

DWORD dwFlags = 0;

TCHAR szAppName[] = TEXT("asce.exe");

STARTUPINFO si;

PROCESS_INFORMATION pi;

BOOL fSuccess;

//将环境变量字符串拷贝到环境变量内存块中

lpszCurrentVariable = (LPTSTR)chNewEnv;

if(FAILED(StringCchCopy(lpszCurrentVariable, BUFSIZE, TEXT("AsceSetting=Luffy"))))

{

printf("String copy failed/n");

return FALSE;

}

lpszCurrentVariable += lstrlen(lpszCurrentVariable) + 1;

if(FAILED(StringCchCopy(lpszCurrentVariable, BUFSIZE, TEXT("AsceVersion=2.0"))))

{

printf("String copy failed/n");

return FALSE;

}

//使环境变量内存块以NULL结尾

lpszCurrentVariable += lstrlen(lpszCurrentVariable) + 1;

*lpszCurrentVariable = (TCHAR)0;

//创建子进程,指定一个新的环境变量内存块

SecureZeroMemory(&si, sizeof(STARTUPINFO));

si.cb = sizeof(STARTUPINFO);

#ifdef UNICODE

dwFlags = CREATE_UNICODE_ENVIRONMENT;

#endif

fSuccess = CreateProcess(szAppName, NULL, NULL, NULL,

TRUE, dwFlags, (LPVOID)chNewEnv, //新的环境变量内存块

NULL, &si, &pi);

if(!fSuccess)

{

printf("CreateProcess failed(%d)/n", GetLastError());

return FALSE;

}

WaitForSingleObject(pi.hProcess, INFINITE);

system("pause");

return TRUE;

}

实例三:在子进程创建过程中改变子进程的环境变量是一个进程改变另一个进程环境变量的唯一方式。一个进程绝不能直接改变另一个进程(非子进程)的环境变量。下面代码实现子进程继承符进程环境变量的方法:

#include<Windows.h>

#include<tchar.h>

#include<stdio.h>

#define BUFSIZE 4096

#define VARNAME TEXT("AsceVariable")

int _tmain()

{

DWORD dwRet, dwErr;

LPTSTR pszOldVal;

TCHAR szAppName[] = TEXT("asce.exe");

DWORD dwFlags = 0;

STARTUPINFO si;

PROCESS_INFORMATION pi;

BOOL fExist, fSuccess;

// Retrieves the current value of the variable if it exists.

// Sets the variable to a new value, creates a child process,

// then uses SetEnvironmentVariable to restore the original

// value or delete it if it did not exist previously.

pszOldVal = (LPTSTR)malloc(BUFSIZE*sizeof(TCHAR));

if(NULL == pszOldVal)

{

printf("Out of memory/n");

return FALSE;

}

dwRet = GetEnvironmentVariable(VARNAME, pszOldVal, BUFSIZE);

if(0 == dwRet)

{

dwErr = GetLastError();

if(ERROR_ENVVAR_NOT_FOUND == dwErr)

{

printf("Environment variable does not exist/n");

fExist = FALSE;

}

}

elseif(BUFSIZE < dwRet)

{

pszOldVal = (LPTSTR)realloc(pszOldVal, dwRet*sizeof(TCHAR));

if(NULL == pszOldVal)

{

printf("Out of memory/n");

return FALSE;

}

dwRet = GetEnvironmentVariable(VARNAME, pszOldVal, dwRet);

if(!dwRet)

{

printf("GetEnvironmentVariable failed(%d)/n", GetLastError());

return FALSE;

}

else

fExist = TRUE;

}

else

fExist = TRUE;

//Set a value for the child process to inherit

if(!SetEnvironmentVariable(VARNAME, TEXT("ASCE")))

{

printf("SetEnvironmentVariable failed(%d)/n", GetLastError());

return FALSE;

}

//Create a child process

SecureZeroMemory(&si, sizeof(STARTUPINFO));

si.cb = sizeof(STARTUPINFO);

#ifndef UNICODE

dwFlags = CREATE_UNICODE_ENVIRONMENT;

#endif

fSuccess = CreateProcess(szAppName, NULL, NULL, NULL,

TRUE, dwFlags, NULL,//inherit parent's environment

NULL, &si, &pi);

if(!fSuccess)

{

printf("CreateProcess failed(%d)/n", GetLastError());

}

WaitForSingleObject(pi.hProcess, INFINITE);

//Restore the original environment variable

if(fExist)

{

if(!SetEnvironmentVariable(VARNAME, pszOldVal))

{

printf("SetEnvironmentVariable failed(%d)/n", GetLastError());

return FALSE;

}

}

else

SetEnvironmentVariable(VARNAME, NULL);

free(pszOldVal);

system("pause");

return fSuccess;

}

文章转载http://blog.csdn.net/ace1985/article/details/5706087
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: