目录
一、引言
在财务数据处理场景中,数字金额的大写转换 是确保票据准确性和防伪性的重要环节。以跨平台GUI框架PyQt 为主体,再将用户输入的数字通过规范化处理,自动转换为符合规定的标准大写格式(如壹、贰、佰、万等),同时支持零、整等特殊情况的分类判断。这种自动化转换不仅避免了人工输入错误的风险,还能显著提升财务人员的工作效率,有效助力于适合报销单、支票、合同等需要严格数字规范的场景。
二、GUI界面设计
使用PyQt5进行界面的搭建,界面如下:

初始界面搭建采用极简风格。用户通过输入数字,再点击"转换!",即可在下方弹出相应的大写形式。
1.效果演示

当用户输入存在小数时,且在实际生活中,小数点后两位分别表示金额中的"角"和"分" ,因此程序对小数点以后的位数最多设置为2 ;当用户输入为纯整数时,转换后的结果要符合"XXXX元整"的格式。
2.相关提示
针对输入为空和输入为0这两种无实义的情况,均会给出"请输入有效数字"的提示。
**3.**界面设计.py
通过pyuic5产生的GUI界面代码UI_CaiwuDaxie.py如下:
python
# -*- coding: utf-8 -*-
# Form implementation generated from reading ui file 'UI_CaiwuDaxie.ui'
#
# Created by: PyQt5 UI code generator 5.15.11
#
# WARNING: Any manual changes made to this file will be lost when pyuic5 is
# run again. Do not edit this file unless you know what you are doing.
from PyQt5 import QtCore, QtGui, QtWidgets
class Ui_Form(object):
def setupUi(self, Form):
Form.setObjectName("Form")
Form.setEnabled(True)
Form.resize(500, 390)
Form.setMinimumSize(QtCore.QSize(500, 390))
Form.setMaximumSize(QtCore.QSize(500, 390))
icon = QtGui.QIcon()
icon.addPixmap(QtGui.QPixmap(":/image1.png"), QtGui.QIcon.Normal, QtGui.QIcon.Off)
Form.setWindowIcon(icon)
self.label_5 = QtWidgets.QLabel(Form)
self.label_5.setGeometry(QtCore.QRect(330, 50, 141, 31))
font = QtGui.QFont()
font.setFamily("Adobe Arabic")
font.setPointSize(16)
self.label_5.setFont(font)
self.label_5.setObjectName("label_5")
self.label_8 = QtWidgets.QLabel(Form)
self.label_8.setGeometry(QtCore.QRect(90, 10, 371, 31))
font = QtGui.QFont()
font.setFamily("Adobe Arabic")
font.setPointSize(18)
font.setBold(False)
font.setWeight(50)
self.label_8.setFont(font)
self.label_8.setObjectName("label_8")
self.label_10 = QtWidgets.QLabel(Form)
self.label_10.setGeometry(QtCore.QRect(410, 40, 71, 51))
self.label_10.setText("")
self.label_10.setPixmap(QtGui.QPixmap(":/image1.png"))
self.label_10.setObjectName("label_10")
self.lineEdit = QtWidgets.QLineEdit(Form)
self.lineEdit.setGeometry(QtCore.QRect(40, 110, 181, 20))
self.lineEdit.setObjectName("lineEdit")
self.label = QtWidgets.QLabel(Form)
self.label.setGeometry(QtCore.QRect(40, 150, 121, 16))
self.label.setObjectName("label")
self.label_2 = QtWidgets.QLabel(Form)
self.label_2.setGeometry(QtCore.QRect(40, 80, 91, 21))
self.label_2.setObjectName("label_2")
self.label_3 = QtWidgets.QLabel(Form)
self.label_3.setGeometry(QtCore.QRect(40, 180, 421, 191))
font = QtGui.QFont()
font.setFamily("Adobe Arabic")
font.setPointSize(16)
self.label_3.setFont(font)
self.label_3.setFrameShape(QtWidgets.QFrame.StyledPanel)
self.label_3.setText("")
self.label_3.setTextFormat(QtCore.Qt.PlainText)
self.label_3.setScaledContents(False)
self.label_3.setAlignment(QtCore.Qt.AlignCenter)
self.label_3.setWordWrap(True)
self.label_3.setIndent(-1)
self.label_3.setObjectName("label_3")
self.pushButton = QtWidgets.QPushButton(Form)
self.pushButton.setGeometry(QtCore.QRect(260, 110, 191, 23))
self.pushButton.setObjectName("pushButton")
self.retranslateUi(Form)
QtCore.QMetaObject.connectSlotsByName(Form)
def retranslateUi(self, Form):
_translate = QtCore.QCoreApplication.translate
Form.setWindowTitle(_translate("Form", "CaiwuDaxie"))
self.label_5.setText(_translate("Form", "Designed By"))
self.label_8.setText(_translate("Form", "财务办公------数字金额转大写"))
self.label.setText(_translate("Form", "大写如下:"))
self.label_2.setText(_translate("Form", "请输入金额:"))
self.pushButton.setText(_translate("Form", "转换"))
import ziyuan_rc
三、主要程序详解
1.导入模块和全局变量
python
import sys
from UI_CaiwuDaxie import *
from PyQt5.QtWidgets import QApplication, QWidget
# 正则表达式
from PyQt5.QtGui import QRegExpValidator
# 保持窗口大小和qtdesigner中的一致
from PyQt5 import QtCore
QtCore.QCoreApplication.setAttribute(QtCore.Qt.AA_EnableHighDpiScaling)
shuzi_dict = {'0': '零',
'1': '壹',
'2': '贰',
'3': '叁',
'4': '肆',
'5': '伍',
'6': '陆',
'7': '柒',
'8': '捌',
'9': '玖'}
导入前端页面设计文件:UI_CaiwuDaxie .py; 因需要对用户输入作规范要求 ,故导入QRegExpValidator ;;不懂为啥需要导入Qtcore的,请看。最后定义数字与大写数字的对应字典为全局变量,方便后续使用。****
2.初始化设置
python
def __init__(self):
super(QWidget, self).__init__()
self.setupUi(self)
# 允许整数或最多两位小数
validator = QRegExpValidator(QtCore.QRegExp("^(?!0+\d)\d*\.?\d{0,2}$"))
self.lineEdit.setValidator(validator)
self.pushButton.clicked.connect(self.daxie)
首先通过正则表达式 定义输入格式:输入内容必须为数字 ,其他字符输不进去;输入整数和小数均可以,但要求小数部分的位数最多不超过两位 ,即\d{0,2};且要求输入的整数部分 可以以"0"开头,但不能出现两个及两个以上"0"连续开头的情况,目的是避免类似于"00.XX"的不规范情况出现 。最后将其应用于lineedit,这里巧妙使用正则表达式提取定义用户输入格式,如不熟悉正则表达式,请看这位博主关于正则表达式的详解:正则表达式全解析+常用示例_正则解析-CSDN博客(个人认为是写得比较好且全面的)。
最后将pushbutton绑定于daxie。
3.大写转换
python
def daxie(self):
shuzi_str = self.lineEdit.text()
if (shuzi_str == '0') or (shuzi_str == ''):
# 无效输入
QtWidgets.QMessageBox.critical(self, "提示", "请输入有效数字!")
else:
# 有效输入
if '.' in shuzi_str:
# 用户输入有小数
zhengshu_str, xiaoshu_str = shuzi_str.split('.')
if zhengshu_str=='0':
# 整数部分为0,只处理小数部分
self.label_3.setText(xiaoshu(xiaoshu_str))
else:
# 用户输入有零有整时
neirong_zhengshu = zhengshu(zhengshu_str)
neirong_xiaoshu = xiaoshu(xiaoshu_str)
self.label_3.setText(neirong_zhengshu+'元'+neirong_xiaoshu)
else:
# 用户输入为纯整数
neirong = zhengshu(shuzi_str)
self.label_3.setText(neirong+'元整')
从lineedit获取用户输入并将其存储在shuzi_str中;判断shuzi_str是否是有效输入(非零、空值),若为无效输入,给出"请输入有效数字"的提示,反之将其细分为三种情况;判断用户输入是否存在小数点".",可将输入分为纯整数输入和小数输入;对于纯整数输入,直接调用自定义函数zhengshu得到大写结果,最后在末尾加上"元整"setText给label_3即可;对于小数,若整数部分为0,按照日常生活,一般不读"零元X角X分",而直接是"X角X分",故直接调用自定义函数xiaoshu;最后对于有整有小数的输入,分别调用自定义函数zhengshu和xiaoshu,再在两者结果中间添加衔接词"元"setText给label_3即可。
4.自定义函数
python
def zhengshu(shu):
jieguo = []
for i in range(-len(shu), 0, 1):
jieguo.append(shuzi_dict[shu[i]])
if (-i)%4 == 0:
jieguo.append('仟')
elif (-i)%4 == 3:
jieguo.append('佰')
elif (-i)%4 == 2:
jieguo.append('拾')
elif (-i)%4 == 1:
# 每4位为一组,判断是否需要增加"万"、"亿"
if (-i)//4 == 0:
pass
elif (-i)//4 == 1:
jieguo.append('万')
elif (-i)//4 == 2:
jieguo.append('亿')
else:
# 最大只到"亿",可满足日常财务基本使用需求,若想扩大进制,可在此继续增加,与上同理
pass
return ''.join(jieguo)
def xiaoshu(shu):
return shuzi_dict[shu[0]]+'角'+shuzi_dict[shu[1]]+'分'
对于自定义函数zhengshu:首先需要明确如何将数字转换为对应的大写汉字,这里采用负索引 进行遍历。对任意数字,最低位在最右,最高位在最左,按负索引遍历契合我们日常习惯,只不过是多了一个负号"---"而已;再通过字典shuzi_dict获取每个数字对应的大写,并将其添加至lst中。
对于在何处分别添加"亿"、"万"、"仟"、"佰"、"拾"这些量词,首先需要找到规律:对于任何一个数字,从低位到高位 (及从右到左,也印证了最优解是使用负索引)每四位一组 ,分别添加的是"万"和"亿"(如比"亿"更大,同理添加,这里只到"亿",足以满足日常财务转换需求),若对负索引取反后再对4取整 ,为1则说明添加"万",为2则说明添加"亿"(如比"亿"更大,同理添加即可);而对于"仟"、"佰"、"拾",需要对负索引取反后再对4取余,若余数为0,则添加"仟";若余数为3,则添加"佰";若余数为2,则添加"拾";若余数为1,则说明下一位即将到4位一组的分界点,需要添加更大的计量单位("万"、"亿")。
对于自定义函数xiaoshu,直接通过查询字典shuzi_dict并返回拼接结果即可。
四、总程序代码
python
import sys
from UI_CaiwuDaxie import *
from PyQt5.QtWidgets import QApplication, QWidget
# 正则表达式
from PyQt5.QtGui import QRegExpValidator
# 保持窗口大小和qtdesigner中的一致
from PyQt5 import QtCore
QtCore.QCoreApplication.setAttribute(QtCore.Qt.AA_EnableHighDpiScaling)
shuzi_dict = {'0': '零',
'1': '壹',
'2': '贰',
'3': '叁',
'4': '肆',
'5': '伍',
'6': '陆',
'7': '柒',
'8': '捌',
'9': '玖'}
class login_interface(QWidget, Ui_Form):
def __init__(self):
super(QWidget, self).__init__()
self.setupUi(self)
# 允许整数或最多两位小数
validator = QRegExpValidator(QtCore.QRegExp("^(?!0+\d)\d*\.?\d{0,2}$"))
self.lineEdit.setValidator(validator)
self.pushButton.clicked.connect(self.daxie)
def daxie(self):
shuzi_str = self.lineEdit.text()
if (shuzi_str == '0') or (shuzi_str == ''):
# 无效输入
QtWidgets.QMessageBox.critical(self, "提示", "请输入有效数字!")
else:
# 有效输入
if '.' in shuzi_str:
# 用户输入有小数
zhengshu_str, xiaoshu_str = shuzi_str.split('.')
if zhengshu_str=='0':
# 整数部分为0,只处理小数部分
self.label_3.setText(xiaoshu(xiaoshu_str))
else:
# 用户输入有零有整时
neirong_zhengshu = zhengshu(zhengshu_str)
neirong_xiaoshu = xiaoshu(xiaoshu_str)
self.label_3.setText(neirong_zhengshu+'元'+neirong_xiaoshu)
else:
# 用户输入为纯整数
neirong = zhengshu(shuzi_str)
self.label_3.setText(neirong+'元整')
def zhengshu(shu):
jieguo = []
for i in range(-len(shu), 0, 1):
jieguo.append(shuzi_dict[shu[i]])
if (-i)%4 == 0:
jieguo.append('仟')
elif (-i)%4 == 3:
jieguo.append('佰')
elif (-i)%4 == 2:
jieguo.append('拾')
elif (-i)%4 == 1:
# 每4位为一组,判断是否需要增加"万"、"亿"
if (-i)//4 == 0:
pass
elif (-i)//4 == 1:
jieguo.append('万')
elif (-i)//4 == 2:
jieguo.append('亿')
else:
# 最大只到"亿",可满足日常财务基本使用需求,若想扩大进制,可在此继续增加,与上同理
pass
return ''.join(jieguo)
def xiaoshu(shu):
return shuzi_dict[shu[0]]+'角'+shuzi_dict[shu[1]]+'分'
if __name__ == '__main__':
app = QApplication(sys.argv)
w = login_interface()
w.show()
sys.exit(app.exec_())
欢迎留言讨论交流!