人民幣數字大寫轉換
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;
}
}
`