Java常用数字工具类 数字转汉字(1)
程序员文章站
2024-02-22 11:34:05
本人是从事互联网金融行业的,所以会接触到一些金融类的问题,常见的一种就是数字转汉字大小写的问题。所以抽空就写了一个小小的工具类,实现了数字转汉字、大数相加、相减、相乘的工具...
本人是从事互联网金融行业的,所以会接触到一些金融类的问题,常见的一种就是数字转汉字大小写的问题。所以抽空就写了一个小小的工具类,实现了数字转汉字、大数相加、相减、相乘的工具类,希望能帮助有需求的同行们。本篇就分享一下数字转化为汉字的思路吧。
数字转汉字的原理:
- 拆分:由于整数部分要加权值,而小数部分直接转换即可,所以首先要将数字拆分成整数+小数;
- 整数处理:按照我们的中国人的习惯,把数字格式化成4位一组,不足4位前面补0。每次处理4位,按位匹配数组中的汉字+权值。即按照数值找数字数组(num_lower 、num_upper )中对应位置的汉字,按照在4位中的偏移量在单位权值数组(unit_lower 、unit_upper )中找。比如21,转化4位为0021,前面的0不用管,2对应数字“二”,权值是“十”,1对应数字“一”,权值是“(个)”用空字符串代替。即得到“二十一”。每4位处理完后,还要整体对应一个权值,比如“万、亿、兆”等;
- 小数处理:小数部分直接按位对应汉字数组和权值即可。
废话了这么多,可能云里雾里的,看看具体代码吧:
//num 表示数字,lower表示小写,upper表示大写 private static final string[] num_lower = { "零", "一", "二", "三", "四", "五", "六", "七", "八", "九" }; private static final string[] num_upper = { "零", "壹", "贰", "叁", "肆", "伍", "陆", "柒", "捌", "玖" }; //unit 表示单位权值,lower表示小写,upper表示大写 private static final string[] unit_lower = { "", "十", "百", "千" }; private static final string[] unit_upper = { "", "拾", "佰", "仟"}; private static final string[] unit_common = {"","万", "亿","兆","京","垓","秭","穰","沟","涧","正","载"}; //允许的格式 private static final list<string> promisstypes = arrays.aslist("integer","int","long","decimal","float","double","string","byte","type","short"); /** * 数字转化为小写的汉字 * * @param num 将要转化的数字 * @return */ public static string tochineselower(object num){ return format(num, num_lower, unit_lower); } /** * 数字转化为大写的汉字 * * @param num 将要转化的数字 * @return */ public static string tochineseupper(object num){ return format(num, num_upper, unit_upper); } /** * 格式化数字 * * @param num 原数字 * @param numarray 数字大小写数组 * @param unit 单位权值 * @return */ private static string format(object num,string[] numarray,string[] unit){ if(!promisstypes.contains(num.getclass().getsimplename().touppercase())){ throw new runtimeexception("不支持的格式类型"); } //获取整数部分 string intnum = getint(string.valueof(num)); //获取小数部分 string decimal = getfraction(string.valueof(num)); //格式化整数部分 string result = formatintpart(intnum,numarray,unit); if(!"".equals(decimal)){//小数部分不为空 //格式化小数 result += "点"+formatfractionalpart(decimal, numarray); } return result; } /** * 格式化整数部分 * * @param num 整数部分 * @param numarray 数字大小写数组 * @return */ private static string formatintpart(string num,string[] numarray,string[] unit){ //按4位分割成不同的组(不足四位的前面补0) integer[] intnums = split2intarray(num); boolean zero = false; stringbuffer sb = new stringbuffer(); for(int i=0;i<intnums.length;i++){ //格式化当前4位 string r = formatint(intnums[i], numarray,unit); if("".equals(r)){// if((i+1)==intnums.length){ sb.append(numarray[0]);//结果中追加“零” }else{ zero=true; } }else{//当前4位格式化结果不为空(即不为0) if(zero || (i>0 && intnums[i]<1000)){//如果前4位为0,当前4位不为0 sb.append(numarray[0]);//结果中追加“零” } sb.append(r); sb.append(unit_common[intnums.length-1-i]);//在结果中添加权值 zero=false; } } return sb.tostring(); } /** * 格式化小数部分 * * @param decimal 小数部分 * @param numarray 数字大小写数组 * @return */ private static string formatfractionalpart(string decimal,string[] numarray) { char[] val = string.valueof(decimal).tochararray(); int len = val.length; stringbuilder sb = new stringbuilder(); for (int i = 0; i < len; i++) { int n = integer.valueof(val[i] + ""); sb.append(numarray[n]); } return sb.tostring(); }
拆分整数和小数的方法在这里:
/** * 获取整数部分 * * @param num * @return */ private static string getint(string num){ //检查格式 checknum(num); char[] val = string.valueof(num).tochararray(); stringbuffer sb = new stringbuffer(); int t , s = 0; for (int i = 0; i < val.length; i++) { if(val[i]=='.') { break; } t = integer.parseint(val[i]+"",16); if(s+t==0){ continue; } sb.append(t); s+=t; } return (sb.length()==0? "0":sb.tostring()); } /** * 获取小数部分 * * @param num * @return */ private static string getfraction(string num){ int i = num.lastindexof("."); if(num.indexof(".") != i){ throw new runtimeexception("数字格式不正确,最多只能有一位小数点!"); } string fraction =""; if(i>=0){ fraction = getint(new stringbuffer(num).reverse().tostring()); if(fraction.equals("0")){ return ""; } } return new stringbuffer(fraction).reverse().tostring(); } /** * 检查数字格式 * * @param num */ private static void checknum(string num) { if(num.indexof(".") != num.lastindexof(".")){ throw new runtimeexception("数字["+num+"]格式不正确!"); } if(num.indexof("-") != num.lastindexof("-") || num.lastindexof("-")>0){ throw new runtimeexception("数字["+num+"]格式不正确!"); } if(num.indexof("+") != num.lastindexof("+")){ throw new runtimeexception("数字["+num+"]格式不正确!"); } if(num.indexof("+") != num.lastindexof("+")){ throw new runtimeexception("数字["+num+"]格式不正确!"); } if(num.replaceall("[\\d|\\.|\\-|\\+]", "").length()>0){ throw new runtimeexception("数字["+num+"]格式不正确!"); } }
通过这种分而治之的思路,处理起来就简单多了。写个main函数调用一下:
public static void main(string[] args) { short s = 10; byte b=10; char c='a'; object[] nums = {s, b, c, 0, 1001, 100100001l, 21., 205.23f, 205.23d, "01000010", "1000000100105.0123", ".142", "20.00", "1..2", true}; system.out.println("将任意数字转化为汉字(包括整数、小数以及各种类型的数字)"); system.out.println("--------------------------------------------"); for(object num :nums){ try{ system.out.print("["+num.getclass().getsimplename()+"]"+num); for(int i=0;i<25-string.valueof(num+num.getclass().getsimplename()).length();i+=4){ system.out.print("\t"); } //调用转化为小写和大写 system.out.print(" format:"+tochineselower(num)); system.out.println("【"+tochineseupper(num)+"】"); }catch(exception e){ system.out.println(" 错误信息:"+e.getmessage()); } } }
看看结果吧:
从上述代码和运行结果中,我们可以看到该功能支持多种数据类型的转换、支持转化为一般汉字和财务专用大写汉字。还可以智能处理非正常逻辑的数字。比如“20”会转化为“二十”而非“二十零”;“1 0000 0001” 转换成“一亿零一”而非“一亿零万零一”。
这里只分享了一个转换汉字的功能,下篇将分享一下大数相乘、相加、相减的方法。支持小数和负数的运算,敬请期待。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。