1. 场景描述
BMI指数(身体质量指数,英文Body Mass Index)是用体重公斤数除以身高米数的平方得出的数字,是目前国际上通用的衡量人体胖瘦程度以及是否健康的一个标准。“身体质量指数”这个概念,是由19世纪中期的比利时数学家凯特勒最先提出来的,BMI指数的计算公式:
B
M
I
=
w
e
i
g
h
t
/
h
e
i
g
h
t
2
BMI = weight / height ^2
BMI=weight/height2
其中weight体重的度量单位是公斤,height身高的度量单位是米。
BMI指数的设计初衷是一个用于公共健康研究的统计工具。如果我们需要知道肥胖与某一疾病的致病原因是否相关时,就可以把病人的身高和体重换算为BMI值,再找出BMI值与病发率是否有线性关系。不过,随着医学科技的发展和进步,现在BMI值只是一个参考值,要检测一个人是否肥胖有了更加科学和先进的检测手段。因此BMI的角色也在慢慢地改变,从医学上的用途逐步变为一般大众的纤体指标。事实上,对于健康达人或者是普通的健身爱好者来说,BMI指数已经成为了衡量健身成效的重要标准。
接下来,我们使用Python来编写一个基于图形界面的BMI计算器。
2. 编程思路
2.1 计算公式
B M I = w e i g h t / h e i g h t 2 BMI = weight / height^2 BMI=weight/height2
2.2 判断标准
BMI 值中国标准含义如下:BMI <= 18.4 偏瘦;18.4 < BMI < 24 正常;24.0 <= BMI <28 过重; >= 28 肥胖。
2.3 健康建议
我们可以针对各种形体人群给出相关的健康建议。需要强调的是这些健康建议均取材于互联网,为编写程序的需要,仅供参考:
偏瘦:适度参加体育运动,特别是要多吃鱼、蛋和奶类等各类健康食品。
正常:注重适度运动,平衡的饮食结构,食物多样化,讲究粗细合理搭配。
超重:建议加强体育锻炼,注重食物搭配,特别是少吃高脂肪、高蛋白、含糖高食品,每天40分钟运动。
肥胖:加强体育锻炼,调整饮食结构,少吃或者不吃高脂肪、高蛋白、含糖量高食品,每天40分钟运动。
我们要把这些健康建议保存到文本文件suggestions.txt中,便于以后可以简单地使用Windows Notepad记事本程序就可以修改和完善这些健康建议。而不必修改Python源代码程序,实现数据与程序的分离。为了与我们的程序配套,这个文件保存时需要使用utf8的文件格式。
3. 代码编写
本程序由4个文件组成,它们是:
bmi.py :BMI计算器的图形界面程序。
bmi_base.py:BMI计算器的一些基础函数模块。
suggestions.txt:存放一些与形体相关的健康建议。
calculator.png:BMI计算器的图片文件。
bmi.py 模块的源代码程序
"""
bmi.py : BMI的前世今生
"""
from tkinter import *
import tkinter.messagebox as msg_box
from common.bmi_base import *
def calculate():
""" 计算 BMI值"""
try:
height = float(v_height.get())
weight = float(v_weight.get())
except ValueError:
msg_box.showinfo('提示', '检查数据:身高或体重!')
else:
bmi = calc_bmi(height, weight)
report = check_report(bmi)
advice = make_suggestion(report)
v_bmi.set(str(round(bmi, 1)) + ' ' + report)
info_area.delete("1.0", END)
info_area.insert("1.0", advice)
def reset():
"""清除输入域内容"""
v_height.set('')
v_weight.set('')
v_bmi.set('')
info_area.delete("1.0", END)
lbl_height.focus()
def initialize():
"""初始化屏幕内容"""
v_height.set('1.70')
v_weight.set('65')
lbl_height.focus()
calculate()
if __name__ == '__main__':
root = Tk()
root.title('BMI 计算器')
root.geometry('350x350+500+300')
# 设置初值
v_height = StringVar() # ①
v_weight = StringVar()
v_bmi = StringVar()
v_bmi.set('')
# 定义图片
photo = PhotoImage(file='calculator.png') # ②
lbl_banner = Label(root, image=photo, compound='top', width=340, height=120)
lbl_banner.grid(row=0, column=0, columnspan=4, sticky=N + S + E + W)
# 显示身高提示和输入身高
Label(root, text='身高 (m)').grid(row=1, column=1, pady=5, sticky=E) # ③
lbl_height = Entry(root, textvariable=v_height) # ④
lbl_height.grid(row=1, column=2) # ⑤
# 定义体重提示和输入体重
Label(root, text='体重 (kg)').grid(row=2, column=1, sticky=E)
Entry(root, textvariable=v_weight).grid(row=2, column=2)
# 定义功能按钮
Button(root, text='计算', width=8, command=calculate).grid(row=6, column=1, pady=6, sticky=E) # ⑥
Button(root, text='清空', width=8, command=reset).grid(row=6, column=2)
# 定义结果显示字段
Label(root, text='BMI').grid(row=5, column=1, pady=5, sticky=E)
Entry(root, textvariable=v_bmi).grid(row=5, column=2)
Label(text='健康建议').grid(row=8, column=1, pady=5, sticky=E)
info_area = Text(width=20, height=4, font=("微软雅黑", 9)) # ⑦
info_area.grid(row=8, column=2)
# 防止屏幕开"天窗"
initialize() # ⑧
# 进入事件驱动模式
root.mainloop() # ⑨
重要函数说明如下:
函数calculate():计算bmi指数,获取形体诊断结果,提取相关的健康建议。
函数reset():清除BMI计算器界面中的输入域的内容。
函数initialize():初始化界面内容,自动计算一个案例,填充屏幕内容。
重要语句说明如下:
语句①定义与界面控件联动的变量内容,也就是说当屏幕界面控件Entry输入域内容发生变化,将自动映射到关联的变量中,实现同步更新。
语句②加载图片以便与有关屏幕控件关联。
语句③定义有关标签控件,以便存放文本字符串。
语句④定义一个输入域控件,并与变量关联,实现数据同步更新。
语句⑤把控件放到屏幕中的指定位置,这里使用tkinter 的grid布局方式。
语句⑥定义按钮实现与函数的关联绑定。意味着当你鼠标点击这个按钮,将自动执行在参数command中指定的函数代码。
语句⑦使用控件Text定义多行的文本输入框,既能输入内容,又可显示文本。
语句⑧执行初始化的案例计算,并使用计算结果填充屏幕内容,避免在图形界面上开“天窗”,留下空白内容,它会给人造成界面不友好的印象。
语句⑨程序进入事件驱动模式。也就是说当用户点击屏幕界面中的控件时,将执行不同的程序代码。
bmi_base.py 模块的源代码
"""
bmi_base.py : BMI计算器可重用代码
"""
__all__ = ['calc_bmi', 'check_report', 'make_suggestion'] # ①
suggestions = {}
def decorate_bmi(func): # ②
"""
扩展calc_bmi()函数功能的装饰器
"""
def wrapper(height, weight):
if is_valid(height, weight):
return func(height, weight)
else:
return -1
return wrapper
@decorate_bmi # ③
def calc_bmi(height, weight):
"""
height : 计量单位是米
weight : 计量单位是公斤
"""
bmi = weight / height ** 2
bmi = round(bmi, 1)
return abs(bmi)
def is_number(number):
"""
判断 number 是数字
"""
if isinstance(number, int) or isinstance(number, float): # ④
return True
else:
return False
def is_valid(height, weight):
"""
判断 height,weight有效性
"""
if height and is_number(height) and is_number(weight):
return True
else:
return False
def check_report(bmi):
"""
bmi : BMI 指数值
report : 形体评测结果
"""
report = '正常'
if bmi < 18.5:
report = '偏瘦'
elif 18.5 <= bmi < 24:
report = '正常'
elif 24 <= bmi < 28:
report = '超重'
elif bmi >= 28:
report = '肥胖'
return report
def load_suggestions(file_name):
"""加载健康建议"""
with open(file_name, encoding='utf-8') as file: # ⑤
for line in file:
key, value = line.split(':') # ⑥
suggestions[key.strip()] = value.strip() # ⑦
def make_suggestion(report):
"""形成健康建议"""
if not suggestions:
# 一次性加载健康建议
load_suggestions('suggestions.txt')
if report in suggestions:
return suggestions[report]
else:
return None
重要函数说明:
函数load_suggestions():从文本文件suggestions.txt中读入健康建议,以字典的方式保存,供后续程序访问。
函数make_suggestion():针对不同的体型,生成不同的健康建议。
函数def calc_bmi(height, weight):计算BMI值的函数,需要注意的是调用此函数,将自动执行该函数的装饰器函数decorate_bmi(),以便对height和weight参数的有效性进行检测。
函数decorate_bmi():是一个装饰器函数,用于对calc_bmi()函数的参数进行有效性检测,以增强程序执行的容错能力。
函数check_report(bmi):通过对bmi值进行判断,获得形体类型的结论报告。
函数is_number(number):判断参数number是否是数字。
函数is_valid(height, weight):判断身高height和体重weight参数的有效性。
重要语句说明:
语句①定义本模块中提供的公共接口函数,即是可由其他程序模块调用的函数,以此实现代码重用。
语句②定义装饰器函数。
语句③将装饰器函数decorate_bmi()与被装饰函数calc_bmi(height, weight)建立绑定关系,当程序调用函数calc_bmi()时,首先要调用装饰器函数decorate_bmi(),然后再执行calc_bmi()函数。使用装饰器功能,我们可以在不修改calc_bmi()函数代码的情况下,扩展其功能。这里相当于利用装饰器是实现了对函数calc_bmi()的输入参数height和参数weight进行的合法性检查。
语句④判断某个变量是否是整数或者浮点数。
语句⑤以utf8的方式打开文件,供后续处理。这里需要强调的是文件suggestions.txt是以utf8的格式保存的文本文件。
语句⑥以冒号为分隔符,解析形体类型和健康建议的内容。
语句⑦以形体类型为键,健康建议为值,创建字典suggestions有存放健康建议,供后续代码使用。
4. 执行效果
首先说明程序结构:模块bmi.py、suggestions.txt和calculator.png存放在当前目录:D:\cases\BMI指数的前世今生,而bmi_base.py则是以Python包的形式,存放在目录common中。
D:\cases\BMI指数与健身达人>dir
2022/12/08 11:47 2,442 bmi.py
2020/12/04 17:49 75,504 calculator.png
2022/12/08 14:33 <DIR> common
2022/12/08 11:59 473 suggestions.txt
D:\cases\BMI指数与健身达人>
查询Python包文件内容
D:\cases\BMI指数与健身达人>cd common
D:\cases\BMI指数与健身达人\common>
D:\cases\BMI指数与健身达人\common>dir
2022/12/08 14:33 1,867 bmi_base.py
2022/04/01 07:34 106 __init__.py
D:\cases\BMI指数与健身达人\common>
执行程序结果显示
D:\cases\BMI指数与健身达人>python bmi.py
将显示以下界面,你可以根据提示进行操作。
经过测试表明,BMI计算器完全达到甚至超越了设计要求。更为重要的是BMI计算器提供了比较完善的容错机制,那就是对输入参数height和weight的有效性检测。如果留意的话,你会发现这样一个事实:往往一个程序的容错处理程序代码量不少于10%。如果一个程序尤其是具有商业用途的产品没有适当的容错机制的话,容易引发程序执行中的意外终止。这些现象通常会被客户或者用户认为程序中存在bugs,程序很low,最起码会认为程序员不够专业!