c++11实现一个简单的lexical_cast
来源:互联网 发布:云顶酒店闹鬼 知乎 编辑:程序博客网 时间:2024/06/05 18:27
boost中有一个lexical_cast可以用统一的方式来做基本类型之间的转换,比如字符串到数字,数字到字符串,bool和字符串及数字之间的相互转换。boost::lexical_cast的用法比较简单:
#include <boost/lexical_cast.hpp>#include <iostream>#include <string> #define ERROR_LEXICAL_CAST 1 int main(){ using boost::lexical_cast; int a = 0; double b = 0.0; std::string s = ""; int e = 0; try { // ----- 字符串 --> 数值 a = lexical_cast<int>("123"); b = lexical_cast<double>("123.12"); // ----- 数值 --> 字符串 s = lexical_cast<std::string>("123456.7"); // ----- 异常处理演示 e = lexical_cast<int>("abc"); } catch(boost::bad_lexical_cast& e) { // bad lexical cast: source type value could not be interpreted as target std::cout << e.what() << std::endl; return ERROR_LEXICAL_CAST; } std::cout << a << std::endl; // 输出:123 std::cout << b << std::endl; // 输出:123.12 std::cout << s << std::endl; // 输出:123456.7 return 0;}
c++11中缺少lexical_cast方法,但是c++11已经提供了一些基本类型转换的方法,比如to_string, atoi, atof等等,但是我们不能通过一种通用的方式来做基本类型转换,因此我希望做一个类似boost的lexical_cast做基本类型的转换,这也是我们的c++社区的一个开发计划。
由于c++11已经提供了一些便利的方法,我要做的事情就变得很简单了,就是把他们糅合在一起并提供一个统一的lexical_cast的方法即可。
实现思路也很简单,转换主要有这几种:1.数字到字符串的转换;2.字符串到数字的转换;3.bool与字符串的相互转换;4.数字转换为bool;具体的实现代码如下:
#include <type_traits>#include <string>#include <cstdlib>#include <algorithm>#include <stdexcept>#include <cctype>#include <cstring>using namespace std;namespace detail{ const char* strue = "true"; const char* sfalse = "false"; template <typename To, typename From> struct Converter { }; //to numeric template <typename From> struct Converter<int, From> { static int convert(const From& from) { return std::atoi(from); } }; template <typename From> struct Converter<long, From> { static long convert(const From& from) { return std::atol(from); } }; template <typename From> struct Converter<long long, From> { static long long convert(const From& from) { return std::atoll(from); } }; template <typename From> struct Converter<double, From> { static double convert(const From& from) { return std::atof(from); } }; template <typename From> struct Converter<float, From> { static float convert(const From& from) { return (float)std::atof(from); } }; //to bool template <typename From> struct Converter<bool, From> { static typename std::enable_if<std::is_integral<From>::value, bool>::type convert(From from) { return !!from; } }; bool checkbool(const char* from, const size_t len, const char* s) { for (size_t i = 0; i < len; i++) { if (from[i] != s[i]) { return false; } } return true; } static bool convert(const char* from) { const unsigned int len = strlen(from); if (len != 4 && len != 5) throw std::invalid_argument("argument is invalid"); bool r = true; if (len == 4) { r = checkbool(from, len, strue); if (r) return true; } else { r = checkbool(from, len, sfalse); if (r) return false; } throw std::invalid_argument("argument is invalid"); } template <> struct Converter<bool, string> { static bool convert(const string& from) { return detail::convert(from.c_str()); } }; template <> struct Converter<bool, const char*> { static bool convert(const char* from) { return detail::convert(from); } }; template <> struct Converter<bool, char*> { static bool convert(char* from) { return detail::convert(from); } }; template <unsigned N> struct Converter<bool, const char[N]> { static bool convert(const char(&from)[N]) { return detail::convert(from); } }; template <unsigned N> struct Converter<bool, char[N]> { static bool convert(const char(&from)[N]) { return detail::convert(from); } }; //to string template <typename From> struct Converter<string, From> { static string convert(const From& from) { return std::to_string(from); } };}template <typename To, typename From>typename std::enable_if<!std::is_same<To, From>::value, To>::type lexical_cast(const From& from){ return detail::Converter<To, From>::convert(from);}template <typename To, typename From>typename std::enable_if<std::is_same<To, From>::value, To>::type lexical_cast(const From& from){ return from;}
前后花了一个多小时,一个基本的类型转换类就完成了,再测试一下吧,测试代码:
void test(){ cout<<lexical_cast<int>(1)<<endl; cout << lexical_cast<int>("1") << endl; cout << lexical_cast<long>("1") << endl; cout << lexical_cast<string>(1) << endl; cout << lexical_cast<bool>(1) << endl; cout << lexical_cast<double>("1.2") << endl; cout << lexical_cast<float>("1.2") << endl; string s = "true"; cout << lexical_cast<bool>(s) << endl; char* p = "false"; cout << lexical_cast<bool>(p) << endl; const char* q = "false"; cout << lexical_cast<bool>(q) << endl; cout << lexical_cast<bool>("false") << endl; cout << lexical_cast<bool>("test") << endl;}int main(){ try { test(); } catch (const std::exception& e) { cout << e.what() << endl; } return 0;}
0 0
- c++11实现一个简单的lexical_cast
- boost::lexical_cast简单实现
- boost/lexical_cast.hpp的简单使用方法
- boost/lexical_cast.hpp的简单使用方法
- C++流实现内幕---由boost::lexical_cast引发的一个问题
- C++流实现内幕---由boost::lexical_cast引发的一个问题
- 一个简单的http_server的c实现
- 一个简单的HashMap C语言实现
- 一个简单的HashMap C语言实现
- BloomFilter的一个简单实现(C语言)
- C语言实现一个简单的计算器
- 【C++】一个简单栈的实现
- 一个简单定时器的实现(C++)
- 【C语言】实现一个简单的通讯录
- 一个简单C顺序栈的实现
- 一个简单定时器的实现(C++)
- C语言实现一个简单的栈
- C语言实现一个简单的服务器
- Linux下搭建xmpp ejabberd 服务器
- C#获取泛型集合中元素的类型
- cookie和session
- JavaScript之DOM样式
- java基础之注释
- c++11实现一个简单的lexical_cast
- 游标变量操作
- 即时搜索input propertychange的bug
- 20160219 .NET中App.config加密实现
- 自己动手实现图的BFS和DFS
- thinkphp 与 tornado 性能测试比较
- NSJSONSerialization介绍
- leetcode 219: Contains Duplicate II
- ThinkPHP跳转页success及error模板实例教程