C#基于純數(shù)學(xué)方法遞歸實(shí)現(xiàn)貨幣數(shù)字轉(zhuǎn)換中文功能詳解
本文實(shí)例講述了C#基于純數(shù)學(xué)方法遞歸實(shí)現(xiàn)貨幣數(shù)字轉(zhuǎn)換中文功能。分享給大家供大家參考,具體如下:
最近由于項(xiàng)目的原因,需要寫(xiě)一個(gè)貨幣數(shù)字轉(zhuǎn)換中文的算法,先在網(wǎng)了找了一下,結(jié)果發(fā)現(xiàn)無(wú)一列外都是用(Replace)替換的方式來(lái)實(shí)現(xiàn)的,所以想寫(xiě)個(gè)另外的算法;因?yàn)楸救耸菍W(xué)數(shù)學(xué)出身的,所以用純數(shù)學(xué)的方法實(shí)現(xiàn)。
注意:本文中的算法支持小于1023 (也就是9999億兆)貨幣數(shù)字轉(zhuǎn)化。
貨幣中文說(shuō)明: 在說(shuō)明代碼之前,首先讓我們回顧一下貨幣的讀法。
10020002.23 讀為 壹仟零貳萬(wàn)零貳元貳角叁分
1020 讀為 壹仟零貳拾元整。
100000 讀為 拾萬(wàn)元整
0.13 讀為 壹角叁分
代碼:
測(cè)試工程
static void Main(string[] args)
{
Console.WriteLine("請(qǐng)輸入金額");
string inputNum = Console.ReadLine();
while (inputNum != "exit")
{
//貨幣數(shù)字轉(zhuǎn)化類(lèi)
NumCast nc = new NumCast();
if (nc.IsValidated<string>(inputNum))
{
try
{
string chineseCharacter = nc.ConvertToChinese(inputNum);
Console.WriteLine(chineseCharacter);
}
catch (Exception er)
{
Console.WriteLine(er.Message);
}
}
else
{
Console.WriteLine("不合法的數(shù)字或格式");
}
Console.WriteLine("\n請(qǐng)輸入金額");
inputNum = Console.ReadLine();
}
Console.ReadLine();
}
貨幣轉(zhuǎn)化類(lèi)(NumCast類(lèi))功能介紹
1. 常量的規(guī)定
/// <summary>
/// 數(shù)位
/// </summary>
public enum NumLevel { Cent, Chiao, Yuan, Ten, Hundred, Thousand, TenThousand, hundredMillon, Trillion };
/// <summary>
/// 數(shù)位的指數(shù)
/// </summary>
private int[] NumLevelExponent = new int[] { -2, -1, 0, 1, 2, 3, 4, 8, 12 };
/// <summary>
/// 數(shù)位的中文字符
/// </summary>
private string[] NumLeverChineseSign = new string[] { "分", "角", "元", "拾", "佰", "仟", "萬(wàn)", "億", "兆" };
/// <summary>
/// 大寫(xiě)字符
/// </summary>
private string[] NumChineseCharacter = new string[] {"零","壹","貳","叁","肆","伍","陸","柒","捌","玖"};
/// <summary>
/// 整(當(dāng)沒(méi)有 角分 時(shí))
/// </summary>
private const string EndOfInt = "整";
2. 數(shù)字合法性驗(yàn)證,采用正則表達(dá)式驗(yàn)證
/// <summary>
/// 正則表達(dá)驗(yàn)證數(shù)字是否合法
/// </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;
}
return false;
}
3. 獲取數(shù)位 例如 1000的數(shù)位為 NumLevel.Thousand
/// <summary>
/// 獲取數(shù)字的數(shù)位使用log
/// </summary>
/// <param name="Num"></param>
/// <returns></returns>
private NumLevel GetNumLevel(double Num)
{
double numLevelLength;
NumLevel NLvl = new NumLevel();
if (Num > 0)
{
numLevelLength = Math.Floor(Math.Log10(Num));
for (int i = NumLevelExponent.Length - 1; i >= 0; i--)
{
if (numLevelLength >= NumLevelExponent[i])
{
NLvl = (NumLevel)i;
break;
}
}
}
else
{
NLvl = NumLevel.Yuan;
}
return NLvl;
}
4. 判斷數(shù)字之間是否有跳位,也就是中文中間是否要加零,例如1020 就應(yīng)該加零。
/// <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);
if(postfixNun> 0)
nextLevel = GetNumLevel(postfixNun);
if (currentLevel != null && nextLevel != null)
{
if (currentLevel > nextLevel + 1)
{
return true;
}
}
}
return false;
}
5. 把長(zhǎng)數(shù)字分割為兩個(gè)較小的數(shù)字?jǐn)?shù)組,例如把9999億兆,分割為9999億和0兆,因?yàn)橛?jì)算機(jī)不支持過(guò)長(zhǎng)的數(shù)字。
/// <summary>
/// 是否大于兆,如果大于就把字符串分為兩部分,
/// 一部分是兆以前的數(shù)字
/// 另一部分是兆以后的數(shù)字
/// </summary>
/// <param name="Num"></param>
/// <returns></returns>
private bool IsBigThanTillion(string Num)
{
bool isBig = false;
if (Num.IndexOf('.') != -1)
{
//如果大于兆
if (Num.IndexOf('.') > NumLevelExponent[(int)NumLevel.Trillion])
{
isBig = true;
}
}
else
{
//如果大于兆
if (Num.Length > NumLevelExponent[(int)NumLevel.Trillion])
{
isBig = true;
}
}
return isBig;
}
/// <summary>
/// 把數(shù)字字符串由‘兆'分開(kāi)兩個(gè)
/// </summary>
/// <returns></returns>
private double[] SplitNum(string Num)
{
//兆的開(kāi)始位
double[] TillionLevelNums = new double[2];
int trillionLevelLength;
if (Num.IndexOf('.') == -1)
trillionLevelLength = Num.Length - NumLevelExponent[(int)NumLevel.Trillion];
else
trillionLevelLength = Num.IndexOf('.') - NumLevelExponent[(int)NumLevel.Trillion];
//兆以上的數(shù)字
TillionLevelNums[0] = Convert.ToDouble(Num.Substring(0, trillionLevelLength));
//兆以下的數(shù)字
TillionLevelNums[1] = Convert.ToDouble(Num.Substring(trillionLevelLength ));
return TillionLevelNums;
}
6. 是否以“壹拾”開(kāi)頭,如果是就可以把它變?yōu)椤笆啊?/strong>
bool isStartOfTen = false;
while (Num >=10)
{
if (Num == 10)
{
isStartOfTen = true;
break;
}
//Num的數(shù)位
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. 合并大于兆連個(gè)數(shù)組轉(zhuǎn)化成的貨幣字符串
/// <summary>
/// 合并分開(kāi)的數(shù)組中文貨幣字符
/// </summary>
/// <param name="tillionNums"></param>
/// <returns></returns>
private string ContactNumChinese(double[] tillionNums)
{
string uptillionStr = CalculateChineseSign(tillionNums[0], NumLevel.Trillion, true, IsStartOfTen(tillionNums[0]));
string downtrillionStr = CalculateChineseSign(tillionNums[1], null, true,false);
string chineseCharactor = string.Empty;
//分開(kāi)后的字符是否有跳位
if (GetNumLevel(tillionNums[1] * 10) == NumLevel.Trillion)
{
chineseCharactor = uptillionStr + NumLeverChineseSign[(int)NumLevel.Trillion] + downtrillionStr;
}
else
{
chineseCharactor = uptillionStr + NumLeverChineseSign[(int)NumLevel.Trillion];
if (downtrillionStr != "零元整")
{
chineseCharactor += NumChineseCharacter[0] + downtrillionStr;
}
else
{
chineseCharactor += "元整";
}
}
return chineseCharactor;
}
8. 遞歸計(jì)算貨幣數(shù)字的中文
/// <summary>
/// 計(jì)算中文字符串
/// </summary>
/// <param name="Num">數(shù)字</param>
/// <param name="NL">數(shù)位級(jí)別 比如1000萬(wàn)的 數(shù)位級(jí)別為萬(wàn)</param>
/// <param name="IsExceptTen">是否以‘壹拾'開(kāi)頭</param>
/// <returns>中文大寫(xiě)</returns>
public string CalculateChineseSign(double Num, NumLevel? NL ,bool IsDump,bool IsExceptTen)
{
Num = Math.Round(Num, 2);
bool isDump = false;
//Num的數(shù)位
NumLevel? currentLevel = GetNumLevel(Num);
int numExponent = this.NumLevelExponent[(int)currentLevel];
string Result = string.Empty;
//整除后的結(jié)果
int prefixNum;
//余數(shù) 當(dāng)為小數(shù)的時(shí)候 分子分母各乘100
double postfixNun ;
if (Num >= 1)
{
prefixNum = Convert.ToInt32(Math.Floor(Num / Math.Pow(10, numExponent)));
postfixNun = Math.Round(Num % (Math.Pow(10, numExponent)), 2);
}
else
{
prefixNum = Convert.ToInt32(Math.Floor(Num*100 / Math.Pow(10, numExponent+2)));
postfixNun = Math.Round(Num * 100 % (Math.Pow(10, numExponent + 2)), 2);
postfixNun *= 0.01;
}
if (prefixNum < 10 )
{
//避免以‘壹拾'開(kāi)頭
if (!(NumChineseCharacter[(int)prefixNum] == NumChineseCharacter[1]
&& currentLevel == NumLevel.Ten && IsExceptTen))
{
Result += NumChineseCharacter[(int)prefixNum];
}
else
{
IsExceptTen = false;
}
//加上單位
if (currentLevel == NumLevel.Yuan )
{
////當(dāng)為 “元” 位不為零時(shí) 加“元”。
if (NL == null)
{
Result += NumLeverChineseSign[(int)currentLevel];
//當(dāng)小數(shù)點(diǎn)后為零時(shí) 加 "整"
if (postfixNun == 0)
{
Result += EndOfInt;
}
}
}
else
{
Result += NumLeverChineseSign[(int)currentLevel];
}
//當(dāng)真正的個(gè)位為零時(shí)加上“元”
if (NL == null && postfixNun < 1 && currentLevel > NumLevel.Yuan && postfixNun > 0)
{
Result += NumLeverChineseSign[(int)NumLevel.Yuan];
}
}
else
{
//當(dāng) 前綴數(shù)字未被除盡時(shí), 遞歸下去
NumLevel? NextNL = null;
if ((int)currentLevel >= (int)(NumLevel.TenThousand))
NextNL = currentLevel;
Result += CalculateChineseSign((double)prefixNum, NextNL, isDump, IsExceptTen);
if ((int)currentLevel >= (int)(NumLevel.TenThousand))
{
Result += NumLeverChineseSign[(int)currentLevel];
}
}
//是否跳位
// 判斷是否加零, 比如302 就要給三百 后面加零,變?yōu)?三百零二。
if (IsDumpLevel(Num))
{
Result += NumChineseCharacter[0];
isDump = true;
}
//余數(shù)是否需要遞歸
if (postfixNun > 0)
{
Result += CalculateChineseSign(postfixNun, NL, isDump, false);
}
else if (postfixNun == 0 && currentLevel > NumLevel.Yuan )
{
//當(dāng)數(shù)字是以零元結(jié)尾的加上 元整 比如1000000一百萬(wàn)元整
if (NL == null)
{
Result += NumLeverChineseSign[(int)NumLevel.Yuan];
Result += EndOfInt;
}
}
return Result;
}
9. 外部調(diào)用的轉(zhuǎn)換方法。
/// <summary>
/// 外部調(diào)用的轉(zhuǎn)換方法
/// </summary>
/// <param name="Num"></param>
/// <returns></returns>
public string ConvertToChinese(string Num)
{
if (!IsValidated<string>(Num))
{
throw new OverflowException("數(shù)值格式不正確,請(qǐng)輸入小于9999億兆的數(shù)字且最多精確的分的金額!");
}
string chineseCharactor = string.Empty;
if (IsBigThanTillion(Num))
{
double[] tillionNums = SplitNum(Num);
chineseCharactor = ContactNumChinese(tillionNums);
}
else
{
double dNum = Convert.ToDouble(Num);
chineseCharactor = CalculateChineseSign(dNum, null, true, IsStartOfTen(dNum));
}
return chineseCharactor;
}
小結(jié):
個(gè)人認(rèn)為程序的靈魂是算法,大到一個(gè)系統(tǒng)中的業(yè)務(wù)邏輯,小到一個(gè)貨幣數(shù)字轉(zhuǎn)中文的算法,處處都體現(xiàn)一種邏輯思想。
是否能把需求抽象成一個(gè)好的數(shù)學(xué)模型,直接關(guān)系到程序的實(shí)現(xiàn)的復(fù)雜度和穩(wěn)定性。在一些常用功能中想些不一樣的算法,對(duì)我們開(kāi)拓思路很有幫助。
更多關(guān)于C#相關(guān)內(nèi)容感興趣的讀者可查看本站專(zhuān)題:《C#數(shù)學(xué)運(yùn)算技巧總結(jié)》、《C#窗體操作技巧匯總》、《C#常見(jiàn)控件用法教程》、《WinForm控件用法總結(jié)》、《C#程序設(shè)計(jì)之線程使用技巧總結(jié)》、《C#數(shù)據(jù)結(jié)構(gòu)與算法教程》、《C#數(shù)組操作技巧總結(jié)》及《C#面向?qū)ο蟪绦蛟O(shè)計(jì)入門(mén)教程》
希望本文所述對(duì)大家C#程序設(shè)計(jì)有所幫助。
上一篇:C# 實(shí)現(xiàn)QQ式截圖功能實(shí)例代碼
欄 目:C#教程
下一篇:詳解C#用new和override來(lái)實(shí)現(xiàn)抽象類(lèi)的重寫(xiě)區(qū)別
本文標(biāo)題:C#基于純數(shù)學(xué)方法遞歸實(shí)現(xiàn)貨幣數(shù)字轉(zhuǎn)換中文功能詳解
本文地址:http://www.jygsgssxh.com/a1/C_jiaocheng/5910.html
您可能感興趣的文章
- 01-10基于C#實(shí)現(xiàn)簡(jiǎn)單離線注冊(cè)碼生成與驗(yàn)證
- 01-10C#基于UDP實(shí)現(xiàn)的P2P語(yǔ)音聊天工具
- 01-10C#實(shí)現(xiàn)基于加減按鈕形式控制系統(tǒng)音量及靜音的方法
- 01-10C#代碼實(shí)現(xiàn)PDF文檔操作類(lèi)
- 01-10C#基于WebBrowser獲取cookie的實(shí)現(xiàn)方法
- 01-10C#基于委托實(shí)現(xiàn)多線程之間操作的方法
- 01-10基于C#對(duì)用戶密碼使用MD5加密與解密
- 01-10基于C#實(shí)現(xiàn)簡(jiǎn)單的隨機(jī)抽獎(jiǎng)小程序
- 01-10C#基于cookie實(shí)現(xiàn)的購(gòu)物車(chē)功能
- 01-10基于C#實(shí)現(xiàn)12306的動(dòng)態(tài)驗(yàn)證碼變成靜態(tài)驗(yàn)證碼的方法


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹(shù)的示例代碼(圣誕
- 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問(wèn)題方法
- 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語(yǔ)言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 01-10C#通過(guò)反射獲取當(dāng)前工程中所有窗體并
- 01-10關(guān)于ASP網(wǎng)頁(yè)無(wú)法打開(kāi)的解決方案
- 01-10WinForm限制窗體不能移到屏幕外的方法
- 01-10WinForm繪制圓角的方法
- 01-10C#實(shí)現(xiàn)txt定位指定行完整實(shí)例
- 01-10WinForm實(shí)現(xiàn)仿視頻播放器左下角滾動(dòng)新
- 01-10C#停止線程的方法
- 01-10C#實(shí)現(xiàn)清空回收站的方法
- 01-10C#通過(guò)重寫(xiě)Panel改變邊框顏色與寬度的
- 01-10C#實(shí)現(xiàn)讀取注冊(cè)表監(jiān)控當(dāng)前操作系統(tǒng)已
隨機(jī)閱讀
- 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子
- 01-10delphi制作wav文件的方法
- 01-11Mac OSX 打開(kāi)原生自帶讀寫(xiě)NTFS功能(圖文
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 04-02jquery與jsp,用jquery
- 01-10C#中split用法實(shí)例總結(jié)
- 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 01-10SublimeText編譯C開(kāi)發(fā)環(huán)境設(shè)置


