Java實現阿拉伯數字轉換成中文大寫數字,以及中文大寫數字到阿拉伯數字的轉換。
阿新 • • 發佈:2019-02-08
學習王曉華老師的《演算法的樂趣》一書中第四章之後,用Java重寫並實現這一功能。
該文章是在學習完該章之後的一個學習總結,以供自己複習使用。
由於現在剛開始學習Java,為了熟悉Java的變成規範,因此,用Java實現這一功能。
public class NumberToChn { static String CHN_NUMBER[] = {"零", "一", "二", "三", "四", "五", "六", "七", "八", "九"}; static String CHN_UNIT[] = {"", "十", "百", "千"}; //權位 static String CHN_UNIT_SECTION[] = {"", "萬", "億", "萬億"}; //節權位 /** * 測試資料的資料型別 */ public static class Test_Data{ int number; String chnNum; public Test_Data(int number,String chnNum){ this.chnNum=chnNum; this.number=number; } } /** * 測試資料 */ static Test_Data testData[]={ new Test_Data(0,"零"), new Test_Data(1,"一"), new Test_Data(2,"二"), new Test_Data(3, "三"), new Test_Data(4, "四"), new Test_Data(5, "五"), new Test_Data(6, "六"), new Test_Data(7, "七"), new Test_Data(8, "八"), new Test_Data(9, "九"), new Test_Data(10, "一十"), new Test_Data(11, "一十一"), new Test_Data(110, "一百一十"), new Test_Data(111, "一百一十一"), new Test_Data(100, "一百"), new Test_Data(102, "一百零二"), new Test_Data(1020, "一千零二十"), new Test_Data(1001, "一千零一"), new Test_Data(1015, "一千零一十五"), new Test_Data(1000, "一千"), new Test_Data(10000, "一萬"), new Test_Data(20010, "二萬零一十"), new Test_Data(20001, "二萬零一"), new Test_Data(100000, "一十萬"), new Test_Data(1000000, "一百萬"), new Test_Data(10000000, "一千萬"), new Test_Data(100000000, "一億"), new Test_Data(1000000000, "一十億"), new Test_Data(1000001000, "一十億一千"), new Test_Data(1000000100, "一十億零一百"), new Test_Data(200010, "二十萬零一十"), new Test_Data(2000105, "二百萬零一百零五"), new Test_Data(20001007, "二千萬一千零七"), new Test_Data(2000100190, "二十億零一十萬零一百九十"), new Test_Data(1040010000, "一十億四千零一萬"), new Test_Data(200012301, "二億零一萬二千三百零一"), new Test_Data(2005010010, "二十億零五百零一萬零一十") // new Test_Data(4009060200, "四十億零九百零六萬零二百"), // new Test_Data(4294967295, "四十二億九千四百九十六萬七千二百九十五") }; /** * 阿拉伯數字轉換為中文數字的核心演算法實現。 * @param num為需要轉換為中文數字的阿拉伯數字,是無符號的整形數 * @return */ public static String NumberToChn(int num) { StringBuffer returnStr = new StringBuffer(); Boolean needZero = false; int pos=0; //節權位的位置 if(num==0){ //如果num為0,進行特殊處理。 returnStr.insert(0,CHN_NUMBER[0]); } while (num > 0) { int section = num % 10000; if (needZero) { returnStr.insert(0, CHN_NUMBER[0]); } String sectionToChn = SectionNumToChn(section); //判斷是否需要節權位 sectionToChn += (section != 0) ? CHN_UNIT_SECTION[pos] : CHN_UNIT_SECTION[0]; returnStr.insert(0, sectionToChn); needZero = ((section < 1000 && section > 0) ? true : false); //判斷section中的千位上是不是為零,若為零應該新增一個零。 pos++; num = num / 10000; } return returnStr.toString(); } /** * 將四位的section轉換為中文數字 * @param section * @return */ public static String SectionNumToChn(int section) { StringBuffer returnStr = new StringBuffer(); int unitPos = 0; //節權位的位置編號,0-3依次為個十百千; Boolean zero = true; while (section > 0) { int v = (section % 10); if (v == 0) { if ((section == 0) || !zero) { zero = true; /*需要補0,zero的作用是確保對連續的多個0,只補一箇中文零*/ //chnStr.insert(0, chnNumChar[v]); returnStr.insert(0, CHN_NUMBER[v]); } } else { zero = false; //至少有一個數字不是0 StringBuffer tempStr = new StringBuffer(CHN_NUMBER[v]);//數字v所對應的中文數字 tempStr.append(CHN_UNIT[unitPos]); //數字v所對應的中文權位 returnStr.insert(0, tempStr); } unitPos++; //移位 section = section / 10; } return returnStr.toString(); } /** * 完成將阿拉伯數字轉換為中文數字的測試 */ public static void TestNumToChn(){ for(int i=0;i<testData.length;i++) { String str=NumberToChn(testData[i].number); System.out.println(testData[i].number+"\t"+testData[i].chnNum+"\t"+str+"\t"+str.equals(testData[i].chnNum)); } } /** * 中文轉換成阿拉伯數字,中文字串除了包括0-9的中文漢字,還包括十,百,千,萬等權位。 * 此處是完成對這些權位的型別定義。 * name是指這些權位的漢字字串。 * value是指權位多對應的數值的大小。諸如:十對應的值的大小為10,百對應為100等 * secUnit若為true,代表該權位為節權位,即萬,億,萬億等 */ public static class Chn_Name_value{ String name; int value; Boolean secUnit; public Chn_Name_value(String name,int value,Boolean secUnit){ this.name=name; this.value=value; this.secUnit=secUnit; } } static Chn_Name_value chnNameValue[]={ new Chn_Name_value("十",10,false), new Chn_Name_value("百",100,false), new Chn_Name_value("千",1000,false), new Chn_Name_value("萬",10000,true), new Chn_Name_value("億",100000000,true) }; /** * 返回中文數字漢字所對應的阿拉伯數字,若str不為中文數字,則返回-1 * @param str * @return */ public static int ChnNumToValue(String str){ for(int i=0;i<CHN_NUMBER.length;i++){ if(str.equals(CHN_NUMBER[i])){ return i; } } return -1; } /** * 返回中文漢字權位在chnNameValue陣列中所對應的索引號,若不為中文漢字權位,則返回-1 * @param str * @return */ public static int ChnUnitToValue(String str){ for(int i=0;i<chnNameValue.length;i++){ if(str.equals(chnNameValue[i].name)){ return i; } } return -1; } /** * 返回中文數字字串所對應的int型別的阿拉伯數字 * @param str * @return */ public static int ChnStringToNumber(String str){ int returnNumber=0; int section=0; int pos=0; int number=0; while (pos<str.length()){ int num=ChnNumToValue(str.substring(pos,pos+1)); //若num>=0,代表該位置(pos),所對應的是數字不是權位。若小於0,則表示為權位 if(num>=0){ number=num; pos++; //pos是最好一位,直接將number加入到section中。 if(pos>=str.length()){ section+=number; returnNumber+=section; break; } }else{ int chnNameValueIndex=ChnUnitToValue(str.substring(pos,pos+1)); //chnNameValue[chnNameValueIndex].secUnit==true,表示該位置所對應的權位是節權位, if(chnNameValue[chnNameValueIndex].secUnit){ section=(section+number)*chnNameValue[chnNameValueIndex].value; returnNumber+=section; section=0; }else{ section+=number*chnNameValue[chnNameValueIndex].value; } pos++; number=0; if(pos>=str.length()){ returnNumber+=section; break; } } } return returnNumber; } /** * 完成對中文數字字串轉換成阿拉伯數字函式的測試 */ public static void TestChnStringToNumber(){ for(int i=0;i<testData.length;i++){ int number=ChnStringToNumber(testData[i].chnNum); System.out.println(testData[i].chnNum+"\t"+number+"\t"+testData[i].number+"\t"+(number==testData[i].number)); } } public static void main(String[] args) { TestNumToChn(); System.out.println("--------------------------------------------------------"); TestChnStringToNumber(); } }