人民币数字大小写转换

人民币数字大写转换


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();
  //  cout <<1<< "" << str <<endl;
    return str;
}

/**
* 判断用户输入的数据是否合法,用户只能输入在0~9之间的数字,不能输入其它字符 ,第一个符号位必须为¥
* @param str string
* @return 如果用户输入数据合法,返回 true,否则返回 false
*/
bool checkNum(string str) {
     string str_ = "¥";
     int flag=0,j=0,length=str.size();        //设置标记符j
     if(str[0]== str_[0]){           //判断第一个输入字符是否为¥,如果不是"¥",提示输入错误,重新输入

          int flag=0,j=0,length=str.size();        //设置标记符j
          for( int i=1;i < length; i++){                     //判断输入字符串除第一个字符为¥,是否还有其他字母

              if((str[i] >= '0')&&(str[i] <= '9'))           //如果输入字符串除第一个字符每一位都是数字,则j++
                 j++;
              if(str[i] == '.'){                             //如果输入字符串除第一个字符有小数点,则j++
                 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) {
    // 将这个数转换成 double 类型,并对其进行四舍五入操作
    // 先转换这个数的整数部分
    // atof(s.c_str()) 方法的功能是将字符串 s 转换成一个双精度数值并返回结果
    // c_str()函数返回一个指向正规C字符串的指针, 内容与本字符串相同

    string s = str.substr(2, str.size()-1);
     double d = atof(s.c_str());
    // 将这个数进行四舍五入,保留到小数点后两位
    // 再将这个数转换成字符串,等待转换

   //  cout << "d = " << d << endl;

    int dec, sign;
    // 注意:当这个数转换成字符串以后不会显示小数点,并且会以四舍五入的形式只保留小数点后两位
    s = fcvt(d, 2, &dec, &sign);
    // 规定数值的最大长度只能是15位(到万亿位)
    //  cout << "3.3 " <<  s <<endl;

    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 "";
    }
    // 截取输入数字的整数部分,调用Convert_formatChinese()转换成对应汉字
    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;

    // cout <<5 << " " << newS;
    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 字符串中从第一个匹配 regex 的字符串索引开始的 length 个字符为 replacement 字符串
        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, "亿万", "亿");
   //  cout <<7 << " " << 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;
    }

}
`
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值