一些在xp下設(shè)計(jì)的程序,轉(zhuǎn)到vista\7下就不能運(yùn)行了。尤其是那種寫入注冊表的,寫入program file文件夾的程序。
在msdn上給出了解釋。uac還有這樣子的保護(hù),相當(dāng)于虛擬機(jī)了。
[來自MSDN]
很多應(yīng)用程序都設(shè)計(jì)為把文件寫入Program Files,Windows目錄,或者系統(tǒng)根(一般是C盤)文件夾。一些應(yīng)用程序被設(shè)計(jì)為更新Microsoft Windows 注冊表的值,特別是HKLM/Software中的值。但是這樣會(huì)有存在一個(gè)問題:文件或注冊表的值并沒有被創(chuàng)建或更新。您可能會(huì)問,“怎么回事?我的應(yīng)用程序運(yùn)行良好并沒有報(bào)錯(cuò)啊。我的文件到底去哪兒了?”
您可能遇到過下面的一種或多種情況:
??? 您的應(yīng)用程序?qū)懭隤rogram Files,Windows目錄,或者系統(tǒng)根(一般是C盤)文件夾,但是您在這些地方并沒有找到您的文件
??? 您的應(yīng)用程序?qū)懭隬indows注冊表,特別是HKLM/Software中,但是您沒有看到注冊表進(jìn)行了更新
??? 您切換到了另一個(gè)帳戶,并且您的應(yīng)用程序沒有辦法找到已寫入Program Files,Windows目錄,或者系統(tǒng)根(一般是C盤)文件夾的文件,或者找到了這些文件的老版本
??? 在啟用或禁用用戶帳戶控制(UAC)后,您的應(yīng)用程序都沒有辦法找到Program Files或Windows目錄中的文件
如果這些出現(xiàn)在了您的應(yīng)用程序中,都是由于UAC虛擬化的原因。下面所提供的信息為您介紹了所有關(guān)于發(fā)現(xiàn)這個(gè)應(yīng)用程序兼容性問題,解決方案,以及特殊兼容性問題的一些附加信息。
真正的問題:UAC 虛擬化
在Windows Vista中,一般由管理員來運(yùn)行應(yīng)用程序。這樣,應(yīng)用程序可以自由的讀寫系統(tǒng)文件和注冊表的值。如果標(biāo)準(zhǔn)用戶可以運(yùn)行這些應(yīng)用程序,那他們可能會(huì)由于沒有足夠的權(quán)限而導(dǎo)致失敗。Windows Vista通過重定向?qū)懭耄ㄒ约昂罄m(xù)文件或者注冊表操作)對應(yīng)用戶的配置文件(profile)位置,來對標(biāo)準(zhǔn)用戶運(yùn)行應(yīng)用程序的能力進(jìn)行了提升。
例如,如果一個(gè)應(yīng)用程序試圖寫入C:\Program Files\Contoso\Settings.ini,并且用戶沒有權(quán)限來對該文件夾進(jìn)行寫入操作,寫入操作將會(huì)被重定向到C:\Users\Username\AppData\Local\VirtualStore\Program Files\Contoso\settings.ini。如果應(yīng)用程序試圖在注冊表中寫入HKEY_LOCAL_MACHINE\Software\Contoso\,這將會(huì)被自動(dòng)重定向到HKEY_CURRENT_USER\Software\Classes\VirtualStore\MACHINE\Software\Contoso或HKEY_USERS\UserSID_Classes\VirtualStore\Machine\Software\Contoso。
下圖顯示了Windows 虛擬化進(jìn)程的兩個(gè)組件:文件虛擬化和注冊表虛擬化

??重要注意
當(dāng)為Windows Vista開發(fā)應(yīng)用程序時(shí),請確定在對應(yīng)的requestedExecutionLevel元素中嵌入應(yīng)用程序清單文件。這將會(huì)關(guān)閉文件和注冊表虛擬化,減少虛擬化的文件和注冊表值的復(fù)雜性。
更多關(guān)于UAC虛擬化和新UAC技術(shù),請參閱“Windows Vista中的新UAC技術(shù)”http://msdn.microsoft.com/en-us/library/bb756960.aspx .
解決方案
虛擬化是為了幫助現(xiàn)存的應(yīng)用程序解決兼容性問題而出現(xiàn)的。為Windows 7而設(shè)計(jì)的新的應(yīng)用程序不應(yīng)該對敏感的系統(tǒng)區(qū)域進(jìn)行寫入操作,不應(yīng)該依靠虛擬化來對不正確的應(yīng)用程序行為進(jìn)行修正。開發(fā)應(yīng)用程序的時(shí)候,應(yīng)該始終考慮使其運(yùn)行在標(biāo)準(zhǔn)用戶的權(quán)限下,而不是在管理員權(quán)限下運(yùn)行。測試您的應(yīng)用程序時(shí),也要用標(biāo)準(zhǔn)用戶權(quán)限而不是管理員權(quán)限。
如果您正在使用UAC虛擬化來為Windows 7開發(fā)應(yīng)用程序,那么請將您的應(yīng)用程序重新設(shè)計(jì)為將文件寫入合適位置。當(dāng)更新現(xiàn)有的代碼以在Windows 7上運(yùn)行的時(shí)候,您應(yīng)該:
??? 確定在運(yùn)行時(shí),應(yīng)用程序存儲(chǔ)數(shù)據(jù)只是在對應(yīng)用戶的位置或者在對訪問控制列表(ACL)進(jìn)行了設(shè)置的%alluserprofile%系統(tǒng)位置。更多關(guān)于ACLs的信息,請參閱訪問控制列表.
??? 使用已知的文件夾來寫入數(shù)據(jù)文件。所有用戶可以使用的一般數(shù)據(jù)應(yīng)該被寫入一個(gè)對所有人共享的公共位置。所有其他的數(shù)據(jù)都應(yīng)該被寫入對應(yīng)用戶的位置。
o??一般數(shù)據(jù)文件可以包含,但不限制,日志文件,配置文件(INI/XML),狀態(tài)存儲(chǔ)應(yīng)用程序例如保存的游戲等等。
o??用戶文件則不同;它們應(yīng)該被保存到Documents文件夾(或者用戶指定的位置)
??? 確保您沒有在代碼中指定你覺得合適的路徑。建議您使用下面的模型和APIs來獲取特定已知的Windows的正確路徑:
o??C/C++ native 應(yīng)用程序: 使用SHGetKnownFolderPath方法來獲取已知文件夾的整個(gè)路徑,它們可以用文件夾的KNOWNFOLDERID來識(shí)別。它是一個(gè)標(biāo)識(shí)您想要獲取的已知位置的GUID參數(shù):
??????????? FOLDERID_ProgramData – 向所有人共享項(xiàng)目數(shù)據(jù)的目錄
??????????? FOLDERID_LocalAppData – 對應(yīng)用戶項(xiàng)目數(shù)據(jù)的目錄(不可移動(dòng)的)
??????????? FOLDERID_RoamingAppData –對應(yīng)用戶項(xiàng)目數(shù)據(jù)的目錄(可移動(dòng)的)
o??托管代碼: 使用System.Environment.GetFolderPath方法。GetFolderPath包含一個(gè)標(biāo)識(shí)您想要獲取的已知位置的參數(shù):
??????????? Environment.SpecialFolder.CommonApplicationData –向所有人共享項(xiàng)目數(shù)據(jù)的目錄
??????????? Environment.SpecialFolder.LocalApplicationData –對應(yīng)用戶項(xiàng)目數(shù)據(jù)的目錄(不可移動(dòng)的)
??????????? Environment.SpecialFolder.ApplicationData –對應(yīng)用戶項(xiàng)目數(shù)據(jù)的目錄(可移動(dòng)的)
??? 如果上面提到的方法都不起作用,則請您使用環(huán)境變量:
o??%ALLUSERSPROFILE% –向所有人共享項(xiàng)目數(shù)據(jù)的目錄
o??%LOCALAPPDATA% –對應(yīng)用戶項(xiàng)目數(shù)據(jù)的目錄(不可移動(dòng)的) - Windows Vista或更高版本
o??%APPDATA% –對應(yīng)用戶項(xiàng)目數(shù)據(jù)的目錄(可移動(dòng)的) - Windows Vista或更高版本