C++ 字符串转浮点数,包括整数、小数和科学记数法

简介: C++ 字符串转浮点数,包括整数、小数和科学记数法

字符串转浮点数,用stoi(),stod(),stof()等库函数,好像转换精度不够,还特别容易出错。于是就写了一个自定义函数double str2num(string),可以转换包括整数、小数和科学记数法浮点数。代码用了double型变量,整数最长可以保留16位。如果觉得还不够,可以在所有的 double 前添上 long 就能增加到20位精度。还有一点科学记数法只用了e,大写E注意先转小写。小数向0趋近时直接输出0,大于DBL_MAX的输出inf错;输入非数字的情况,人为用pow()函数实现输出nan错。如果取得的数字串已用正则表达式检验过,可以删除掉两个注释符之间的代码。


#include <iostream>
#include <iomanip>
#include <vector>
#include <cmath>
using namespace std;
#define isNum(c) (isdigit(c)?c-48:(c=='e'?10:(c=='.'?11:(c=='-'?12:(c=='+'?13:-1)))))
double str2num(string s)
{ //字符串转数字,包括整数、小数和科学记数法 
  int i,j,k,negative=0;
  double n=0;
  string s1,s2;
  if (s.empty()) return 0;
  if (s[0]=='-') negative=1; //设置负数标记 
  if (s[0]=='+'||s[0]=='-') s=s.substr(1,s.size());
  //--------------- 
  for (i=0;i<s.size();i++) //排除不需要的字符 
    if (isNum(s[i])==-1) return pow(-1.1,1.1);
  if (s[0]=='e'||s[0]=='.'||s[s.size()-1]=='e'||s[s.size()-1]=='.')
    return pow(-1.1,1.1); //排除 e或. 出现在首尾 
  i=-1;j=0;
  while ((i=s.find('.',++i))!=s.npos) j++;
  if (j>1) return pow(-1.1,1.1); //排除多个小数点 
  i=-1;j=0;
  while ((i=s.find('e',++i))!=s.npos) j++;
  if (j>1) return pow(-1.1,1.1); //排除多个字母e 
  if (s.find('e')==s.npos) //没有e时排除加减
    if (s.find('+')!=s.npos||s.find('-')!=s.npos) return pow(-1.1,1.1);
  //---------------
  if ((i=s.find('e'))!=s.npos){
    s1=s.substr(0,i); //尾数部分 
    s2=s.substr(i+1,s.size()); //阶码 
    if (s2[0]=='+') s2=s2.substr(1,s2.size()); //阶码为正数,去掉+ 
    if (s2.find('.')!=s2.npos) return pow(-1.1,1.1); //阶码不准出现小数
    n=str2num(s1)*pow(10.0,str2num(s2)); //尾数和阶码分别递归调用 
    return negative?-n:n;
  }
  i=0;k=1; 
    if ((i=s.find('.'))!=s.npos){
      for (j=i+1;j<s.length();j++,k++)
        n+=isNum(s[j])/pow(10.0,(double)k);
      n+=str2num(s.substr(0,i));  //整数部分递归调用 
    }
    else
      for (j=0;j<s.size();j++)
        n=n*10+isNum(s[j]);
  return negative?-n:n; //负数返回-n 
}
int main(void)
{
  vector<string>a={"-12","0","+12.345","123456789012345","1.23456789012345e+20",
    "-1.5e-2","1e2","3e1.1","1.7977e+308","-4.95e-324","1.1.1","1e2e2","abc","1+2"};
  for (auto s:a)
    cout<<s<<"->"<<setprecision(15)<<str2num(s)<<endl;
}


测试结果如下:

E:\>test
-12->-12
0->0
+12.345->12.345
123456789012345->123456789012345
1.23456789012345e+20->1.23456789012345e+020
-1.5e-2->-0.015
1e2->100
3e1.1->nan
1.7977e+308->inf
4.95e-324->0
1.1.1->nan
1e2e2->nan
abc->nan
1+2->nan
E:\>


最后,还是觉得字符串与数字的互转还用<sstream>库函数最方便:

#include <iostream>
#include <iomanip>
#include <sstream>
#include <string>
using namespace std;
string double2str(double d)
{
  string s;
  stringstream ss;
  ss<<setprecision(16)<<d;
  s=ss.str();
  ss.clear();
  return s;
}
double str2double(string s)
{
  double d;
  stringstream ss;
  ss<<s;
  ss>>setprecision(16)>>d;
  ss.clear();
  return d;
}
int main()
{
    double x;
    string str;
    for(;;) {
    cin >> str;
      x=str2double(str);
      cout<<setprecision(16)<<x<<"->|"+double2str(x)+"|"<<endl;
      if (x==-1) return 0;
  }
}


- 编译器名: TDM-GCC 4.9.2 64-bit Release

处理 C++ 源文件...

--------

- C++ 编译器: C:\Program Files (x86)\Dev-Cpp\MinGW64\bin\g++.exe

- 命令: g++.exe "E:\test.cpp" -o "E:\test.exe" -std=c++11 ......................



目录
相关文章
|
2月前
|
搜索推荐 编译器 C语言
【C++核心】特殊的元素集合-数组与字符串详解
这篇文章详细讲解了C++中数组和字符串的基本概念、操作和应用,包括一维数组、二维数组的定义和使用,以及C风格字符串和C++字符串类的对比。
80 4
|
1月前
|
人工智能 C++
第十四届蓝桥杯省赛大学B组(C/C++)整数删除
第十四届蓝桥杯省赛大学B组(C/C++)整数删除
|
1月前
|
缓存 网络协议 API
C/C++ StringToAddress(字符串转 boost::asio::ip::address)
通过上述步骤和示例代码,你可以轻松地在C++项目中实现从字符串到 `boost::asio::ip::address`的转换,从而充分利用Boost.Asio库进行网络编程。
52 0
|
1月前
|
编译器 C语言 C++
C/C++数字与字符串互相转换
C/C++数字与字符串互相转换
|
2月前
|
C++
HTML+JavaScript构建一个将C/C++定义的ANSI字符串转换为MASM32定义的DWUniCode字符串的工具
HTML+JavaScript构建一个将C/C++定义的ANSI字符串转换为MASM32定义的DWUniCode字符串的工具
|
2月前
|
存储 C++
C++(五)String 字符串类
本文档详细介绍了C++中的`string`类,包括定义、初始化、字符串比较及数值与字符串之间的转换方法。`string`类简化了字符串处理,提供了丰富的功能如字符串查找、比较、拼接和替换等。文档通过示例代码展示了如何使用这些功能,并介绍了如何将数值转换为字符串以及反之亦然的方法。此外,还展示了如何使用`string`数组存储和遍历多个字符串。
|
3月前
|
C++
C++产生N以内的随机整数
C++产生N以内的随机整数
19 0
|
4月前
|
算法 C++
2730. 找到最长的半重复子字符串(c++,滑动窗口)
2730. 找到最长的半重复子字符串(c++,滑动窗口)
|
4月前
|
C++
567. 字符串的排列(c++)滑动窗口
567. 字符串的排列(c++)滑动窗口
|
4月前
|
编译器 C++
【C++】string类的使用④(字符串操作String operations )
这篇博客探讨了C++ STL中`std::string`的几个关键操作,如`c_str()`和`data()`,它们分别返回指向字符串的const char*指针,前者保证以&#39;\0&#39;结尾,后者不保证。`get_allocator()`返回内存分配器,通常不直接使用。`copy()`函数用于将字符串部分复制到字符数组,不添加&#39;\0&#39;。`find()`和`rfind()`用于向前和向后搜索子串或字符。`npos`是string类中的一个常量,表示找不到匹配项时的返回值。博客通过实例展示了这些函数的用法。