描述符<下>轉換
業余有時候把一個事情當作任務來做的時候,往往會很受限制,就象這篇文檔,上次寫了個上篇,關于描述符的下篇,自己就遲遲沒有勇氣和時間寫完,好幾次都想靜下心來好好完成它,但是都未能如愿,可如果不寫顯然我也不好意思寫其它的一些東西,閑話提到這里,花了幾天業余時間總算是完成了,該文質量不足之處還望讀者您見諒。
通過前面關于描述符概念和使用的簡單描述,下面我們直接進入描述符轉換的主題。
雖然描述符有五類,但是作為描述符變量用的,只有三類即棧描述符、堆描述符、指針描述符。所以本文所述描述符的轉換也只是在這三種類型間展開,這三類描述符之間的互相轉換其實可以用下面這種偽代碼方式給出:
TBuf<20> iBuf;
TPtr iPtr((const_cast<TUint16*>(iBuf.Ptr())),iBuf.Length(), iBuf.Length());
HBufC *iHbuf;
iBuf = iPtr;
iBuf = iHbuf->Des();
iPtr.Set((const_cast<TUint16*>(iBuf.Ptr())),iBuf.Length(), iBuf.Length());
由上面可知指針型可以起堆型和棧型描述符的中間過渡的橋梁作用。
不可修改向可修改描述符的轉換
原則1:通過不可修改描述符類內的Des()函數,將不可修改的描述符轉換成可修改的指針描述符
示例1:TBufC轉換成TPtr
_LIT(KText, "Test Data");
TBufC<10> NBuf ( KText );
TPtr Pointer = NBuf.Des();
示例2:HBufC轉換成TPtr
HBufC * Buf = HBufC::NewL(15);
_LIT (KText , "Test Text");
*Buf = KText;
TPtr Pointer = Buf->Des();
原則2:通過TPtr的構造函數或Set()函數可以將TPtrC描述轉換為可修改的指針描述符
示例3:TPtrC到TPtr
const TText * text1 = _S("Hello World\n");
TPtrC Ptr1(text1);
TPtrC Ptr2(Ptr1);
//可以通過構造函數
TPtr Ptr3((TUint16 *)(Ptr1.Ptr()), Ptr1.Length());
//也可以通過Set()函數
Ptr3.Set((TUint16 *)(Ptr1.Ptr()),Ptr1.Length(), Ptr1.Length());
8位和16位之間的描述符轉換
8位到16位的描述符轉換
原則1:通過將兩個8位描述符的內容合并為一個16位描述符的方法實現內存不動,內存塊類型強轉(我稱其為保留描述符字節大小方法)。
示例4:8bit To 16bit保留描述符字節大小方法1
_LIT8(KText , "Test Text");
TBuf8<20> iBuf8(KText);//描述符實際占用了9個字節9個8位描述符
TBuf<20> iBuf16;
//實現內存塊的強轉
TPtrC16 ptr16(reinterpret_cast<const TUint16*>(iBuf8.Ptr()), (iBuf8.Length()/2));
//真實的內存不動已經結束,這里是重新申請了一塊內存并實現內存賦值拷貝
iBuf16 = ptr16;//描述符實際占用了8個字節4個16位描述符,導致數據丟失
由上面的這個例子明顯是丟失了數據了,原因就出在設置長度時用了(iBuf8.Length()/2),為了剔除錯誤,我后來改了一個新的示例6
示例6:8bit To 16bit保留描述符字節大小方法2(由于上例用了TPtrC,所以該例使用TPtr指針,以擴展應用)
_LIT8(KText , "Test Text");
TBuf8<20> iBuf8;
iBuf8.FillZ(20);
iBuf8 = KText;//實際占用9字節9描述符
TBuf<20> iBuf16;
TPtr ptr16(reinterpret_cast<TUint16*>(const_cast<TUint8*>(iBuf8.Ptr())),
((iBuf8.Length()+1)/2),((iBuf8.Length()+1)/2));
iBuf16 = ptr16;//實際占用10字節5描述符,但是第10個字節為0零值,也即9字節大小有用
注:其實這里用堆描述符來說明可能更好些,但是本人在調試過程中為了能夠看到調試的效果,所以用了TBuf;而且個人覺得通常這種方法主要在文件讀取等實際過程中用到,如果要轉換為16位的話,肯定是偶數大小的8位描述符。
原則2:重新構建內存塊使用描述符的copy函數將8位字符的高字節置為0低字節不變,進行拷貝轉換(我稱其為保留描述符長度方法)。
示例7
_LIT8(KText , "Test Text");
TBuf8<20> iBuf8(KText);//實際占用了9個描述符9個字節
TBuf<20> iBuf16;
iBuf16.Copy(iBuf8);//實際占用了9個描述符18個字節
原則3:使用charconv.lib內的API函數將8位的UTF8描述符轉換為16位的Unicode 描述符
CnvUtfConverter::ConvertToUnicodeFromUtf8(iBuf, iBuf8);
注:在使用這種方法時除了包含charconv.lib庫,還應該包含utf.h頭文件
原則4:使用GBK轉Unicode的方法
注:原則3和4是我在實際中常用的方法,涉及到各種編碼的問題,暫時不做大的展開,下次使用中文的整理時再展開。
16位到8位的描述符轉換
原則1:通過將一個16位描述符拆分為兩個8位描述符的方法實現內存不動,內存塊類型強轉(我稱其為保留描述符字節大小方法)
示例8
_LIT(KText , "Test 文本");
TBuf<20> iBuf(KText);//描述符實際占用了14個字節7個16位描述符 //實現內存塊的強轉
TPtrC8 ptr8(reinterpret_cast<const TUint8*>(iBuf.Ptr()), (iBuf.Size()));
//真實的內存不動已經結束,這里是重新申請了一塊內存并實現內存賦值拷貝
TBuf8<20> iBuf8;
iBuf8 = ptr8;//描述符實際占用了14個字節14個8位描述符
以上
原則2:重新構建內存塊使用描述符的copy函數將16位描述符的,進行拷貝轉換(我稱其為保留描述符長度方法)。
示例9:
_LIT(KText , "Test 文本");
TBuf<20> iBuf(KText);// 描述符實際占用了14個字節7個16位描述符
TBuf8<20> iBuf8;
iBuf8.Copy(iBuf);//描述符實際占用了7個字節7個描述符,非ASCII字符值轉為1
注:該種方法在純ASCII碼的轉換時可行,其它數據大于255的時候就會丟失數據。
原則3:使用charconv.lib內的API函數將16位的Unicode描述符轉換為8位的UTF8描述符。
CnvUtfConverter::ConvertFromUnicodeToUtf8(iBuf8, iBuf);
原則4:使用Unicode轉GBK的方法
同前面理,以后我會再講到。
由于最近做通信模塊時組解包用的比較多,為此經常會將memcpy,strcpy,sprintf,sscanf等函數在char字符串和描述符串內存塊之間的直接使用,個人覺得描述符一旦取得了后面數據區的首指針,那么內存塊的這些操作沒有什么好展開,直接用函數大家都會,當然本人也推薦讀者使用文后的char字符串和Symbian描述符串的轉換方法。
在整理過程種,也有其它類型與描述符的轉換,我就摘錄在本文后面作為mark和備查:
1. TTime轉TBuf型
TBuf<32> theTime;//存儲轉換后的時間
TTime tt;
tt.HomeTime();
_LIT(KTimeFormat,"%Y%M%D%1-%2-%3 %H:%T:%S");//格式為:2006-03-04 12:12:12
tt.FormatL(theTime,KTimeFormat);
2. TDateTime轉TBuf型
TTime currentTime;//聲明一個TTime類型
currentTime.HomeTime();//設置TTime為當前時間
TDateTime tdt=currentTime.DateTime();//TTime ---> TdateTime
TBuf<32> tmp;//存儲轉換完的Buf
tmp.AppendNum(tdt.Year());//用AppendNum()方法將一個Tint加入到TBuf中。
_LIT(gang,"-");//聲明一個橫線分隔年月日,同樣可聲明冒號分隔小時分秒
tmp.Append(gang);
tmp.AppendNum(tdt.Month());
tmp.Append(gang);
tmp.AppendNum(tdt.Day());//…………時分秒的轉換同上
3. TBuf和Tint互轉型
// 15位數字
TInt iNum1(123456789009876);
// TInt轉TBuf
iBuf.Num(iNum1);//當buf只用來轉Tint時可以使用AppendNum,但是性質是不一樣的
// 使用iBuf包含的內容創建TLex對象
TLex iLex(iBuf);
TInt iNum2;
//TBuf轉TInt
iLex.Val(iNum2);// Num2現在包含了15位數字
4. TBuf轉TDateTime型
將長的TBuf截成小段,分別是年月日時分秒,通過下面TBuf轉TInt ,再分別把轉換成TInt的年月日取出,通過TDateTime的setYear(),setMonth()等方法將時間set進TdateTime。
5. .將symbian串轉換成char串
char* p = NULL;
TBuf8<20> buf( _L( "aaaaa" ) );
p = (char *)buf.Ptr();
6.將char串轉換成symbian串
char* cc = "aaaa";
TPtrC8 a;
a.Set( (const TUint8*)cc , strlen(cc) );
本文參考了一些論壇資料,至于資料索引,我也無法給出,望讀者見諒,有錯誤之處還望指出。
posted on 2007-11-28 22:51
frank.sunny 閱讀(4960)
評論(15) 編輯 收藏 引用 所屬分類:
symbian 開發