将数字转为中文金额的大写方式(C++版)

  1. /** 
  2.  * 程序目的: 
  3.  *   从命令行接收一个数,并将其转化为中文金额的大写方式(C++版) 
  4.  * 例如 123.45 --> 壹佰贰拾叁元肆角伍分 
  5.  * @author LovinChan 
  6.  * 
  7.  *   前一段时间做了个 Java 版的。突然有天心血来潮做个 C++ 版本的,实现的功能 
  8.  * 跟 Java 版本的没什么区别,不过由于我对 C++ 的了解还不是很多,还是学习阶段, 
  9.  * 写出来的东西还有很多问题和不合规范的地方,希望各位批评指出来。 
  10.  *  
  11.  *   程序的注释我尽量写得详细一点,如果觉得这个程序哪里有问题或者是哪里有改进的 
  12.  * 地方欢迎随时跟我交流。 
  13.  *  
  14.  *   我附上了编译以后的 .exe 文件跟 .bat 文件,还有项目源码,供大家测评。 
  15.  * 
  16.  * 工具:Microsoft Visual Studio 2005 
  17.  * 编译环境:Microsoft Visual Studio 2005 
  18.  * 
  19.  * 我的msn:egg.chenlw@gmail.com 
  20.  *    QQ:372133556(注上为什么加我就可以了) 
  21.  * 我的blog:http://hi.baidu.com/egg_chen 
  22.  * 欢迎交流 
  23.  */  
  24. #include <iostream>  
  25. #include <string>  
  26.   
  27. using namespace std;  
  28.   
  29. // 表示整数部分的标志  
  30. const int INT_ONLY = 1;  
  31. // 表示小数部分的标志  
  32. const int SMALL_ONLY = 2;  
  33.   
  34. /** 
  35.  * 从命令行接收一个数,在其中调用 checkNum() 方法对其进行 
  36.  * 验证,并返回相应的值 
  37.  * @return 如果输入合法,返回输入的这个数 
  38.  */  
  39. string getNum() {  
  40.     string s;  
  41.     cout << "请输入一个数字(精确到小数点后两位):" << endl;  
  42.     // 从命令行输入这个浮点数  
  43.     cin >> s;  
  44.     // 清除输入流状态标志  
  45.     cin.clear();  
  46.     return s;  
  47. }  
  48.   
  49. /** 
  50. * 判断用户输入的数据是否合法,用户只能输入大于零的数字,不能输入其它字符 
  51. * @param s string 
  52. * @return 如果用户输入数据合法,返回 true,否则返回 false 
  53. */  
  54. bool checkNum(string s) {  
  55.     // atof(s.c_str()) 方法的功能是将字符串 s 转换成一个双精度数值并返回结果  
  56.     double d = atof(s.c_str());  
  57.     // 只有当用户输入一个大于0的数时,才会返回true  
  58.     if(d > 0) {  
  59.         return true;  
  60.     }  
  61.     return false;  
  62. }  
  63.   
  64. /** 
  65.  * 对传入的数进行四舍五入操作 
  66.  * @param s string,从命令行输入的那个数 
  67.  * @return 四舍五入后的新值 
  68.  */  
  69. string roundString(string s) {  
  70.     // 将这个数转换成 double 类型,并对其进行四舍五入操作  
  71.     // 先转换这个数的整数部分  
  72.     // atof(s.c_str()) 方法的功能是将字符串 s 转换成一个双精度数值并返回结果  
  73.     // c_str()函数返回一个指向正规C字符串的指针, 内容与本字符串相同  
  74.     double d = atof(s.c_str());  
  75.     // 将这个数进行四舍五入,保留到小数点后两位  
  76.     // 再将这个数转换成字符串,等待转换  
  77.     int dec, sign;   
  78.     // 注意:当这个数转换成字符串以后不会显示小数点,并且会以四舍五入的形式只保留小数点后两位  
  79.     s = fcvt(d, 2, &dec, &sign);  
  80.     // 规定数值的最大长度只能是15位(到万亿位)  
  81.     if(s.length() > 15) {  
  82.         cout << "输入数据过大!(整数部分最多13位!)" << endl;  
  83.         return "";  
  84.     }  
  85.     return s;  
  86. }  
  87.   
  88. /** 
  89.  * 把传入的数转换为中文金额大写形式 
  90.  * @param flag int 标志位,1 表示转换整数部分,0 表示转换小数部分 
  91.  * @param s string 要转换的字符串 
  92.  * @return 转换好的带单位的中文金额大写形式 
  93.  */  
  94. string formatChinese(int flag, string s) {  
  95.     int sLength = s.length();  
  96.     // 货币大写形式  
  97.     string bigLetter[] = {"零""壹""贰""叁""肆""伍""陆""柒""捌""玖"};  
  98.     // 货币单位  
  99.     string unit[] = {"元""拾""佰""仟""万",   
  100.                 // 拾万位到仟万位  
  101.                 "拾""佰""仟",  
  102.                 // 亿位到万亿位  
  103.                 "亿""拾""佰""仟""万"};  
  104.     string small[] = {"分""角"};  
  105.     // 用来存放转换后的新字符串  
  106.     string newS = "";  
  107.     // 逐位替换为中文大写形式  
  108.     for(int i = 0; i < sLength; i ++) {  
  109.         if(flag == INT_ONLY) {  
  110.             // 转换整数部分为中文大写形式(带单位)  
  111.             newS = newS + bigLetter[s.at(i) - 48] + unit[sLength - i - 1];  
  112.         } else if(flag == SMALL_ONLY) {  
  113.             // 转换小数部分(带单位)  
  114.             newS = newS + bigLetter[s.at(i) - 48] + small[sLength - i - 1];  
  115.         }  
  116.     }  
  117.     return newS;  
  118. }  
  119.   
  120. /** 
  121.  * 把用户输入的数以小数点为界分割开来,并调用 numFormat() 方法 
  122.  * 进行相应的中文金额大写形式的转换 
  123.  * 注:传入的这个数应该是经过 roundString() 方法进行了四舍五入操作的 
  124.  * @param s string 
  125.  * @return 转换好的中文金额大写形式的字符串 
  126.  */  
  127. string splitNum(string s) {  
  128.     // 如果传入的是空串则继续返回空串  
  129.     if("" == s) {  
  130.         return "";  
  131.     }  
  132.     // 截取输入数字的整数部分  
  133.     string intOnly = s.substr(0, s.size() - 2);  
  134.     string intPart = formatChinese(1, intOnly);  
  135.   
  136.     // 截取这个数的小数部分  
  137.     string smallOnly = s.substr(s.size() - 2, s.size());  
  138.     string smallPart = formatChinese(2, smallOnly);  
  139.   
  140.     // 把转换好了的整数部分和小数部分重新拼凑一个新的字符串  
  141.     string newS = intPart + smallPart;  
  142.   
  143.     return newS;  
  144. }  
  145.   
  146. /** 
  147.  * 使用给定的 replacement 替换此字符串所有匹配给定的 regex 的子字符串。 
  148.  * @param src - 待操作的源字符串 
  149.  * @param regex - 用来匹配此字符串的正则表达式 
  150.  * @param replacement - 用来替换每个匹配项的字符串  
  151.  * @return 替换后的字符串  
  152.  */  
  153. string replaceAll(string src, string regex, string replacement) {  
  154.     int length = regex.length();  
  155.     while(src.find(regex) < src.length()) {  
  156.         // 替换 src 字符串中从第一个匹配 regex 的字符串索引开始的 length 个字符为 replacement 字符串          
  157.         src.replace(src.find(regex), length, replacement);  
  158.     }  
  159.     return src;  
  160. }  
  161.   
  162. /** 
  163.  * 把已经转换好的中文金额大写形式加以改进,清理这个字 
  164.  * 符串里面多余的零,让这个字符串变得更加可观 
  165.  * 注:传入的这个数应该是经过 splitNum() 方法进行处理,这个字 
  166.  * 符串应该已经是用中文金额大写形式表示的 
  167.  * @param s string 已经转换好的字符串 
  168.  * @return 改进后的字符串 
  169.  */  
  170. string cleanZero(string s) {  
  171.     // 如果传入的是空串则继续返回空串  
  172.     if("" == s) {  
  173.         return "";  
  174.     }  
  175.     // 字符串中存在多个'零'在一起的时候只读出一个'零',并省略多余的单位  
  176.     /* 由于本人对算法的研究太菜了,只能用4个正则表达式去转换了,各位大虾别介意哈... */  
  177.     string regex1[] = {"零仟""零佰""零拾"};  
  178.     string regex2[] = {"零亿""零万""零元"};  
  179.     string regex3[] = {"亿""万""元"};  
  180.     string regex4[] = {"零角""零分"};  
  181.   
  182.     // 第一轮转换把 "零仟", 零佰","零拾"等字符串替换成一个"零"  
  183.     for(int i = 0; i < 3; i ++) {  
  184.         s = replaceAll(s, regex1[i], "零");  
  185.     }  
  186.     // 第二轮转换考虑 "零亿","零万","零元"等情况  
  187.     // "亿","万","元"这些单位有些情况是不能省的,需要保留下来  
  188.     for(int i = 0; i < 3; i ++) {  
  189.         // 当第一轮转换过后有可能有很多个零叠在一起  
  190.         // 要把很多个重复的零变成一个零  
  191.         s = replaceAll(s, "零零零""零");  
  192.         s = replaceAll(s, "零零""零");  
  193.         s = replaceAll(s, regex2[i], regex3[i]);  
  194.     }  
  195.     // 第三轮转换把"零角","零分"字符串省略  
  196.     for(int i = 0; i < 2; i ++) {  
  197.         s = replaceAll(s, regex4[i], "");  
  198.     }  
  199.     // 当"万"到"亿"之间全部是"零"的时候,忽略"亿万"单位,只保留一个"亿"  
  200.     s = replaceAll(s, "亿万""亿");  
  201.     return s;  
  202. }  
  203.   
  204. int main() {  
  205.     cout << "\n------------将数字转换成中文金额的大写形式(C++)------------\n" << endl;  
  206.     string s = getNum();  
  207.     if(checkNum(s)) {  
  208.         s = roundString(s);  
  209.         s = splitNum(s);  
  210.         s = cleanZero(s);  
  211.         cout << "转换成中文后为:" << s << endl;  
  212.     } else {  
  213.         cout << "非法输入,程序即将退出" << endl;  
  214.     }  
  215.     cout << "\n--------------------------------------------------------------" << endl;  
  216. }  
  • 10
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值