最近由于项目的原因,需要写一个货币数字转换中文的算法,先在网了找了一下,结果发现无一列外都是用(Replace)替换的方式来实现的,所以想写个另外的算法;因为本人是学数学出身的,所以用纯数学的方法实现。
注意:本文中的算法支持小于1023 (也就是9999亿兆)货币数字转化。
货币中文说明: 在说明代码之前,首先让我们回顾一下货币的读法。
10020002.23 读为 壹仟零贰万零贰元贰角叁分
1020 读为 壹仟零贰拾元整。
100000 读为 拾万元整
0.13 读为 壹角叁分
代码:
测试工程 static void Main(string[] args) { Console.WriteLine("请输入金额"); //货币数字转化类
Console.WriteLine("\n请输入金额"); } |
#p#副标题#e#
测试结果如下:
货币转化类(NumCast类)功能介绍
1 常量的规定 /// <summary> /// 数位 /// </summary> public enum NumLevel { Cent, Chiao, Yuan, Ten, Hundred, Thousand, TenThousand, hundredMillon, Trillion }; /// <summary> /// <summary> /// <summary>
private const string EndOfInt = "整"; |
2:数字合法性验证,采用正则表达式验证 /// <summary> /// 正则表达验证数字是否合法 /// </summary> /// <param name="Num"></param> /// <returns></returns> public bool IsValidated<T>(T Num) { Regex reg = new Regex(@"^(([0])|([1-9]\d{0,23}))(\.\d{1,2})?$"); if (reg.IsMatch(Num.ToString())) { return true; } |
3: 获取数位 例如 1000的数位为 NumLevel.Thousand
/// <summary> } |
4:判断数字之间是否有跳位,也就是中文中间是否要加零,例如1020 就应该加零。 /// <summary> /// 是否跳位 /// </summary> /// <returns></returns> private bool IsDumpLevel(double Num) { if (Num > 0) { NumLevel? currentLevel = GetNumLevel(Num); NumLevel? nextLevel = null; int numExponent = this.NumLevelExponent[(int)currentLevel]; double postfixNun = Math.Round(Num % (Math.Pow(10, numExponent)),2); } |
5 把长数字分割为两个较小的数字数组,例如把9999亿兆,分割为9999亿和0兆,
因为计算机不支持过长的数字。 /// <summary> |
#p#副标题#e#
6 是否以“壹拾”开头,如果是就可以把它变为“拾” bool isStartOfTen = false; while (Num >=10) { if (Num == 10) { isStartOfTen = true; break; } //Num的数位 NumLevel currentLevel = GetNumLevel(Num); int numExponent = this.NumLevelExponent[(int)currentLevel]; Num = Convert.ToInt32(Math.Floor(Num / Math.Pow(10, numExponent))); if (currentLevel == NumLevel.Ten && Num == 1) { isStartOfTen = true; break; } } return isStartOfTen; |
7 合并大于兆连个数组转化成的货币字符串
/// <summary> } |
8:递归计算货币数字的中文 /// <summary> /// 计算中文字符串 /// </summary> /// <param name="Num">数字</param> /// <param name="NL">数位级别 比如1000万的 数位级别为万</param> /// <param name="IsExceptTen">是否以‘壹拾’开头</param> /// <returns>中文大写</returns> public string CalculateChineseSign(double Num, NumLevel? NL ,bool IsDump,bool IsExceptTen) { Num = Math.Round(Num, 2); bool isDump = false; //Num的数位 NumLevel? currentLevel = GetNumLevel(Num); int numExponent = this.NumLevelExponent[(int)currentLevel]; string Result = string.Empty; //整除后的结果 if (prefixNum < 10 ) }
Result += CalculateChineseSign((double)prefixNum, NextNL, isDump, IsExceptTen); //是否跳位 } //余数是否需要递归 return Result; |
#p#副标题#e#
9:外部调用的转换方法。
/// <summary> if (!IsValidated<string>(Num)) |
小结:
个人认为程序的灵魂是算法,大到一个系统中的业务逻辑,小到一个货币数字转中文的算法,处处都体现一种逻辑思想。
是否能把需求抽象成一个好的数学模型,直接关系到程序的实现的复杂度和稳定性。在一些常用功能中想些不一样的算法,对我们开拓思路很有帮助。