记一次bug解决过程(数字转化成中文)
栏目: JavaScript · 发布时间: 5年前
内容简介:由于公司业务周期较短,时常是几个项目一起做,或是加上bug修复。上午的时候接到个任务,正式上的网站发现如下图的错误。找到项目中对应页面,很快发现了问题,原来之前的哥们通过定义了个数组,通过序号来取对应的中文。这样的方式缺点很明显,个数有限。导致现在出现不够用的状况。经过考虑觉得可以抽出一个公共的方法代码如下:
由于公司业务周期较短,时常是几个项目一起做,或是加上bug修复。上午的时候接到个任务,正式上的网站发现如下图的错误。
问题原因
找到项目中对应页面,很快发现了问题,原来之前的哥们通过定义了个数组,通过序号来取对应的中文。这样的方式缺点很明显,个数有限。导致现在出现不够用的状况。
解决方式
经过考虑觉得可以抽出一个公共的方法代码如下:
/** * 根据输入的数字返回对应是中文,格式如:三十一 * @param {Number} index 序号 */ export function covertNumberTochinese(index) { const multiple = parseInt((index + 1) / 10) const remainder = (index + 1) % 10 const weekheadNum = [ '一', '二', '三', '四', '五', '六', '七', '八', '九', '十' ] let text = '' const textMap = new Map([ [/^1_[0-9]$/, () => `十${weekheadNum[remainder - 1]}`], [/^1_0$/, () => `十`], [ /^[2-9]_[1-9]$/, () => `${weekheadNum[multiple - 1]}十${weekheadNum[remainder - 1]}` ], [/^[2-9]_0$/, () => `${weekheadNum[multiple - 1]}十`], [/^0_\w/, () => `${weekheadNum[remainder - 1]}`] ]) const textList = [...textMap].filter(([reg]) => { return reg.test(multiple + '_' + remainder) }) textList.forEach(([reg, callBack]) => { text = callBack() }) return text } 复制代码
- multiple用来判断有几个10,remainder表示余几。
- 这里使用Map数据结构的原因是因为将对象作为键名,这样可以将正则与函数关联起来。
- 使用正则匹配,可以匹配多种情况,使用正则的test方法来校验是否匹配成功。
需要用到的知识点
- new Map([['键名':'键值']])
- ... 扩展运算符 遍历 Iterator 接口(这里是Map)
- [reg] = [/^1_[0-9]$/, () => `十${weekheadNum[remainder - 1]}`] 数组结构赋值
分析
- covertNumberTochinese函数接受一个数值,对数值进行取余 remainder(用来当作个位数)和商取整的multiple(用来当作十位数)。
- 将个位数十位数拼接成用_相连的字符串,然后用正则取匹配这个字符串。然后两个中文字拼接起来。
- 需要注意的是当multiple为1,remainder为0时本应该返回'十零'但是我们习惯不是注意的叫法,多加一个判断条件。
- 判断下remainder(个位数)为0时,省去'几十零'后面的零
小结
用上面的方法可以将数字匹配到中午 1-99位,虽然项目中是够用了,但是局限性还是很大,而且判断的条件很多,逻辑不直观。
寻求其他解决方案
/** * 根据输入的数字返回对应是中文,格式如:一千零一 * @param {Number|String} index 序号或者数字开头的字符串 */ export numberToChinese(number) { number = String(parseInt(number)) const ChineseText = '零一二三四五六七八九' const smallUnit = '十百千' const length = number.length let n = length - 2 let string = '' for (var i = 0; i < length; i++) { let num = number.charAt(i) string += ChineseText.charAt(num) string += num > 0 ? smallUnit.charAt(n) : '' n-- } return string } 复制代码
分析
- 首先将传入的数字取整处理,然后将其转换成字符串,为了是使用字符串的
.charAt
方法取对应位数的中文。 - 定义好中文的0-9,以及单位'十百千' (为了简单起见,只讨论
四位数以下
的转换,位数增加方法是类似的) - 定义好string是最终返回的中文字符串,是通过
拼接
的方式得到一个中文的数字,这里n代表取第几位单位也就是取smallUnit
的第几位
循环体中做的事
- 去除number(也就是数字字符串如1001)的第一位,就是数字1类型是string类型,然后通过这个1取
ChineseText
中的中文得到中文的一 - 然后通过n取一对应的单位,这里n=length-2是因为再定义单位时是
从十开始
的而不是从个位数开始,并且.charAt(0)取的是字符串的第一位
。并且当数字为零的时候不需要单位,所以加了num > 0的限定条件。
完善
就此数字转换成中文的方法就写完了,在浏览器中打印如下
有多个零的时候我们习惯是一千零一,并且结尾为零的时候是省略的 所以 我们还得对返回的string做一下去零处理
/** * 根据输入的字符串返回按照规则去零后的字符串,格式如:一千零一 * @param {string} str 中文数字 如 一千零零一 */ export clearZero(str){ const regMiddle = /零{2}/g const regEnd = /零?零$/ str = str.replace(regMiddle, '零') return str.replace(regEnd, '') } 复制代码
- 通过clearZero函数首先将匹配中间有两个零相连的情况,转为一个零
- 然后判断结尾是否有一个至两个零的情况,将它们清空
- 注意点: replace 不会改变原来的值,需要将操作后的结果重新赋值
将 numberToChinese 函数改为如下
export numberToChinese(number) { number = String(parseInt(number)) const ChineseText = '零一二三四五六七八九' const smallUnit = '十百千' const length = number.length let n = length - 2 let string = '' for (var i = 0; i < length; i++) { let num = number.charAt(i) string += ChineseText.charAt(num) string += num > 0 ? smallUnit.charAt(n) : '' n-- } return clearZero(string) } 复制代码
总结
- 四位数以下的数字转换成中文的方法已经完成了 ,更多位数的转换原理相同,有个方法就是将多位数的数字截取成 四位为一组,分组处理,最后将结果拼接起来,这样可能会减少一些逻辑判断。
- 核心部分就是通过
charAt
取数字字符串,然后通过数组去找定义好的ChineseText对应的中文,定义了一个n
来取当前位数对应的单位
以上所述就是小编给大家介绍的《记一次bug解决过程(数字转化成中文)》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!
猜你喜欢:- [译]从输入URL到页面呈现的超详细过程——第二步:Tags转化成DOM的过程
- js的类型转化三两事儿
- archTIS:将数据安全转化为经济增长
- 如何将JavaScript转化成Swift?(一)
- python3 第十章 - 如何进行进制转化
- 用Golang将图片转化成ASCII码
本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。