数字的英文表达和中文表达

2019-04-17  本文已影响0人  chengcongyue

题目

给定一个32位的整数num,写两个函数分别返回它的英文表达和中文表达字符串.
然后我们来看看几个例子.


图片.png

笔者已经将最复杂的数字进行了分析,这个值是Integer.MIN_VALUE,如图Two Billion,One Hundred Forty Seven Million,.......Thousand,Six Hundred Forty Eight.我们注意到其中基本都是1---999的数字然后在加上Million,Thousand这些单位,这样的话,我们的首要目的就是找出1---999的数字表达.

1---19的英文表达

我们可以通过枚举的方式,将所有19个可能性放入到一个String数组中去,十分简单,如下图:

public static String num1To19(int num)
    {
        if(num<1||num>19)
        {
            return "";
        }
        String[] names={"One ","Two ","Three ","Four ","Five ","Six ","Seven ","Eight ","Nine ",
        "Ten ","Eleven ","Twelve ","Thirteen ","Fourteen ","Fifteen ","Sixteen ","Seventeen ",
        "Eightteen ","Nineteen "
        };
        return names[num-1];
    }

当我们传入16是,就找出这个数组中16-1,下标为15的表达.十分简单,现在完成了1---19的表达,然后我们来完成1---99的表达,我们通过/运算得到十位,用一个数组来承担,然后通过%运算得到低位,然后使用num1to19来表达

public static String num1To99(int num)
    {
        if(num<1||num>99)
        {
            return "";
        }
        if(num<20)
        {
            return num1To19(num);
        }
        int high=num/10;
        String[] tyNames={"Twenty ","Thirty ","Forty ","Fifty ","Sixty ","Seventy ","Eighty "
                ,"Ninety "};
        return tyNames[high-2]+num1To19(num%10);
    }

依次类推,我们现在知道了1---99的表达,然后我们通过/和%运算就可以得到1---999的表达了,但是要注意,在中间加上,"Hundred".

//1---999
    public static String num1To999(int num)
    {
        if(num<1||num>999)
        {
            return "";
        }
        if(num<100)
        {
            return num1To99(num);
        }
        int high=num/100;
        return num1To19(high)+"hundred "+num1To99(num%100);
    }

然后就是主函数了.
我们先把代码贴上来,然后再去分析为什么这样写

public static String getNumEngExp(int num)
    {
        if(num==0)
        {
            return "Zero";
        }
        String res="";
        if(num<0)
        {
            res="Negative, ";
        }
        if(num==Integer.MIN_VALUE)
        {
            res+="Two Billion,";
            num%=-2000000000;
        }
        num=Math.abs(num);//求绝对值
        int high=1000000000;
        int highIndex=0;
        String[] names={"Billion","Million","Thousand",""};
        while(num!=0)
        {
            int cur=num/high;
            System.out.println(cur);
            num%=high;
            if(cur!=0)
            {
                res+=num1To999(cur);
                res+=names[highIndex]+(num==0?" ":", ");
            }
            high/=1000;
            highIndex++;
        }
        return res;
    }
        System.out.println(Integer.MAX_VALUE);
        System.out.println(Integer.MIN_VALUE);

就是简单了两句代码,我们知道MAX_VALUE指的是32位有符号正整数的最大位置,如果这个位置加1的话就会溢出,结果运行结果如下:


图片.png

所以这就是我们要对Integer.MIN_VALUE进行单独处理的原因.
先得到他的最高位,然后取模,得到他除了最高位的数,然后进行循环.

while(num!=0)
        {
            int cur=num/high;//1000000000
            System.out.println(cur);
            num%=high;
            if(cur!=0)
            {
                res+=num1To999(cur);
                res+=names[highIndex]+(num==0?" ":", ");
            }
            high/=1000;
            highIndex++;
        }
        return res;

我们以Integer.MAX_VALUE为例,第一次遍历得到的cur为2,然后得到它的次高位等待.每次都要选出三位来,如果我们每次要选出一位出来,那就/10,如果一次要选出两位出来那就/100,所以这里我们要/1000,每次/1000,就可以选出3位,同时最开始的high的设值,0的个数一定是3的倍数,这道题就是9个零.
同时我们要注意/和%的协作运算.加深印象,我们在这道题中,在重新写一下

num=Math.abs(num);
int high=1000000000;
while(num!=0)
{
      cur=num/high;
      num=num%high;
      if(cur!=0)
     {
          res+=num1To999(cur);
          加上位数 
     } 
     high/=1000;
     highIndex++;
}

然后就是中文的表达

图片.png

我们注意到中文时4位为一次重复,然后修改其后面的单位即可.
直接贴代码了

public static String num1To9(int num)
    {
        if(num<1||num>9)
        {
            return "";
        }
        String[] names={"一","二","三","四","五","六","七","八","九"};
        return names[num-1];
    }
    
    public static String num1To99(int num,boolean hasBai)
    {
        if(num<1||num>99)
        {
            return "";
        }
        if(num<10)
        {
            return num1To9(num);
        }
        //得到高位
        int shi=num/10;
        if(shi==1&&(!hasBai))//如果有百位的情况  三百一十九
        {
            return "十"+num1To9(num%10);
        }else
        {
            return num1To9(shi)+"十"+num1To9(num%10);
        }
    }
    
    public static String num1To999(int num)
    {
        if(num<1||num>999)
        {
            return "";
        }
        if(num<100)
        {
            return num1To99(num,false);
        }
        String res=num1To9(num/100)+"百";
        int rest=num%100;
        if(rest==0)
        {
            return res;
        }else if(rest>=10)
        {
            res+=num1To99(rest,true);
        }else
        {
            res+="零"+num1To9(rest);
        }
        return res;
    }
    
    public static String num1To9999(int num)
    {
        if(num<1||num>9999)
        {
            return "";
        }
        if(num<1000)
        {
            return num1To999(num);
        }
        String res=num1To9(num/1000)+"千";
        int rest=num%1000;
        if(rest==0)
        {
            return res;
        }else if(rest>=100)
        {
            res+=num1To999(rest);
        }else
        {
            res+="零"+num1To99(rest, false);
        }
        return res;
    }
    
    public static String num1To99999999(int num)
    {
        if(num<1||num>99999999)
        {
            return "";
        }
        int wan=num/10000;
        int rest=num%10000;
        if(wan==0)
        {
            return num1To9999(rest);
        }
        String res=num1To9999(wan)+"万";
        if(rest==0)
        {
            return res;
        }else
        {
            if(rest<1000)
            {
                return res+"零"+num1To999(rest);
            }else
            {
                return res+num1To9999(rest);
            }
        }
    }
    public static String getNumChiExp(int num)
    {
        if(num==0)
        {
            return "零";
        }
        String res=num<0?"负":"";
        int yi=Math.abs(num/100000000);
        int rest=Math.abs(num%100000000);
        if(yi==0)
        {
            return res+num1To99999999(rest);
        }
        res+=num1To9999(yi)+"亿";
        if(rest==0)
        {
            return res;
        }else
        {
            if (rest < 10000000) {
                return res + "零" + num1To99999999(rest);
            } else {
                return res + num1To99999999(rest);
            }
        }
    }
    public static String getNumChiExp2(int num)
    {
        if(num==0)
        {
            return "零";
        }
        String res=num<0?"负":"";
        int high=100000000;
        int highIndex=0;
        if(num==Integer.MIN_VALUE)
        {
            res+=num1To9999(num/high*-1)+"亿";
            num=num%high;
            highIndex++;
        }
        num=Math.abs(num);
        String[] names={"亿","万",""};
        while(num!=0)
        {
            int cur=num/high;
            num=num%high;
            if(cur!=0)
            {
                res+=num1To9999(cur)+names[highIndex];
            }
            high/=10000;
            highIndex++;
        }
        return res;
    }
    public static void main(String[] args) {
        System.out.println(getNumChiExp2(323232));
        System.out.println(getNumChiExp(323232));
    }

最后还有几句想说的,关于/和%的配合使用问题,如果想获得当前的位,那么就是用一个10000.....(总位数等于被除数),这是做/运算,如果是做%运算,上面的情况就是获取除当前位的其他位数.
然后就是Integer.MAX_VALUE和Integer.MIN_VALUE,当处理比较大的数据时,就要考虑到溢出的情况,同时MAX_VALUE比MIN_VALUE的绝对值小1.

上一篇 下一篇

猜你喜欢

热点阅读