人民币数字大写转换
include <iostream>
include <iostream>
include <string>
include <stdlib.h>
using namespace std;
const int INT_ONLY = 1;
const int SMALL_ONLY = 2;
/**
* 从命令行接收一个字符串
*/
string getNum() {
string str;
cin >> str;
cin.clear();
return str;
}
/**
* 判断用户输入的数据是否合法,用户只能输入在0~9之间的数字,不能输入其它字符 ,第一个符号位必须为¥
* @param str string
* @return 如果用户输入数据合法,返回 true,否则返回 false
*/
bool checkNum(string str) {
string str_ = "¥";
int flag=0,j=0,length=str.size();
if(str[0]== str_[0]){
int flag=0,j=0,length=str.size();
for( int i=1;i < length; i++){
if((str[i] >= '0')&&(str[i] <= '9'))
j++;
if(str[i] == '.'){
j++;
flag=i;
}
}
return true;
}
else return false;
}
/**
* 原因:输入小数点后可能有多位,只需要对小数点后两位进行操作,故需要四舍五入
* 目标:对传入的数进行四舍五入操作只保留小数点后两位数字
* 方法:先将字符串转换为浮点数,然后对浮点数进行四舍五入,只保留小数点后两位,
* 再将处理过的浮点数转换为字符串进行操作。
atof(s.c_str()) 方法的功能是将字符串 s 转换成一个双精度数值并返回结果
c_str()函数返回一个指向正规C字符串的指针, 内容与本字符串相同
fcvt,函数名,把一个浮点数转换为字符串。
用 法: char *fcvt(double value, int ndigit, int *decpt, int *sign);
参数:
value:要转换的浮点数,输入参数
ndigit:取小数的位数,输入参数
decpt:表示小数点的位置,输出参数
sign:表示value的符号,0为正数,1为负数,输出参数
* @param s string,从命令行输入的那个数
* @return 四舍五入后的新值
*/
string roundString(string & str) {
string s = str.substr(2, str.size()-1);
double d = atof(s.c_str());
int dec, sign;
s = fcvt(d, 2, &dec, &sign);
if(s.length() > 15) {
cout << "输入数据过大!(整数部分最多13位!)" << endl;
return "";
}
return s;
}
/**
* 目标:把传入的数转换为中文金额大写形式
* 原因:
* @param flag int 标志位,1 表示转换整数部分,0 表示转换小数部分
* @param s string 要转换的字符串
* @return 转换好的带单位的中文金额大写形式
*/
string Convert_formatChinese(int flag, string str) {
int sLength = str.length();
string bigLetter[] = {"零", "壹", "贰", "叁", "肆", "伍", "陆", "柒", "捌", "玖"};
string unit[] = {"元", "拾", "佰", "仟", "万",
"拾", "佰", "仟",
"亿", "拾", "佰", "仟", "万"};
string small[] = {"分", "角"};
string newS = "";
for(int i = 0; i < sLength; i ++) {
if(flag == INT_ONLY) {
newS = newS + bigLetter[str.at(i) - 48] + unit[sLength - i - 1];
} else if(flag == SMALL_ONLY) {
newS = newS + bigLetter[str.at(i) - 48] + small[sLength - i - 1];
}
}
return newS;
}
/**
* 把用户输入的数以小数点为界分割开来,并调用 Convert_formatChinese() 方法
* 进行相应的中文金额大写形式的转换
* 注:传入的这个数应该是经过 roundString() 方法进行了四舍五入操作的字符串
* @param s string
* @return 转换好的中文金额大写形式的字符串
*/
string splitNum(string str) {
if("" == str) {
return "";
}
string intOnly = str.substr(0, str.size() - 2);
string intPart = Convert_formatChinese(1, intOnly);
string smallOnly = str.substr(str.size() - 2, str.size());
string smallPart = Convert_formatChinese(2, smallOnly);
string newS = intPart + smallPart;
return newS;
}
/**
* 使用给定的 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.replace(src.find(regex), length, replacement);
}
return src;
}
/** 原因:从数字转换成汉字是采用数组的匹配,故会有会有多个零存在,不符合汉字表达方式
* 目标:把已经转换好的中文金额大写形式加以改进,清理这个字
* 符串里面多余的零,让这个字符串变得符合汉语表达
* 注:传入的这个数应该是经过 splitNum() 方法进行处理,这个字
* 符串应该已经是用中文金额大写形式表示的
* @param s string 已经转换好的字符串
* @return 改进后的字符串
*/
string cleanZero(string str) {
if("" == str) {
return "";
}
string regex1[] = {"零仟", "零佰", "零拾"};
string regex2[] = {"零亿", "零万", "零元"};
string regex3[] = {"亿", "万", "元"};
string regex4[] = {"零角", "零分"};
for(int i = 0; i < 3; i ++) {
str = replaceAll(str, regex1[i], "零");
}
for(int i = 0; i < 3; i ++) {
str = replaceAll(str, "零零零", "零");
str = replaceAll(str, "零零", "零");
str = replaceAll(str, regex2[i], regex3[i]);
}
str = replaceAll(str, "零角零分", "整");
for(int i = 0; i < 2; i ++) {
str = replaceAll(str, regex4[i], "");
}
str = replaceAll(str, "亿万", "亿");
return str;
}
int main(){
cout <<"请输入要转换的人民币金额:" ;
string str = getNum();
if(checkNum( str)){
str = roundString(str);
str = splitNum(str);
str = cleanZero(str);
cout <<"转换结果为:人民币" << str << endl;
}else{
cout << "输入错误!请输入人民币,按如下格式:¥14242.34" <<endl;
}
}
`