Windows CE" title="Windows CE">Windows CE是典型的使用電池供電的系統(tǒng),。這使得正確操作系統(tǒng)十分關(guān)鍵,,應(yīng)用程序大多數(shù)時(shí)間都不需要關(guān)注Windows CE 設(shè)備的電源損耗,但是在某些時(shí)候,,你可能要注意這些損耗,。
當(dāng)用戶關(guān)閉了一個(gè)使用電池的Windows CE 設(shè)備,電源系統(tǒng)不會(huì)關(guān)閉PC電源,,事實(shí)上,,只是系統(tǒng)被掛起(譯者注:這里就像有些PocketPC把關(guān)閉電源放在拔SIM卡的位置,拔出SIM卡才真正關(guān)閉電源,。但是,,目前包括Smartphone在內(nèi),因?yàn)橛布O(shè)備,,比如CPU無(wú)法進(jìn)入低功耗,,所以為了省電,需要做到關(guān)閉應(yīng)用處理器及大部分設(shè)備供電,,然后需要喚醒時(shí),,再通過(guò)定時(shí)器或無(wú)線模塊喚醒。所以不關(guān)閉電源的情況不是絕對(duì)的,。)當(dāng)用戶打開(kāi)設(shè)備電源,,設(shè)備不會(huì)像PC一樣重新啟動(dòng),而是被喚醒,,返回到與系統(tǒng)掛起前一樣的狀態(tài),。這樣導(dǎo)致一個(gè)應(yīng)用程序在喚醒后會(huì)像掛起前一樣運(yùn)行。事實(shí)上,,應(yīng)用程序根本不知道它被掛起,,除非它明確地請(qǐng)求當(dāng)系統(tǒng)掛起時(shí)通知它。從應(yīng)用程序的角度看,,電源管理" title="電源管理">電源管理有三種方式,,查詢電源狀態(tài),,改變電源狀態(tài),和防止電源狀態(tài)改變,。
查詢電源狀態(tài)
要查詢系統(tǒng)當(dāng)前的電源狀態(tài),,你必須調(diào)用
DWORD GetSystemPowerStatusEx2 (PSYSTEM_POWER_STATUS_EX2 pSystemPowerStatusEx2, DWORD dwLen,, BOOL fUpdate);
函數(shù)帶了三個(gè)參數(shù):一個(gè)指向SYSTEM_POWER_ STATUS_EX2結(jié)構(gòu)的指針,,結(jié)構(gòu)的長(zhǎng)度,和一個(gè)布爾值,,表示告訴操作系統(tǒng)是否應(yīng)該查詢電池驅(qū)動(dòng)來(lái)得到最后的信息或者直接返回電池緩存中的信息,。系統(tǒng)大約每5秒查詢一次電池狀態(tài),因此,,如果第三個(gè)差數(shù)是FALSE,,得到的數(shù)據(jù)不會(huì)太舊。結(jié)構(gòu)SYSTEM_POWER_STATUS_EX2被定義為
typedef struct _SYSTEM_POWER_STATUS_EX2 {
BYTE ACLineStatus;
BYTE BatteryFlag;
BYTE BatteryLifePercent;
BYTE Reserved1;
DWORD BatteryLifeTime;
DWORD BatteryFullLifeTime;
BYTE Reserved2;
BYTE BackupBatteryFlag;
BYTE BackupBatteryLifePercent;
BYTE Reserved3;
DWORD BackupBatteryLifeTime;
DWORD BackupBatteryFullLifeTime;
WORD BatteryVoltage;
DWORD BatteryCurrent;
DWORD BatteryAverageCurrent;
DWORD BatteryAverageInterval;
DWORD BatterymAHourConsumed;
DWORD BatteryTemperature;
DWORD BackupBatteryVoltage;
BYTE BatteryChemistry;
} SYSTEM_POWER_STATUS_EX2;
在我描述的這個(gè)巨大的結(jié)構(gòu)之前,,我必須告誡你,,這個(gè)結(jié)構(gòu)返回的數(shù)據(jù)精確程度和電池驅(qū)動(dòng)一樣。同樣的結(jié)構(gòu)被傳給電池驅(qū)動(dòng)來(lái)查詢它的狀態(tài),。Windows CE不驗(yàn)證電池驅(qū)動(dòng)返回的數(shù)據(jù),。這個(gè)函數(shù)返回來(lái)的數(shù)據(jù)依賴于電池驅(qū)動(dòng),因此不同的系統(tǒng)有不同的變化,。舉個(gè)例子,,許多系統(tǒng)在使用AC電源時(shí)不報(bào)告精確的電源級(jí)數(shù);另一些系統(tǒng)則相反,。應(yīng)用程序使用GetSystemPowerStatusEx2來(lái)自動(dòng)預(yù)防和檢測(cè)系統(tǒng)是否可能運(yùn)行應(yīng)用程序,。
第一個(gè)區(qū)域,ACLineStatus,,包含一個(gè)標(biāo)志,,表示系統(tǒng)是否連接到AC 電源,。如果值是AC_LINE_OFFLINE,,表示系統(tǒng)沒(méi)有使用AC 電源;AC_LINE_ONLINE,,表示系統(tǒng)使用了AC 電源,;AC_LINE_BACKUP_POWER和AC_LINE_UNKNOWN,表示備用電源和未知電源,。BatteryFlag區(qū)域,,提供了一個(gè)總的標(biāo)識(shí),表示當(dāng)前系統(tǒng)的電池狀態(tài),,可以有以下值:
BATTERY_FLAG_HIGH
電池被充滿或接近充滿,。
BATTERY_FLAG_LOW
電池還有一點(diǎn)剩余,。
BATTERY_FLAG_CRITICAL
電池電量處在一個(gè)臨界狀態(tài)。
BATTERY_FLAG_CHARGING
電池當(dāng)前正在充電,。
BATTERY_FLAG_NO_BATTERY
系統(tǒng)無(wú)電池
BATTERY_FLAG_UNKNOWN
電池狀態(tài)未知
BatteryLifePercent區(qū)域包含估計(jì)的電池電量能夠維持的百分比,。數(shù)值可能是0到100之間的一個(gè),或用255表示百分比未知,。BatteryLifeTime區(qū)域表示電池耗盡之前可以維持的秒數(shù),。如果該值不能估計(jì),區(qū)域填入BATTERY_LIFE_UNKNOWN,。BatteryFullLifeTime區(qū)域包含完全充滿電池需要的時(shí)間,。如果該值不能估計(jì),填入BATTERY_LIFE_UNKNOWN,。注意,,在許多系統(tǒng)中,這些值可能難以測(cè)量,。大多數(shù)OEM 廠商簡(jiǎn)單地在每個(gè)區(qū)域內(nèi)填入BATTERY_LIFE_UNKNOWN,。
接下來(lái)的第四個(gè)區(qū)域(不計(jì)算保留區(qū)域)重復(fù)了前面的表述,只不過(guò)是對(duì)系統(tǒng)備份電池來(lái)說(shuō),。因?yàn)檫@些值大多數(shù)難以測(cè)量,,許多系統(tǒng)簡(jiǎn)單地返回“unknown”給這些區(qū)域。
剩下的區(qū)域描述了電池和備用電池的電力狀態(tài),,因?yàn)樵S多系統(tǒng)缺少測(cè)量這些值的能力,,這些區(qū)域也被簡(jiǎn)單地默認(rèn)為“unknown”。最后一個(gè)區(qū)域,,BatteryChemistry,,包含一個(gè)標(biāo)志,表示系統(tǒng)中電池的類型,。當(dāng)前已定義的值包括
BATTERY_CHEMISTRY_ALKALINE
BATTERY_CHEMISTRY_NICD
BATTERY_CHEMISTRY_NIMH
BATTERY_CHEMISTRY_LION
BATTERY_CHEMISTRY_LIPOLY
BATTERY_CHEMISTRY_UNKNOWN
Windows CE是典型的使用電池供電的系統(tǒng),。這使得正確操作系統(tǒng)十分關(guān)鍵,應(yīng)用程序大多數(shù)時(shí)間都不需要關(guān)注Windows CE 設(shè)備的電源損耗,,但是在某些時(shí)候,,你可能要注意這些損耗。
當(dāng)用戶關(guān)閉了一個(gè)使用電池的Windows CE 設(shè)備,,電源系統(tǒng)不會(huì)關(guān)閉PC電源,,事實(shí)上,只是系統(tǒng)被掛起(譯者注:這里就像有些PocketPC把關(guān)閉電源放在拔SIM卡的位置,,拔出SIM卡才真正關(guān)閉電源,。但是,目前包括Smartphone在內(nèi),因?yàn)橛布O(shè)備,,比如CPU無(wú)法進(jìn)入低功耗,,所以為了省電,需要做到關(guān)閉應(yīng)用處理器及大部分設(shè)備供電,,然后需要喚醒時(shí),,再通過(guò)定時(shí)器或無(wú)線模塊喚醒。所以不關(guān)閉電源的情況不是絕對(duì)的,。)當(dāng)用戶打開(kāi)設(shè)備電源,,設(shè)備不會(huì)像PC一樣重新啟動(dòng),而是被喚醒,,返回到與系統(tǒng)掛起前一樣的狀態(tài),。這樣導(dǎo)致一個(gè)應(yīng)用程序在喚醒后會(huì)像掛起前一樣運(yùn)行。事實(shí)上,,應(yīng)用程序根本不知道它被掛起,,除非它明確地請(qǐng)求當(dāng)系統(tǒng)掛起時(shí)通知它。從應(yīng)用程序的角度看,,電源管理有三種方式,,查詢電源狀態(tài),改變電源狀態(tài),,和防止電源狀態(tài)改變,。
查詢電源狀態(tài)
要查詢系統(tǒng)當(dāng)前的電源狀態(tài),你必須調(diào)用
DWORD GetSystemPowerStatusEx2 (PSYSTEM_POWER_STATUS_EX2 pSystemPowerStatusEx2,, DWORD dwLen,, BOOL fUpdate);
函數(shù)帶了三個(gè)參數(shù):一個(gè)指向SYSTEM_POWER_ STATUS_EX2結(jié)構(gòu)的指針,結(jié)構(gòu)的長(zhǎng)度,,和一個(gè)布爾值,,表示告訴操作系統(tǒng)是否應(yīng)該查詢電池驅(qū)動(dòng)來(lái)得到最后的信息或者直接返回電池緩存中的信息。系統(tǒng)大約每5秒查詢一次電池狀態(tài),,因此,,如果第三個(gè)差數(shù)是FALSE,得到的數(shù)據(jù)不會(huì)太舊,。結(jié)構(gòu)SYSTEM_POWER_STATUS_EX2被定義為
typedef struct _SYSTEM_POWER_STATUS_EX2 {
BYTE ACLineStatus;
BYTE BatteryFlag;
BYTE BatteryLifePercent;
BYTE Reserved1;
DWORD BatteryLifeTime;
DWORD BatteryFullLifeTime;
BYTE Reserved2;
BYTE BackupBatteryFlag;
BYTE BackupBatteryLifePercent;
BYTE Reserved3;
DWORD BackupBatteryLifeTime;
DWORD BackupBatteryFullLifeTime;
WORD BatteryVoltage;
DWORD BatteryCurrent;
DWORD BatteryAverageCurrent;
DWORD BatteryAverageInterval;
DWORD BatterymAHourConsumed;
DWORD BatteryTemperature;
DWORD BackupBatteryVoltage;
BYTE BatteryChemistry;
} SYSTEM_POWER_STATUS_EX2;
在我描述的這個(gè)巨大的結(jié)構(gòu)之前,,我必須告誡你,這個(gè)結(jié)構(gòu)返回的數(shù)據(jù)精確程度和電池驅(qū)動(dòng)一樣,。同樣的結(jié)構(gòu)被傳給電池驅(qū)動(dòng)來(lái)查詢它的狀態(tài),。Windows CE不驗(yàn)證電池驅(qū)動(dòng)返回的數(shù)據(jù),。這個(gè)函數(shù)返回來(lái)的數(shù)據(jù)依賴于電池驅(qū)動(dòng),,因此不同的系統(tǒng)有不同的變化。舉個(gè)例子,許多系統(tǒng)在使用AC電源時(shí)不報(bào)告精確的電源級(jí)數(shù),;另一些系統(tǒng)則相反,。應(yīng)用程序使用GetSystemPowerStatusEx2來(lái)自動(dòng)預(yù)防和檢測(cè)系統(tǒng)是否可能運(yùn)行應(yīng)用程序。
第一個(gè)區(qū)域,,ACLineStatus,,包含一個(gè)標(biāo)志,表示系統(tǒng)是否連接到AC 電源,。如果值是AC_LINE_OFFLINE,,表示系統(tǒng)沒(méi)有使用AC 電源;AC_LINE_ONLINE,,表示系統(tǒng)使用了AC 電源,;AC_LINE_BACKUP_POWER和AC_LINE_UNKNOWN,表示備用電源和未知電源,。BatteryFlag區(qū)域,,提供了一個(gè)總的標(biāo)識(shí),表示當(dāng)前系統(tǒng)的電池狀態(tài),,可以有以下值:
BATTERY_FLAG_HIGH
電池被充滿或接近充滿,。
BATTERY_FLAG_LOW
電池還有一點(diǎn)剩余。
BATTERY_FLAG_CRITICAL
電池電量處在一個(gè)臨界狀態(tài),。
BATTERY_FLAG_CHARGING
電池當(dāng)前正在充電,。
BATTERY_FLAG_NO_BATTERY
系統(tǒng)無(wú)電池
BATTERY_FLAG_UNKNOWN
電池狀態(tài)未知
BatteryLifePercent區(qū)域包含估計(jì)的電池電量能夠維持的百分比。數(shù)值可能是0到100之間的一個(gè),,或用255表示百分比未知,。BatteryLifeTime區(qū)域表示電池耗盡之前可以維持的秒數(shù)。如果該值不能估計(jì),,區(qū)域填入BATTERY_LIFE_UNKNOWN,。BatteryFullLifeTime區(qū)域包含完全充滿電池需要的時(shí)間。如果該值不能估計(jì),,填入BATTERY_LIFE_UNKNOWN,。注意,在許多系統(tǒng)中,,這些值可能難以測(cè)量,。大多數(shù)OEM 廠商簡(jiǎn)單地在每個(gè)區(qū)域內(nèi)填入BATTERY_LIFE_UNKNOWN。
接下來(lái)的第四個(gè)區(qū)域(不計(jì)算保留區(qū)域)重復(fù)了前面的表述,,只不過(guò)是對(duì)系統(tǒng)備份電池來(lái)說(shuō),。因?yàn)檫@些值大多數(shù)難以測(cè)量,許多系統(tǒng)簡(jiǎn)單地返回“unknown”給這些區(qū)域,。
剩下的區(qū)域描述了電池和備用電池的電力狀態(tài),,因?yàn)樵S多系統(tǒng)缺少測(cè)量這些值的能力,,這些區(qū)域也被簡(jiǎn)單地默認(rèn)為“unknown”。最后一個(gè)區(qū)域,,BatteryChemistry,,包含一個(gè)標(biāo)志,表示系統(tǒng)中電池的類型,。當(dāng)前已定義的值包括
BATTERY_CHEMISTRY_ALKALINE
BATTERY_CHEMISTRY_NICD
BATTERY_CHEMISTRY_NIMH
BATTERY_CHEMISTRY_LION
BATTERY_CHEMISTRY_LIPOLY
BATTERY_CHEMISTRY_UNKNOWN
改變電源狀態(tài)
應(yīng)用程序能通過(guò)一系列的方式改變系統(tǒng)的電源狀態(tài),。在基于Windows CE.NET系統(tǒng)的較新系統(tǒng)中,首選的方式是使用電源管理程序,,在之后的章節(jié)將會(huì)討論,。可是無(wú)論如何,,還有大量的基于早期Windows CE版本的系統(tǒng)以及Windows CE.NET不包含電源管理程序版本,。對(duì)這些系統(tǒng)來(lái)說(shuō),下面的技術(shù)會(huì)很方便,。
關(guān)閉電源
應(yīng)用程序可以通過(guò)調(diào)用一個(gè)少有資料的GwesPowerOffSystem函數(shù)掛起系統(tǒng),。這個(gè)函數(shù)可以在大多數(shù)版本W(wǎng)indows CE中使用,但是最近才被公開(kāi),。事實(shí)上,,大多數(shù)SDK沒(méi)有包含這個(gè)函數(shù)的原型,你可能要提供原型,。這個(gè)函數(shù)定義為 void GwesPowerOffSystem(void);
GwesPowerOffSystem的使用很簡(jiǎn)單:簡(jiǎn)單調(diào)用,,系統(tǒng)就會(huì)掛起。
如果你想避免使用很少資料的函數(shù),,你可以通過(guò)簡(jiǎn)單地模擬用戶按關(guān)閉按鈕來(lái)關(guān)閉系統(tǒng),。你可以通過(guò)使用keybd_event函數(shù)很容易地允許你的應(yīng)用程序掛起系統(tǒng),如下:
keybd_event (VK_OFF,, 0,, KEYEVENTF_SILENT, 0);
keybd_event (VK_OFF,, 0,, KEYEVENTF_SILENT │ KEYEVENTF_KEYUP, 0);
這兩個(gè)keybd_event調(diào)用模擬了按和釋放電源按鈕,,電源按鈕的虛擬鍵值是VK_OFF,。執(zhí)行前面的兩行代碼將掛起系統(tǒng)。因?yàn)樘摂M鍵代碼在執(zhí)行時(shí)會(huì)由GWES表現(xiàn),,兩個(gè)函數(shù)可能在系統(tǒng)掛起前有一些狀態(tài)的表現(xiàn)(譯者注:屏幕上會(huì)有關(guān)閉對(duì)話框之類的圖像,,和真實(shí)按下按鈕的畫(huà)面一樣)。如果你的程序無(wú)法在keybd_event函數(shù)之前停止工作,,添加一個(gè)Sleep調(diào)用來(lái)使應(yīng)用程序暫停一些毫秒來(lái)讓GWES真實(shí)地掛起系統(tǒng),。
關(guān)閉系統(tǒng)
如果系統(tǒng)有有色背光顯示,,主要的電源消耗不是CPU而是背光。在一些環(huán)境下,,一個(gè)應(yīng)用程序需要運(yùn)行卻不需要顯示在屏幕上。一個(gè)例子是音樂(lè)播放器應(yīng)用程序,,當(dāng)用戶聽(tīng)音樂(lè)的時(shí)候,,不關(guān)注屏幕。在這些情形下,,有能力關(guān)閉背光將意味著提高電池壽命,。
當(dāng)然,當(dāng)用戶想看屏幕時(shí),,任何關(guān)閉背光應(yīng)用程序的需要一個(gè)簡(jiǎn)單的用戶友好的方式來(lái)重新打開(kāi)屏幕,。同樣,記得用戶典型的想法是屏幕變黑時(shí)會(huì)認(rèn)為被關(guān)閉了,,因此要考慮這點(diǎn),。舉個(gè)例子,一個(gè)用戶可能在系統(tǒng)已經(jīng)運(yùn)行時(shí)試圖打開(kāi)系統(tǒng)電源,,并且這樣做了,,卻很意外地發(fā)現(xiàn),設(shè)備電源被關(guān)閉了,。同樣,,當(dāng)系統(tǒng)在這種情況下關(guān)閉顯示,它同時(shí)也關(guān)閉了觸摸屏,。這意味著你不能告訴用戶敲擊屏幕來(lái)打開(kāi),。而是,你需要使用一些其他的事件,,比如設(shè)置時(shí)間,,任務(wù)完成,或用戶按了一個(gè)按鈕,。最后,,這里討論的方式對(duì)大多數(shù)基于Windows CE 3.0或更新的版本比較有用,并且被Windows CE .NET 4.0中的電源管理程序所替代,。對(duì)于較新的系統(tǒng),,先看看是否電源管理程序可用,然后通過(guò)它來(lái)控制屏幕,。如果失敗了,,ExtEscape方式也許能行。
在Windows CE中,,顯示的控制是通過(guò)ExtEscape函數(shù),。這是一個(gè)顯示和打印機(jī)驅(qū)動(dòng)的后門(mén),。Windows CE顯示驅(qū)動(dòng)支持許多設(shè)備轉(zhuǎn)義代碼(escape codes),這些被公布在Platform Builder中,。對(duì)于我們的目的來(lái)說(shuō),,只有兩個(gè)轉(zhuǎn)義代碼被用到:SETPOWERMANAGEMENT來(lái)設(shè)置顯示的電源狀態(tài)和QUERYESCSUPPORT來(lái)查詢是否SETPOWERMANAGEMENT被驅(qū)動(dòng)支持。下面的例子打開(kāi)或關(guān)閉系統(tǒng)顯示通過(guò)顯示驅(qū)動(dòng),,并且支持完全的轉(zhuǎn)義代碼:
//
// Defines and structures taken from pwingdi.h in the Platform Builder
//
#define QUERYESCSUPPORT 8
#define SETPOWERMANAGEMENT 6147
#define GETPOWERMANAGEMENT 6148
typedef enum _VIDEO_POWER_STATE {
VideoPowerOn = 1,,
VideoPowerStandBy,
VideoPowerSuspend,,
VideoPowerOff
} VIDEO_POWER_STATE,, *PVIDEO_POWER_STATE;
typedef struct _VIDEO_POWER_MANAGEMENT {
ULONG Length;
ULONG DPMSVersion;
ULONG PowerState;
} VIDEO_POWER_MANAGEMENT, *PVIDEO_POWER_MANAGEMENT;
//----------------------------------------------------------------------
// SetVideoPower - Turns on or off the display
//
int SetVideoPower (BOOL fOn) {
VIDEO_POWER_MANAGEMENT vpm;
int rc,, fQueryEsc;
HDC hdc;
// Get the display dc.
hdc = GetDC (NULL);
// See if supported.
fQueryEsc = SETPOWERMANAGEMENT;
rc = ExtEscape (hdc,, QUERYESCSUPPORT, sizeof (fQueryEsc),,
?。↙PSTR)&fQueryEsc, 0,, 0);
if (rc == 0) {
// No support,, fail.
ReleaseDC (NULL, hdc);
return -1;
}
// Fill in the power management structure.
vpm.Length = sizeof (vpm);
vpm.DPMSVersion = 1;
if (fOn)
vpm.PowerState = VideoPowerOn;
else
vpm.PowerState = VideoPowerOff;
// Tell the driver to turn on or off the display.
rc = ExtEscape (hdc,, SETPOWERMANAGEMENT,, sizeof (vpm),
?。↙PSTR)&vpm,, 0, 0);
// Always release what you get.
ReleaseDC (NULL,, hdc);
return 0;
}
前面的代碼通過(guò)調(diào)用ExtEscape和QUERYESCSUPPORT命令來(lái)查詢是否支持轉(zhuǎn)移代碼,。被查詢的命令首先交給輸入緩沖,如果SETPOWERMANAGEMENT命令被支持,,程序就填充VIDEO_POWER_MANAGEMENT結(jié)構(gòu)并再次調(diào)用ExtEscape設(shè)置電源狀態(tài),。
雖然這些轉(zhuǎn)義代碼允許應(yīng)用程序打開(kāi)或關(guān)閉顯示,Windows CE沒(méi)有一個(gè)統(tǒng)一的方式來(lái)控制背光的亮度,。每個(gè)系統(tǒng)都有它自己的OEM特有方式來(lái)控制背光亮度,。如果將來(lái)有一種標(biāo)準(zhǔn)的背光亮度控制方式,它將很可能放在ExtEscape函數(shù)中,。
打開(kāi)電源系統(tǒng)
當(dāng)系統(tǒng)被掛起,,應(yīng)用程序?qū)⒉辉龠\(yùn)行,因此當(dāng)系統(tǒng)喚醒時(shí),,應(yīng)用程序看起來(lái)沒(méi)有被控制,。然而,,有一些方式來(lái)喚醒一個(gè)掛起的設(shè)備。首先,,一個(gè)應(yīng)用程序通過(guò)給定一個(gè)時(shí)間,,并使用11章提到的消息API(Notification API)做系統(tǒng)被喚醒的計(jì)劃。在一般情況下,,OEM廠商會(huì)分配一些中斷條件,,以便管理系統(tǒng)電源打開(kāi),或喚醒,。這種方式的一個(gè)例子是一個(gè)系統(tǒng)當(dāng)防止了一個(gè)同步架(synchronization cradle)時(shí)被喚醒,。
防止系統(tǒng)關(guān)閉電源
相反的情況,,防止系統(tǒng)掛起也是一個(gè)問(wèn)題,。Windows CE系統(tǒng)通常被設(shè)置為當(dāng)一段時(shí)間沒(méi)有用戶輸入就自動(dòng)掛起。要防止自動(dòng)掛起,,一個(gè)應(yīng)用程序可以周期性地調(diào)用一下函數(shù):
void WINAPI SystemIdleTimerReset (void);
這個(gè)函數(shù)重設(shè)Windows CE用來(lái)監(jiān)視用戶輸入的定時(shí)器,。如果定時(shí)器到達(dá)預(yù)先的沒(méi)有用戶輸入的間隔,系統(tǒng)會(huì)自動(dòng)掛起,。因?yàn)閽炱鸪瑫r(shí)值可以被改變,,一個(gè)應(yīng)用程序需要知道超時(shí)值,這樣就要多一點(diǎn)調(diào)用SystemIdleTimerReset,。系統(tǒng)維護(hù)三個(gè)超時(shí)值,,這些都能夠使用SystemParametersInfo來(lái)查詢。傳遞給SystemParametersInfo的常量的不同表現(xiàn),,顯示如下:
SPI_GETBATTERYIDLETIMEOUT
當(dāng)系統(tǒng)運(yùn)行在電池電源狀態(tài)下,,離用戶最后輸入的時(shí)間
SPI_GETEXTERNALIDLETIMEOUT
當(dāng)系統(tǒng)運(yùn)行在AC電源狀態(tài)下,離用戶最后輸入的時(shí)間
SPI_GETWAKEUPIDLETIMEOUT
在系統(tǒng)再次掛起時(shí)離系統(tǒng)被自動(dòng)喚醒的時(shí)間
要防止電源被自動(dòng)掛起,,你需要查詢這三個(gè)值,,并在最短時(shí)間內(nèi)返回之前調(diào)用SystemIdleTimerReset。如果超時(shí)值被設(shè)置為0,,表示超時(shí)值被禁止,。
電源管理程序
一個(gè)新的,獨(dú)立的電源管理組件在Windows CE .NET 4.0中被引入了,。這個(gè)電源管理程序替代了許多GWES以前完成的函數(shù),。電源管理程序定義了一系列的電源狀態(tài),如D0,,D1,,D2,和D3,。這些看起來(lái)神秘的名字被對(duì)應(yīng)于一些友好的系統(tǒng)級(jí)別名稱,。
對(duì)嵌入式" title="嵌入式">嵌入式系統(tǒng)來(lái)說(shuō),,OEM廠商定義了系統(tǒng)的電源狀態(tài)。例如,,電源狀態(tài)可能是打開(kāi)(On),,空閑(Idle)和掛起(Suspend)。其他電源狀態(tài)也被定義了,,像ScreenOff,, InCradle, 和 OnBattery,。
從應(yīng)用程序的觀點(diǎn)看,,新的電源管理程序提供了通知電源狀態(tài)改變的能力以及通過(guò)一系列的函數(shù)統(tǒng)一改變電源狀態(tài)的能力。
系統(tǒng)的電源狀態(tài)被定義在注冊(cè)表中,,SDK定義了PWRMGR_REG_KEY,,以致你不得不知道注冊(cè)表的字符串,但是當(dāng)常量沒(méi)定義的時(shí)間,,電源管理程序注冊(cè)數(shù)據(jù)被保留在HKEY_LOCAL_MACHINE\System\CurrentControlSet\Control\Power,。電源狀態(tài)被定義作為子鍵,位于Key State,。
電源通知
電源管理程序一個(gè)十分受歡迎的特點(diǎn)是,,可以在系統(tǒng)電源狀態(tài)改變時(shí)通知應(yīng)用程序。這可以讓?xiě)?yīng)用程序從手動(dòng)檢測(cè)電源狀態(tài)中解脫出來(lái),。一個(gè)應(yīng)用程序可以通過(guò)調(diào)用RequestPowerNotifications請(qǐng)求電源管理程序當(dāng)電源狀態(tài)改變的時(shí)候發(fā)送一個(gè)通知給應(yīng)用程序,。電源管理程序會(huì)通過(guò)一個(gè)由應(yīng)用程序前面建立的消息隊(duì)列發(fā)送通知。
RequestPowerNotifications原型如下,。
HANDLE RequestPowerNotifications (HANDLE hMsgQ,, DWORD Flags);
第一個(gè)參數(shù)是一個(gè)應(yīng)用程序在之前建立的消息隊(duì)列的句柄。第二個(gè)參數(shù)是一系列參數(shù),,表示應(yīng)用程序想接收的通知,。
PBT_TANSITION
接受系統(tǒng)電源狀態(tài)改變的通知。例如,,當(dāng)系統(tǒng)從On到Suspend,。
PBT_RESUME
當(dāng)系統(tǒng)resume的時(shí)候接收通知。
PBT_POWERSTATUSCHANGE
當(dāng)系統(tǒng)在AC和電池之間切換的時(shí)候接收通知,。
PBT_POWERINFOCHANGE
當(dāng)系統(tǒng)電池級(jí)數(shù)變化時(shí)接收通知,。
POWER_NOTIFY_ALL
接收所有的通知。
RequestPowerNotifications函數(shù)返回一個(gè)電源通知的句柄,,失敗返回NULL,。消息隊(duì)列建立的時(shí)候必須使應(yīng)用程序有讀權(quán)限,因?yàn)閼?yīng)用程序?qū)南㈥?duì)列中讀取電源通知。
要接收通知,,應(yīng)用程序必須使用WaitForSingleObject來(lái)阻塞消息句柄,。像第10章所討論的,當(dāng)通知被放在隊(duì)列中時(shí),,句柄將被signaled,。實(shí)際的通知將由結(jié)構(gòu)POWER_BROADCAST表中被接收到。
typedef struct _POWER_BROADCAST {
DWORD Message;
DWORD Flags;
DWORD Length;
WCHAR SystemPowerState[1];
} POWER_BROADCAST,, *PPOWER_BROADCAST;
第一個(gè)要注意的是,,這個(gè)結(jié)構(gòu)長(zhǎng)度是可變的。最后一個(gè)字段,,SystemPowerState,,是被定義為WCHARs類型,但是可以填上非字符串?dāng)?shù)據(jù),。第一個(gè)字段是通知自己的標(biāo)識(shí),,這個(gè)字段可以填前面PBT_標(biāo)志列表之一。Flags區(qū)可以包括以下標(biāo)志,,依賴于被接收的通知:
POWER_STATE_ON
系統(tǒng)處于on狀態(tài),。
POWER_STATE_OFF
系統(tǒng)處于off狀態(tài),。
POWER_STATE_CRITICAL
系統(tǒng)進(jìn)入了一個(gè)臨界off狀態(tài),。
POWER_STATE_BOOT
系統(tǒng)正在啟動(dòng)。
POWER_STATE_IDLE
系統(tǒng)進(jìn)入idle狀態(tài),。
POWER_STATE_SUSPEND
系統(tǒng)被掛起,。
POWER_STATE_RESET
系統(tǒng)被復(fù)位。
最后兩個(gè)字段是相互關(guān)聯(lián)的,。Length字段是SystemPowerState字段數(shù)據(jù)的長(zhǎng)度,。SystemPowerState中包含的數(shù)據(jù)依賴于被發(fā)送的通知。對(duì)于PBT_TRANSITION通知來(lái)說(shuō),,SystemPowerState字段包含一個(gè)新電源狀態(tài)的標(biāo)識(shí)字符串,。這個(gè)字符串是以非0結(jié)尾的。為了結(jié)束字符串,,使用Length字段來(lái)指出字符串的長(zhǎng)度,。注意,Length字段是以字節(jié)為單位的,,當(dāng)字符是雙字節(jié)的Uncode字符時(shí),,需要獲得字符串字符的長(zhǎng)度,就需要用Length字段去除TCHAR的size,。
對(duì)于PBT_POWERINFOCHANGE通知來(lái)說(shuō),,SystemPowerState字段包含一個(gè)PPOWER_BROADCAST_POWER_INFO結(jié)構(gòu):
typedef struct _POWER_BROADCAST_POWER_INFO {
DWORD dwNumLevels;
DWORD dwBatteryLifeTime;
DWORD dwBatteryFullLifeTime;
DWORD dwBackupBatteryLifeTime;
DWORD dwBackupBatteryFullLifeTime;
BYTE bACLineStatus;
BYTE bBatteryFlag;
BYTE bBatteryLifePercent;
BYTE bBackupBatteryFlag;
BYTE bBackupBatteryLifePercent;
} POWER_BROADCAST_POWER_INFO, *PPOWER_BROADCAST_POWER_INFO;
注意,這里有一些字段的名字和函數(shù)十分相似于前面討論的SYSTEM_POWER_STATUS_EX2結(jié)構(gòu),。
設(shè)置電源狀態(tài)
電源管理程序提供的函數(shù)也允許應(yīng)用程序來(lái)控制電源狀態(tài),。有兩個(gè)方式來(lái)控制電源。第一個(gè)方式是應(yīng)用程序給定一個(gè)電源設(shè)定,。第二個(gè)方式是應(yīng)用程序請(qǐng)求電源狀態(tài)不要低于給定的級(jí)別,。
一個(gè)應(yīng)用程序通過(guò)調(diào)用函數(shù)SetSystemPowerState可以請(qǐng)求特定的電源狀態(tài)。這個(gè)函數(shù)原型如下,。
DWORD SetSystemPowerState (LPCWSTR psState,, DWORD StateFlags,
DWORD Options);
電源狀態(tài)可以被請(qǐng)求通過(guò)指定前兩個(gè)參數(shù),。如果第一個(gè)參數(shù)是非零值,,它指向一個(gè)字符串標(biāo)識(shí)被請(qǐng)求的狀態(tài)。這個(gè)字符串必須和注冊(cè)表中列出的電源狀態(tài)之一相匹配,。
如果psState 為 NULL,,第二個(gè)參數(shù)StateFlags,定義了請(qǐng)求的電源狀態(tài),。這個(gè)參數(shù)是從POWER_STATE_ON直到POWER_STATE_RESET狀態(tài)其中之一,,這些在前面提到的POWER_BROADCAST結(jié)構(gòu)有描述。
比較特別的是POWER_STATE_RESET標(biāo)志,。這個(gè)標(biāo)志請(qǐng)求系統(tǒng)重起,,使用SetSystemPowerState的方法重起比通過(guò)直接使用IOCTL_HAL_REBOOT命令來(lái)調(diào)用KernelIoControl的方法更好。調(diào)用 SetSystemPowerState 會(huì)讓系統(tǒng)在重起設(shè)備之前任何還在緩沖中的數(shù)據(jù)保存到文件系統(tǒng),。
調(diào)用SetSystemPowerState是一個(gè)直接改變電源狀態(tài)的方法,。更巧妙的方法是通過(guò)調(diào)用SetPowerRequirement來(lái)請(qǐng)求系統(tǒng)維持應(yīng)用程序所需最低限度的電源狀態(tài)。SetSystemPowerState是假定應(yīng)用程序知道所需狀態(tài),,而調(diào)用SetPowerRequirement是允許系統(tǒng)對(duì)電源設(shè)定做優(yōu)化以滿足應(yīng)用程序的需要,。一個(gè)使用SetPowerRequirement會(huì)比較方便的例子是,一個(gè)使用串口的應(yīng)用程序需要串口在進(jìn)行通信時(shí)保持住電源狀態(tài),。SetPowerRequirement被定義如下,。
HANDLE SetPowerRequirement (PVOID pvDevice,
CEDEVICE_POWER_STATE DeviceState,,
ULONG DeviceFlags,, PVOID pvSystemState,
ULONG StateFlags);
第一個(gè)參數(shù)指定了應(yīng)用程序需要維護(hù)電源狀態(tài)的設(shè)備,。DeviceState參數(shù)定義了設(shè)備的電源狀態(tài),。CEDEVICE_POWER_STATE指定了狀態(tài)范圍是從D0(意味著設(shè)備是處于最大功耗狀態(tài))到D4(表示設(shè)備被關(guān)閉)(譯者注:其實(shí)D0到D4的狀態(tài)的具體表現(xiàn),完全是由OEM廠商可自定義的,,對(duì)應(yīng)用程序開(kāi)發(fā)者來(lái)說(shuō),,比如是在D1關(guān)LCD背光還是在D2,都是不確定的,微軟只給出標(biāo)準(zhǔn)定義,,而不是實(shí)際定義),。DeviceFlags參數(shù)由兩個(gè)標(biāo)志合并而成:POWER_NAME,表示設(shè)備名有效,;POWER_FORCE,,表示設(shè)備應(yīng)當(dāng)維持當(dāng)前狀態(tài)甚至當(dāng)系統(tǒng)掛起時(shí)。如果pvSystemState不為NULL,,它表示只有對(duì)于在pvSystemState中已命名的電源請(qǐng)求才是有效的,。設(shè)備可能無(wú)法更改請(qǐng)求的狀態(tài)。
應(yīng)用程序應(yīng)當(dāng)注銷通過(guò)調(diào)用ReleasePowerRequirement來(lái)注銷請(qǐng)求,,原型如下,。
DWORD ReleasePowerRequirement (HANDLE hPowerReq);
這里唯一的參數(shù)是從SetPowerRequirement里返回的句柄。