一起学习Boost标准库--Boost.texical_cast&format库
今天接续介绍有关字符串表示相关的两个boost库:
- lexical_cast 将数值转换成字符串
- format 字符串输出格式化
首先,介绍下lexical_cast ,闻其名,知其意。类似C中的atoi 函数,可以进行字符串与整数/浮点数之间的字面转换
Boost::lexical_cast库
前期准备
lexical_cast库位于boost命名空间下,使用需要引入头文件
#include <boost/lexical_cast.hpp>
using namespace boost;
函数声明
lexical_cast使用类似C++标类型操作符的形式进行通用的语法,其声明如下:
// 1. 标准形式,转换数字和字符串
template <typename Target, typename Source>
inline Target lexical_cast(const Source &arg);
// 2. 转换C字符串
template <typename Target>
inline Target lexical_cast(const char* chars, std::size_t count);
inline Target lexical_cast(const unsigned char* chars, std::size_t count);
inline Target lexical_cast(const signed char* chars, std::size_t count);
inline Target lexical_cast(const wchar_t* chars, std::size_t count);
inline Target lexical_cast(const char16_t* chars, std::size_t count);
inline Target lexical_cast(const char32_t* chars, std::size_t count);
- 第一种形式有两个模版参数,Target 为需要转换的目标类型,通常是数字类型或者std::string,第二个参数Source 则不用写,可以通过函数参数推到出来,调用形式如下
lexical_cast<int>("123");
lexical_cast<double>("234.123");
lexical_cast<std::string>(567.789);
- 第二种形式主要用来处理C字符串,支持多种类型,只接受一个模板参数Target,指明转换后的目标类型,函数参数chars 和count 则标记了要转换的字符串范围
const char* double_str = "123.456";
double d1 = lexical_cast<double>(double_str, strlen(double_str));
使用样例
通过上文的介绍,您大概已经知道如何使用了,此处,我们就通过一个简单Demo来使用lexical_cast库
#include <boost/lexical_cast.hpp>
#include <iostream>
#include <string>
#include <iomanip>
using namespace std;
using namespace boost;
int main()
{
int age = lexical_cast<int>("29");
int money = lexical_cast<long>("10000000");
float pai_f = lexical_cast<float>("3.1415926535");
double pai_d = lexical_cast<double>("3.14159265358979323846264338324990", 20);
cout << "age: " << age << endl
<< "money: " << money << endl
<< setiosflags(ios::fixed)
<< "pai_f: " << setprecision(8) << pai_f << endl
<< "pai_d: " << setprecision(16) << pai_d << endl;
cout << lexical_cast<string>(age) << endl
<< lexical_cast<string>(money) << endl
<< lexical_cast<string>(pai_d) << endl;
return 0;
}
Output:
age: 29
money: 10000000
pai_f: 3.14159274
pai_d: 3.1415926535897931
29
10000000
3.1415926535897931
注意: 使用lexical_cast时要注意,转换成数字的字符串中只能有数字和小数点,不能出现字母或其他非数字字符,同时也不支持高级的格式控制,如果要进行复杂的格式控制可以使用std::stringstram 和 boost::format (后面介绍该库)
错误处理
当lexcial_cast无法执行转换操作的时候会抛出异常bad_lexical_cast ,他是std::bad_cast 的派生类,此处以上文中注意 来说明
try
{
int age = lexical_cast<int>("0x64");
//int age = lexical_cast<int>("100L");
//bool f = lexical_cast<bool>("false");
cout << "age: " << age << endl;
}
catch (bad_lexical_cast &e)
{
cout << "cast error: " << e.what() << endl;
}
Output:
cast error: bad lexical cast: source type value could not be interpreted as target
如果每次都通过异常捕获来处理,就比较麻烦了,还好lexical_cast为我们想到了,再命名空间boost::conversion 提供方法try_lexical_convert() 函数来避免抛出异常,通过返回bool值来表示是否转换成功。具体使用如下:
using namespace boost::conversion;
int x = 0;
if (!try_lexical_convert<int>("0x64", x))
cout << "convert failed" << endl;
else
cout << "x: " << x << endl;
Boost.format库
C++标准库提供了强大的输入输出流处理,可以通过设置输出各种各样的格式,精度控制、填充、对齐等。但是唯一缺点就是太复杂了,真心记不住这么多,还是怀恋printf() 可以通过规定的样式输出想要的格式。虽然C++中可以继续使用printf() 但它缺乏类型安全检查等其他缺点,重点就是boost.format库实现了类似于printf() 的格式化对象,可以把参数格式化到一个字符串,而且是类型安全的,是一个header-only 的函数库,只要准备好头文件,不用预先编译就可以使用了,最主要的是用着还挺顺手。
前期准备
format库位于boost命名空间中,需引入头文件:
#include <boost/format.hpp>
using namespace boost;
类声明
template <class Ch,
class Tr = BOOST_IO_STD char_traits<Ch>, class Alloc = std::allocator<Ch> >
class basic_format;
typedef basic_format<char > format;
template<class Ch, class Tr, class Alloc>
class basic_format
{
public:
explicit basic_format(const Ch* str = NULL);
explicit basic_format(const string_type& s);
basic_format(const basic_format& x);
basic_format& operator= (const basic_format& x);
basic_format& clear(); // 清空缓存
basic_format& parse(const string_type&); // 重新格式化
size_type size() const; // 获取字符串长度
string_type str() const; // 获取格式化后的字符串
template<class T>
basic_format& operator%(const T& x); //重载操作符%
template<class Ch2, class Tr2, class Alloc2>
friend std::basic_ostream<Ch2, Tr2> &
operator<<(std::basic_ostream<Ch2, Tr2> &,
const basic_format<Ch2, Tr2, Alloc2>&); //流输出
}; // class basic_format
- str: 返回format对象内部已经格式化号的字符串
- size : 返回format对象格式化号的字符串长度,可以直接从str()返回的string.size()
- parse :重新格式化,清空format对象内部缓存,改用一个新的格式化字符串,如果只是想清空缓存,则使用clear(),它把format对象恢复到初始化状态
- operator% : 可以接受待格式化的任意参数,%输入的参数个数必须等于格式化字符串中要求的个数,过多或者过少都会抛出异常
- operator<<:重载流输入操作符,可以直接输出格式化好的字符串
不过要注意的是,透过operator%传给boost::format对象的变量是会储存在对象内部的,所以可以分批的传入变数;但是如果变量的数量不符合的话,在编译阶段虽然不会出现错误,可是到了执行阶段还是会让程序崩溃,所以在使用上必须小心一点。 不过,在有输出后,是可以再重新传入新的变量、重复使用同一个boost::format 对象的。
格式化语法
format基本继承了printf的格式化语法,格式化选项以%开始,后面是格式规则
%[标志][输出最少宽度][.精度][长度]类型
详细请参考printf格式输出
除了支持printf格式化外,还新增了格式:
- %|spec| :与printf格式选项功能相同,但是两边增加竖线分隔,更好的区分格式化选项与普通字符
- %N% :标记第N个参数,相当于占位符,不带任何其他格式化的选项
通过以下使用竖线分隔,更加清楚明了格式化参数
format fmt("%05d\n%-8.3f\n% 10s\n%05X\n");
format fmt("%|05d|\n%|-8.3f|\n%| 10s|\n%|05X|\n");
使用样例
#include <boost/format.hpp>
#include <iostream>
using namespace std;
using namespace boost;
int main()
{
cout << "-------------"<<endl;
format fmt("%d + %d = %d");
fmt % 2 % 3 % 5;
cout << fmt.str() << endl;
cout << "-------------"<<endl;
format fmt2("%05d\n%-8.3f\n% 10s\n%05X\n");
fmt2 % 123;
fmt2 % 456.7;
fmt2 %"boost" % 100;
cout << fmt2 << endl;
cout << "-------------"<<endl;
cout << boost::format("x=%1%, y=%2% ,z= %3%") % "format" % 40.2 % 134 << endl;
cout << "-------------"<<endl;
return 0;
}
Output:
-------------
2 + 3 = 5
-------------
00123
456.700
boost
00064
-------------
x=format, y=40.2 ,z= 134
-------------
一起学习Boost标准库--Boost.texical_cast&format库的更多相关文章
- 一起学习Boost标准库--Boost.StringAlgorithms库
概述 在未使用Boost库时,使用STL的std::string处理一些字符串时,总是不顺手,特别是当用了C#/Python等语言后trim/split总要封装一个方法来处理.如果没有形成自己的com ...
- c++学习书籍推荐《超越C++标准库:Boost库导论》下载
<超越C++标准库Boost库导论>不仅介绍了Boost库的功能.使用方法及注意事项,而且还深入讨论了Boost库的设计理念.解决问题的思想和技巧以及待处理的问题.因此,本书是一本了解Bo ...
- C++三大库boost、loki、stlport
转: STL是一个标准,各商家根据这个标准开发了各自的STL版本.而在这形形色色的STL版本中,SGI STL无疑是最引人瞩目的一个.这当然是因为这个STL产品系出名门,其设计和编写者名单中,Alex ...
- [Boost]图形处理库Boost::Polygon
Background 工作中经师傅指导学习应用到了Boost::Polygon这个库,相对于Boost::Geometry,Polygon出自Intel.抽象于芯片流程,于是更贴近于芯片设计流程应用. ...
- [Boost基础]并发编程——asio网络库——同步socket处理
网络通信简述 asio库支持TCP,UDP和ICMP通信协议,它在名字空间boost::asio::ip里提供了大量的网络通信方面的函数和类,很好的封装了原始的Berkeley Socket API, ...
- [Boost基础]并发编程——asio网络库——定时器deadline_timer
asio库基于操作系统提供的异步机制,采用前摄器设计模式(Proactor)实现了可移植的异步(或者同步)IO操作,而且并不要求使用多线程和锁定,有些的避免了多线程编程带来的诸多有害副作用(如条件竞争 ...
- Dev-C++安装第三方库boost
Dev-C++安装第三方库boost 转 https://www.jianshu.com/p/111571e4d6f5?utm_source=oschina-app 之前鉴于codeblocks界面 ...
- [Boost基础]并发编程——asio网络库——异步socket处理
异步服务器端 #include <conio.h> #include <iostream> using namespace std; #include <boost/as ...
- 值得学习的C语言开源项目和库
收集一些C/C++相关的源码,如有更高效的库,请提醒我 补充上去 C/C++相关交流Q群 1414577 - 1. Webbench Webbench是一个在linux下使用的非常简单的网站压测工具. ...
随机推荐
- ASP.NET MVC View中的标签(tag)
在编辑View的时候会用到各种HTML标签,如<a>,<input>,<p>等待,这些标签在ASP.NET MVC中都有对应的编程语法,它叫Razor,它是帮助我们 ...
- Tomcat学习总结(6)——Tomca常用配置详解
注:Tomcat 8需要JRE7以上的JRE 1. Tomcat环境变量设置 1.1 Java环境变量设置 右键计算机—属性—高级系统设置—环境变量,在”系统环境变量”,设置如下三个变量(如果变量已有 ...
- 【LeetCode题解】347_前K个高频元素(Top-K-Frequent-Elements)
目录 描述 解法一:排序算法(不满足时间复杂度要求) Java 实现 Python 实现 复杂度分析 解法二:最小堆 思路 Java 实现 Python 实现 复杂度分析 解法三:桶排序(bucket ...
- Java设计模式学习记录-简单工厂模式、工厂方法模式
前言 之前介绍了设计模式的原则和分类等概述.今天开启设计模式的学习,首先要介绍的就是工厂模式,在介绍工厂模式前会先介绍一下简单工厂模式,这样由浅入深来介绍. 简单工厂模式 做法:创建一个工厂(方法或类 ...
- [NOI 2016]优秀的拆分
Description 题库链接 给你一个长度为 \(n\) 的只含小写字母的字符串 \(S\) ,计算其子串有多少优秀的拆分. 如果一个字符串能被表示成 \(AABB\) 的形式,其中 \(A,B\ ...
- Git 使用SSH密钥操作
git使用ssh密钥 git支持https和git两种传输协议,github分享链接时会有两种协议可选: git协议链接图例 : ↓ https协议链接图例:↓ git使用https协议,每次pull ...
- BloomFilter布隆过滤器
BloomFilter 简介 当一个元素被加入集合时,通过K个散列函数将这个元素映射成一个位数组中的K个点,把它们置为1.检索时,我们只要看看这些点是不是都是1就(大约)知道集合中有没有它了:如果这些 ...
- 当堆遇到STL 代码焕发光芒
来自度娘的释义,堆的含义大概是这样的: 感性理解: 堆(英语:heap)是计算机科学中一类特殊的数据结构的统称.堆通常是一个可以被看做一棵树的数组对象.堆总是满足下列性质: 堆中某个节点的值总是不大于 ...
- 通过这些示例快速学习Java lambda语法
对于那些不熟悉函数式编程的人来说,基本的Java lambda语法起初可能有点令人生畏.但是,一旦将lambda表达式分解为它们的组成部分,语法很快就会变得有意义并变得非常自然. Java中lambd ...
- Chinese remainder theorem again(中国剩余定理)
C - Chinese remainder theorem again Time Limit:1000MS Memory Limit:32768KB 64bit IO Format:% ...