字符串数字转换成中国的元角分

雨燕双飞 提交于 2020-04-05 17:26:27
/**
 * 程序目的:
 *   从命令行接收一个数,并将其转化为中文金额的大写方式(C++版)
 * 例如 123.45 --> 壹佰贰拾叁元肆角伍分
 * @author LovinChan
 *
 *   前一段时间做了个 Java 版的。突然有天心血来潮做个 C++ 版本的,实现的功能
 * 跟 Java 版本的没什么区别,不过由于我对 C++ 的了解还不是很多,还是学习阶段,
 * 写出来的东西还有很多问题和不合规范的地方,希望各位批评指出来。
 *
 *   程序的注释我尽量写得详细一点,如果觉得这个程序哪里有问题或者是哪里有改进的
 * 地方欢迎随时跟我交流。
 *
 *   我附上了编译以后的 .exe 文件跟 .bat 文件,还有项目源码,供大家测评。
 *
 * 工具:Microsoft Visual Studio 2005
 * 编译环境:Microsoft Visual Studio 2005
 *
 * 我的msn:egg.chenlw@gmail.com
 *    QQ:372133556(注上为什么加我就可以了)
 * 我的blog:http://hi.baidu.com/egg_chen
 * 欢迎交流
 */

#define _CRT_SECURE_NO_WARNINGS

#include <iostream>  
#include <string>
#include <stdlib.h>

using namespace std;

/**
 * 从命令行接收一个数,在其中调用 checkNum() 方法对其进行
 * 验证,并返回相应的值
 * @return 如果输入合法,返回输入的这个数
 */
string getNum() {
	string s;
	cout << "请输入一个数字(精确到小数点后两位):" << endl;
	// 从命令行输入这个浮点数  
	cin >> s;
	// 清除输入流状态标志  
	cin.clear();
	return s;
}

// 判断数字是否合法,是否大于零
bool checkNum(string& s) {

	int decpt = 0;
	int sign = 0;
	double d = 0.00;
	string str;

	d = atof(s.c_str());

	str = _fcvt(d, 2, &decpt, &sign);

	// 输入数据过大!(整数部分最多13位!)
	if (str.length() > 15) return false;
	// 数据为负数
	if (1 == sign) return false;

	s = str;

	return true;
}

// 把传入的数转换为中文金额大写形式
string formatChinese(bool bint, string s) {
	// ASCII 48~57 对应 0~9
	
	// 货币大写形式  
	string bigLetter[] = { "零", "壹", "贰", "叁", "肆", "伍", "陆", "柒", "捌", "玖" };
	// 货币单位  
	string unit[] = { "元", "拾", "佰", "仟", "万",
		// 拾万位到仟万位  
		"拾", "佰", "仟",
		// 亿位到万亿位  
		"亿", "拾", "佰", "仟", "万" };
	string small[] = { "分", "角" };
	// 用来存放转换后的新字符串
	string tmp;
	string newS;
	int sLength = 0;
	int i = 0;
	
	sLength = s.length();
	if (0 == sLength) return "";
	// 逐位替换为中文大写形式  
	for (i = 0; i < sLength; i++) {
		if (true == bint) 
		{
			// 转换整数部分为中文大写形式(带单位)  
			tmp = bigLetter[s.at(i) - 48] + unit[sLength - i - 1];
		}
		else
		{
			// 转换小数部分(带单位)  
			tmp = bigLetter[s.at(i) - 48] + small[sLength - i - 1];
		}

		newS = newS + tmp;
	}

	return newS;
}

// 数字转换成字符串
bool splitNum(string& s) 
{
	string tmp;
	string intpart;
	string smallpart;

	if (0 == s.length()) return false;
	
	if (s.length() > 2)
	{
		// 截取整数部分
		tmp = s.substr(0, s.size() - 2);
		intpart = formatChinese(true, tmp);

		// 截取小数部分  
		tmp = s.substr(s.size() - 2, s.size());
		smallpart = formatChinese(false, tmp);
	}
	else
	{
		// 截取小数部分
		tmp = s;
		smallpart = formatChinese(false, tmp);
	}

	// 把转换好了的整数部分和小数部分重新拼凑一个新的字符串  
	s = intpart + smallpart;

	return true;
}

/**
 * 使用给定的 replacement 替换此字符串所有匹配给定的 regex 的子字符串。
 * @param src - 待操作的源字符串
 * @param regex - 用来匹配此字符串的正则表达式
 * @param replacement - 用来替换每个匹配项的字符串
 * @return 替换后的字符串
 */
string replaceAll(string src, string regex, string replacement) {
	int length = regex.length();
	while (src.find(regex) < src.length()) {
		// 替换 src 字符串中从第一个匹配 regex 的字符串索引开始的 length 个字符为 replacement 字符串          
		src.replace(src.find(regex), length, replacement);
	}
	return src;
}

/**
 * 把已经转换好的中文金额大写形式加以改进,清理这个字
 * 符串里面多余的零,让这个字符串变得更加可观
 * 注:传入的这个数应该是经过 splitNum() 方法进行处理,这个字
 * 符串应该已经是用中文金额大写形式表示的
 * @param s string 已经转换好的字符串
 * @return 改进后的字符串
 */
string cleanZero(string s) {
	// 如果传入的是空串则继续返回空串  
	if ("" == s) {
		return "";
	}
	// 字符串中存在多个'零'在一起的时候只读出一个'零',并省略多余的单位  
	/* 由于本人对算法的研究太菜了,只能用4个正则表达式去转换了,各位大虾别介意哈... */
	string regex1[] = { "零仟", "零佰", "零拾" };
	string regex2[] = { "零亿", "零万", "零元" };
	string regex3[] = { "亿", "万", "元" };
	string regex4[] = { "零角", "零分" };

	// 第一轮转换把 "零仟", 零佰","零拾"等字符串替换成一个"零"  
	for (int i = 0; i < 3; i++) {
		s = replaceAll(s, regex1[i], "零");
	}
	// 第二轮转换考虑 "零亿","零万","零元"等情况  
	// "亿","万","元"这些单位有些情况是不能省的,需要保留下来  
	for (int i = 0; i < 3; i++) {
		// 当第一轮转换过后有可能有很多个零叠在一起  
		// 要把很多个重复的零变成一个零  
		s = replaceAll(s, "零零零", "零");
		s = replaceAll(s, "零零", "零");
		s = replaceAll(s, regex2[i], regex3[i]);
	}
	// 第三轮转换把"零角","零分"字符串省略  
	for (int i = 0; i < 2; i++) {
		s = replaceAll(s, regex4[i], "");
	}
	// 当"万"到"亿"之间全部是"零"的时候,忽略"亿万"单位,只保留一个"亿"  
	s = replaceAll(s, "亿万", "亿");
	return s;
}

int main() {
	

	while (true)
	{
		cout << "\n------------将数字转换成中文金额的大写形式(C++)------------\n" << endl;
		string s = getNum();
		if (true == checkNum(s) && true == splitNum(s)) {
			s = cleanZero(s);
			cout << "转换成中文后为:" << s << endl;
		}
		else {
			cout << "非法输入,程序即将退出" << endl;
		}
		cout << "\n--------------------------------------------------------------" << endl;
	}

	
}

  

标签
易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!