今天遇到一個bug,折騰了半天才解決掉,分享給大家。
Bug描述
一位開發人員調用下面的代碼來創建一個注冊表鍵值:
代碼如下:
HKEY hKey;
if(::RegOpenKeyEx(HKEY_CURRENT_USER, DemoRegKey, 0, KEY_WRITE, &hKey) != ERROR_SUCCESS)
{
RegCreateKeyEx(HKEY_CURRENT_USER, DemoRegKey, 0, NULL, REG_OPTION_VOLATILE
, KEY_ALL_ACCESS , NULL , &hKey, NULL);
}
執行後,成功的在注冊表編輯器中看到了鍵值,後續從中獲取值等代碼沒有任何問題。
提交測試後發現,當系統重啟後,上面的程序創建的注冊表鍵值都沒有了,導致後面讀取鍵值的代碼都報錯。
Why?
隨後我調整了代碼,發現下面的代碼在重啟後得到的錯誤碼是2
代碼如下:
LONG lRet;
lRet = ::RegOpenKeyEx(HKEY_CURRENT_USER, DemoRegKey, 0, KEY_WRITE, &hKey);
執行這段代碼lRet的值是2. 錯誤碼2的意思是:系統找不到指定的文件。 對於注冊表函數而已,就是對應的鍵不存在。
調試後發現,創建注冊表的代碼的確成功執行了,在注冊表中也可以看得到。同時,再調用讀取鍵值的代碼,也沒有任何問題。但是一旦重啟電腦,路徑就不存在了,裡面保存的鍵值都沒有了。
看來是某些程序刪掉了我的鍵值,於是查找所有刪除鍵值的代碼,沒有找到。
在重啟前將所有的程序都刪除掉。但是重啟後,該死的鍵值還是沒有出現。
看來是系統或者其它程序刪除的。難道我的鍵值和某些程序的鍵值重名了?好,這次用guid做鍵名,還是重啟後沒有了。
哇哇,真是抓狂啊。
這時候,我回過頭來再細看所有創建注冊表相關的代碼,因為必定是我的程序導致鍵值消失的。
我注意到了REG_OPTION_VOLATILE 這個奇怪的參數。
MSDN之後,我終於找到了原因,就是這個參數造成的。
原因
REG_OPTION_VOLATILE 這個參數的意思是創建的注冊表鍵值都位於內存中,不會保存到相應的注冊表文件中。
英文如下:
代碼如下:
All registry keys are created as volatile, and the information is stored in memory and is not preserved when the corresponding registry hive is unloaded. For HKEY_LOCAL_MACHINE, this occurs when the OS is shut down. The RegSaveKey function does not save volatile registry keys. This flag is ignored for keys that already exist.
所以,重啟後這些鍵值當然就沒有了。
解決辦法
很簡單,使用REG_OPTION_NON_VOLATILE 即可
代碼如下:
RegCreateKeyEx(HKEY_CURRENT_USER, DemoRegKey, 0, NULL, REG_OPTION_NON_VOLATILE
, KEY_ALL_ACCESS , NULL , &hKey, NULL);
這個可以用在測試上。一重啟,之前創建的鍵值都沒了。