學(xué)習Symbian的基本概念
一、基本數據類(lèi)型
本文引用地址:http://dyxdggzs.com/article/201609/303391.htm這個(gè)比較簡(jiǎn)單,都在e32def.h中寫(xiě)著(zhù)呢。也就是以下這幾個(gè)類(lèi)型要注意一下,以后照著(zhù)寫(xiě)罷了,含義也很明了,不用多說(shuō)。
typedef void TAny;
typedef signed char TInt8;
typedef unsigned char TUint8;
typedef short int TInt16;
typedef unsigned short int TUint16;
typedef long int TInt32;
typedef unsigned long int TUint32;
typedef signed int TInt;
typedef unsigned int TUint;
typedef float TReal32;
typedef double TReal64;
typedef double TReal;
typedef unsigned char TText8;
typedef unsigned short int TText16;
typedef int TBool;
typedef TUint32 TLinAddr; //Defines a linear (virtual) address type.
二、描述符
這個(gè)東西比較有趣,其實(shí)說(shuō)白了,也就是兩個(gè)我們以前常用的玩意兒:String和malloc。不過(guò)在symbian中把描述符分為三類(lèi):緩沖描述符、指針描述符和堆描述符。
A.緩沖描述符:TBuf、TBufC
類(lèi)似于char[],也就是說(shuō)它是一個(gè)字符串的表示方法,比如:TBuf20> str;與我們以前寫(xiě)的char str[20];意思基本一樣。
不過(guò),描述符可以包含一些方法,就象我們用String主要就是沖著(zhù)它的方便的字串處理方法去的。
B.指針描述符:TPtr、TPtrC
類(lèi)似于char *,也就是說(shuō)這是一個(gè)字符(字節)指針的另類(lèi)表示罷了。
C.堆描述符:HBufC
類(lèi)似于我們用malloc開(kāi)辟的一塊空間,比如:HBufC * buf = HBufC::NewL(128);與 byte * buf = (byte*)malloc(128);的意思也是基本一樣的。
還有一種抽象的描述符TDes和TDesC,是其它描述符的基類(lèi)。
所有描述符名稱(chēng)后面的C表示它是一個(gè)不可修改的描述符。換句話(huà)說(shuō)就是,所有不帶C的描述符是在帶C描述符的基礎上增加了一些進(jìn)行修改操作的函數。
要記住的是描述符帶給我們的便利。
比如這幾個(gè)函數:
Length() 得到字串的真實(shí)長(cháng)度(元素個(gè)數),而Size()則是得到它所占的字節數。
Left()/Right()/Mid() 是用來(lái)獲取子字符串的函數。
Compare() 比較函數。
Locate()/LocateReverse()/Find()/Match() 則可以查找子串或字符。
Copy()/Delte()/Insert()/Replace()/Trim()/Append()/Zero() 則可以對描述符的內容進(jìn)行修改操作。
Num() 可以將數值轉成字符串。
Format() 類(lèi)似于sprintf,比較常用,格式化輸出。不過(guò)還有同系的其它函數也許更方便,如AppendFormat()/AppendNum()等等。
對于堆描述符,需要注意的地方有三點(diǎn):
一是Des(),因為HBufC帶C是不可修改的描述符,所以如果我們要修改它,則需要用buf.Des()得到一個(gè)指向它的指針描述符。如下句:
_LIT(KHello,hello china);
HBufC * buf = HBufC::NewL(64);
*buf=KHello;
TPtr p = buf->Des();
p[0]='H';
二是在TDesC中有一個(gè)AllocLC()可以分配內存得到HBufC描述符,與HBufC::NewL()是一樣的。而且HBufC中也有ReAllocL可以重新分配內存,就象realloc一樣。
三是區分一下兩句話(huà)的含義:
TPtr p=buf->Des();
TPtr p(buf->Des());
第一句只是根據buf當前的真實(shí)長(cháng)度得到一個(gè)指針(p的最大長(cháng)度與當前的實(shí)際長(cháng)度一樣,就是buf此時(shí)的真實(shí)長(cháng)度11),而第二句則完全用buf的信息來(lái)構造了p,所以它的最大長(cháng)度應該是64,雖然當前的真實(shí)長(cháng)度也是11。
還有一個(gè)與描述符相關(guān)的宏很常用,需要注意一下:_LIT(常量名稱(chēng),字串值)。比如:
_LIT(KSayHelloSTR,Hello world.);
而那個(gè)_L宏不提倡用了,因為效率太低的原因。
這里的KSayHelloSTR是另一種描述符TLitC。而TLitC提供兩個(gè)運算符要注意:
操作符能得到它的const TDesC*,而()操作符則得到它的const TDesC。
KSayHelloSTR().Length(); //得到這個(gè)字串的長(cháng)度
TBuf256> str;
str.Format(KFormatSTR,KSayHelloSTR); //得到這個(gè)字串的引用地址
具體關(guān)于描述符的信息可以參考H文件:e32des16.h
三、錯誤處理
有三個(gè)概念,一是Leave。最常見(jiàn)的地方是對new操作符的重載new(ELeave),表示此時(shí)的構造會(huì )產(chǎn)生內存不足的現象。
有了new(ELeave),我們就可以放心地在new一個(gè)對象之后直接使用它而不需要去判斷是否構造成功,因為如果不成功會(huì )拋出錯誤并返回上層。
第二個(gè)概念配合Leave,有一個(gè)宏TRAPD(error,Func)類(lèi)似于try...catch...。
也就是說(shuō)如果Func函數中發(fā)生了Leave,則error能得到錯誤碼。一般在程序中可以用User::Leave()來(lái)拋出錯誤,類(lèi)似于throw new Exception的操作。
第三個(gè)概念也是最常用的,就是清理棧CleanupStack的使用。
取代TRAPD宏的使用,我們可以在可能發(fā)生Leave之前將指針push到cleanupstack中,在正確完成之后再將它pop出來(lái),如果萬(wàn)一不成功,系統會(huì )幫我們將cleanupstack中的東西銷(xiāo)掉。這樣就方便了很多。
一般的代碼類(lèi)似于:
CMyCls * mc=new (ELeave) CMyCls;
CleanupStack::PushL(mc);
me->doSth1L();
me->doSth2L();
CleanupStack::PopAndDestroy();
一般我們是將局部變量用cleanupstack來(lái)保護一下,但是對于類(lèi)的成員變量則不能這樣做(否則會(huì )二次銷(xiāo)毀,產(chǎn)生嚴重錯誤)。
此外,在pop時(shí)超出范圍了,壓幾個(gè)就彈幾個(gè),如果不小心把其它的內容給彈出來(lái),也可能會(huì )引起嚴重錯誤的。所以,Pop有一個(gè)重載Pop(3,pA)這里表示彈出三個(gè)對象并且比較一下最后出棧的是不是pA。
四、兩階段構造
一般Symbian的類(lèi)不會(huì )提供public的構造函數(不建議這么做),因為它提倡所謂的“兩階段構造”方法。定義為:
A)構造函數是protected或者privted的,并且不能包含引起Leave的操作。
B)實(shí)現兩個(gè)靜態(tài)函數NewL和NewLC,來(lái)代替構造函數提供給用戶(hù)使用。
C)實(shí)現一個(gè)ContructL函數實(shí)現第二階段的構造,其實(shí)大多數初始化的工作可以放在這里進(jìn)行。
兩階段構造的代碼相當地格式化了,比如兩個(gè)靜態(tài)函數NewL和NewLC的代碼一般是這樣的:
CMyClass * CMyClass::NewL()
{
CMyClass * self=NewLC();
CleanupStack::Pop();
return self;
}
CMyClass * CMyClass:NewLC()
{
CMyClass * self = new(ELeave) CMyClass;
CleanupStack::PushL(self);
self->ContructL();
return self;
}
而在ContructL中一般可以做一些真正的構造函數里的操作,例如分配內存、創(chuàng )建窗體之類(lèi)的活兒。
五、命名規則
這也是一個(gè)需要注意的地方,按教材上所言列舉如下:
類(lèi)的名稱(chēng)前綴有T、C、M和R四個(gè),分述如下:
T表示基本類(lèi),它位于棧里,就當作是一個(gè)結構吧。
C表示常規的類(lèi),繼承于CBase的,這是C++標準的類(lèi)的概念,所以有構造要析構。
M表示是一個(gè)接口,很好理解,它肯定含有純虛函數。
R表示是一個(gè)系統資源,比如文件、網(wǎng)絡(luò )等等,所以它肯定有Open有Close。
還有,K開(kāi)頭表示常量、E開(kāi)頭表示枚舉也要記得。
形參用a開(kāi)頭,類(lèi)成員變量用i開(kāi)頭,這此規則我們在自動(dòng)生成的代碼中也能看到。
還有函數的命名上也有講究,不過(guò)不是開(kāi)頭而是結尾:象L表示可能會(huì )有Leave,LC表示不但可能有Leave而且它會(huì )被自動(dòng)放在CleanupStack中。
還有兩個(gè)二階段構造又有三個(gè)函數名稱(chēng)固定了:NewL、NewLC和ContructL。
評論