亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > C++ > 正文

C++中輸出十六進制形式的字符串

2020-01-26 14:19:52
字體:
來源:轉載
供稿:網友

前言

在進行 i18n 相關的開發時,經常遇到字符編碼轉換的錯誤。這時如果能把相關字符串用十六進制的形式打印出來,例如,"abc" 輸出成 "http://x61//x62//x63" 這對于 i18n 的除錯來說是很有幫助的。Python 里面,只需要使用 repr() 函數就行了??稍?C++ 中如何做到這點呢?

下面是用 ostream 的格式化功能的一個簡單的實現:

std::string get_raw_string(std::string const& s){ std::ostringstream out; out << '/"'; out << std::hex; for (std::string::const_iterator it = s.begin(); it != s.end(); ++it) { out << "http://x" << *it; } out << '/"'; return out.str();}

看上去簡單直接,但很可惜這段代碼不能實現我們的意圖。它還是按字面輸出了每個字符??晌覀兠髅髦付耸褂?std::hex 來格式化輸出啊?。繂栴}原來是出在 std::hex 只是一個針對整數類型的輸出格式設置,當輸出字符類型時,C++ 流還是按照字面輸出。到 ostream 的文檔去細查才知,原來 C++ 標準輸出流對于格式化輸出的控制很弱,只能提供有限的幾種格式定制,而且大部分都是針對整數和浮點數類型的,對于字符類型完全沒有參數可以控制。有點諷刺的是, ostream 利用了 C++ 的函數重載和強類型機制做到了在表達力不輸于 C 的同時,又杜絕了臭名昭著的 printf 帶來的無窮的麻煩,大大增加了安全??稍谶@里,強類型安全反而是我們達到目的的障礙:我就是想讓 ostream 把字符當成整數打印啊!還好,C++ 還有類型強轉這招可以讓我們繞過強類型匹配這道安全閘門:

out << std::hex << "http://x" << static_cast<int>(*it);

好了,這下字符都按整數來輸出了,而 std::hex 又指示 ostream 用十六進制表示去輸出整數。問題解決了。且慢,為什么輸出 UTF-8 中文編碼的時候會變成這樣:

"/xffffffe4/xffffffb8/xffffffad" // get_raw_string("中")

這么多的 F word 太影響市容了。能不能把它們去掉?其實原因在于,我們輸出的是強制類型轉換成 int 的整形數值,而 int 是 32 bit 長,所以會多出前面這么多位來。如果要去掉,只要轉成 8 bit 的整數不就行了嗎??上?C/C++ 中沒有 8 bit 的整數,你唯一能做到的是

typedef char int8_t;

可是用這樣得來的 int8_t 去轉也還是不行,因為在 C++ 中,typedef 并沒有產生一個新的類型,而只是定義了一個原來類型的別名。而這個別名是不參與到函數重載的匹配計算當中的。換言之,ostream 說了,別以為你披上件 int8_t 的馬甲我就不認識你了,我還是把你當 char 來輸出。此路不通!

那我們就放棄利用 ostream 了嗎?且慢,其實 ostream 默認是不會輸出前面的 0 的,那只要把最后 8 bit 之前的位都抹成 0 不就能達到我們的要求了嗎。

好了,下面就是無錯最終版:

std::string get_raw_string(std::string const& s){ std::ostringstream out; out << '/"'; out << std::hex; for (std::string::const_iterator it = s.begin(); it != s.end(); ++it) { // AND 0xFF will remove the leading "ff" in the output, // So that we could get "/xab" instead of "/xffab" out << "http://x" << (static_cast<short>(*it) & 0xff); } out << '/"'; return out.str();}

經歷了幾番波折,終于成功利用了 ostream 提供的十六進制輸出的功能實現了打印字符串十六進制的功能。其實細究起來,之所以那么繞,還是因為 ostream 本身在格式化輸出控制方面太弱了。進一步的,C++ 里還有更好的工具做這件事嗎? boost::format 看起來象是,但它依然不能正確處理我們上面遇到的兩難境地。好在,另一個 boost 庫給出了合適的答案: boost::spirit::karma

Karma 是 boost::spirit 庫的一部分。大家可能比較熟悉的是用 spirit 庫做 parser 來解析字符串。而 spirit 通過 Karma 提供的功能就恰好相反,它是專門用來將 C++ 數據結構格式化為字符流的。

我們恰好就需要它,下面就是用 karma 庫重寫的代碼:

template <typename OutputIterator>bool generate_raw(OutputIterator sink, std::string s){ using boost::spirit::karma::hex; using boost::spirit::karma::generate; return generate(sink, '/"' << *("http://x" << hex) << '/"', s);}std::string get_raw_string_k(std::string const& s){ std::string result; if (!generate_raw(std::back_inserter(result), s)) { throw std::runtime_error("parse error"); } return result;}

這里面最主要就是利用了 karma 內置的一個輸出模塊 karam::hex 來幫我們完成工作,而這個 hex 是一個多態的生成器。它不象 ostream 的類型重載,只能針對某些類型輸出 hex 格式,而是針對所有類型都能輸出 hex 格式,包括 char 。還有一個優點,代碼的表達力更強了,輸出的格式完全在一行代碼中體現:

// 輸出格式為 "/x61/x62/x63",方便直接貼到 python 或 C++ 的代碼中'/"' << *("http://x" << hex) << '/"'

如果想要改變輸出格式,只需要改這行代碼即可,例如:

// 輸出格式變為 "0x61 0x62 0x63 "'/"' << *("0x" << hex << " ") << '/"'

那么效率方面有沒有任何性能損失呢?下面是一段測試代碼,分別用兩種算法轉換相同的字符串:

#include "boost/test/unit_test.hpp"#include "boost/../libs/spirit/optimization/measure.hpp"#include "string.hpp" // The function for teststatic std::string const message = "hex output performance test data 中文";struct using_karma : test::base{ void benchmark() { this->val += get_raw_string_c(message).size(); }};struct using_ostream : test::base{ void benchmark() { this->val += get_raw_string(message).size(); }};BOOST_AUTO_TEST_CASE(TestStringPerformance){ BOOST_SPIRIT_TEST_BENCHMARK( 100, (using_karma) (using_ostream) ); BOOST_CHECK_NE(0, live_code);}

下面是運行的結果,分別是兩種算法需要的時間,值越小越好:

算法 耗時(s)
karma 6.97
ostream 14.24

可能出乎意料,大致來說 karma 比 ostream 快了一倍。這也與 spirit 官方給出的性能數據差不多。這里的函數返回值是通過 std::string 值拷貝返回的,消耗了不少時間,如果純從格式化輸出來說,猜測 karma 的性能優勢只會更大。另一份測試 表明,karma 應該是 C/C++ 里面你能找到的速度最快的格式化字符流方案了。

對于這么簡單的功能來說,這篇文章已經顯得太長了,慶幸的是,我們最終還是找到了一個表達力強,性能高的十六進制輸出方案。人說好事難雙,可 C++ 這門復雜的語言,卻經常能找執行飛快又高度抽象的代碼方案。只是有些過于復雜了 ...

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作能帶來一定的幫助,如果有疑問大家可以留言交流。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲最新视频在线| 成人夜晚看av| 国产精品亚洲网站| 少妇高潮 亚洲精品| 欧美久久精品一级黑人c片| 人人做人人澡人人爽欧美| 中文字幕亚洲色图| 中文字幕欧美精品在线| 日本高清久久天堂| 日韩欧美第一页| 97精品国产97久久久久久春色| 久久久久久高潮国产精品视| 国产这里只有精品| 91日韩在线视频| 国产在线观看精品一区二区三区| 国产美女精品免费电影| 成人啪啪免费看| 一本大道香蕉久在线播放29| 海角国产乱辈乱精品视频| 欧美激情xxxx性bbbb| 最近的2019中文字幕免费一页| 久久91精品国产91久久久| 精品久久久一区二区| 欧美日韩精品二区| 亚洲欧美日韩一区二区在线| 国产亚洲激情在线| 91成人天堂久久成人| 欧美专区第一页| 国产成人亚洲综合91精品| 亚洲国产婷婷香蕉久久久久久| 久久精品久久久久电影| 性日韩欧美在线视频| 国产综合久久久久| 国产日韩av在线播放| 精品成人乱色一区二区| 久久亚洲综合国产精品99麻豆精品福利| 国产精品专区第二| 欧美野外猛男的大粗鳮| 在线观看91久久久久久| 午夜精品福利视频| 911国产网站尤物在线观看| 欧美激情女人20p| 国产亚洲xxx| 国产精品av免费在线观看| 91福利视频在线观看| 日韩电影大全免费观看2023年上| 国产日韩在线免费| 欧美高清视频一区二区| 日本久久久久久久久| 正在播放欧美一区| 亚洲男人天天操| 国产欧美一区二区三区四区| 亚洲视频电影图片偷拍一区| 黄色一区二区在线| 中文字幕在线精品| 精品欧美一区二区三区| 亚洲午夜av电影| 成人午夜高潮视频| 国产精品一区二区久久久| 中文字幕亚洲欧美日韩在线不卡| 久久久久久久久中文字幕| 亚洲无线码在线一区观看| 亚洲欧美自拍一区| 欧美日韩国产成人在线观看| 亚洲国产精品一区二区三区| 国产在线视频2019最新视频| 欧美日韩中文字幕| 日韩av电影中文字幕| 精品国产欧美一区二区三区成人| 高清欧美性猛交xxxx黑人猛交| 久久精品国产免费观看| 亚洲自拍在线观看| 91国产高清在线| 麻豆国产va免费精品高清在线| 91在线视频导航| 亚洲国产精品人久久电影| 亚洲大胆人体视频| 欧美午夜女人视频在线| 欧美在线视频在线播放完整版免费观看| 国产一区二区三区毛片| 欧美成人在线免费| 久久视频在线免费观看| 91精品久久久久久综合乱菊| 91久久精品国产91久久性色| 狠狠色噜噜狠狠狠狠97| 欧美亚洲第一区| 国产精品第一第二| 久久综合免费视频| 亚洲伊人一本大道中文字幕| 亚洲欧美精品一区二区| 欧美中文在线观看| 日韩美女写真福利在线观看| 午夜精品久久久久久久白皮肤| xxxxx成人.com| 国产性猛交xxxx免费看久久| 欧美激情在线观看视频| 国产精品丝袜久久久久久高清| 久久国产精品久久久| 国产99在线|中文| 欧美日韩中国免费专区在线看| 欧美精品久久久久| 国产精品高清网站| 日本一区二区三区在线播放| 97在线免费观看视频| 日本sm极度另类视频| 亚洲欧美日韩精品久久亚洲区| 欧美日韩在线免费| 欧美俄罗斯乱妇| 国产精品成人av在线| 5278欧美一区二区三区| 亚洲成人精品av| 久久免费视频在线| 亚洲天堂av女优| 国产啪精品视频网站| 国产成人一区二区三区小说| 欧美精品videos性欧美| 96精品久久久久中文字幕| 欧美精品videos性欧美| 欧美性xxxx| 亚洲高清av在线| 日韩在线视频观看正片免费网站| 国产精自产拍久久久久久蜜| 日韩欧美国产骚| 亚洲综合日韩中文字幕v在线| 久久男人资源视频| 国产日韩欧美日韩| 久久这里只有精品视频首页| 欧美成人午夜影院| www.国产一区| 亚洲图片在区色| 国产91精品不卡视频| 国产丝袜精品视频| 亚洲精品美女久久| 亚洲欧美中文另类| 亚洲性生活视频在线观看| 成人黄色午夜影院| 亚洲色在线视频| 亚洲第一国产精品| 亚洲护士老师的毛茸茸最新章节| 欧美成人激情在线| 亚洲免费伊人电影在线观看av| 亚洲aaaaaa| 热99精品只有里视频精品| 亚洲国产精品视频在线观看| 欧美日韩国产在线播放| 欧美激情中文网| 成人综合网网址| 久热精品视频在线观看| 欧美在线免费视频| 亚洲国产99精品国自产| 性色av一区二区三区在线观看| 欧美性生交xxxxxdddd| 国产不卡av在线| 国产精品久久久久久久久久三级| 高跟丝袜一区二区三区| 亚洲欧美色图片| 91午夜理伦私人影院| 亚洲黄色成人网| 日韩精品高清在线| 欧美日韩高清区| 亚洲精品日韩激情在线电影| 精品久久久久国产| 国产精品激情av在线播放| 日韩精品视频免费在线观看|