• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            franksunny的個人技術空間
            獲得人生中的成功需要的專注與堅持不懈多過天才與機會。 ——C.W. Wendte

             

            描述符<>轉換

             

            業(yè)余有時候把一個事情當作任務來做的時候,往往會很受限制,就象這篇文檔,上次寫了個上篇,關于描述符的下篇,自己就遲遲沒有勇氣和時間寫完,好幾次都想靜下心來好好完成它,但是都未能如愿,可如果不寫顯然我也不好意思寫其它的一些東西,閑話提到這里,花了幾天業(yè)余時間總算是完成了,該文質量不足之處還望讀者您見諒。

             

            通過前面關于描述符概念和使用的簡單描述,下面我們直接進入描述符轉換的主題。

            雖然描述符有五類,但是作為描述符變量用的,只有三類即棧描述符、堆描述符、指針描述符。所以本文所述描述符的轉換也只是在這三種類型間展開,這三類描述符之間的互相轉換其實可以用下面這種偽代碼方式給出:

            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()函數,將不可修改的描述符轉換成可修改的指針描述符

            示例1TBufC轉換成TPtr

                   _LIT(KText, "Test Data");

                   TBufC<10> NBuf ( KText );

                   TPtr     Pointer = NBuf.Des();

            示例2HBufC轉換成TPtr

            HBufC * Buf = HBufC::NewL(15);

                   _LIT (KText , "Test Text");

                   *Buf = KText;

                   TPtr Pointer = Buf->Des();

             

            原則2:通過TPtr的構造函數或Set()函數可以將TPtrC描述轉換為可修改的指針描述符

            示例3TPtrCTPtr

            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位描述符的方法實現內存不動,內存塊類型強轉(我稱其為保留描述符字節(jié)大小方法)。

            示例48bit To 16bit保留描述符字節(jié)大小方法1

            _LIT8(KText , "Test Text");

            TBuf8<20> iBuf8(KText);//描述符實際占用了9個字節(jié)98位描述符

            TBuf<20> iBuf16;

            //實現內存塊的強轉

            TPtrC16 ptr16(reinterpret_cast<const TUint16*>(iBuf8.Ptr()), (iBuf8.Length()/2));

            //真實的內存不動已經結束,這里是重新申請了一塊內存并實現內存賦值拷貝

            iBuf16 = ptr16;//描述符實際占用了8個字節(jié)416位描述符,導致數據丟失

            由上面的這個例子明顯是丟失了數據了,原因就出在設置長度時用了(iBuf8.Length()/2),為了剔除錯誤,我后來改了一個新的示例6

            示例68bit To 16bit保留描述符字節(jié)大小方法2(由于上例用了TPtrC,所以該例使用TPtr指針,以擴展應用)

            _LIT8(KText , "Test Text");

            TBuf8<20> iBuf8;

            iBuf8.FillZ(20);

            iBuf8 = KText;//實際占用9字節(jié)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字節(jié)5描述符,但是第10個字節(jié)為0零值,也即9字節(jié)大小有用

            注:其實這里用堆描述符來說明可能更好些,但是本人在調試過程中為了能夠看到調試的效果,所以用了TBuf;而且個人覺得通常這種方法主要在文件讀取等實際過程中用到,如果要轉換為16位的話,肯定是偶數大小的8位描述符。

             

            原則2:重新構建內存塊使用描述符的copy函數將8位字符的高字節(jié)置為0低字節(jié)不變,進行拷貝轉換(我稱其為保留描述符長度方法)。

            示例7

            _LIT8(KText , "Test Text");

            TBuf8<20> iBuf8(KText);//實際占用了9個描述符9個字節(jié)

            TBuf<20> iBuf16;

            iBuf16.Copy(iBuf8);//實際占用了9個描述符18個字節(jié)

             

            原則3:使用charconv.lib內的API函數8位的UTF8描述符轉換為16位的Unicode 描述符

            CnvUtfConverter::ConvertToUnicodeFromUtf8(iBuf, iBuf8);

            注:在使用這種方法時除了包含charconv.lib庫,還應該包含utf.h頭文件

             

            原則4:使用GBKUnicode的方法

            注:原則34是我在實際中常用的方法,涉及到各種編碼的問題,暫時不做大的展開,下次使用中文的整理時再展開。

             

            16位到8位的描述符轉換

            原則1:通過將一個16位描述符拆分為兩個8位描述符的方法實現內存不動,內存塊類型強轉(我稱其為保留描述符字節(jié)大小方法)

            示例8

            _LIT(KText , "Test 文本");

            TBuf<20> iBuf(KText);//描述符實際占用了14個字節(jié)716位描述符                            //實現內存塊的強轉

            TPtrC8 ptr8(reinterpret_cast<const TUint8*>(iBuf.Ptr()), (iBuf.Size()));

            //真實的內存不動已經結束,這里是重新申請了一塊內存并實現內存賦值拷貝

            TBuf8<20> iBuf8;

            iBuf8 = ptr8;//描述符實際占用了14個字節(jié)148位描述符

            以上

            原則2:重新構建內存塊使用描述符的copy函數將16位描述符的,進行拷貝轉換(我稱其為保留描述符長度方法)。

            示例9

            _LIT(KText , "Test 文本");

            TBuf<20> iBuf(KText);// 描述符實際占用了14個字節(jié)716位描述符

            TBuf8<20> iBuf8;

            iBuf8.Copy(iBuf);//描述符實際占用了7個字節(jié)7個描述符,非ASCII字符值轉為1

            注:該種方法在純ASCII碼的轉換時可行,其它數據大于255的時候就會丟失數據。

             

            原則3:使用charconv.lib內的API函數16位的Unicode描述符轉換為8位的UTF8描述符。

            CnvUtfConverter::ConvertFromUnicodeToUtf8(iBuf8, iBuf);

             

            原則4:使用UnicodeGBK的方法

            同前面理,以后我會再講到。

             

            由于最近做通信模塊時組解包用的比較多,為此經常會將memcpystrcpysprintfsscanf等函數在char字符串和描述符串內存塊之間的直接使用,個人覺得描述符一旦取得了后面數據區(qū)的首指針,那么內存塊的這些操作沒有什么好展開,直接用函數大家都會,當然本人也推薦讀者使用文后的char字符串和Symbian描述符串的轉換方法。

             

            在整理過程種,也有其它類型與描述符的轉換,我就摘錄在本文后面作為mark和備查:

            1.    TTimeTBuf

            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.    TDateTimeTBuf

            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.    TBufTint互轉型

            // 15位數字

            TInt iNum1(123456789009876);

            // TIntTBuf

            iBuf.Num(iNum1);//buf只用來轉Tint可以使用AppendNum,但是性質是不一樣的

             

            // 使用iBuf包含的內容創(chuàng)建TLex對象

            TLex iLex(iBuf);

            TInt iNum2;

            //TBufTInt

            iLex.Val(iNum2);// Num2現在包含了15位數字

             

            4.    TBufTDateTime

            將長的TBuf截成小段,分別是年月日時分秒,通過下面TBufTInt ,再分別把轉換成TInt的年月日取出,通過TDateTimesetYear(),setMonth()等方法將時間setTdateTime

             

            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 開發(fā)

            FeedBack:
            # re: 描述符轉換
            2007-11-29 22:25 | helixapp
            真的挺全的 贊一個  回復  更多評論
              
            # re: 描述符轉換
            2008-03-03 15:37 | myh
            博主的示例6:8bit To 16bit保留描述符字節(jié)大小方法2(由于上例用了TPtrC,所以該例使用TPtr指針,以擴展應用)


            _LIT8(KText , "Test Text");

            TBuf8<20> iBuf8;

            iBuf8.FillZ(20);

            iBuf8 = KText;//實際占用9字節(jié)9描述符

            TBuf<20> iBuf16;

            TPtr ptr16(reinterpret_cast<TUint16*>(const_cast<TUint8*>(iBuf8.Ptr())),
            ((iBuf8.Length()+1)/2),((iBuf8.Length()+1)/2));

            iBuf16 = ptr16;//

            // 本人在Draw函數中引用了你上面的代碼,,同時調用下面語句
            gc.DrawText(iBuf16,TPoint(0,20));
            在模擬器上顯示的是亂碼,,不知為什么,,
              回復  更多評論
              
            # re: 描述符轉換
            2008-03-04 10:50 | frank.sunny
            @myh
            好久沒來更新了,今天在郵箱里看到來自博客園的郵件感覺好親切

            謝謝myh兄,還自己去做過實驗,其實8位到16位的前兩個原則只是保證在內存塊中數據一致,不放你可以自己調試看看內存里面內容,但是你在顯示的時候8位和16位的編碼不一樣自然就會出現亂碼了

            其實這個問題還是涉及編碼的問題,由于后來項目緊就一直沒有更新博客了,myh兄可以采用原則3和4,肯定不會出現亂碼的,并且如果是中文,在模擬器上如果采用了中文SDK是不會出現亂碼的,在手機上可能還會出現亂碼,這個只能以后詳細解說了

            再次感謝myh兄參與討論


              回復  更多評論
              
            # re: 描述符轉換
            2008-03-20 11:26 | myh
            多謝回復,,最近在學習真機上中文顯示問題,,感覺比較復雜,,

            用兩種方法實現了真機上中文顯示,,
            1、CnvUtfConverter::ConvertToUnicodeFromUtf8(iBuf, iBuf8);

            2、CCnvCharacterSetConverter中的ConvertToUnicode

            但第一種方法存在文件保存時的格式問題,,選擇UTF-8 無 BOM格式,,當文件中的內容改變時又會出問題,,
            能不能把整個項目發(fā)到你的郵箱里,,幫我看一下,,

            另外第二種方法比較耗時,,但比較方便,,  回復  更多評論
              
            # re: 描述符轉換
            2008-03-20 11:58 | frank.sunny
            閣下連耗時與否都做過比較了,非常感激,但是不知道具體是怎樣耗時法,本人未做過這方面的實驗。
            照理轉成utf8應該無問題啊,不知道怎么聯系閣下,我的郵箱是:frank.sunny@163.com
              回復  更多評論
              
            # re: 描述符轉換
            2008-03-20 12:03 | myh

            第一種方法,,
            _LIT8(KOpentoMap8, "返回我的最愛");
            會報以下錯
            error C2001: 常數中有換行符
            fatal error C1057: 宏展開中遇到意外的文件結束

            但是將文字改成以下幾種情況又不會報錯
            1、_LIT8(KOpeap8, "返回我的最");
            2、_LIT8(KOpeap8, "城市設置成功");
            3、_LIT8(KOpeap8, "返回我的最w");


              回復  更多評論
              
            # re: 描述符轉換
            2008-03-20 12:12 | frank.sunny
            不知道你是怎么考慮的,照理“_LIT8(KOpentoMap8, "返回我的最愛");”這個語句應該是“_LIT(KOpentoMap8, "返回我的最愛");”吧 ?

            我不是很清楚你要實現什么目的


              回復  更多評論
              
            # re: 描述符轉換
            2008-03-20 12:50 | myh
            我定義了一個專門存放字符常量的頭文件,,Common.h

            其中有_LIT8(KOpentoMap8, "返回我的最愛");
            ccp文件中調用
            CnvUtfConverter::ConvertToUnicodeFromUtf8(iBuf, KOpentoMap8);


            最后調用DrawText; 以顯示中文,,“返回我的最愛”
              回復  更多評論
              
            # re: 描述符轉換
            2008-03-20 13:27 | frank.sunny
            不好意思,剛剛午睡下
            你這樣是不對的,“_LIT8(KOpentoMap8, "返回我的最愛");"中的"返回我的最愛"是GBK編碼的,而_LIT8并非轉化為utf-8的,仍然是GBK編碼的。這個在symbian中現在一般不推薦使用,只針對ASCII碼時是適用的。
            關于GBK轉化到Unicode另外有代碼實現的,也怪自己偷懶老是沒有整理這個東西,我有時間整理下吧。  回復  更多評論
              
            # re: 描述符轉換
            2008-03-20 13:58 | myh
            沒事,,



            _LIT8(KOpentoMap8, "返回我的最愛");"中的"返回我的最愛"是GBK編碼的,

            我已經將文件Commom.h頭文件存為UTF-8 無 BOM 格式 了,,所以

            返回我的最愛 應該是UTF-8 無 BOM 編碼了,,  回復  更多評論
              
            # re: 描述符轉換
            2008-03-20 14:28 | frank.sunny
            不知道閣下如何聯系,想問下無bom和有bom具體是什么關系啊
            我不懂呢,查了下有bom多了內容為“FFFE”,對吧,但是有什么用,能告知下嗎  回復  更多評論
              
            # re: 描述符轉換
            2008-03-20 15:20 | myh
            644621379

            其實我也不太懂,,昨天看到的,,你加我吧,,咱們QQ聊,,  回復  更多評論
              
            # re: 描述符轉換
            2008-06-25 19:42 | wangfsun
            symbian學習中,借用了你的一些資料,很感謝  回復  更多評論
              
            # re: 描述符轉換
            2008-11-28 09:36 | liuhz
            很好得文章,非常感謝分享。再接再厲,呵呵。  回復  更多評論
              
            # re: 描述符轉換
            2011-01-13 11:24 | longteng9
            // 15位數字

            TInt iNum1(123456789009876);

            // TInt轉TBuf

            iBuf.Num(iNum1);//當buf只用來轉Tint時可以使用AppendNum,但是性質是不一樣的
            我用寫在文件中
            TBuf8<20> buf=iBuf.Num(iNum1);
            iFile.Write(buf);  回復  更多評論
              

            常用鏈接

            留言簿(13)

            隨筆分類

            個人其它博客

            基礎知識鏈接

            最新評論

            閱讀排行榜

            評論排行榜

            亚洲国产精品成人久久| 久久久精品久久久久久| 久久天天躁狠狠躁夜夜avapp| 狠狠色丁香久久综合五月| 国内精品人妻无码久久久影院| 精品国产99久久久久久麻豆| 久久99久久成人免费播放| 成人精品一区二区久久| 人人狠狠综合久久亚洲婷婷| 99久久综合狠狠综合久久止| 久久99精品久久久久久动态图 | 久久精品日日躁夜夜躁欧美| 亚洲欧美日韩久久精品| 思思久久好好热精品国产| 精品久久久久久无码专区| 久久偷看各类wc女厕嘘嘘| 久久99国产精品久久| 久久99精品免费一区二区| 香蕉99久久国产综合精品宅男自| 亚洲欧美一级久久精品| 精品久久久久久无码专区不卡 | 久久精品国产秦先生| 91精品国产色综久久| 亚洲国产精品一区二区三区久久 | 久久人妻少妇嫩草AV蜜桃| 久久精品国产欧美日韩99热| 少妇内射兰兰久久| 97久久精品人人澡人人爽| 四虎影视久久久免费观看| 久久无码人妻一区二区三区| 久久久久亚洲AV成人网人人网站 | 久久综合九色欧美综合狠狠| 蜜桃麻豆WWW久久囤产精品| 国产精品久久久久…| 免费精品久久久久久中文字幕| 成人综合久久精品色婷婷| 亚洲精品高清国产一久久| 久久精品国产亚洲av麻豆图片| 伊人久久大香线蕉精品| 久久综合给久久狠狠97色| 香蕉aa三级久久毛片|