合规国际互联网加速 OSASE为企业客户提供高速稳定SD-WAN国际加速解决方案。 广告
### 5.3.1 统计图表 图形的一个重要用途是为数据提供可视的表示,这在统计、汇总性质的应用程序中尤其 重要,因为汇总数据几乎都可以利用图形来改善表示。下面我们编写一个简单的统计汇总程 序,以演示图形编程在数据可视化方面的应用。 假设某高校的老师在考试后需要根据学生的考试成绩来分析试卷,以判断试卷是偏难、 偏容易还是适中。难度适中的试卷应该导致正态分布的成绩。为帮助老师完成试卷分析,我 们编写一个统计汇总程序,其功能是:老师输入考试分数(百分制),然后程序将分数换算 成等级制(分为 A、B、C、D、F 五等)并统计各等级分数的人数,最后画一个饼图来直观 地给出各等级人数的比例。 程序规格 输入:考试分数。 输出:以饼图表示的各分数段所占比例。 算法设计 本程序在算法上很简单,属于典型的 IPO(输入-处理-输出)模式。不过虽然算法很 简单,但是在绘制图形方面需要花费大量精力,因为绘图涉及精确的坐标、形状、颜色等细 节,还需要整个图形画面看上去整齐、匀称、美观。可以说,图形编程中大量时间都花在了 这类“美工”任务之上。 首先,由用户输入每个学生的分数(百分制)。然后根据该分数所对应的等级去累加各 等级人数变量。输入结束后,总人数和各等级人数就确定了。 其次,计算各分数等级人数占总人数的比例。 然后,根据比例绘制饼图。在 Tkinter 编程中,这需要先创建窗口和画布,然后利用画 布的 create_arc()方法绘制代表五个等级的五个扇形。扇形的角度反映了各分数等级的 比例,扇形具有不同填充色以相互区分。为了显示各扇形对应的等级,还需要绘制图例。 最后,用户通过饼图各扇形的大小只能看出各分数等级所占的大致比例。精确的比例值 当然可以固定显示在画面中,不过我们采用另一种更有趣的设计:当用户将鼠标指针移入某 个扇形中时,画布上就显示该扇形所代表的比例值。 以上步骤还需要进一步明确细节,最主要的就是窗口、画布的大小和各图形项的精确位 置等。通过用草图等手段做一些计算和试验,最终确定如图 5.18 所示的设计: ![](https://box.kancloud.cn/2016-02-22_56cafce18aa6f.png) 图 5.18 画布图形项设计 至此,可以写出本程序的算法伪代码。 ``` 算法: 用户输入考试分数 mark,并根据 mark 对应的等级累加各等级的人数 a、b、c、d、f; 创建窗口和大小为 300x200 的画布; 计算各分数等级的比例(a/n 等),并据此确定每个扇形的起止角度(sA、eA 等); 绘制各个扇形; 绘制图例; 为各扇形绑定“鼠标进入”事件,并定义事件处理函数(inPieA()等); 进入主事件循环。 ``` 代码实现 从上面的算法很容易翻译成 Python 代码。程序 5.2 中所用到的知识都在前面介绍过, 只有“鼠标进入”事件的处理需要说明一下。 当鼠标指针移到某个图形项上面时即发生事件"<Enter>",这时系统触发所绑定的事 件处理函数(如 inPieA),这些函数的功能是计算该图形项对应的比例值,然后显示在画 布上的指定位置。另外由于事件处理函数中需要引用画布对象和各图形项,所以我们将这些 函数的定义放在了 main()函数内部,以便它们能引用 main()中定义的变量,即 cv、 piepct、a、b、c、d、f 和 n。 【程序 5.2】piechart.py ``` from Tkinter import * def getMarks(): a,b,c,d,f = 0,0,0,0,0 mark = input("Enter a mark: ") while mark >= 0: if mark >= 90: a = a + 1 elif mark >= 80: b = b + 1 elif mark >= 70: c = c + 1 elif mark >= 60: d = d + 1 else: f = f + 1 mark = input("Enter a mark: ") return a,b,c,d,f def main(): a,b,c,d,f = getMarks() win = Tk() cv = Canvas(win,width=300,height=200,bg="white") cv.pack() n = a+b+c+d+f eA,sA = 360.0*a/n,0 eB,sB = 360.0*b/n,eA eC,sC = 360.0*c/n,eA+eB eD,sD = 360.0*d/n,eA+eB+eC eF,sF = 360.0*f/n,eA+eB+eC+eD bb = (90,40,210,160) pieA = cv.create_arc(bb,start=sA,extent=eA,fill="yellow") pieB = cv.create_arc(bb,start=sB,extent=eB,fill="green") pieC = cv.create_arc(bb,start=sC,extent=eC,fill="black") pieD = cv.create_arc(bb,start=sD,extent=eD,fill="gray") pieF = cv.create_arc(bb,start=sF,extent=eF,fill="red") cv.create_rectangle(240,40,260,50,fill="yellow") cv.create_rectangle(240,40+24,260,50+24,fill="green") cv.create_rectangle(240,40+48,260,50+48,fill="black") cv.create_rectangle(240,40+72,260,50+72,fill="gray") cv.create_rectangle(240,40+96,260,50+96,fill="red") cv.create_text(270,40,text="A",anchor=N) cv.create_text(270,40+24,text="B",anchor=N) cv.create_text(270,40+48,text="C",anchor=N) cv.create_text(270,40+72,text="D",anchor=N) cv.create_text(270,40+96,text="F",anchor=N) piepct = cv.create_text(40,100,text="") def inPieA(event): pct = "%5.1f%%" % (100.0*a/n) cv.itemconfig(piepct,text=pct) def inPieB(event): pct = "%5.1f%%" % (100.0*b/n) cv.itemconfig(piepct,text=pct) def inPieC(event): pct = "%5.1f%%" % (100.0*c/n) cv.itemconfig(piepct,text=pct) def inPieD(event): pct = "%5.1f%%" % (100.0*d/n) cv.itemconfig(piepct,text=pct) def inPieF(event): pct = "%5.1f%%" % (100.0*f/n) cv.itemconfig(piepct,text=pct) cv.tag_bind(pieA,"<Enter>",inPieA) cv.tag_bind(pieB,"<Enter>",inPieB) cv.tag_bind(pieC,"<Enter>",inPieC) cv.tag_bind(pieD,"<Enter>",inPieD) cv.tag_bind(pieF,"<Enter>",inPieF) win.mainloop() main() ``` 程序 5.2 的一次运行结果如图 5.19 所示。 ![](https://box.kancloud.cn/2016-02-22_56cafce19a1c1.png) 图 5.19 程序 5.2 的一次执行结果