java简单题:一组金额数据,用人民币大写显示出来

如12.5输出一十二元五角整,应该是大写壹…

/**
 * 程序目的:
 *   从命令行接收一个数,并将其转化为中文金额的大写方式
 * 例如 123.45 --> 壹佰贰拾叁元肆角伍分
 * @author gdb
 *
 * 实在是不符合规范,程序里面的算法没有让人明白得很清楚的注释,读上去觉得有
 * 点难度,可读性不强。而且很多程序还存在不少bug,随便一测就测出来了。
 *   所以本人还是决定重新写一下这个程序,并且尽量做到消除不必要的bug。这个程
 * 序我没有用什么很精妙的算法,不过用了一些Java类库中的类,像是正则表达式之类
 * 
 */
public class Trans2RMB {

/**
 * 测试程序的可行性
 * @param args
 */
public static void main(String[] args) {
System.out.println("\n--------将数字转换成中文金额的大写形式------------\n");
Trans2RMB t2r = new Trans2RMB();
String s = t2r.cleanZero(t2r.splitNum(t2r.roundString(t2r.getNum())));
// 如果转换过后是一个空串,则不输出屏幕
if(!"".equals(s)) {
System.out.println("转换成中文后为:" + s);;
}
System.out.println("\n---------------------------------------------");
}

/**
 * 从命令行接收一个数,在其中调用 checkNum() 方法对其进行
 * 验证,并返回相应的值
 * @return 如果输入合法,返回输入的这个数
 */
private String getNum() {
String s = null;
System.out.println("请输入一个数字(精确到小数点后两位):");
// 从命令行输入这个浮点数
java.util.Scanner scanner = new java.util.Scanner(System.in);
s = scanner.next();
// 关闭这个Scanner
scanner.close();
// 判断用户输入是否合法
// 若合法,返回这个值;若非法返回 "0"
if(this.checkNum(s)) {
return s;
} else {
return "";
}
}

/**
 * 判断用户输入的数据是否合法,用户只能输入大于零的数字,不能输入其它字符
 * @param s String
 * @return 如果用户输入数据合法,返回 true,否则返回 false
 */
private boolean checkNum(String s) {
// 如果用户输入的数里有非数字字符,则视为非法数据,返回 false
try {
float f = Float.valueOf(s);
// 如果这个数小于零则视为非法数据,返回 false
if(f < 0) {
System.out.println("非法数据,请检查!");
return false;
}else {
return true;
}
} catch (NumberFormatException e) {
System.out.println("非法数据,请检查!");
return false;
}
}

/**
 * 把用户输入的数以小数点为界分割开来,并调用 numFormat() 方法
 * 进行相应的中文金额大写形式的转换
 * 注:传入的这个数应该是经过 roundString() 方法进行了四舍五入操作的
 * @param s String
 * @return 转换好的中文金额大写形式的字符串
 */
private String splitNum(String s) {
// 如果传入的是空串则继续返回空串
if("".equals(s)) {
return "";
}
// 以小数点为界分割这个字符串
int index = s.indexOf(".");
// 截取并转换这个数的整数部分
String intOnly = s.substring(0, index);
String part1 = this.numFormat(1, intOnly);
// 截取并转换这个数的小数部分
String smallOnly = s.substring(index + 1);
String part2 = this.numFormat(2, smallOnly);
// 把转换好了的整数部分和小数部分重新拼凑一个新的字符串
String newS = part1 + part2;
return newS;
}

/**
 * 对传入的数进行四舍五入操作
 * @param s String 从命令行输入的那个数
 * @return 四舍五入后的新值
 */
private String roundString(String s) {
// 如果传入的是空串则继续返回空串
if("".equals(s)) {
return "";
}
// 将这个数转换成 double 类型,并对其进行四舍五入操作
double d = Double.parseDouble(s);
// 此操作作用在小数点后两位上
d = (d * 100 + 0.5) / 100;
// 将 d 进行格式化
s = new java.text.DecimalFormat("##0.000").format(d);
// 以小数点为界分割这个字符串
int index = s.indexOf(".");
// 这个数的整数部分
String intOnly = s.substring(0, index);
// 规定数值的最大长度只能到万亿单位,否则返回 "0"
if(intOnly.length() > 13) {
System.out.println("输入数据过大!(整数部分最多13位!)");
return "";
}
// 这个数的小数部分
String smallOnly = s.substring(index + 1);
// 如果小数部分大于两位,只截取小数点后两位
if(smallOnly.length() > 2) {
String roundSmall = smallOnly.substring(0, 2);
// 把整数部分和新截取的小数部分重新拼凑这个字符串
s = intOnly + "." + roundSmall;
}
return s;
}

/**
 * 把传入的数转换为中文金额大写形式
 * @param flag int 标志位,1 表示转换整数部分,0 表示转换小数部分
 * @param s String 要转换的字符串
 * @return 转换好的带单位的中文金额大写形式
 */
private String numFormat(int flag, String s) {
int sLength = s.length();
// 货币大写形式
String bigLetter[] = {"零", "壹", "贰", "叁", "肆", "伍", "陆", "柒", "捌", "玖"};
// 货币单位
String unit[] = {"元", "拾", "佰", "仟", "万", 
// 拾万位到仟万位
"拾", "佰", "仟",
// 亿位到万亿位
"亿", "拾", "佰", "仟", "万"};
String small[] = {"分", "角"};
// 用来存放转换后的新字符串
String newS = "";
// 逐位替换为中文大写形式
for(int i = 0; i < sLength; i ++) {
if(flag == 1) {
// 转换整数部分为中文大写形式(带单位)
newS = newS + bigLetter[s.charAt(i) - 48] + unit[sLength - i - 1];
} else if(flag == 2) {
// 转换小数部分(带单位)
newS = newS + bigLetter[s.charAt(i) - 48] + small[sLength - i - 1];
}
}
return newS;
}

/**
 * 把已经转换好的中文金额大写形式加以改进,清理这个字
 * 符串里面多余的零,让这个字符串变得更加可观
 * 注:传入的这个数应该是经过 splitNum() 方法进行处理,这个字
 * 符串应该已经是用中文金额大写形式表示的
 * @param s String 已经转换好的字符串
 * @return 改进后的字符串
 */
private String cleanZero(String s) {
// 如果传入的是空串则继续返回空串
if("".equals(s)) {
return "";
}
// 如果用户开始输入了很多 0 去掉字符串前面多余的'零',使其看上去更符合习惯
while(s.charAt(0) == '零') {
// 将字符串中的 "零" 和它对应的单位去掉
s = s.substring(2);
// 如果用户当初输入的时候只输入了 0,则只返回一个 "零"
if(s.length() == 0) {
return "零";
}
}
// 字符串中存在多个'零'在一起的时候只读出一个'零',并省略多余的单位
/* 由于本人对算法的研究太菜了,只能用4个正则表达式去转换了,各位大虾别介意哈... */
String regex1[] = {"零仟", "零佰", "零拾"};
String regex2[] = {"零亿", "零万", "零元"};
String regex3[] = {"亿", "万", "元"};
String regex4[] = {"零角", "零分"};
// 第一轮转换把 "零仟", 零佰","零拾"等字符串替换成一个"零"
for(int i = 0; i < 3; i ++) {
s = s.replaceAll(regex1[i], "零");
}
// 第二轮转换考虑 "零亿","零万","零元"等情况
// "亿","万","元"这些单位有些情况是不能省的,需要保留下来
for(int i = 0; i < 3; i ++) {
// 当第一轮转换过后有可能有很多个零叠在一起
// 要把很多个重复的零变成一个零
s = s.replaceAll("零零零", "零");
s = s.replaceAll("零零", "零");
s = s.replaceAll(regex2[i], regex3[i]);
}
// 第三轮转换把"零角","零分"字符串省略
for(int i = 0; i < 2; i ++) {
s = s.replaceAll(regex4[i], "");
}
// 当"万"到"亿"之间全部是"零"的时候,忽略"亿万"单位,只保留一个"亿"
s = s.replaceAll("亿万", "亿");
return s;
}
}

温馨提示:答案为网友推荐,仅供参考
第1个回答  2009-03-04
只能自己写解析程序了,JDK没有提供这个功能。也可以搜下是否有开源类包可以用。
第2个回答  推荐于2017-10-13
01.import java.math.BigDecimal;
02.
03.public class ConvertNumber {
04. /** 定义数组存放数字对应的大写 */
05. private final static String[] STR_NUMBER = { "零", "壹", "贰", "叁", "肆", "伍", "陆", "柒", "捌", "玖" };
06.
07. /** 定义数组存放位数的大写 */
08. private final static String[] STR_MODIFY = { "", "拾", "佰", "仟", "万", "拾", "佰", "仟", "亿", "拾", "佰", "仟" };
09.
10. /**
11. * 转化整数部分
12. *
13. * @param tempString
14. * @return 返回整数部分
15. */
16. private static String getInteger(String tempString) {
17. /** 用来保存整数部分数字串 */
18. String strInteger = null;//
19. /** 记录"."所在位置 */
20. int intDotPos = tempString.indexOf(".");
21. int intSignPos = tempString.indexOf("-");
22. if (intDotPos == -1)
23. intDotPos = tempString.length();
24. /** 取出整数部分 */
25. strInteger = tempString.substring(intSignPos + 1, intDotPos);
26. strInteger = new StringBuffer(strInteger).reverse().toString();
27. StringBuffer sbResult = new StringBuffer();
28. for (int i = 0; i < strInteger.length(); i++) {
29. sbResult.append(STR_MODIFY[i]);
30. sbResult.append(STR_NUMBER[strInteger.charAt(i) - 48]);
31. }
32.
33. sbResult = sbResult.reverse();
34. replace(sbResult, "零拾", "零");
35. replace(sbResult, "零佰", "零");
36. replace(sbResult, "零仟", "零");
37. replace(sbResult, "零万", "万");
38. replace(sbResult, "零亿", "亿");
39. replace(sbResult, "零零", "零");
40. replace(sbResult, "零零零", "零");
41. /** 这两句不能颠倒顺序 */
42. replace(sbResult, "零零零零万", "");
43. replace(sbResult, "零零零零", "");
44. /** 这样读起来更习惯. */
45. replace(sbResult, "壹拾亿", "拾亿");
46. replace(sbResult, "壹拾万", "拾万");
47. /** 删除个位上的零 */
48. if (sbResult.charAt(sbResult.length() - 1) == '零' && sbResult.length() != 1)
49. sbResult.deleteCharAt(sbResult.length() - 1);
50. if (strInteger.length() == 2) {
51. replace(sbResult, "壹拾", "拾");
52. }
53. /** 将结果反转回来. */
54. return sbResult.toString();
55. }
56.
57. /**
58. * 转化小数部分 例:输入22.34返回叁肆
59. *
60. * @param tempString
61. * @return
62. */
63. private static String getFraction(String tempString) {
64. String strFraction = null;
65. int intDotPos = tempString.indexOf(".");
66. /** 没有点说明没有小数,直接返回 */
67. if (intDotPos == -1)
68. return "";
69. strFraction = tempString.substring(intDotPos + 1);
70. StringBuffer sbResult = new StringBuffer(strFraction.length());
71. for (int i = 0; i < strFraction.length(); i++) {
72. sbResult.append(STR_NUMBER[strFraction.charAt(i) - 48]);
73. }
74. return sbResult.toString();
75. }
76.
77. /**
78. * 判断传入的字符串中是否有.如果有则返回点
79. *
80. * @param tempString
81. * @return
82. */
83. private static String getDot(String tempString) {
84. return tempString.indexOf(".") != -1 ? "点" : "";
85. }
86.
87. /**
88. * 判断传入的字符串中是否有-如果有则返回负
89. *
90. * @param tempString
91. * @return
92. */
93. private static String getSign(String tempString) {
94. return tempString.indexOf("-") != -1 ? "负" : "";
95. }
96.
97. /**
98. * 将一个数字转化为金额
99. *
100. * @param tempNumber 传入一个double的变量
101. * @return 返一个转换好的字符串
102. */
103. public static String numberToChinese(double tempNumber) {
104. java.text.DecimalFormat df = new java.text.DecimalFormat("#.#########");
105. String pTemp = String.valueOf(df.format(tempNumber));
106. StringBuffer sbResult = new StringBuffer(getSign(pTemp) + getInteger(pTemp) + getDot(pTemp) + getFraction(pTemp));
107. return sbResult.toString();
108. }
109.
110. public static String numberToChinese(BigDecimal tempNumber) {
111. return numberToChinese(tempNumber.doubleValue());
112. }
113.
114. /**
115. * 替代字符
116. *
117. * @param pValue
118. * @param pSource
119. * @param pDest
120. */
121. private static void replace(StringBuffer pValue, String pSource, String pDest) {
122. if (pValue == null || pSource == null || pDest == null)
123. return;
124. /** 记录pSource在pValue中的位置 */
125. int intPos = 0;
126. do {
127. intPos = pValue.toString().indexOf(pSource);
128. /** 没有找到pSource */
129. if (intPos == -1)
130. break;
131. pValue.delete(intPos, intPos + pSource.length());
132. pValue.insert(intPos, pDest);
133. } while (true);
134. }
135.
136. /**
137. * @param args
138. */
139. public static void main(String[] args) {
140. System.out.println(numberToChinese(1230400567.8934));
141. }
142.}本回答被提问者采纳