?? 00000029.htm
字號(hào):
<?xml version="1.0" encoding="gb2312"?><!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"><html xmlns="http://www.w3.org/1999/xhtml"><head><meta http-equiv="Content-Type" content="text/html; charset=gb2312"/><title>Microsoft Windows CE 編程的十點(diǎn)忠告 turbolinux </title></head><body><center><h1>BBS 水木清華站∶精華區(qū)</h1></center><a name="top"></a>發(fā)信人: encarta (知更鳥(niǎo)), 信區(qū): Embedded <br />標(biāo) 題: Microsoft Windows CE 編程的十點(diǎn)忠告 <br />發(fā)信站: BBS 水木清華站 (Thu Aug 10 21:37:50 2000) <br /> <br />以下轉(zhuǎn)載自微軟網(wǎng)站 <br /> <br /> <br />Microsoft Windows CE 編程的十點(diǎn)忠告 <br />---------------------------------------------------------------------------- <br />---- <br /> 最近兩周我們花了大部分時(shí)間將已有的應(yīng)用程序移植到Microsoft Windows CE中。 <br />一般說(shuō)來(lái),這個(gè)計(jì)劃不是太難。我們起步于Microsoft Win32代碼,當(dāng)然 Windows CE是 <br />基于Win32應(yīng)用程序接口(API)的。有利的是,我們的應(yīng)用程序(即Raima 數(shù)據(jù)管理器 <br />)有方便的使用接口,并包含一個(gè)大約由150個(gè)子函數(shù)組成的庫(kù),這些函數(shù)都是由C語(yǔ)言 <br />寫(xiě)成,可以用來(lái)創(chuàng)建、管理和訪問(wèn)數(shù)據(jù)庫(kù)。 <br /> 按建立應(yīng)用程序的方式來(lái)說(shuō),我們?cè)詾閷⑺浦驳絎indows CE中是一項(xiàng)相對(duì)簡(jiǎn)單 <br />的C語(yǔ)言編程練習(xí)。然而,我們不久便遇到好些困難。從粗心大意的錯(cuò)誤開(kāi)始,比如在基 <br />于Windows NT 的Windows CE仿真器上使用Microsoft Windows NT庫(kù),接著又違背Windo <br />ws CE的編程戒律,如"千萬(wàn)不要給Unicode(國(guó)際標(biāo)準(zhǔn)組織10646標(biāo)準(zhǔn))字符分配奇數(shù)內(nèi) <br />存地址"。 <br /> 大約有百分之九十的問(wèn)題或多或少地與Unicode有關(guān)。盡管Unicode編程不難,但是 <br />,當(dāng)給單字節(jié)字符編寫(xiě)代碼時(shí),很容易出錯(cuò)(我有過(guò)許多次錯(cuò)誤)。 <br /> 下面這些忠告是根據(jù)我們?cè)赪indows CE上編寫(xiě)Raima 數(shù)據(jù)管理器的經(jīng)驗(yàn)總結(jié)出來(lái)的 <br />,但我相信,在做任何其它Windows CE程序之前,它們都值得借鑒。畢竟大多數(shù)Window <br />s開(kāi)發(fā)者,當(dāng)他們創(chuàng)建第一個(gè)Windows CE應(yīng)用程序時(shí),真正運(yùn)用的是已掌握的Win32知識(shí) <br />。 <br />1. 不要在仿真器上使用Windows NT庫(kù) <br /> 這里所討論的第一個(gè)錯(cuò)誤實(shí)在太愚蠢了,但我還是陷了進(jìn)去,也許你也會(huì)。當(dāng)用Mi <br />crosoft VC++(5.0版)創(chuàng)建一個(gè)Windows CE程序時(shí),你會(huì)發(fā)現(xiàn),包含路徑(include) <br />、 庫(kù)路徑(library)、及可執(zhí)行程序路徑被自動(dòng)調(diào)整以匹配反應(yīng)目標(biāo)環(huán)境的選擇。因 <br />此,比如說(shuō)為Windows CE模擬器建立應(yīng)用程序時(shí),你會(huì)發(fā)現(xiàn),include路徑?jīng)]有指向Win <br />32的包含文件(在VC目錄下),而是指向Windows CE包含文件(在WCE目錄下)。千萬(wàn)別 <br />去修改。 <br /> 由于Windows CE在Windows NT下運(yùn)行,所以仿真器上運(yùn)行的程序能夠調(diào)用任一Wind <br />ows NT動(dòng)態(tài)鏈接庫(kù)(DLL)中的函數(shù),即使這個(gè)DLL不是模擬器的成員也一樣。顯然,這不 <br />是很好的事,因?yàn)橄嗤暮瘮?shù)也許在手持PC(H/PC)或Windows CE設(shè)備上不可用,而你的 <br />軟件最終要能在這些設(shè)備上運(yùn)行。 <br /> 第一次將非Unicode應(yīng)用程序裝入Windows CE仿真器時(shí),你會(huì)發(fā)現(xiàn),許多正在使用的 <br />函數(shù)它都不支持,例如美國(guó)國(guó)家標(biāo)準(zhǔn)協(xié)會(huì)(ANSI)定義的字符函數(shù)strcpy()。這也許引誘 <br />你去鏈接Windows NT 運(yùn)行時(shí)間庫(kù),以便能解決所有問(wèn)題。 <br /> 如果你是剛開(kāi)始用Windows CE編程,可能你能用的包含文件和庫(kù)文件是明顯的。答 <br />案就是,你不要采用那些在寫(xiě)普通Win32或非Windows CE程序時(shí)使用的包含文件和庫(kù)文件 <br />。 <br />2. 不要混淆TCHARs和bytes <br /> 如果你正在Windows CE上寫(xiě)非Unicode應(yīng)用程序,你或許要將所有的字符串從單個(gè)字 <br />符(chars)轉(zhuǎn)換為寬字符(widechars)(例如,C變量類(lèi)型whcar_t)。幾乎所有Windows <br />CE支持的Win32和運(yùn)行時(shí)間庫(kù)函數(shù)都要求寬字符變量。Windows 95不支持Unicode,然而 <br />,為了使程序代碼具有可移植性,你要盡可能采用tchar.h中定義的TCHAR類(lèi)型,不要直 <br />接使用wchar_t。 <br /> TCHAR是定義為wchar_t還是char,取決于預(yù)處理器的符號(hào)UNICODE是否定義。同樣, <br />所有有關(guān)字符串處理函數(shù)的宏,如_tcsncpy宏,它是定義為Unicode函數(shù)wcsncpy還是定 <br />義為ANSI函數(shù)strncpy,取決于UNICODE是否定義。 <br /> 在現(xiàn)存的Windows應(yīng)用程序中,有些代碼也許暗示字符長(zhǎng)為單字節(jié)。這在給字符串分 <br />配內(nèi)存時(shí)經(jīng)常用到,例如: <br />int myfunc(char *p) <br />{ <br />char *pszFileName; <br />pszFileName = malloc(MAXFILELEN); <br />if(pszFileName) <br />strncpy(pszFileName, p, MAXFILELEN); <br />/*etc*/ <br /> 在這段代碼中,分配的內(nèi)存塊應(yīng)該寫(xiě)作(MAXFILELEN * sizeof(char)),但是大多數(shù) <br />程序員喜歡將它簡(jiǎn)化為MAXFILELEN,因?yàn)閷?duì)于所有的平臺(tái)來(lái)說(shuō)sizeof(char)的值等于1。 <br />然而,當(dāng)你用TCHARS代替多個(gè)字符時(shí),很容易忘記這種固有的概念,于是將代碼編寫(xiě)成 <br />下面的形式: <br />int myfunc(TCHAR *p) <br />{ <br />TCHAR *pszFileName; <br />PszFileName = (TCHAR*)malloc(MAXFILELEN); <br />If (pszFileName) <br />tcsncpy(pszFileName, p, MAXFILELEN); <br />/*etc*/ <br /> 這是不行的。它馬上會(huì)導(dǎo)致出錯(cuò)。這里的錯(cuò)誤在于malloc函數(shù)中指定變量大小為by <br />tes,然而_tcsncpy函數(shù)中使用的第三個(gè)變量卻指定為T(mén)CHARs而不是bytes。當(dāng)UNICODE被 <br />定義時(shí),一個(gè)TCHAR等于兩個(gè)字節(jié)數(shù)(bytes)。 <br />上述代碼段應(yīng)該改寫(xiě)為: <br />int myfunc(TCHAR *p) <br />{ <br />TCHAR *pszFileName; <br />PszFileName = (TCHAR*)malloc(MAXFILELEN * sizeof(TCHAR)); <br />if(pszFileName) <br />tcsncpy(pszFileName, p, MAXFILELEN); <br />/*etc*/ <br />3. 不要將Unicode 字符串放入奇數(shù)內(nèi)存地址 <br /> 在Intel系列處理器上,你可以在一奇數(shù)內(nèi)存地址儲(chǔ)存任何變量或數(shù)組,不會(huì)導(dǎo)致任 <br />何致命的錯(cuò)誤影響。但在H/PC上,這一點(diǎn)不一定能行 ? 你必須對(duì)大于一個(gè)字節(jié)的數(shù)據(jù)類(lèi) <br />型小心謹(jǐn)慎,包括定義為無(wú)符號(hào)短型(unsigned short) 的wchar_t。當(dāng)你設(shè)法訪問(wèn)它 <br />們的時(shí)候,將它們置于奇地址會(huì)導(dǎo)致溢出。 <br /> 編輯器經(jīng)常在這些問(wèn)題上提醒你。你無(wú)法管理堆棧變量地址,并且編輯器會(huì)檢查確 <br />定這些地址與變量類(lèi)型是否相匹配。同樣,運(yùn)行時(shí)間庫(kù)必須保證從堆中分配的內(nèi)存總是 <br />滿足一個(gè)word邊界 ,所以你一般不必?fù)?dān)心那兩點(diǎn)。但是,如果應(yīng)用程序含有用memcpy( <br />)函數(shù)拷貝內(nèi)存區(qū)域的代碼,或者使用了某種類(lèi)型的指針?biāo)阈g(shù)以確定內(nèi)存地址,問(wèn)題也許 <br />就出現(xiàn)了。考慮下面的例子: <br />int send_name (TCHAR * pszName) <br />{ <br />char *p, *q; <br />int nLen=(_tcslen(pszName) + 1) * sizeof(TCHAR); <br />p=maloc(HEADER_SIZE + nLen); <br />if(p) <br />{ <br />q = p + HEADER_SIZE; <br />_tcscpy((TCHAR*)q, pszName); <br />} <br />/* etc */ <br /> 這段代碼是從堆中分配內(nèi)存并復(fù)制一個(gè)字符串,在字符串的開(kāi)頭留一個(gè)HEADER_SIZ <br />E的大小。假設(shè)UNICODE定義了,那么該字符串就是一個(gè)widechar字符串。如果HEADER_S <br />IZE是一個(gè)偶數(shù),這段代碼就會(huì)正常工作,但如果HEADER_SIZE為奇數(shù),這段代碼就會(huì)出 <br />錯(cuò),因?yàn)閝指向的地址也將為奇數(shù)。 <br /> 注意,當(dāng)你在Intel系列處理器中的Windows CE仿真器上測(cè)試這段代碼時(shí),這個(gè)問(wèn)題 <br />是不會(huì)發(fā)生的。 <br /> 在這個(gè)例子中,只要確保HEADER_SIZE為偶數(shù),你就可以避免問(wèn)題的發(fā)生。然而,在 <br />某些情況下你也許不能這么做。例如,如果程序是從一臺(tái)式PC輸入數(shù)據(jù),你也許不得不 <br />采用事先定義過(guò)的二進(jìn)制格式,盡管它對(duì)H/PC不適合。在這種情況下,你必須采用函數(shù) <br />,這些函數(shù)用字符指針控制字符串而不是TCHAR指針。如果你知道字符串的長(zhǎng)度,就可以 <br />用memcpy()復(fù)制字符串。因此,采用逐個(gè)字節(jié)分析Unicode字符串的函數(shù)也許足以確定字 <br />符串在widechars中的長(zhǎng)度。 <br />4. 在ANSI和Unicode字符串之間進(jìn)行翻譯 <br /> 如果你的Windows CE應(yīng)用程序接口于臺(tái)式PC,也許你必須操作PC機(jī)中的ANSI字符串 <br />
?? 快捷鍵說(shuō)明
復(fù)制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號(hào)
Ctrl + =
減小字號(hào)
Ctrl + -