doclist 阅读(11) 评论(0)

昨天的某时某刻突发奇想,想用自己现阶段所学的python知识来制作一个小程序。大致功能为:开机时像某安全软件一样,弹出窗口,窗口上能提示你,“距xxxx年xx月xx日还有多少天”。

        在开始前,因为涉及到日期的计算(如标题所说),首先我想到了标准库中的datetime模块,该模块有个today()函数能返回今天的日期(格式:xxxx-xx-xx),将函数返回的日期转换成字符串后,日期就成了一个长度为10的字符串,对字符串进行切片后分别转换为整型数,就可得到year,month,day参数,具体代码如下

 1 from datetime import date
 2 
 3 
 4 def get_date():
 5     today_date = date.today()
 6     str_t_d = str(today_date)
 7     year = int(str_t_d[:4])
 8     month = int(str_t_d[5:7])
 9     day = int(str_t_d[-2:])
10     return year, month, day

调用get_date()后会返回元组(year, month, day)

下一步开始计算天数差了,因为考虑到对于每一年来说,每个月份天数除了2月以外都是固定的,2月天数的变化关系受到年份影响,因此需要判断闰年或平年,代码如下

 1 def is29days(year=get_date()[0]):
 2     months = [31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31]
 3     if year % 100 == 0:
 4         if year % 400 == 0:
 5             months[1] = 29
 6             return months
 7         else:
 8             return months
 9     elif year % 4 == 0:
10         months[1] = 29
11         return months
12     else:
13         return months

在上述代码中,将一年的十二个月对应天数按顺序放在列表months中,这个列表中二月对应的天数为28天,即默认为平年,用 if-elif-else语句判断是否为闰年,是就改变months[1]的值为29,返回新的months列表,否返回默认列表,在调用is29days()时将返回月份列表

         有了月份列表,就可也计算了,假定,给定一个日期:3月6日,今天的日期:get_date()[1] 月get_date()[2](今天日期为2月12日,没错我是今天写的)日,利用两个日期的月份对months列表切片,结合列表索引值将其与月份对应,如果当月为2月,则对应的months列表索引值为2-1=1!!!,切片为:months[2-1 : 3],这样列表中就包含了2月,3月两个月,用sum计算切片后的列表元素和后,减去两头的天数后就可以得到3月6日与2月12日之间相差的天数了,代码如下

 1 def count_days(num_month=3, d_day=6):
 2     if num_month < get_date()[1] and d_day < get_date()[2]:
 3         '''在上一月的情况'''
 4         msg = 'error'
 5         month = 'error'
 6         year = 'error'
 7         x_day = 'error'
 8         return msg, month, year, x_day
 9     elif num_month == get_date()[1] and d_day < get_date()[2]:
10         '''在当月前些天的情况'''
11         msg = 'error'
12         month = 'error'
13         year = 'error'
14         x_day = 'error'
15         return msg, month, year, x_day
16     else:
17         months = is29days()[get_date()[1]-1:num_month]
18         x_day = d_day
19         year = get_date()[0]
20         month = num_month
21         days = sum(months) - get_date()[2] - months[-1] + x_day
22         msg = '%3s天' % days
23         return msg, month, year, x_day

考虑到会有选择的日期在当日日期之前的情况,加入判断语句,调用count_days()后返回元组(msg,  month,  year,  x_day),(若不传递实参 ,则默认实参为num_month=3, d_day=6)

现在就完成了天数差的计算,但开头所讲我们还需一个能弹出窗口(GUI界面),鉴于我仅仅接触过 tkinter 所以我使用了它,代码如下

 1 deadline = Tk()
 2 deadline.title('DeadLine')
 4 deadline.geometry('220x300+1678+732')
 6 label = Label(deadline, text='距%4s年%2s月%2s日还有' % (count_days()[2], count_days()[1], count_days()[3])
 7 label.grid(row=0, columnspan=2)
 8 label_1 = Label(deadline, justify='left', text="""
 9 Don`t waste time any more !
10
11 " So do you want to take a
12 leap of faith,or become an
13 old man,filled with regret
14 waiting to die alone? "
15 """)
16 label_1.grid(row=2, columnspan=2)
17 display = Listbox(deadline, font=100, width=5, height=1)
18 display.insert(END, count_days()[0])
19 display.grid(row=1, columnspan=2)
22 button_2 = Button(deadline, text='Exit', command=deadline.destroy)
23 button_2.grid(columnspan=2)
24 deadline.mainloop()

效果如下

 

好像差不多了啊

        然后开始将其封装成一个.exe的文件,在这里使用第三方库PyInstaller,因为不是python自带的需要自己安装,安装完成后,在控制台输入命令pyinstaller -F -w filename.py,这里需要注意如果不是在filename.py文件所在的目录下打开的控制台,是需要给定路径的,同时F 大写,w小写,如果不加上-w选项,在运行程序时会弹出控制器界面,同时py文件名不能用汉字,否则会报错,完成后在当前目录下,会有一个dist文件件,打开会发现一个filename.exe文件,大功告成!!

还没完,我想每天打开电脑都能看到上图的GUI界面,就像某安全软件在屏幕右下角弹出界面一样(好像安全软件都弹),上网查了一下,找到一个方法,如下

https://jingyan.baidu.com/article/54b6b9c0d103662d593b474e.html

完整的代码如下:

 1 #!/usr/bin/env python3
 2 # coding=utf-8
 3 # Date: 2019/2/12
 4 
 5 from tkinter import *
 6 from datetime import date
 7 
 8 
 9 def get_date():
10     today_date = date.today()
11     str_t_d = str(today_date)
12     year = int(str_t_d[:4])
13     month = int(str_t_d[5:7])
14     day = int(str_t_d[-2:])
15     return year, month, day
16 
17 
18 def is29days(year=get_date()[0]):
19     months = [31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31]
20     if year % 100 == 0:
21         if year % 400 == 0:
22             months[1] = 29
23             return months
24         else:
25             return months
26     elif year % 4 == 0:
27         months[1] = 29
28         return months
29     else:
30         return months
31 
32 
33 def count_days(num_month=3, d_day=6):
34     if num_month < get_date()[1] and d_day < get_date()[2]:
35         '''在上一月的情况'''
36         msg = 'error'
37         month = 'error'
38         year = 'error'
39         x_day = 'error'
40         return msg, month, year, x_day
41     elif num_month == get_date()[1] and d_day < get_date()[2]:
42         '''在当月前些天的情况'''
43         msg = 'error'
44         month = 'error'
45         year = 'error'
46         x_day = 'error'
47         return msg, month, year, x_day
48     else:
49         months = is29days()[get_date()[1]-1:num_month]
50         x_day = d_day
51         year = get_date()[0]
52         month = num_month
53         days = sum(months) - get_date()[2] - months[-1] + x_day
54         msg = '%3s天' % days
55         return msg, month, year, x_day
56 
57 
58 deadline = Tk()
59 deadline.title('DeadLine')
60 deadline.geometry('220x300+1678+732')
61 label = Label(deadline, text=
62 '距%4s年%2s月%2s日还有' % (count_days()[2], count_days()[1], count_days()[3]))
63 label.grid(row=0, columnspan=2)
64 label_1 = Label(deadline, justify='left', text="""
65 Don`t waste time any more !
66 
67 " So do you want to take a
68 leap of faith,or become an
69 old man,filled with regret
70 waiting to die alone? "
71 """)
72 label_1.grid(row=2, columnspan=2)
73 display = Listbox(deadline, font=100, width=5, height=1)
74 display.insert(END, count_days()[0])
75 display.grid(row=1, columnspan=2)
76 button_2 = Button(deadline, text='Exit', command=deadline.destroy)
77 button_2.grid(columnspan=2)
78 deadline.mainloop()

 

写在最后:

  上述代码只是实现了最简单的情况,我想以此为基础制作一个能计算任意两个日期天数差的程序,一步一步来,day day up,Come on ,code newbie!