Chinaunix首页 | 论坛 | 博客
  • 博客访问: 588453
  • 博文数量: 752
  • 博客积分: 40000
  • 博客等级: 大将
  • 技术积分: 5005
  • 用 户 组: 普通用户
  • 注册时间: 2008-10-13 14:47
文章分类

全部博文(752)

文章存档

2011年(1)

2008年(751)

我的朋友

分类:

2008-10-13 16:54:07

不重起Windows直接更改IP地址
作者:浙江省温岭电信局 王骏

注:本文适用于WINDOWS NT/2000/XP/2003



源代码运行效果图如下:


设置IP地址只需要更改注册表中关于适配器的相应设置,但更改后需要重新启动系统才能生效,而AddIPAddress函数只能添加IP而不是更改当前的IP,我们在Windows NT/2000界面上操作不需要重新启动就可以生效,那系统到底做了什么额外的工作才使IP设置直接生效呢?笔者通过跟踪explorer.exe中API的调用发现在netcfgx.dll中调用了dhcpcsvc.dll中一个未公开的API:DhcpNotifyConfigChange,现将不重新启动WINDOWS直接更改IP地址的详细方法介绍如下:

一、获取适配器名称

这里指的适配器名称要区别于适配器描述,比如我的一块网卡,适配器描述是:Realtek RTL8139(A) PCI Fast Ethernet Adapter,适配器名称为:{66156DC3-44A4-434C-B8A9-0E5DB4B3EEAD}。获取适配器名称的方法有多种:

1.1 调用IP helper API取得适配器名称

ULONG ulAdapterInfoSize = sizeof(IP_ADAPTER_INFO);
IP_ADAPTER_INFO *pAdapterInfoBkp, *pAdapterInfo = (IP_ADAPTER_INFO*)new char[ulAdapterInfoSize];
if( GetAdaptersInfo(pAdapterInfo, &ulAdapterInfoSize) == ERROR_BUFFER_OVERFLOW ) // 缓冲区不够大
{
	delete pAdapterInfo;
	pAdapterInfo = (IP_ADAPTER_INFO*)new char[ulAdapterInfoSize];
	pAdapterInfoBkp = pAdapterInfo;
}
if( GetAdaptersInfo(pAdapterInfo, &ulAdapterInfoSize) == ERROR_SUCCESS )
{
	do{ // 遍历所有适配器
		if(pAdapterInfo->Type == MIB_IF_TYPE_ETHERNET)	// 判断是否为以太网接口
		{
			// pAdapterInfo->Description 是适配器描述
			// pAdapterInfo->AdapterName 是适配器名称
		}
		pAdapterInfo = pAdapterInfo->Next;
	}while(pAdapterInfo);
}
delete pAdapterInfoBkp;
1.2 读取注册表取得适配器名称

在Windows2000中可以通过遍历 HKEY_LOCAL_MACHINE\System\CurrentControlSet\Control\Class\{4d36e972-e325-11ce-bfc1-08002be10318}\000n\ (n是从0开始编号的数字)所有接口, 在Windows NT中可以读取HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows NT\CurrentVersion\NetworkCards中的信息,下面以Windows2000为例:
HKEY hKey, hSubKey, hNdiIntKey;

if(RegOpenKeyEx(HKEY_LOCAL_MACHINE,
			"System\\CurrentControlSet\\Control\\Class\\{4d36e972-e325-11ce-bfc1-08002be10318}",
			0,
			KEY_READ,
			&hKey) != ERROR_SUCCESS)
	return FALSE;

DWORD dwIndex = 0;
DWORD dwBufSize = 256;
DWORD dwDataType;
char szSubKey[256];
unsigned char szData[256];

while(RegEnumKeyEx(hKey, dwIndex++, szSubKey, &dwBufSize, NULL, NULL, NULL, NULL) == ERROR_SUCCESS)
{
	if(RegOpenKeyEx(hKey, szSubKey, 0, KEY_READ, &hSubKey) == ERROR_SUCCESS)
	{		
		if(RegOpenKeyEx(hSubKey, "Ndi\\Interfaces", 0, KEY_READ, &hNdiIntKey) == ERROR_SUCCESS)
		{
			dwBufSize = 256;
			if(RegQueryValueEx(hNdiIntKey, "LowerRange", 0, &dwDataType, szData, &dwBufSize) == ERROR_SUCCESS)
			{
				if(strcmp((char*)szData, "ethernet") == 0)		//	判断是不是以太网卡
				{
					dwBufSize = 256;
					if(RegQueryValueEx(hSubKey, "DriverDesc", 0, &dwDataType, szData, &dwBufSize) == ERROR_SUCCESS)
					{
						// szData 中便是适配器详细描述
						dwBufSize = 256;
						if(RegQueryValueEx(hSubKey, "NetCfgInstanceID", 0, &dwDataType, szData, &dwBufSize) == ERROR_SUCCESS)
						{
							// szData 中便是适配器名称
						}
					}
				}
			}
			RegCloseKey(hNdiIntKey);
		}
		RegCloseKey(hSubKey);
	}

	dwBufSize = 256;
}	/* end of while */
		
RegCloseKey(hKey);
二、将IP信息写入注册表

代码如下:
BOOL RegSetIP(LPCTSTR lpszAdapterName, LPCTSTR pIPAddress, LPCTSTR pNetMask, LPCTSTR pNetGate)
{
	HKEY hKey;
	string strKeyName = "SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters\\Interfaces\\";
	strKeyName += lpszAdapterName;
	if(RegOpenKeyEx(HKEY_LOCAL_MACHINE,
				strKeyName.c_str(),
				0,
				KEY_WRITE,
				&hKey) != ERROR_SUCCESS)
		return FALSE;
	
	char mszIPAddress[100];
	char mszNetMask[100];
	char mszNetGate[100];

	strncpy(mszIPAddress, pIPAddress, 98);
	strncpy(mszNetMask, pNetMask, 98);
	strncpy(mszNetGate, pNetGate, 98);

	int nIP, nMask, nGate;

	nIP = strlen(mszIPAddress);
	nMask = strlen(mszNetMask);
	nGate = strlen(mszNetGate);

	*(mszIPAddress + nIP + 1) = 0x00;	// REG_MULTI_SZ数据需要在后面再加个0
	nIP += 2;

	*(mszNetMask + nMask + 1) = 0x00;
	nMask += 2;

	*(mszNetGate + nGate + 1) = 0x00;
	nGate += 2;
	
	RegSetValueEx(hKey, "IPAddress", 0, REG_MULTI_SZ, (unsigned char*)mszIPAddress, nIP);
	RegSetValueEx(hKey, "SubnetMask", 0, REG_MULTI_SZ, (unsigned char*)mszNetMask, nMask);
	RegSetValueEx(hKey, "DefaultGateway", 0, REG_MULTI_SZ, (unsigned char*)mszNetGate, nGate);

	RegCloseKey(hKey);

	return TRUE;
}

三、调用DhcpNotifyConfigChange通知配置的改变

未公开函数DhcpNotifyConfigChange位于 dhcpcsvc.dll中,原型如下:
BOOL DhcpNotifyConfigChange(
    LPWSTR lpwszServerName, // 本地机器为NULL
    LPWSTR lpwszAdapterName, // 适配器名称
    BOOL bNewIpAddress, // TRUE表示更改IP
    DWORD dwIpIndex, // 指明第几个IP地址,如果只有该接口只有一个IP地址则为0
    DWORD dwIpAddress, // IP地址
    DWORD dwSubNetMask, // 子网掩码
    int nDhcpAction ); // 对DHCP的操作 0:不修改, 1:启用 DHCP,2:禁用 DHCP
具体调用代码如下:
BOOL NotifyIPChange(LPCTSTR lpszAdapterName, int nIndex, LPCTSTR pIPAddress, LPCTSTR pNetMask)
{
	BOOL			bResult = FALSE;
	HINSTANCE		hDhcpDll;
	DHCPNOTIFYPROC	pDhcpNotifyProc;
	WCHAR wcAdapterName[256];
	
	MultiByteToWideChar(CP_ACP, 0, lpszAdapterName, -1, wcAdapterName,256);

	if((hDhcpDll = LoadLibrary("dhcpcsvc")) == NULL)
		return FALSE;

	if((pDhcpNotifyProc = (DHCPNOTIFYPROC)GetProcAddress(hDhcpDll, "DhcpNotifyConfigChange")) != NULL)
		if((pDhcpNotifyProc)(NULL, wcAdapterName, TRUE, nIndex, inet_addr(pIPAddress), inet_addr(pNetMask), 0) == ERROR_SUCCESS)
			bResult = TRUE;

	FreeLibrary(hDhcpDll);
	return bResult;
}

(全文完)
--------------------next---------------------

DHCPNOTIFYPROC需要引用什么头文件? ( liuwei520500 发表于 2008-5-14 9:09:00)
 
大侠,我需要一份源代码,有些东西不懂,发份
597091880@qq.com ( liuwei520500 发表于 2008-5-14 9:07:00)
 
在windows xp home edition sp2下运行例程DhcpNotifyConfigChange返回错误15,不知道是什么错误请大家指教,不胜感激 ( gmmy 发表于 2006-7-31 21:14:00)
 
17146738@163.com ( wgm001 发表于 2006-5-7 1:42:00)
 
moonstone:能否给俺一份源码,非常感谢! ( wgm001 发表于 2006-5-7 1:42:00)
 
moonstone:做了两天,终于把Win98下不重启直接修改IP生效的程序写出来,我的方法比较笨:先在注册表中修改IP地址,然后禁用/使用网卡使之生效。我测试的确可以,只是速度不是很快,不知谁有更好的方法?

是不是可以不用手动去禁网卡的啊,我要一份源码 先谢了
127470@2299.com ( shahai 发表于 2006-1-22 11:30:00)
 
其实用iphelper提供的CreateIpForwardEntry/SetIpForwardEntry就可以了,这两个函数是用来设置路由表的 ( JiaxWang 发表于 2005-11-30 6:30:00)
 
在WINXP sp2下无法正确转换,似乎此未公开函数并没有完全起作用,在连接属性里可以看到ip地址和掩码都转过来了,可网关却没有,连点2次修复后才看到了正确的网关地址 ( nyzj2 发表于 2005-4-15 18:26:00)
 
我在VB里调用DhcpNotifyConfigChange,2000和2003下都成功了,可xp下却不成功,在TCP/IP属性高级设置里,ip地址出现了乱码,为何会出现乱码呢?
我的VB已升级到最新的SP6版。 ( nyzj2 发表于 2005-4-3 11:21:00)
 
多谢了先 ( thsqming 发表于 2005-3-20 11:38:00)
 
.......................................................

--------------------next---------------------

阅读(352) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~