欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

利用Python實(shí)現(xiàn)圖書超期提醒

 更新時間:2016年08月02日 15:49:06   投稿:daisy  
很多人喜歡逛圖書館,時不時去借本書,但每本書可能只可以借兩個月,一旦超期不還就會進(jìn)行相應(yīng)的處罰!為什么不寫個腳本來通知自己圖書超期呢?說了這么多廢話,我們就進(jìn)入主題吧?。?!

一、模擬登錄圖書館管理系統(tǒng)

我們可以先看一下登錄頁面(很多學(xué)校這些管理系統(tǒng)頁面就是很low):

兩種方式去模擬登錄圖書館:

1. 構(gòu)造登錄表單進(jìn)行模擬登錄

這種方式模擬登錄似乎是很可靠的,但有時候就是在驗(yàn)證碼獲取上很困難,如果簡單的網(wǎng)站,有的會利用當(dāng)前時間戳來構(gòu)造驗(yàn)證碼,這種就很容易從網(wǎng)頁上觀察出來,但比如我們這次要模擬登錄的網(wǎng)站似乎是不能這樣做,因?yàn)樗鞘褂肑avaScript標(biāo)準(zhǔn)庫里的Math函數(shù)直接隨機(jī)生成的驗(yàn)證碼鏈接,可以從下面圖片上觀察驗(yàn)證碼處的代碼:

它使用Math.random()函數(shù)返回 [0-1) 的浮點(diǎn)值偽隨機(jī)數(shù)(大于等于0,小于1)
好吧!我們換用一種比這個更簡單的方式模擬登錄吧!

2. 通過Cookie登錄圖書館

Cookie,指某些網(wǎng)站為了辨別用戶身份、進(jìn)行session跟蹤而儲存在用戶本地終端上的數(shù)據(jù)(通常經(jīng)過加密)。

這里我們使用Requests庫來進(jìn)行模擬登錄過程,在這之前我們還有個問題,怎么獲取Cookie呢??
如果你使用的是谷歌瀏覽器,那你可以通過按F12就可以看到下圖里面有個Cookie的內(nèi)容,這就是你要的東西:

再上個圖分析一下,希望大家能有耐心讀下去:

通過圖片我們知道可以獲取借閱日期和應(yīng)還日期,獲取日期后根據(jù)應(yīng)還日期和當(dāng)前日期比較,就可以得出是否超期的結(jié)果。不多說,先貼代碼再說:

import requests
session = requests.Session()  # 會話對象讓你能夠跨請求保持某些參數(shù),它也會在同一個Session實(shí)例發(fā)出的所有請求之間保持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, 如果我們需要跳轉(zhuǎn)到其它頁面,我們不用每次都模擬登錄,因?yàn)閏ookie已經(jīng)保存了我們的登錄狀態(tài)。

會不會有人疑問,不是要說模擬登錄的嗎??怎么沒有這過程呢??

其實(shí)我們上面代碼中的Cookie已經(jīng)保存了我們的登錄狀態(tài),相當(dāng)于我們已經(jīng)模擬登錄過了,這樣子模擬登錄是不是簡單多了,但缺點(diǎn)是我們需要手動在登錄頁面輸入一遍,然后再從登錄頁面找到cookie粘貼到代碼中來

二、獲取所借書籍信息

通過分析頁面,我們可以使用BeautifulSoup來提取我們需要的內(nèi)容,我們需要的是書籍的條形碼、題名和作者、借閱日期、應(yīng)還日期,其實(shí)我們只需要應(yīng)還日期就行,但為了以后需要,先獲取書籍的所有信息并保存進(jìn)數(shù)據(jù)庫里面:

 

定義了一個數(shù)據(jù)庫操作的函數(shù),方便以后調(diào)用

def get_mysql():
  conn = pymysql.connect(host = 'localhost', user = 'root', passwd = '2014081029', db = 'mysql', charset = 'utf8')  # user為數(shù)據(jù)庫的名字,passwd為數(shù)據(jù)庫的密碼,一般把要把字符集定義為utf8,不然存入數(shù)據(jù)庫容易遇到編碼問題
  cur = conn.cursor()  # 獲取操作游標(biāo)
  cur.execute('use book')  # 使用book這個數(shù)據(jù)庫
  return (cur, conn)

定義一個函數(shù)來獲取圖書信息并保存:

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 = []  # 書籍的條形碼列表,用來判斷要存入數(shù)據(jù)庫的書籍是否已經(jīng)存在

  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()) # 移除字符串頭尾指定的字符(默認(rèn)為空格)
    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(條形碼, 題名和作者, 借閱日期, 應(yīng)還日期, 續(xù)借量, 館藏地, 附件) 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項(xiàng)內(nèi)容,因此我們使用一個判斷語句:

if len(book_every) == 7:
  book_list.append(book_every)
  if book_every[0] not in book_title:
    sql = 'insert book_list(條形碼, 題名和作者, 借閱日期, 應(yīng)還日期, 續(xù)借量, 館藏地, 附件) 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()  # 如果存入數(shù)據(jù)庫失敗,執(zhí)行回滾操作
  book_every = []  

也就是說,如果判斷出book_every已經(jīng)達(dá)到7項(xiàng)內(nèi)容,就執(zhí)行存入數(shù)據(jù)庫的操作,然后在把book_every重置為空列表

三、發(fā)送郵件提醒功能

先貼上代碼:

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()) # 獲取當(dāng)前時間
  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('已經(jīng)超期了%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('已經(jīng)超期了%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('已經(jīng)超期了%d天' % extend_day)
          number += 1
          send_email(day, number, i[2])

      else:
        print('還沒有超期的書籍')

      print(pattern[2])
    time.sleep(3600 * 24)

我們來分析代碼吧,首先我們判斷是否超期是根據(jù)當(dāng)前時間和應(yīng)還日期的相加減得到的,所以我們考慮到:

    1.如果應(yīng)還日期是上個月,這里我們就要進(jìn)行月份的相加減,因?yàn)殚c年和平年的月份不一樣,所以我們定義了day_num和day_num1兩個列表來表示閏年和平年的月份天數(shù)。

    2.然后我們使用月份當(dāng)做判斷條件來比較超期天數(shù)

月份判斷,如果當(dāng)前月份等于應(yīng)還月份,就執(zhí)行下面操作,注意里面已經(jīng)包含發(fā)送郵件函數(shù),下面會貼出發(fā)送郵件函數(shù),大家也許會想,為什么沒有判斷年份,因?yàn)槲乙话憬钑粫谶@么久,所以沒有加上這個判斷

      if times[1] == pattern[1]:
        day = int(times[2]) - int(pattern[2])
        if day > 0:
          print('已經(jīng)超期了%d天' % day)
          number += 1
          send_email(day, number, i[2])

然后是當(dāng)前月份大于應(yīng)還月份時,這時候就有閏年和平年的判斷了

      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('已經(jīng)超期了%d天' % extend_day)
          number += 1
          send_email(day, number, i[2])

下面貼出發(fā)送郵件的代碼:

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》的書籍超期了' \
      ',而且已經(jīng)超期了%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()

最后把把發(fā)送郵件的截圖發(fā)出來:

 

以上就是利用Python實(shí)現(xiàn)圖書超期提醒的全部內(nèi)容,這個功能還是挺實(shí)用的,感興趣的小伙伴可以自己動手實(shí)踐起來。希望對大家學(xué)習(xí)Python有所幫助。

相關(guān)文章

  • 精確查找PHP WEBSHELL木馬的方法(1)

    精確查找PHP WEBSHELL木馬的方法(1)

    今天,我想了下,現(xiàn)在把查找PHP WEBSHELL木馬思路發(fā)出來,需要的朋友可以參考下。
    2011-04-04
  • Python使用pandas將表格數(shù)據(jù)進(jìn)行處理

    Python使用pandas將表格數(shù)據(jù)進(jìn)行處理

    這篇文章主要介紹了Python使用pandas將表格數(shù)據(jù)進(jìn)行處理,文章圍繞主題展開詳細(xì)的內(nèi)容介紹,具有一定的參考價值,感興趣的小伙伴可以參考一下
    2022-08-08
  • Python matplotlib可視化繪圖詳解

    Python matplotlib可視化繪圖詳解

    這篇文章主要介紹了Python matplotlib繪圖可視化知識點(diǎn)整理(小結(jié)),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2021-09-09
  • Python實(shí)現(xiàn)點(diǎn)陣字體讀取與轉(zhuǎn)換的方法

    Python實(shí)現(xiàn)點(diǎn)陣字體讀取與轉(zhuǎn)換的方法

    今天小編就為大家分享一篇Python實(shí)現(xiàn)點(diǎn)陣字體讀取與轉(zhuǎn)換的方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2019-01-01
  • python基礎(chǔ)教程之獲取本機(jī)ip數(shù)據(jù)包示例

    python基礎(chǔ)教程之獲取本機(jī)ip數(shù)據(jù)包示例

    本文主要介紹了python獲取本機(jī)ip數(shù)據(jù)包的示例,大家參考使用吧
    2014-02-02
  • python將處理好的圖像保存到指定目錄下的方法

    python將處理好的圖像保存到指定目錄下的方法

    今天小編就為大家分享一篇python將處理好的圖像保存到指定目錄下的方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2019-01-01
  • 詳解Python Celery和RabbitMQ實(shí)戰(zhàn)教程

    詳解Python Celery和RabbitMQ實(shí)戰(zhàn)教程

    這篇文章主要介紹了詳解Python Celery和RabbitMQ實(shí)戰(zhàn)教程,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-01-01
  • 如何使用python計算復(fù)雜三角函數(shù)

    如何使用python計算復(fù)雜三角函數(shù)

    當(dāng)涉及到計算復(fù)雜的三角函數(shù)時,Python 提供了強(qiáng)大的數(shù)學(xué)庫和函數(shù)來幫助我們進(jìn)行計算,在本篇博客中,我將介紹如何使用 Python 來計算復(fù)雜的三角函數(shù),需要的朋友可以參考下
    2023-08-08
  • Python使用Matplotlib繪制三維散點(diǎn)圖詳解流程

    Python使用Matplotlib繪制三維散點(diǎn)圖詳解流程

    matplotlib是基建立在python之上,適用于創(chuàng)建靜態(tài),動畫和交互式可視化,通常與數(shù)據(jù)分析模塊pandas搭配使用,用于數(shù)據(jù)的分析和展示,適用于主流的操作系統(tǒng),如Linux、Win、Mac
    2022-11-11
  • matplotlib bar()實(shí)現(xiàn)百分比堆積柱狀圖

    matplotlib bar()實(shí)現(xiàn)百分比堆積柱狀圖

    這篇文章主要介紹了matplotlib bar()實(shí)現(xiàn)百分比堆積柱狀圖,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-02-02

最新評論