浏览 16785 次
锁定老帖子 主题:将数字转换成中文金额大写形式(Java版)
精华帖 (0) :: 良好帖 (0) :: 新手帖 (0) :: 隐藏帖 (2)
|
|
---|---|
作者 | 正文 |
发表时间:2008-06-13
/** * 程序目的: * 从命令行接收一个数,并将其转化为中文金额的大写方式 * 例如 123.45 --> 壹佰贰拾叁元肆角伍分 * @author LovinChan * * 看到网上有很多这样的例子程序,不过觉得很不满意。有些程序我从命名上就觉得 * 实在是不符合规范,程序里面的算法没有让人明白得很清楚的注释,读上去觉得有 * 点难度,可读性不强。而且很多程序还存在不少bug,随便一测就测出来了。 * 所以本人还是决定重新写一下这个程序,并且尽量做到消除不必要的bug。这个程 * 序我没有用什么很精妙的算法,不过用了一些Java类库中的类,像是正则表达式之类 * 的东西。由于本人对算法不怎么在行,在做转换操作的时候用的是很笨的方法。望各位 * 大虾海涵,呵呵。 * * 程序的注释我尽量写得详细一点,如果觉得这个程序哪里有问题或者是哪里有改进的 * 地方欢迎随时跟我交流。 * 我的msn:egg.chenlw@gmail.com * QQ:543669516(注上为什么加我就可以了) * 我的blog:http://hi.baidu.com/egg_chen * 欢迎交流 */ 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; } } 声明:ITeye文章版权属于作者,受法律保护。没有作者书面许可不得转载。
推荐链接
|
|
返回顶楼 | |
发表时间:2009-01-24
你好,感谢你的代码,省了我很多时间.
根据我的需求,在你的基础上作了一些修改: 1.当数值为0时,大写为"零元".(而你原来的是"零") 在196行:return "零"; 改为 tag2=true;return "零元"; 2.当小数部分为零时,大写为"XXXXX元整".(而你原来的是"XXXXX元") 增加两个全局变量: private boolean tag=false;//无小数或小数为0标志 private boolean tag2=false; //数值为0标志 在139行加入: if(roundSmall.equals("00")) tag=true; 在33行加入: if(tag&&(tag2 ==false)){ chineseNum=chineseNum+"整"; } 请楼主帮我看一下这样改是否会出现重大的BUG,暂时试了几下没看到有 |
|
返回顶楼 | |
发表时间:2009-01-24
不好意思上面写错了:chineseNum=chineseNum+"整";
改为:s=s+"整"; |
|
返回顶楼 | |
发表时间:2009-02-02
太感谢了 很实用的程序!
|
|
返回顶楼 | |
发表时间:2009-02-02
改成我需要的了~~谢谢
|
|
返回顶楼 | |