人民幣大寫轉換
問題:????商業領域,開發票,支票等金錢相關項目通常要求轉換成大寫,諸如“壹拾圓陸角玖分”.編程過程中,一般的錢幣都是double型.
當前看到的方法:之所以寫這個方法,因為看到的方法設計的不是很好,通常的做法就是逐個字符處理,通過冗長的switch...case結構判斷,來決定當前是否應該輸出某某漢字,總體來說,完成這件任務是簡單的,但是代碼和設計質量卻不令人滿意.
設計方法如下:
需要注意的問題:
????1.各個阿拉伯數字可以通過一個數組'壹','貳','叁'....表示.
????2.對于大于10000和大于100000000的數字,可能出現'萬','億'字樣
????3.對于中間連續為0的數字,正確出現'零'的字樣,但是有幾種不同的情況需要處理
????4.對于某個段的數字全零的情況,例如,整個萬段都是0的情況-100000101,中間的0如何出現
????5.角分的處理,如果不存在角分的話,應該出現'圓整'的字樣
????6.整數部分不存在的情況,即只有角分,應該沒有'圓'的字樣
設計框架:
????1.把數字轉化成字符串處理,使用Java的時候,把一個double類型轉化成一個字符串類型很簡單,
????調用????String.valueOf(double_var)即可得到,但是有一個問題,當你的數字大于10個位的時候,
????也就是達億的時候,他會轉換成科學計數法的字串,解決方法就是把他轉化成整形long.
????2.把數字分割成整數部分和小數部分分別處理,根據上面的方法,我們索性把double乘上100,
????取后兩位為小數部分,前面的為整數部分,得到
????long l = (long)(d*100);
????String strVal = String.valueOf(l);
????String head = strVal.substring(0,strVal.length()-2);???????? //整數部分
????String end = strVal.substring(strVal.length()-2);????????????? //小數部分
????
????3.我們應該把錢數分成段,每四個一段,實際上得到的是一個二維數組,如下:
????????????仟????????佰????????拾???? ????'?'
????' '????? $4????????$3????????$2?????????$1
????萬???? $8??? ????$7?? ????$6????? ????$5
????億???? $12? ????$11 ????$10??? ????$9
????
????其中$i表示這個數字的第i個位置的數字,我們并不實際設定二維數組,我們得到的是數字的位置,
????要處理的該產生什么樣的表示法,很簡單這種處理方式往往就是:設pos表示數字位置,pos/4 在那一個段
????萬以下段,萬段,億段.pos%4表示某一個段的段內位置,仟,佰,拾,由于疊加的緣故,即會有千萬,百萬,千億等
????出現,因此這種設計是成立的.這里面隱含了一個問題就是,我們當前的處理的最大數字達千億位,
????更大的數字用這種結構是不妥的,因為可能會有萬億,這時候推薦的想法是把這些設計成單維的數組結構,
????從而取得疊加的表示.
????4.循環處理各個位的過程中,我們可以預想到,零的問題是最難解決的.
????因為我們多個連續的零你只能出現一個表示,更有甚者,當某段全為0時,'零'還不能出現.
????因此這些問題綜合考慮得到以下代碼.
代碼:(JAVA描述)
???????? char []?hunit = { ' 拾 ' , ' 佰 ' , ' 仟 ' };?????????????????????????????????? // 段內位置表示
???????? char []?vunit = { ' 萬 ' , ' 億 ' };??????????????????????????????????????? // 段名表示
???????? char []?digit = { ' 零 ' , ' 壹 ' , ' 貳 ' , ' 叁 ' , ' 肆 ' , ' 伍 ' , ' 陸 ' , ' 柒 ' , ' 捌 ' , ' 玖 ' };?? // 數字表示
???????? long ?midVal? = ?( long )(value * 100 );??????????????????????????????? // 轉化成整形
????????String?valStr = String.valueOf(midVal);?????????????????????????? // 轉化成字符串
????????String?head = valStr.substring( 0 ,valStr.length() - 2 );????????????? // 取整數部分
????????String?rail = valStr.substring(valStr.length() - 2 );??????????????? // 取小數部分
????????String?prefix = "" ;?????????????????????????????????????????????? // 整數部分轉化的結果
????????String?suffix = "" ;?????????????????????????????????????????????? // 小數部分轉化的結果
???????? // 處理小數點后面的數
???????? if (rail.equals( " 00 " )){????????????????????????????????????????? // 如果小數部分為0
??????????suffix = " 整 " ;
????????}? else {
??????????suffix = digit[rail.charAt( 0 ) - ' 0 ' ] + " 角 " + digit[rail.charAt( 1 ) - ' 0 ' ] + " 分 " ;? // 否則把角分轉化出來
????????}
???????? // 處理小數點前面的數
???????? char []?chDig = head.toCharArray();????????????????? // 把整數部分轉化成字符數組
???????? char ?zero = ' 0 ' ;??????????????????????????????????? // 標志'0'表示出現過0
???????? byte ?zeroSerNum? = ? 0 ;????????????????????????????? // 連續出現0的次數
???????? for ( int ?i = 0 ;i < chDig.length;i ++ ){????????????????? // 循環處理每個數字
?????????? int ?idx = (chDig.length - i - 1 ) % 4 ;?????????????????? // 取段內位置
?????????? int ?vidx = (chDig.length - i - 1 ) / 4 ;????????????????? // 取段位置
?????????? if (chDig[i] == ' 0 ' ){????????????????????????????? // 如果當前字符是0
????????????zeroSerNum ++ ;???????????????????????????????? // 連續0次數遞增
???????????? if (zero? == ? ' 0 ' ){?????????????????????????????? // 標志
??????????????zero = digit[ 0 ];
????????????}? else ? if (idx == 0 ? && ?vidx? > 0 ? && zeroSerNum? < ? 4 ){
??????????????prefix? += ?vunit[vidx - 1 ];
??????????????zero = ' 0 ' ;
????????????}
???????????? continue ;
??????????}
??????????zeroSerNum? = ? 0 ;???????????????????????????????? // 連續0次數清零
?????????? if (zero? != ? ' 0 ' )?{?????????????????????????????? // 如果標志不為0,則加上,例如萬,億什么的
????????????prefix += zero;
????????????zero = ' 0 ' ;
??????????}
??????????prefix += digit[chDig[i] - ' 0 ' ];?????????????????? // 轉化該數字表示
?????????? if (idx? > ? 0 )?prefix? += ?hunit[idx - 1 ];??????????????????
?????????? if (idx == 0 ? && ?vidx > 0 ){
????????????prefix += vunit[vidx - 1 ];????????????????????? // 段結束位置應該加上段名如萬,億
??????????}
????????}
???????? if (prefix.length()? > ? 0 )?prefix? += ? ' 圓 ' ;???????? // 如果整數部分存在,則有圓的字樣
???????? return ?prefix + suffix;????????????????????????? // 返回正確表示
??????}