一、模擬登錄圖書館管理系統
我們可以先看一下登錄頁面(很多學校這些管理系統頁面就是很low):
兩種方式去模擬登錄圖書館:
1. 構造登錄表單進行模擬登錄
這種方式模擬登錄似乎是很可靠的,但有時候就是在驗證碼獲取上很困難,如果簡單的網站,有的會利用當前時間戳來構造驗證碼,這種就很容易從網頁上觀察出來,但比如我們這次要模擬登錄的網站似乎是不能這樣做,因為它是使用JavaScript標準庫里的Math函數直接隨機生成的驗證碼鏈接,可以從下面圖片上觀察驗證碼處的代碼:
它使用
Math.random()
函數返回 [0-1) 的浮點值偽隨機數(大于等于0,小于1)
好吧!我們換用一種比這個更簡單的方式模擬登錄吧!
2. 通過Cookie登錄圖書館
Cookie,指某些網站為了辨別用戶身份、進行session跟蹤而儲存在用戶本地終端上的數據(通常經過加密)。
這里我們使用
Requests
庫來進行模擬登錄過程,在這之前我們還有個問題,怎么獲取Cookie呢??
如果你使用的是谷歌瀏覽器,那你可以通過按F12就可以看到下圖里面有個Cookie的內容,這就是你要的東西:
再上個圖分析一下,希望大家能有耐心讀下去:
通過圖片我們知道可以獲取借閱日期和應還日期,獲取日期后根據應還日期和當前日期比較,就可以得出是否超期的結果。不多說,先貼代碼再說:
import requests session = requests.Session() # 會話對象讓你能夠跨請求保持某些參數,它也會在同一個Session實例發出的所有請求之間保持cookie session.headers = { 'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/51.0.2704.84 Safari/537.36', 'Cookie': 'ASP.NET_SessionId=1qri0rmoylpyrs45rurzme55; Hm_lvt_ed06d5e5f94d85932b82e4aac94d0c68=1467535679,1469713840; Hm_lpvt_ed06d5e5f94d85932b82e4aac94d0c68=1469713840; PHPSESSID=ev339udv0rrhqg6tfdvfukqos1' }
上述代碼使用了
requests
的會話對象來保存Cookie, 如果我們需要跳轉到其它頁面,我們不用每次都模擬登錄,因為cookie已經保存了我們的登錄狀態。
會不會有人疑問,不是要說模擬登錄的嗎??怎么沒有這過程呢??
其實我們上面代碼中的Cookie已經保存了我們的登錄狀態,相當于我們已經模擬登錄過了,這樣子模擬登錄是不是簡單多了,但缺點是我們需要手動在登錄頁面輸入一遍,然后再從登錄頁面找到cookie粘貼到代碼中來
二、獲取所借書籍信息
通過分析頁面,我們可以使用
BeautifulSoup
來提取我們需要的內容,我們需要的是書籍的條形碼、題名和作者、借閱日期、應還日期,其實我們只需要應還日期就行,但為了以后需要,先獲取書籍的所有信息并保存進數據庫里面:
定義了一個數據庫操作的函數 ,方便以后調用
def get_mysql(): conn = pymysql.connect(host = 'localhost', user = 'root', passwd = '2014081029', db = 'mysql', charset = 'utf8') # user為數據庫的名字,passwd為數據庫的密碼,一般把要把字符集定義為utf8,不然存入數據庫容易遇到編碼問題 cur = conn.cursor() # 獲取操作游標 cur.execute('use book') # 使用book這個數據庫 return (cur, conn)
定義一個函數來獲取圖書信息并保存:
def get_book_name(book_url): html = session.get(book_url, cookies = cookie, headers = headers).content.decode('utf-8') soup = BeautifulSoup(html, 'lxml') book_bar = [] # 書籍的條形碼列表,用來判斷要存入數據庫的書籍是否已經存在 cur, conn = get_mysql() sql = 'select * from book_list;' cur.execute(sql) rows = cur.fetchall() for row in rows: book_bar.append(row[1]) book_list = [] # 這個是我測試時使用的,作用是把每本書籍的信息列表放在這個列表中 book_every = [] # 一本書籍的所有信息列表 for book_time in soup.find_all('td', class_="whitetext"): print(book_time.get_text().strip()) # 移除字符串頭尾指定的字符(默認為空格) pattern = re.compile(r'\s') content = re.sub(pattern, r'', book_time.get_text()) # 目的也是匹配任何空白符并去除,貌似對空行去除沒影響 if content != '': book_every.append(content) if len(book_every) == 7: book_list.append(book_every) if book_every[0] not in book_bar: sql = 'insert book_list(條形碼, 題名和作者, 借閱日期, 應還日期, 續借量, 館藏地, 附件) value(' + "\'" \ + book_every[0] + "\'," + "\'" + book_every[1] + "\'," + "\'" + book_every[2] + "\'," + "\'" \ + book_every[3] + "\'," + "\'" + book_every[4] + "\'," + "\'" + book_every[5] + "\'," + "\'" \ + book_every[6] + "\'" + ');' try: cur.execute(sql) conn.commit() except: conn.rollback() book_every = [] print(book_list)
接下來我們分析一下上面代碼中沒有注釋的代碼,首先我們先把處理后的信息加入
book_every
列表中,然后從頁面源代碼(tp9.png)中我們可以知道,一本書信息中只需要前面7項內容,因此我們使用一個判斷語句:
if len(book_every) == 7: book_list.append(book_every) if book_every[0] not in book_title: sql = 'insert book_list(條形碼, 題名和作者, 借閱日期, 應還日期, 續借量, 館藏地, 附件) value(' + "\'" \ + book_every[0] + "\'," + "\'" + book_every[1] + "\'," + "\'" + book_every[2] + "\'," + "\'" \ + book_every[3] + "\'," + "\'" + book_every[4] + "\'," + "\'" + book_every[5] + "\'," + "\'" \ + book_every[6] + "\'" + ');' try: cur.execute(sql) conn.commit() except: conn.rollback() # 如果存入數據庫失敗,執行回滾操作 book_every = []
也就是說,如果判斷出
book_every
已經達到7項內容,就執行存入數據庫的操作,然后在把
book_every
重置為空列表
三、發送郵件提醒功能
先貼上代碼:
def send_message(): day_num = [31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31] day_num1 = [31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31] sql = 'select * from book_list;' cur, conn = get_mysql() cur.execute(sql) rows = cur.fetchall() local_time = time.strftime("%Y-%m-%d", time.localtime()) # 獲取當前時間 local_time = str(local_time) times = re.split(r'-', local_time) year = times[0] number = 0 while(True): for i in rows: print(i[4]) pattern = re.split(r'-', i[4]) if times[1] == pattern[1]: day = int(times[2]) - int(pattern[2]) if day > 0: print('已經超期了%d天' % day) number += 1 send_email(day, number, i[2]) elif times[1] > pattern[1]: if (year % 4 == 0 and year % 100 != 0) or year % 400 == 0: extend_day = day_num1[int(pattern[1]) - 1] - int(pattern[2]) + times[2] print('已經超期了%d天' % extend_day) number += 1 send_email(day, number, i[2]) else: extend_day = day_num[int(pattern[1]) - 1] - int(pattern[2]) + times[2] print('已經超期了%d天' % extend_day) number += 1 send_email(day, number, i[2]) else: print('還沒有超期的書籍') print(pattern[2]) time.sleep(3600 * 24)
我們來分析代碼吧,首先我們判斷是否超期是根據當前時間和應還日期的相加減得到的,所以我們考慮到:
??? 1.如果應還日期是上個月,這里我們就要進行月份的相加減,因為閏年和平年的月份不一樣,所以我們定義了day_num和day_num1兩個列表來表示閏年和平年的月份天數。
??? 2.然后我們使用月份當做判斷條件來比較超期天數
月份判斷,如果當前月份等于應還月份,就執行下面操作,注意里面已經包含發送郵件函數,下面會貼出發送郵件函數,大家也許會想,為什么沒有判斷年份,因為我一般借書不會超期這么久,所以沒有加上這個判斷
if times[1] == pattern[1]: day = int(times[2]) - int(pattern[2]) if day > 0: print('已經超期了%d天' % day) number += 1 send_email(day, number, i[2])
然后是當前月份大于應還月份時,這時候就有閏年和平年的判斷了
elif times[1] > pattern[1]: if (year % 4 == 0 and year % 100 != 0) or year % 400 == 0: extend_day = day_num1[int(pattern[1]) - 1] - int(pattern[2]) + times[2] print('已經超期了%d天' % extend_day) number += 1 send_email(day, number, i[2])
下面貼出發送郵件的代碼:
def send_email(day, number, title): from_addr = '15602200534@163.com' password = '就不告訴你' to_addr = '673411814@qq.com' smtp_server = 'smtp.163.com' text = 'Hello ,郭偉匡, 告訴你一個不好的消息,趕緊帶上你的書,去圖書館交錢吧!你有一本叫《%s》的書籍超期了' \ ',而且已經超期了%d天了,總共有%d書超期了!!!' % (title, day, number) msg = MIMEText(text, 'plain', 'utf-8') msg['From'] = format_addr('圖書館的通知<%s>' % from_addr) msg['To'] = format_addr('管理員<%s>' % to_addr) msg['Subject'] = Header('來著郭偉匡的問候......', 'utf-8').encode() server = smtplib.SMTP(smtp_server, 25) server.set_debuglevel(1) server.login(from_addr, password) server.sendmail(from_addr, [to_addr], msg.as_string()) server.quit()
最后把把發送郵件的截圖發出來:
以上就是利用Python實現圖書超期提醒的全部內容,這個功能還是挺實用的,感興趣的小伙伴可以自己動手實踐起來。希望對大家學習Python有所幫助。
更多文章、技術交流、商務合作、聯系博主
微信掃碼或搜索:z360901061

微信掃一掃加我為好友
QQ號聯系: 360901061
您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對您有幫助就好】元
