最近因故重装了操作系统,随着Win7的发布,Win7系统也越来越流行啦,于是头脑发热的将用了多年的XP系统的换成了Win7。装好系统,安装好常用的程序后,不幸的发现:我原来用VS开发大部分程序,现在在Win7上运行都会或多或少的出现问题。
看来,VS开发的程序有Win7是也同样存在兼容性问题。
经过查找原因和分析,发现问题主要集中在两个方面,一是UAC(用户访问控制)和IP地址上。
在XP时代,到用户访问资源的控制没有像Win7这么严格的权限要求。一般用户的权限都可以完成诸如注册表修改,文件删除等操作。但是到了Win7时代,由于有啦UAC,用户到一些敏感资源的访问都将受到UAC的管理,没有足够权限的操作是不允许的。于是问题就产生啦。
很多程序都有访问注册表的操作,原本在XP上可以自由访问的注册表,现在Win7的UAC将检查用户的权限,一般用户会收到“无权访问该资源”的错误提示,那么,如果才能在Win7上正常的访问需要的资源呢?
答案就是提升用户权限。
提升权限的方法有多种,比如:应用程序提升,管理员协助提升,系统必备组件提升,自定义操作提升等等。这里着重用的是在应用程序中直接提升权限,达到访问资源的目的。其它相关内容可以参考微软的文章:http://msdn.microsoft.com/en-us/bb384154(zh-cn,VS.90).aspx
方法如下:
给应用程序增加一个Application manifest file文件(应用程序清单文件),内容默认如下:
<?xml version="1.0" encoding="utf-8"?> <asmv1:assembly manifestVersion="1.0" xmlns="urn:schemas-microsoft-com:asm.v1" xmlns:asmv1="urn:schemas-microsoft-com:asm.v1" xmlns:asmv2="urn:schemas-microsoft-com:asm.v2" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> <assemblyIdentity version="1.0.0.0" name="MyApplication.app"/> <trustInfo xmlns="urn:schemas-microsoft-com:asm.v2"> <security> <requestedPrivileges xmlns="urn:schemas-microsoft-com:asm.v3"> <!-- UAC 清单选项 如果希望更改 Windows 用户帐户控制级别,请用以下节点之一替换 requestedExecutionLevel 节点。 <requestedExecutionLevel level="asInvoker" uiAccess="false" /> <requestedExecutionLevel level="requireAdministrator" uiAccess="false" /> <requestedExecutionLevel level="highestAvailable" uiAccess="false" /> 如果您希望利用文件和注册表虚拟化提供 向后兼容性,请删除 requestedExecutionLevel 节点。 --> <requestedExecutionLevel level="asInvoker" uiAccess="false" /> </requestedPrivileges> </security> </trustInfo> </asmv1:assembly>
然后将该文件中的asInvoker替换成requireAdministrator就可以啦。一般加了这个文件后,调试程序的时候,Win7系统会要求你以administrator的身份来启动vs,之后再操作注册表就没有问题了。另一种方法是删除 requestedExecutionLevel 节点,利用注册表虚拟化,不过这样做无法达到开机启动之类的效果。有兴趣的朋友也可以参考一一上:MSDN的 在Windows Vista 上进行 Windows Installer 部署 中的相关说明。
关于IP地址的问题
Win7为IPV6提供了良好的支持,同时这也为一些相对较老的IP应用带来的问题。我们通常习惯于用:
IPHostEntry ipHostEntry = Dns.GetHostEntry(Dns.GetHostName());
来获取本机的IP地址,这样在Win7上就会发现,一些地址的形式居然是IPV6的。晕,因此,做兼容Win7开发时,不仅需要获得IP地址,而且还需要判断IP地址的类型。由于我的程序不需要提供IPV6的支持,所以,简单的略去IPV6类型的地址,如下:
string DefaultIP="127.0.0.1"; foreach (IPAddress ip in ipHostEntry.AddressList) { if (ip.AddressFamily == AddressFamily.InterNetwork) { DefaultIP = ip.ToString(); break; } }
这样获取第一个不是IPV6类型的地址,做为DefaultIP的字符串值。这样,去掉啦IPV6格式的地址后,程序可以正常运行起来啦。
《完》
※※※※※※※※※※※※※※--我的程序人生--※※※※※※※※※※※※※※