精品久久看,欧美成人久久一级c片免费,日本加勒比在线精品视频,国产一区二区三区免费大片天美,国产成人精品999在线,97理论三级九七午夜在线观看

當前位置: 首頁IT技術 → 便攜軟件制作的系列教程

便攜軟件制作的系列教程

更多

在NSIS中怎么導入注冊表。

這有何難,用registry插件嘛:

${registry::RestoreKey} file.reg $var

可是,如果你經(jīng)常在 RestoreKey 后面用 ${registry:write} ,就會發(fā)現(xiàn),往往導入注冊表會失敗,或者寫入的鍵值被reg文件中的舊鍵值覆蓋了,這是為什么呢?

原來,${registry::RestoreKey} 這個命令并不會等待導入完成。作者在文檔中寫了:

${registry::RestoreKey} simply exec regedit: regedit /s “[file]“

執(zhí)行的是 Exec 而非 ExecWait 。那么,可能 regedit.exe 尚未啟動,就開始執(zhí)行下一行命令了。制作一般的安裝包問題不大,但便攜軟件對執(zhí)行步驟的順序要求更加精確。所以,有些人的代碼是這樣寫的:

${registry::RestoreKey} file.reg $0

Sleep 200

睡一會。睡多久?睡一秒還是一年,這種盲人摸象的做法,我們完美主義者是不會使用的。因為這個命令,有些朋友凡是用到registry插件,都習慣性地加上個 sleep,這是完全沒有必要的,作者說了:

問:So my question is, what other functions in your plugin behave in the same way (ie do not wait for the registry operation to finish)?

答:registry::RestoreKey is the only one.

那么,用:

ExecWait 'regedit /s "[file]"' $var

不就行了嗎?

你又錯了,我們制作便攜軟件的時候,要對自己嚴格要求,在Vista以上的系統(tǒng)中,不經(jīng)過UAC驗證,是無法執(zhí)行 regedit /s 這個命令的(即使導入HKCU中的鍵值也不行)。難道你的每個軟件都要用戶通過UAC驗證以管理員權(quán)限運行嗎,完全是別有居心!

可是,在UAC環(huán)境的測試中,你會發(fā)現(xiàn),即使不通過UAC驗證,${registry::RestoreKey} 這個命令也可以完成注冊表導入,難道,作者隱瞞了什么?

于是,作為代碼盲的你,充滿狐疑地打開 NSIS\Include\Registry.nsh ,找到這樣一段代碼:

!define registry::RestoreKey !insertmacro registry::RestoreKey

!macro registry::RestoreKey _FILE _ERR

registry::_RestoreKey /NOUNLOAD ${_FILE}

Pop ${_ERR}

IntCmp ${_ERR} -2 0 0 +10 ;REGEDIT4 ansi file

SetDetailsPrint none

IfFileExists "$SYSDIR\reg.exe" 0 +4 ;reg.exe used in Windows2K/XP/Vista/7

nsExec::ExecToStack "$SYSDIR\reg.exe" import "${_FILE}"

Pop ${_ERR}

StrCmp ${_ERR} 0 +5 0

IfFileExists "$WINDIR\regedit.exe" 0 +3 ;regedit.exe used in Wine

ExecWait "$WINDIR\regedit.exe" /s "${_FILE}" ${_ERR}

IfErrors 0 +2

StrCpy ${_ERR} -1

SetDetailsPrint lastused

!macroend

真是狡兔三窟!registry::RestoreKey失敗后,用reg.exe import,失敗后,又用 regedit.exe /s,我們就要有這種不屈不撓的精神,不要讓一次執(zhí)行的失敗變成Bug。

眼尖的你發(fā)現(xiàn),關鍵在于這一行:

nsExec::ExecToStack "$SYSDIR\reg.exe" import "${_FILE}"

原來,雖然regedit /s需要管理員權(quán)限,但reg import命令并不需要,這就是${registry::RestoreKey}成功的秘訣。

但是,${registry::RestoreKey}首先嘗試用插件導入,而插件并不等待導入結(jié)束,所以,我們在應用的時候,要把順序顛倒一下:

nsExec::ExecToStack "$SYSDIR\reg.exe" import "${_FILE}"

Pop $0

${IfNot} $0 == 0

${registry::RestoreKey} "${_FILE}" $0

Sleep 500

${IfNotThen} $0 == 0 ${|} StrCpy ${_OutVar} Error ${|}

${Endif}

nsExec::ExecToStack是等待運行結(jié)束的,首先執(zhí)行,假如失敗,再用${registry::RestoreKey},并暫停0.5秒(比較安全的數(shù)值)。當以上動作始終返回Error的時候,我們就應該考慮做個標記,在便攜軟件結(jié)束的時候跳過這一次軟件運行中的注冊表修改,不覆蓋原先的reg文件了。

不過,當你翻閱 PortableApps.com Launcher 的源代碼時,卻發(fā)現(xiàn)關于注冊表導入,僅僅用了一行:

${registry::RestoreKey} $DataDirectory\settings\$0.reg $R9

可為什么感覺上PAL那么穩(wěn)定,極少出錯呢?我猜是因為PAL的代碼非常繁雜,每個實際動作以前都有一堆工作,又是讀Launcher.ini,又是轉(zhuǎn)換變量,又是檢測PAF平臺,慢悠悠的,慢工出細活吧!

例1:

設想某一天,某妞將可移動磁盤插入電腦A,電腦A為她的U盤分配了盤符 F: 。該妞使用U盤上的便攜軟件打開了儲存于U盤上的幾個文檔:

F:\1.doc

F:\2.doc

……

在拔出U盤的時候,她甚至沒有為最后一個文件存檔,反正所有進度都會自動保存嘛!

過了幾日,該妞試圖在電腦B繼續(xù)她的工作,插入U盤以后,電腦B為她的U盤分配了盤符G: 。當她打開便攜軟件的時候,她會看到“最近文檔”列表那里顯示著什么呢?

仍然是:

F:\1.doc

F:\2.doc

……

當她試圖恢復上一次”自動保存“的文檔,卻提示“找不到……文件”。于是,她發(fā)怒了,發(fā)誓再也不來你的網(wǎng)站。

這正常嗎?太正常了,上次你就是在F盤打開文件的嘛?墒悄氵能完美地使用“最近文件”列表、繼續(xù)上一次的工作嗎?

我們干革命,就是要勇于把正常變?yōu)椴徽。于是,我們需要盤符替換。

什么是盤符替換

所謂盤符替換 (Driver Letter Replacement),就是在檢測到盤符相對上一次運行時改變的時候,將某些文件中的舊盤符替換為新盤符,以實現(xiàn)用戶數(shù)據(jù)的完美銜接。以上述例子為例,就是在盤符轉(zhuǎn)變?yōu)?G: 的時候,將最近文檔列表替換為:

G:\1.doc

G:\2.doc

……

讓用戶體會不到盤符改變帶來的變化,而順利繼續(xù)上一次的工作進程。

什么是路徑替換

例2:

設想某一個軟件,在配置文件中保存了大量包含軟件路徑的數(shù)據(jù),這些數(shù)據(jù)在軟件首次運行時生成,指向軟件的插件、模板等目錄,假如這些目錄設置錯誤,該軟件便無法正常運行。

而某一天,某妞將該便攜軟件從同盤符的一個目錄移動到另一個目錄,例如:從 f:\XXXPortable 移動到 f:\PortableApps\XXXPortable 。

那么,如何保證該軟件正常運行呢?假如包含路徑的設置項不多,我們可以一個個地寫入,而假如類似設置很多(例如ACDSee),或者數(shù)量不定,難道也要一個個寫入嗎?所以,我們需要在檢測到路徑改變時,將所有的 XXXPortable 替換為 PortableApps\XXXPortable 。

目錄格式

在 PortableApps.com Launcher 中,提供了四種類型的目錄格式,分別是:

%VARIABLE% : 正向單斜杠。例如:%PAL:AppDir% = x:\portableapps\xxxportable\app 。主要應對ini、xml等普通配置文件。

%VARIABLE:ForwardSlash% : 反向單斜杠。例如:%PAL:AppDir:ForwardSlash% = x:/portableapps/xxxportable/app

%VARIABLE:DoubleBackslash% : 正向雙斜杠。例如:%PAL:AppDir:DoubleBackslash% = x:\\portableapps\\xxxportable\\app 。主要應用于注冊表(.reg)文件。

%VARIABLE:java.util.prefs% : 反向多斜杠。例如: %PAL:AppDir:java.util.prefs% = /X:///Portable/Apps///App/Name/Portable///App 。主要應用于java程序。

我們需要根據(jù)替換文件的類型選擇相應的目錄形式。假如遇到這四種情況以外的目錄形式,則要靠 Custom Code 解決。

實現(xiàn)原理與流程

為了兼顧例1與例2的兩種情況,避免兩種情況同時發(fā)生,我們要將盤符替換與路徑替換分開,那就是:先替換盤符,再替換不帶盤符的路徑。

在引導過程中,讀取上一次記錄的INI文件,判斷是否盤符/路徑改變。

若改變,則讀取上一次的盤符/路徑,轉(zhuǎn)換為正確形式。

讀取當前的盤符/路徑,轉(zhuǎn)換為正確形式。

在文件中替換舊盤符為新盤符。

在文件中替換舊路徑為新路徑。

將當前的盤符、路徑寫入INI文件,以便下一次讀取。

在 PortableApps.com Launcher 中實現(xiàn):

以ACDSee Portable為例,我們需要在引導過程中替換注冊表文件 HKCU.reg 中的舊盤符\路徑為新。

[FileWrite1]

Type=Replace

File=%PAL:DataDir%\settings\HKCU.reg

Find=%PAL:LastDrive%\\

Replace=%PAL:Drive%\\

[FileWrite2]

Type=Replace

File=%PAL:DataDir%\settings\HKCU.reg

Find=%PAL:LastPackagePartialDir:DoubleBackslash%

Replace=%PAL:PackagePartialDir:DoubleBackslash%

效果如下:

x:\\

替換為:

y:\\

\\xxx\\AppNamePortable

替換為:

\\yyy\\AppNamePortable

請注意,在 [FileWrite1] (盤符替換)中,我在 %PAL:LastDrive% 后面加上了雙斜杠。這是因為,%PAL:LastDrive% 是不帶斜杠的(x:)?赡艹霈F(xiàn)這種情況:替換 D:,把 DWORD: 的最后兩個字母也替換了。難道PortableApps.com的人不擔心這種情況嗎?我認為使用PAL替換盤符的時候都要注意這一點,替換盤符一定要加斜杠。

在 Custom Code 中實現(xiàn):

PortableApps.com Launcher 是一個死板的網(wǎng)站的死板的程序員做出的死板的工具,在險峻難料的革命事業(yè)中,我們要堅決摒棄教條主義思想。許多時候稍有變化,我們就需要用到 Custom Code 。那么,在NSIS語言中怎樣實現(xiàn)呢?

例如,一個程序以這樣的形式在 Data\File.txt 記錄路徑:

F__PortableApps_App_Portable

“:”、“\”、“空格”三種符號都轉(zhuǎn)換為下劃線。讓我們來寫一段 Custom Code 來解決它。

${SegmentPrePrimary}

; 替換盤符

; 首先讀取Ini文件中的記錄

ReadINIStr $0 $DataDirectory\settings\$AppIDSettings.ini $AppIDSettings LastDrive

; 替換三種符號為下劃線

${WordReplace} "$0\" "\" "_" "+" "$R0"

${WordReplace} "$R0" ":" "_" "+" "$R0"

${WordReplace} "$R0" " " "_" "+" "$R0"

; 得到當前盤符

StrCpy $R1 "$AppDirectory" 3

; 替換三種符號為下劃線

${WordReplace} "$R1" "\" "_" "+" "$R1"

${WordReplace} "$R1" ":" "_" "+" "$R1"

${WordReplace} "$R1" " " "_" "+" "$R1"

; 在文件中替換

${ReplaceInFileCS} "$DataDirectory\File.txt" $R0 $R1

; 替換路徑

; 首先讀取Ini文件中的記錄

ReadINIStr $0 $DataDirectory\settings\$AppIDSettings.ini $AppIDSettings LastDirectory

; 替換兩種符號為下劃線

${WordReplace} "$0" "\" "_" "+" "$R0"

${WordReplace} "$R0" " " "_" "+" "$R0"

; 得到當前路徑(不帶盤符)

StrCpy $R1 "$AppDirectory" "" 2

; 替換兩種符號為下劃線

${WordReplace} "$R1" "\" "_" "+" "$R1"

${WordReplace} "$R1" " " "_" "+" "$R1"

; 在文件中替換

${ReplaceInFileCS} "$DataDirectory\File.txt" $R0 $R1

!macroend

請注意文件的編碼,如果是 UTF-16LE 編碼,用 ${ReplaceInFileUTF16LECS} 。若需要忽略大小寫,取消最后的“CS”。

在原始NSIS腳本中使用,需要另外:

!include "TextReplace.nsh"

!include "ReplaceInFileWithTextReplace.nsh"

注意事項

盤符與路徑替換是一種簡單地銜接工作環(huán)境的方法,但我認為,在應用中需要注意以下幾點:

一定要確定你替換的是盤符/路徑,而非別的東西。例如,使用PAL的時候,在盤符后加入斜杠,替換 x:\ 而不是 x: 。

自行撰寫代碼時,注意所替換文件的編碼。

在替換大文件或多次替換之間,加入Sleep。否則可能遇到替換失敗。

對于重要路徑,最好在替換后手動寫入一次,以保障無誤。盤符替換依賴INI文件中的記錄,假如一次記錄與實際銜接不上,可能從此都銜接不上了。

自從去年心血來潮開了個頭,這個教程就一直沒了下文,我要用實際行動粉碎虎頭蛇尾的謠言,同志們,今天來談談 DefaultData。

剛開始制作便攜軟件的朋友常犯的一個錯誤是,將軟件的默認配置保存到 Data 目錄中。何以說是錯誤呢:

PortableApps.com格式便攜軟件在安裝后,Data目錄必須是空的。Data目錄中的文件必須在首次運行后生成。

合格的P.A格式便攜軟件,用戶可以隨時刪除Data目錄,將便攜軟件恢復到初始狀態(tài)。

因此,如果某些默認配置在軟件第一次運行時必須導入,我們應該將它保存到DefaultData目錄中。

DefaultData 的誕生

在一個不可考證的從前,John T. Haller 同志(PortableApps.com 的創(chuàng)始人)開始制作他的第一個便攜軟件:Firefox Portable。在移動介質(zhì)運行的 Firefox 瀏覽器應該有如下調(diào)整:關閉磁盤緩存,不檢測默認瀏覽器,不設置默認下載目錄,同時,他希望在Firefox的默認書簽內(nèi)加入他的網(wǎng)站地址。如何實現(xiàn)以上默認設置的調(diào)整呢?

直接修改程序?吃力不討好。于是,他在 App 目錄下新建了 DefaultData 目錄,將一份配置好的最簡化的配置保存于此。在Firefox首次運行時,DefaultData 目錄的內(nèi)容會被復制到 Data 目錄,以實現(xiàn)設置默認配置的目的。于是,今天的 Firefox Portable(以及所有標準P.A格式便攜軟件)的結(jié)構(gòu),就成了這個樣子:

-\ <--- Directory with FirefoxPortable.exe

+\App\

+\AppInfo\

+\firefox\

+\DefaultData\

+\profile\

+\settings\

+\plugins\

+\Data\

FirefoxPortable.exe

什么是 DefaultData

通過以上敘述我們已經(jīng)知道,DefaultData 是 PortableApps.com 格式便攜軟件的標準部件之一,它位于 App\DefaultData ,是軟件的默認配置。在首次運行時,它被復制到 Data 目錄,DefaultData 內(nèi)部的文件結(jié)構(gòu)應該和 Data 目錄完全一致。

DefaultData 怎樣工作?

DefaultData在首次運行時復制到 Data 目錄,作為初始的程序配置。判斷是否首次運行有幾種不同的方式:

Firefox Portable 的方式:

Firefox Portable 通過 NSIS 語言寫成,它通過檢查 Data\Profile\prefs.js是否存在來判斷是否首次運行,假如 Data\Profile\prefs.js 不存在,則復制默認配置到Data目錄。這種方式較為靈活,可根據(jù)不同軟件的具體情況選擇不同的判斷物:

ProfileWork:

;=== Check for an existing profile

IfFileExists "$PROFILEDIRECTORY\prefs.js" ProfileFound

;=== No profile was found

StrCmp $ISDEFAULTDIRECTORY "true" CopyDefaultProfile CreateProfile

CopyDefaultProfile:

CreateDirectory "$EXEDIR\Data"

CreateDirectory "$EXEDIR\Data\plugins"

CreateDirectory "$EXEDIR\Data\profile"

CreateDirectory "$EXEDIR\Data\settings"

CopyFiles /SILENT $EXEDIR\App\DefaultData\plugins\*.* $EXEDIR\Data\plugins

CopyFiles /SILENT $EXEDIR\App\DefaultData\profile\*.* $EXEDIR\Data\profile

PortableApps.com Launcher 的方式:

PortableApps.com Launcher 通過檢查 Data\settings目錄是否存在判斷首次運行,PAL在運行一次以后必然創(chuàng)建 Data\settings 目錄,如果此目錄不存在,則判斷為首次運行,并復制 DefaultData:

${IfNot} ${FileExists} $EXEDIR\Data\settings

CreateDirectory $EXEDIR\Data\settings

${If} ${FileExists} $EXEDIR\App\DefaultData\*.*

CopyFiles /SILENT $EXEDIR\App\DefaultData\*.* $EXEDIR\Data

${EndIf}

${EndIf}

第三種方式:

在制作具有中國特色的便攜軟件時,有時會碰到更為復雜的情況。為了保證軟件始終從默認配置的基礎上啟動,我們可以分別判斷多個目錄,缺少哪一個,就復制哪一個:

例一,

若Data\Profile不存在則復制DefaultData\Profile,若Data\Plugins不存在則復制DefaultData\Plugins:

${IfNot} ${FileExists} $EXEDIR\Data\Profile

CreateDirectory $EXEDIR\Data\Profile

CopyFiles /Silent $EXEDIR\App\DefaultData\Profile\*.* $EXEDIR\Data\Profile

${EndIf}

${IfNot} ${FileExists} $EXEDIR\Data\Plugins

CreateDirectory $EXEDIR\Data\Plugins

CopyFiles /Silent $EXEDIR\App\DefaultData\Plugins\*.* $EXEDIR\Data\Plugins

${EndIf}

例二,

在迅雷便攜版中應用到的,檢測任何一個 DefaultData 中的目錄,如果在 Data 目錄中不存在,都復制過去:

Section Main

; ......

; CopyDefaultData:

StrLen $R0 "$EXEDIR\App\DefaultData\"

${Locate} "$EXEDIR\App\DefaultData" "/L=D" CopyDefaultData

; ......

SectionEnd

Function CopyDefaultData

StrCpy $R1 $R9 "" $R0

${IfNot} ${FileExists} "$EXEDIR\Data\$R1"

CreateDirectory "$EXEDIR\Data\$R1"

CopyFiles /Silent "$R9\*.*" "$EXEDIR\Data\$R1"

${Endif}

Push $0

FunctionEnd

DefaultData 能做什么

修改默認配置

例如,在 Evernote Portable 中,將以下內(nèi)容保存為 App\DefaultData\settings\EvernotePortable.reg:

Windows Registry Editor Version 5.00

[HKEY_CURRENT_USER\Software\Evernote\Evernote]

"UpdateToPreReleaseVersion"=dword:00000000

"CheckForUpdatesAtLaunch"=dword:00000000

則會在首次運行時復制為 Data\settings\EvernotePortable.reg ,接著導入注冊表,實現(xiàn)默認關閉自動升級的目的。

程序本身的默認/初始配置

通過 Total Uninstall 監(jiān)測軟件安裝,可發(fā)現(xiàn)某些軟件在首次安裝后會在配置目錄中寫入一些文件,這些文件必須放到 DefaultData 目錄,以保證程序的完整性,以及讓用戶隨時可以刪除 Data 目錄恢復軟件初始配置。

通過 DefaultData 新建文件夾

在 PortableApps.com Launcher 中,如果你希望使用 FilesMove 來移動文件,必須保證 Data 目錄中有這個文件的父目錄,否則移動會失敗,例如:

[FilesMove]

config\file.txt=%PAL:AppDir%\AppName

在此例中,假如Data\config目錄不存在,那么file.txt就無法被移動到Data目錄。

解決辦法是,創(chuàng)建 App\DefaultData\config ,那么,首次運行時,App\DefaultData\config 會被復制為 Data\config ,以實現(xiàn)新建文件夾的目的。

不過,在大多數(shù)情況下,將單個文件保存到 Data\settings 中是更好的方法,PAL會自動創(chuàng)建此文件夾,避免了通過 DefaultData 來創(chuàng)建的麻煩。

其它用途

在制作具有中國特色的便攜軟件時,有時我們希望一些軟件配置永遠是“一次性”的(例如廣告目錄),那么,我們將一份干凈的初始配置保存到 DefaultData 中,在每次軟件啟動時復制到配置目錄,在軟件結(jié)束時刪除掉復制的副本。以保證軟件的潔凈。

注意事項

DefaultData 是 Data 目錄的初始狀態(tài),其目錄、文件結(jié)構(gòu)必須和 Data 目錄完全一致。假如你在設計便攜軟件時設定將注冊表導出到 Data\settings\AppNamePortable.reg,那么你應該將默認配置保存為 App\DefaultData\settings\AppNamePortable.reg 。否則無法奏效。

DefaultData 是軟件初始配置的一份存檔,應該盡量保持精簡。僅僅保留最必要的部分。你不應該將整個配置好的 Data 保存為 DefaultData,那樣浪費空間,延長首次啟動的時間,而應該找出真正有必要的、不可缺少的修改部分,保存為 DefaultData 。如果你的 DefaultData 超過1M,那么就該想想辦法了。

DefaultData 應該保留最通用的部分,如果你的 DefaultData 中存在關于你的計算機的信息,例如:installdir=c:\Program Files\AppName ,那么是非常不專業(yè)的,我們要嚴格要求自己。

通過 Total Uninstall 監(jiān)視軟件安裝,可發(fā)現(xiàn)某些軟件自身的默認配置。有時候這些配置很重要(例如一個初始的數(shù)據(jù)庫),請別忘了把它們保存到 DefaultData。

 

熱門評論
最新評論
發(fā)表評論 查看所有評論(0)
昵稱:
表情: 高興 可 汗 我不要 害羞 好 下下下 送花 屎 親親
字數(shù): 0/500 (您的評論需要經(jīng)過審核才能顯示)
主站蜘蛛池模板: 精品久久国产视频 | 开心婷婷激情五月 | 久久免费视频一区 | 日韩欧美精品在线观看 | 国产片a国产片免费看视频 国产拍在线 | 久久综合五月天婷婷伊人 | 久久理论片 | 欧美一区二区三区在线观看 | 欧美一级在线视频 | 玖玖中文| 五月婷婷丁香综合 | 综合热久久 | 四虎成人精品国产一区a | 五月天黄色网址 | 日韩精品视频在线免费观看 | 精品三区 | 国内精品中文字幕 | 少女中文字幕在线视频 | 国产一区二区视频免费 | 热久久视久久精品18国产 | 国产99久久久久久免费看 | 色婷婷精品 | 99久久精品一区二区三区 | 99999久久久久久亚洲 | 韩国三级理论电影青春 | 娼年日剧未删减版 | 精品suv一区二区三区 | 米奇电影院免费观看 | 婷婷激情四月 | 国产亚洲精品激情都市 | 99久久99久久精品 | www.人人草 | 精品国产_亚洲人成在线高清 | 美女视频免费观看网站 | 国产亚洲综合久久 | 国产精品1区 2区 3区 | 天堂视频在线观看免费完整版 | 四虎亚洲 | 深夜视频免费观看 | 日韩欧美久久一区二区 | 国产精品成人四虎免费视频 |