昨天的某時某刻突發奇想,想用自己現階段所學的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!