人民幣數字大小寫轉換
阿新 • • 發佈:2019-01-05
人民幣數字大寫轉換
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;
}
}
`