1. 程式人生 > >人民幣大小寫轉換(阿拉伯數字和漢字轉換)

人民幣大小寫轉換(阿拉伯數字和漢字轉換)

           最近在網上看到一個關於數字轉換人民幣大寫的演算法的一篇部落格,很好,就收藏了,演算法直接copy原作者的。自己做了一點小改動:

                   1.以CString代替,封裝成類。

                   2.修復超過"億"時顯示"億萬"的bug。

                   3.擴大檢索範圍至"千萬億(lg16)"。

          如果各位有更好的演算法,多貢獻出來大家研究研究,廢話補多說,直接貼出來供大家參考!

// numberConvert.h

#ifndef _NUMBERCONVERT_H_
#define _NUMBERCONVERT_H_

#include <atlstr.h>
#include <CString>
#include <String>

#ifndef ArrayCount
	#define	ArrayCount(Array) (sizeof(Array)/sizeof(Array[0]))
#endif // ArrayCount

#ifndef SAFE_DELETE
#	define SAFE_DELETE(p)	{ try { delete p; } catch (...) { } p = nullptr; } 
#endif // SAFE_DELETE

#ifndef SAFE_DELETE_ARRAY
#	define SAFE_DELETE_ARRAY(pArr) { if (pArr) { delete [] (pArr);  (pArr) = nullptr; } }
#endif // SAFE_DELETE_ARRAY

using namespace std;
class NumberConvert
{
public:

	NumberConvert();
	~NumberConvert();
	
	// 將源字串strOrig中第一個匹配strSub的子串部分替換為strReplace
	int ReplaceSubStr(CString &strOrig, CString strSub, CString strReplace);

	// 將人民幣double數值轉化為人民幣漢字CString
	CString NumToChineseStr(double money);

private:

	CString* m_pstrDigit;
	CString* m_pstrUnit1;
	CString* m_pstrUnit2;
};

#endif //_NUMBERCONVERT_H_
// numberConvert.cpp

#include "numberConvert.h"

NumberConvert::NumberConvert()
{
	CString strDigit[]	= { _T("零"), _T("壹"), _T("貳"), _T("叄"), _T("肆")
		, _T("伍"), _T("陸"), _T("柒"), _T("捌"), _T("玖") };

	CString strUnit[]	= { _T("圓"), _T("拾"), _T("佰"), _T("仟"), _T("萬")
		, _T("拾"), _T("佰"), _T("仟"), _T("億")
		, _T("拾"), _T("佰"), _T("仟"), _T("萬"), _T("拾"), _T("佰"), _T("仟") };

	CString strUnit2[]	= { _T("角"), _T("分") };

	int iLen = ArrayCount(strDigit);
	m_pstrDigit = new CString[iLen];
	for (int i = 0; i < iLen; ++i)
	{
		m_pstrDigit[i] = strDigit[i];
	}

	iLen = ArrayCount(strUnit);
	m_pstrUnit1 = new  CString[iLen];
	for (int i = 0; i < iLen; ++i)
	{
		m_pstrUnit1[i] = strUnit[i];
	}

	iLen = ArrayCount(strUnit2);
	m_pstrUnit2 = new  CString[iLen];
	for (int i = 0; i < iLen; ++i)
	{
		m_pstrUnit2[i] = strUnit2[i];
	}
}

NumberConvert::~NumberConvert()
{
	SAFE_DELETE_ARRAY(m_pstrDigit);
	SAFE_DELETE_ARRAY(m_pstrUnit1);
	SAFE_DELETE_ARRAY(m_pstrUnit2);
}

int NumberConvert::ReplaceSubStr(CString &strOrig, CString strSub, CString strReplace)
{
	int pos = (int)strOrig.Find(strSub);
	if (pos >= 0)
	{
		strOrig.Replace(strSub, strReplace);
		return 0;
	}

	return -1;
}

CString NumberConvert::NumToChineseStr(double money)
{
	int i = 0;
	int ret = 0;
	int length = 0;
	char *p = NULL;
	char *pcDecimal = NULL;				//儲存小數部分字元
	char czNumber[MAX_PATH] = { 0 };	//儲存完整數字部分字元
	CString strResult;
	
	//判斷是否為小數
	if (money < 0)
	{
		strResult += _T("負");
		money = ::abs(money);
	}

	//將數字轉為數字字串,利用sprintf_s的正則轉換
	_stprintf_s(czNumber, MAX_PATH, _T("%.16lg"), money);

	//如果數字是太大或太小的數,因為已經轉為科學計數,所以會含有e字元
	p = _tcschr(czNumber, _T('e'));
	if (NULL != p)
	{
		strResult = _T("金額超出計算範疇");
		return strResult;
	}

	p = _tcschr(czNumber, _T('.'));
	if (NULL != p)
	{
		p[0] = 0;
		pcDecimal = p + 1;
	}

	length = (int)_tcslen(czNumber);
	for (i = 0; i < length; ++i)
	{
		if (_T('0') == czNumber[i] && 0 != ((length - 1 - i) % 4))
		{
			strResult += m_pstrDigit[czNumber[i] - _T('0')];
		}
		else
		{
			strResult += m_pstrDigit[czNumber[i] - _T('0')] + m_pstrUnit1[length - i - 1];
		}
	}

	//把strResult中的所有"零零"子串替換為"零"
	while (true)
	{
		ret = ReplaceSubStr(strResult, _T("零零"), _T("零"));
		if (ret < 0)
		{
			break;
		}
	}

	ReplaceSubStr(strResult, _T("零億"), _T("億"));
	ReplaceSubStr(strResult, _T("零萬"), _T("萬"));
	ReplaceSubStr(strResult, _T("億萬"), _T("億"));

	//如果整數部分全為0,則不要去除元單位前面的零
	if (_tcscmp(strResult, _T("零圓")) != 0)
	{
		ReplaceSubStr(strResult, _T("零圓"), _T("圓"));
	}
	
	//小數精確到兩位數,即精確到單位分
	if (NULL != pcDecimal)
	{
		//如果小數部分有數值而整數部分為0,則刪除字串中的零元
		if (_tcscmp(strResult, _T("零圓")) == 0
			|| _tcscmp(strResult, _T("零")) == 0)
		{
			strResult.Empty();
		}

		i = 0;
		while (true)
		{
			if (0 == pcDecimal[i] || i >= 2)
			{
				break;
			}
			strResult += m_pstrDigit[pcDecimal[i] - _T('0')] + m_pstrUnit2[i];
			++i;
		}
	}

	// 剔除掉"圓"字
	strResult.Replace(_T("圓"), _T(""));

	return strResult;
}