boost字符串處理函數(shù)format的用法
用boost::format來格式化字符串
在字符串處理中少不了格式化字符串,C++中傳統(tǒng)的格式化函數(shù)是C語言的sprintf,但它一個很大的問題就是不安全。因此,在stl中引入了stringstream來實現(xiàn)安全格式化,但是stringstream卻遠不如sprintf來得直觀。例如,對如如下代碼:
char text[]="hello";
bool is_all_lower = boost::algorithm::all(text, is_lower());
char output[128];
sprintf(output, "<%s> %s in the lower case", text, (is_all_lower? "is": "is not"));如果把最后兩句format的函數(shù)用stringstream來寫的話,可讀性是遠不如sprintf的。
stringstream output;
output << "<" << text << "> "
<< (is_all_lower)? "is": "is not")
<< " in the lower case";boost引入了一個提供類似.net中的string.format的方式提供格式化字符串的函數(shù),用它來格式化的話就是如下形式:
boost::format fmt = boost::format("<%s> %s in the lower case") % text % (is_all_lower? "is": "is not");
string output = fmt.str();前面的例子中演示的是C風(fēng)格的格式化字符串,boost.format也提供了類似.net風(fēng)格的格式化字符串方式:
boost::format fmt = boost::format("<%1%> %2% in the lower case") % text % (is_all_lower? "is": "is not");
cout << fmt << endl;這種方式更容易看到參數(shù)在格式化字符串中的位置,推薦這種形式。不過它的起始坐標是1而不是0,用慣了.net的string.format的朋友需要注意下。
格式化控制
格式化語法為: [ N$ ] [ flags ] [ width ] [ . precision ] type-char。也提供了C語言和.net兩種風(fēng)格。
//傳統(tǒng)c語言風(fēng)格
cout << boost::format("\n\n%s"
"%1t 十進制 = [%d]\n"
"%1t 格式化的十進制 = [%5d]\n"
"%1t 格式化十進制,前補'0' = [%05d]\n"
"%1t 十六進制 = [%x]\n"
"%1t 八進制 = [%o]\n"
"%1t 浮點 = [%f]\n"
"%1t 格式化的浮點 = [%3.3f]\n"
"%1t 科學(xué)計數(shù) = [%e]\n"
) % "example :\n" % 15 % 15 % 15 % 15 % 15 % 15.01 % 15.01 % 15.01 << endl;
//.net的風(fēng)格
cout << boost::format("%1%"
"%1t 十進制 = [%2$d]\n"
"%1t 格式化的十進制 = [%2$5d]\n"
"%1t 格式化十進制,前補'0' = [%2$05d]\n"
"%1t 十六進制 = [%2$x]\n"
"%1t 八進制 = [%2$o]\n"
"%1t 浮點 = [%3$f]\n"
"%1t 格式化的浮點 = [%3$3.3f]\n"
"%1t 科學(xué)計數(shù) = [%3$e]\n"
) % "example :\n" % 15 % 15.01 << endl;異常處理
既然boost.format函數(shù)是用來代替sprintf的,那么自然就得有異常處理的功能,而不是像sprintf那樣死給你看。boost.format的處理方法是拋異常,它在如下兩種情況家會拋異常:
format字符串非法
format綁定非法
如下代碼演示了這兩種情形:
try
{
boost::format("<%3");
}
catch(std::exception& err)
{
cout << err.what() << endl;
}
boost::format fmt = boost::format("<%3%> %2% in the lower case") % text % (is_all_lower? "is": "is not");
try
{
cout << fmt << endl;
}
catch(std::exception& err)
{
cout << err.what() << endl;
}封裝
boost.format是以一個對象,而不是函數(shù)來實現(xiàn)的,導(dǎo)致其使用和異常處理起來要麻煩不少,不過,利用c++11的可變參數(shù)模板的語法還是可以很容易把它封裝成一個可變參數(shù)的函數(shù)的形式:
string string_fromat(const char* format, …)
需要定義三個重載版本:
template<class TFirst>
void string_format(boost::format& fmt, TFirst&& first)
{
fmt % first;
}
template<class TFirst, class... TOther>
void string_format(boost::format& fmt, TFirst&& first, TOther&&... other)
{
fmt % first;
string_format(fmt, other...);
}
template<class TFirst, class... TOther>
string string_format(const char* format, TFirst&& first, TOther&&... other)
{
boost::format fmt(format);
string_format(fmt, first, other...);
return fmt.str();
}現(xiàn)在就可以這么用了:
auto?output = string_format("<%1%> %2% in the lower case", text, (is_all_lower??"is":?"is not"));所有的異常也都會在該函數(shù)中拋出,雖然效率上相對低點,但用起來要舒服點。
到此這篇關(guān)于boost字符串處理函數(shù)format的文章就介紹到這了。希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
C++線性表深度解析之動態(tài)數(shù)組與單鏈表和棧及隊列的實現(xiàn)
這篇文章主要為大家詳細介紹了C++實現(xiàn)動態(tài)數(shù)組、單鏈表、棧、隊列,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-05-05
C++通過自定義函數(shù)找出一個整數(shù)數(shù)組中第二大數(shù)的方法
這篇文章主要介紹了C++通過自定義函數(shù)找出一個整數(shù)數(shù)組中第二大數(shù)的方法,涉及C++針對數(shù)組的遍歷操作相關(guān)技巧,需要的朋友可以參考下2015-06-06
VS2019 更新MSDN并創(chuàng)建快捷方式的實現(xiàn)
這篇文章主要介紹了VS2019 更新MSDN并創(chuàng)建快捷方式的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-03-03
關(guān)于C++中數(shù)據(jù)16進制輸出的方法
本文主要介紹了關(guān)于C++中數(shù)據(jù)16進制輸出的方法,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-03-03

