• <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>
            隨筆 - 96  文章 - 255  trackbacks - 0
            <2010年6月>
            303112345
            6789101112
            13141516171819
            20212223242526
            27282930123
            45678910

            E-mail:zbln426@163.com QQ:85132383 長期尋找對戰(zhàn)略游戲感興趣的合作伙伴。

            常用鏈接

            留言簿(21)

            隨筆分類

            隨筆檔案

            SDL相關網(wǎng)站

            我的個人網(wǎng)頁

            我的小游戲

            資源下載

            搜索

            •  

            積分與排名

            • 積分 - 494280
            • 排名 - 39

            最新評論

            閱讀排行榜

            評論排行榜

            <本文PDF文檔下載>

            std::locale

            通過前面兩節(jié)的知識,我們知道了在C/C++中,字符(串)和寬字符(串)之間的轉(zhuǎn)換不是簡單的,固定的數(shù)學關系,寬窄轉(zhuǎn)換依賴于本地化策略集(locale)。換句話說,一個程序在運行之前并不知道系統(tǒng)的本地化策略集是什么,程序只有在運行之后才通過locale獲得當時的本地化策略集。
            C有自己的locale函數(shù),我們這里直接介紹C++的locale類。
            先討論locale的構造函數(shù):
            locale() throw();
            這個構造函數(shù)是獲得當前程序的locale,用法如下:
            std::locale app_loc = std::locale();
            或者(這是構造對象的兩種表示方式,后同)
            std::locale app_loc;
            另外一個構造函數(shù)是:
            explicit locale(const char* name);
            這個構造函數(shù)以name的名字創(chuàng)建新的locale。重要的locale對象有:
            std::locale sys_loc("");      //獲得當前系統(tǒng)環(huán)境的locale
            std::locale C_loc("C");      或者      std::locale C_loc = std::locale::classic();      //獲得C定義locale
            std::locale old_loc = std::locale::global(new_loc);      //將new_loc設置為當前全局locale,并將原來的locale返回給old_loc
            除了這些,其它的name具體名字依賴于C++編譯器和操作系統(tǒng),比如Linux下gcc中文系統(tǒng)的locale名字為"zh_CN.UTF-8",中文Windows可以用"chs"(更加完整的名字可以用name()函數(shù)查看)。

            mbstowcs()和wcstombs()

            這兩個C運行時庫函數(shù)依賴于全局locale進行轉(zhuǎn)換,所以,使用前必須先設置全局locale。
            std::locale已經(jīng)包含在<iostream>中了,再加上我們需要用到的C++字符串,所以包含<string>。
            我們先看窄到寬的轉(zhuǎn)換函數(shù):
            const std::wstring s2ws(const std::string& s)
            {
                std::locale old_loc 
            =
                    std::locale::global(std::locale(
            ""));

                
            const char* src_str = s.c_str();
                
            const size_t buffer_size = s.size() + 1;
                wchar_t
            * dst_wstr = new wchar_t[buffer_size];
                wmemset(dst_wstr, 
            0, buffer_size);
                mbstowcs(dst_wstr, src_str, buffer_size);
                std::wstring result 
            = dst_wstr;
                delete []dst_wstr;

                std::locale::global(old_loc);

                
            return result;
            }
            我們將全局locale設置為系統(tǒng)locale,并保存原來的全局locale在old_loc中。
            在制定轉(zhuǎn)換空間緩存大小的時候,考慮如下:char是用1個或多個對象,也就是1個或者多個字節(jié)來表示各種符號:比如,GB2312用1個字節(jié)表示數(shù)字和字母,2個字節(jié)表示漢字;UTF-8用一個字節(jié)表示數(shù)字和字母,3個字節(jié)表示漢字,4個字節(jié)表示一些很少用到的符號,比如音樂中G大調(diào)符號等。wchar_t是用1個對象(2字節(jié)或者4字節(jié))來表示各種符號。因此,表示同樣的字符串,寬字符串的大?。ㄒ簿褪莣char_t對象的數(shù)量)總是小于或者等于窄字符串大小(char對象數(shù)量)的。+1是為了在最后預留一個值為0的對象,以便讓C風格的char或者wchar_t字符串自動截斷——這當然是寬串大小等于窄串大小的時候才會用上的,大部分時候,字符串早在前面某個轉(zhuǎn)換完畢的位置就被0值對象所截斷了。
            最后我們將全局locale設置回原來的old_loc。
            窄串到寬串的轉(zhuǎn)換函數(shù):
            const std::string ws2s(const std::wstring& ws)
            {
                std::locale old_loc 
            =
                    std::locale::global(std::locale(
            ""));

                
            const wchar_t* src_wstr = ws.c_str();
                size_t buffer_size 
            = ws.size() * 4 + 1;
                
            char* dst_str = new char[buffer_size];
                memset(dst_str, 
            0, buffer_size);
                wcstombs(dst_str ,src_wstr, buffer_size);
                std::
            string result = dst_str;
                delete []dst_str;

                std::locale::global(old_loc);

                
            return result;
            }
            這里考慮轉(zhuǎn)換空間緩存大小的策略正好相反,在最極端的情況下,所有的wchar_t都需要4個char來表示,所以最大的可能就是4倍加1。
            這兩個函數(shù)在VC和gcc中都能正常運行(MinGW因為前面說到的原因不支持寬字符的正常使用),在VC中會給出不安全的警告,這是告訴給那些弄不清寬窄轉(zhuǎn)換實質(zhì)的人的警告,對于了解到目前這些知識的你我來說,這就是啰嗦了。
            posted on 2010-06-26 11:17 lf426 閱讀(3000) 評論(1)  編輯 收藏 引用 所屬分類: 語言基礎、數(shù)據(jù)結構與算法

            FeedBack:
            # re: 徹底解密C++寬字符:3、利用C運行時庫函數(shù)轉(zhuǎn)換 2010-06-26 20:38 Sunshine Alike
            這個好,我先學習了~  回復  更多評論
              
            99久久亚洲综合精品成人| 波多野结衣久久一区二区| 国产三级精品久久| 香蕉99久久国产综合精品宅男自| 午夜福利91久久福利| 亚洲精品无码久久久久| 欧美一区二区精品久久| 伊人伊成久久人综合网777| 久久婷婷五月综合97色一本一本| 四虎国产精品免费久久久| 精品国产乱码久久久久久人妻| 狠狠色丁香婷综合久久| 噜噜噜色噜噜噜久久| 狠狠人妻久久久久久综合| 精品熟女少妇av免费久久| 亚洲精品97久久中文字幕无码| 好属妞这里只有精品久久| 亚洲天堂久久久| 精品水蜜桃久久久久久久| 996久久国产精品线观看| 久久精品久久久久观看99水蜜桃| 国产高清美女一级a毛片久久w| 久久久一本精品99久久精品66| 亚洲国产精品综合久久网络 | 91久久精品国产免费直播| 国内精品久久久久影院薰衣草 | 久久久一本精品99久久精品88| 国产精品亚洲综合专区片高清久久久| 久久婷婷五月综合色高清| 天天爽天天狠久久久综合麻豆| 亚洲中文久久精品无码ww16| 久久精品免费一区二区| 中文字幕无码精品亚洲资源网久久| 综合久久一区二区三区 | 久久久久久A亚洲欧洲AV冫| 久久91综合国产91久久精品| 97久久精品午夜一区二区| 色综合久久久久网| 久久久久99精品成人片| 韩国三级中文字幕hd久久精品| 国产精品久久久久久久久久免费|