在小写转大写金额时我们应该注意的是人类在读数的过程进行分析,比如要读“12345.67”,大写读法是:“壹万贰仟叁佰肆拾伍元陆角柒分”,在实际的读数过程中,人必须知道1后面有4位,即是万,2后面有3位,即是仟,依次类推,当然由于人已经习惯了万的下一位是仟,所以不再去数2的后面有几位。
根据上面的识别过程,我们应该采取反相识别实现遇到的数字,即在读取整数部份时,应是“伍肆拾叁佰贰仟壹万”。
另外置零的方式与段的设置,
1. 零主要存在于两个数之间有0的情况,但“拾”位情况除外,如拾万伍仟,因为在大写里面不会念拾万零伍仟,所以这里要特殊处理。
2. 所谓段,中国人的习惯与西方的习惯不同,西方的数字是以千计,而我们的习惯是以万计,即该段与更高一段之间没有非零数字,则这个段名不必读,如壹亿零伍仟,其中的万也就不读了。
下面提供实现的代码及注释
CString GetBigMoney(double dMoney)
{
//这里没有对超出部份作异常,使用者要注意(现实中不会出现如此巨大的金额数)
CString strMoney;
strMoney.Format ("%.2f" , dMoney);
CString strUnit = "元拾佰仟万拾佰仟亿拾佰仟";
CString strNumber = "零壹贰叁肆伍陆柒捌玖";
CString strOtherUnit = "整角分";
//将数字分整数部份与小数部份处理
int nPos = strMoney.Find (".");
int nLength = strMoney.GetLength ();
if(nPos < 0)
nPos = nLength;
CString strReturnValue;
int nCount = 0;
bool bZero = false;
bool bNeedLevel = false; //对段的识别,用于是否需要出现段名,如亿,万等
//对整数部份进行反相识别处理
for(int i = nPos - 1;i >= 0;i --)
{
TCHAR ch = strMoney.GetAt (i);
if(nCount % 4 == 0 && nCount > 0)
{
//如果处理的数字为第四位(万),或第八位(亿)等,则要求置段
bNeedLevel = true;
}
if(ch == '0')
{
//只对拾佰仟位的0进行识别,主要考虑到拾的特殊性,即如10读壹拾,不会读壹拾零
if(nCount % 4 != 0)
bZero = true;
}
else
{
CString strTemp(strReturnValue);
strReturnValue = strNumber.Mid ((ch - 0x30) * 2 , 2);
if(nCount > 0)
{
strReturnValue += strUnit.Mid (nCount * 2 , 2);
if(nCount % 4 != 0 && bNeedLevel)
{
//这里判断是否需要读段名,如万,亿等
strReturnValue += strUnit.Mid (int(nCount / 4) * 8 , 2);
}
bNeedLevel = false;
}
if(bZero)
{
//只有比当前处理的位要低中有数字才补零
if(!strTemp.IsEmpty ())
strReturnValue += strNumber.Left (2);
bZero = false;
}
strReturnValue += strTemp;
}
nCount ++;
}
strReturnValue += strUnit.Left (2);
bool bAllZero = true;
//下面实现对小数点后面的处理
//先判断是否为全零,则不需要继续读
if(nPos < nLength)
{
if(nLength > 2)
nLength = 2;
for(int i = 0;i < nLength;i ++)
if(strMoney.GetAt (nPos + i + 1) != '0')
bAllZero = false;
}
if(bAllZero)
{
strReturnValue += strOtherUnit.Left (2);
}
else
{
//对分角的处理
for(int i = 0;i < nLength;i ++)
{
TCHAR ch = strMoney.GetAt (nPos + 1 + i);
if(ch == '0' && i > 0)
{
}
else
{
strReturnValue += strNumber.Mid ((ch - 0x30) * 2 , 2);
if(ch != '0')
strReturnValue += strOtherUnit.Mid ((i + 1) * 2 , 2);
}
}
}
return strReturnValue;
}
点评:小写转大写金额中,根据理解,我们可以将一个数分成两个层次处理,即段间处理与段内处理,段内处理即为对于小于10,000的数字的转换,段间处理则在段内处理的基础上再进行处理,显然递归是一种可行的算法(参阅递归在C++应用中的利与弊),实现起来可能会简单些。
本算法是笔者在急需要类似算法时写下的,只有参考价值,并无直接应用的实践价值,读者可以对此修改后使用。