国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專(zhuān)欄INFORMATION COLUMN

Python 實(shí)現(xiàn)圖書(shū)超期提醒小幫手(修改版)

susheng / 1003人閱讀

摘要:前期工作準(zhǔn)備得差不多了,開(kāi)始找這個(gè)的提交部分的內(nèi)容了,我們從登錄頁(yè)面應(yīng)該也可以知道我們需要提交學(xué)號(hào)密碼驗(yàn)證碼這三個(gè)。差點(diǎn)忘了把發(fā)送郵件的截圖發(fā)出來(lái)

一、實(shí)現(xiàn)目的

本來(lái)就很喜歡逛圖書(shū)館,時(shí)不時(shí)去借本書(shū)(注:借的都沒(méi)看過(guò)),但我這個(gè)學(xué)期突然發(fā)現(xiàn)了問(wèn)題,每本書(shū)都可以借兩個(gè)月,但不幸的是我最近一學(xué)期借的書(shū)全部超期,一天一毛錢(qián),我心疼這錢(qián)啊!!!靈機(jī)一動(dòng),為什么不寫(xiě)個(gè)腳本來(lái)通知自己圖書(shū)超期呢?說(shuō)了這么多廢話(huà),我們就進(jìn)入主題吧!!!

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

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

兩種方式去模擬登錄圖書(shū)館:

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

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

日了個(gè)狗,它使用Math.random()函數(shù)返回 [0-1) 的浮點(diǎn)值偽隨機(jī)數(shù)(大于等于0,小于1),剛開(kāi)始我以為python的math.random()函數(shù)生成的隨機(jī)數(shù)和JavaScript的有區(qū)別,后來(lái)試了一下,呵呵,原來(lái)兩個(gè)函數(shù)生成的隨機(jī)數(shù)都是[0-1)而且都是16位小數(shù)點(diǎn)的。那樣子我們就可以模擬登錄了。
首先,在模擬登錄先,我們應(yīng)該在瀏覽器上模擬登錄一次,觀(guān)察頁(yè)面變化情況,剛開(kāi)始時(shí)頁(yè)面只有l(wèi)ogin.php頁(yè)面的:

然后我們輸入驗(yàn)證碼后再觀(guān)察一下,頁(yè)面立刻被轉(zhuǎn)向redr_info.php,同時(shí)還有redr_verify.php頁(yè)面出現(xiàn)

然后看看我們的redr_info.php里面的東西,唉,怎么這個(gè)頁(yè)面是GET請(qǐng)求呢??

那驗(yàn)證登錄請(qǐng)求的POST頁(yè)面在哪里去了呢??帶著疑問(wèn)看看redr_verify.php,光是看這個(gè)頁(yè)面的命名就覺(jué)得這是個(gè)驗(yàn)證登錄的頁(yè)面:

果然,POST請(qǐng)求在這里,那我們就可以構(gòu)造登錄表單通過(guò)這個(gè)頁(yè)面來(lái)模擬登錄了。
前期工作準(zhǔn)備得差不多了,開(kāi)始找這個(gè)redr_verify.php的post提交部分的內(nèi)容了,我們從登錄頁(yè)面應(yīng)該也可以知道我們需要提交學(xué)號(hào)、密碼、驗(yàn)證碼這三個(gè)。我們可以去redr_verify.php下看看我們POST表單提交的數(shù)據(jù)

我們只需要填寫(xiě)前面四項(xiàng)就可以了,第四項(xiàng)是什么呢,我回到登錄頁(yè)面看了一下,就是下面圖片的選擇,

然后貼代碼吧

import subprocess
import sys
import os

session = requests.Session()
session.headers = {
    "User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/52.0.2743.82 Safari/537.36"
}


def login(name, password):
    random_num = random.random()  # 生成隨機(jī)數(shù),構(gòu)造獲取驗(yàn)證碼的鏈接
    url = "http://210.38.102.131:86/reader/captcha.php?" + str(random_num)

    get_captcha = session.get(url).content
    with open("captcha.png", "wb") as f:
        f.write(get_captcha)
        f.close()

    """
        這段代碼是為了方便我們打開(kāi)圖片,它可以直接打開(kāi)圖片
        我們就不用去文件夾里去找,里面是判斷使用什么系統(tǒng),
        不同系統(tǒng)打開(kāi)方式有點(diǎn)差異,可以找python文檔了解這部分內(nèi)容
    """
    if sys.platform.find("darwin") >= 0:
        subprocess.call(["open", "captcha.png"])
    elif sys.platform.find("linux") >= 0:
        subprocess.call(["xdg-open", "captcha.png"])
    else:
        os.startfile("captcha.png")

    input_captcha = input("請(qǐng)輸入驗(yàn)證碼:")
    input_captcha = str(input_captcha)
    
    # 構(gòu)造登錄表單,里面就是我們上面提及的四項(xiàng)
    post_data = {
        "number": name,
        "passwd": password,
        "captcha": input_captcha,
        "select": "cert_no"
    }

    login_url = "http://210.38.102.131:86/reader/redr_verify.php"
    html = session.post(login_url, data=post_data).content

    book_hist_url = "http://210.38.102.131:86/reader/book_hist.php"
    content = session.get(book_hist_url).content.decode("utf-8")
    print(content)

這就模擬登錄成功了,
好吧!我們換用一種比這個(gè)更簡(jiǎn)單的方式模擬登錄吧!

2. 通過(guò)Cookie登錄圖書(shū)館

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

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

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

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

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

會(huì)不會(huì)有人疑問(wèn),不是要說(shuō)模擬登錄的嗎??怎么沒(méi)有這過(guò)程呢??

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

三、獲取所借書(shū)籍信息

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

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

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

定義一個(gè)函數(shù)來(lái)獲取圖書(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ū)籍的條形碼列表,用來(lái)判斷要存入數(shù)據(jù)庫(kù)的書(shū)籍是否已經(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 = []    # 這個(gè)是我測(cè)試時(shí)使用的,作用是把每本書(shū)籍的信息列表放在這個(gè)列表中
    book_every = []  # 一本書(shū)籍的所有信息列表
    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())  # 目的也是匹配任何空白符并去除,貌似對(duì)空行去除沒(méi)影響

        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)

接下來(lái)我們分析一下上面代碼中沒(méi)有注釋的代碼,首先我們先把處理后的信息加入book_every列表中,然后從頁(yè)面源代碼(tp9.png)中我們可以知道,一本書(shū)信息中只需要前面7項(xiàng)內(nèi)容,因此我們使用一個(gè)判斷語(yǔ)句:

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ù)庫(kù)失敗,執(zhí)行回滾操作
    book_every = []   

也就是說(shuō),如果判斷出book_every已經(jīng)達(dá)到7項(xiàng)內(nèi)容,就執(zhí)行存入數(shù)據(jù)庫(kù)的操作,然后在把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)前時(shí)間
    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("還沒(méi)有超期的書(shū)籍")

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

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

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

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

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

            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)還月份時(shí),這時(shí)候就有閏年和平年的判斷了

            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 ,郭偉匡, 告訴你一個(gè)不好的消息,趕緊帶上你的書(shū),去圖書(shū)館交錢(qián)吧!你有一本叫《%s》的書(shū)籍超期了" 
           ",而且已經(jīng)超期了%d天了,總共有%d書(shū)超期了!!!" % (title, day, number)
    msg = MIMEText(text, "plain", "utf-8")
    msg["From"] = format_addr("圖書(shū)館的通知<%s>" % from_addr)
    msg["To"] = format_addr("管理員<%s>" % to_addr)
    msg["Subject"] = Header("來(lái)著郭偉匡的問(wèn)候......", "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()
方案二

其實(shí)我們還有一種最簡(jiǎn)單的判斷相差日期的方法,
那就是使用python提供的datetime模塊,就利用方案一里面的東西來(lái)說(shuō)吧

    local_time = time.strftime("%Y-%m-%d", time.localtime())  # 獲取當(dāng)前時(shí)間
    local_time = str(local_time)
    times = re.split(r"-", local_time)

我們通過(guò)split分離出年月日后,就可以很簡(jiǎn)單得使用datetime進(jìn)行日期相加減了,我們datetime相加減日期的用法如下:

    d1 = datetime.datetime(2016, 4, 3)
    d2 = datetime.datetime(2016, 6, 23)
    print((d2 - d1).days)

打印結(jié)果就是相差天數(shù),這樣在判斷日期方面就變得十分簡(jiǎn)單了,所以方案二顯然比方案一好得多了

關(guān)于發(fā)送郵件的知識(shí)。。。我靠,0:22了,還沒(méi)洗澡呢,下次有空再補(bǔ)上這部分知識(shí),還是貼出廖雪峰網(wǎng)站關(guān)于這方面的知識(shí)吧 廖雪峰網(wǎng)站關(guān)于SMTP發(fā)送郵件。

差點(diǎn)忘了把發(fā)送郵件的截圖發(fā)出來(lái):

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://specialneedsforspecialkids.com/yun/38089.html

相關(guān)文章

  • 白看過(guò)來(lái) 讓Python爬蟲(chóng)成為你的好幫手

    摘要:小白看過(guò)來(lái)讓爬蟲(chóng)成為你的好幫手隨著信息化社會(huì)的到來(lái),人們對(duì)網(wǎng)絡(luò)爬蟲(chóng)這個(gè)詞已經(jīng)不再陌生。互動(dòng)活動(dòng)關(guān)于華為云微認(rèn)證的任何問(wèn)題,均可在下方評(píng)論區(qū)留言。華為云微認(rèn)證每期將送出個(gè)免費(fèi)機(jī)會(huì),獎(jiǎng)項(xiàng)公布時(shí)間月日。 小白看過(guò)來(lái) 讓Python爬蟲(chóng)成為你的好幫手 隨著信息化社會(huì)的到來(lái),人們對(duì)網(wǎng)絡(luò)爬蟲(chóng)這個(gè)詞已經(jīng)不再陌生。但什么是爬蟲(chóng),如何利用爬蟲(chóng)為自己服務(wù),這聽(tīng)起來(lái)有些高大上。下面一文帶你走近爬蟲(chóng)世界,讓即...

    darcrand 評(píng)論0 收藏0
  • Python和FFmpeg強(qiáng)強(qiáng)聯(lián)合

    摘要:核心子進(jìn)程運(yùn)行控制。由應(yīng)用來(lái)看,關(guān)鍵是錄制屏幕和錄制攝像頭,以及用快捷鍵控制在這兩者之間切換。限制條件是超過(guò)三個(gè)月快捷鍵失效。實(shí)現(xiàn)分兩步安裝時(shí)在注冊(cè)表特定位置,假如是,寫(xiě)入目錄相關(guān)信息。在程序運(yùn)行時(shí),檢測(cè)當(dāng)前目錄是否存在于注冊(cè)表下。 錄制項(xiàng)目終于做完,不用總是提醒自己抓緊時(shí)間這樣來(lái)想問(wèn)題了。在完成之后帶著一些滿(mǎn)足感,回頭看看哪些地方是需要改進(jìn)的,哪些地方又是有更好的替代方案,自己又有哪...

    ThreeWords 評(píng)論0 收藏0
  • 報(bào)告!7至8月中旬項(xiàng)目總結(jié)!

    摘要:閱讀本文約分鐘序章月至月中旬一直在忙公司新項(xiàng)目,這也是我第一次做技術(shù)領(lǐng)隊(duì)的項(xiàng)目,從面試開(kāi)始就一直在閱讀有關(guān)技術(shù)團(tuán)隊(duì)管理有關(guān)的書(shū)籍,本文將簡(jiǎn)述此項(xiàng)目的總結(jié),從設(shè)計(jì)到編碼實(shí)現(xiàn)到上線(xiàn)測(cè)試用戶(hù)反饋等方面,篇幅略長(zhǎng),建議收藏。 閱讀本文約5.8分鐘 序章 7月至8月中旬一直在忙公司新項(xiàng)目,這也是我第一次做技術(shù)領(lǐng)隊(duì)的項(xiàng)目,從面試開(kāi)始就一直在閱讀有關(guān)技術(shù)團(tuán)隊(duì)管理有關(guān)的書(shū)籍,本文將簡(jiǎn)述此項(xiàng)目的總結(jié),...

    YPHP 評(píng)論0 收藏0
  • 【備戰(zhàn)春招/秋招系列】Java程序員必備書(shū)單

    摘要:相關(guān)推薦,豆瓣評(píng)分,人評(píng)價(jià)本書(shū)介紹了在編程中條極具實(shí)用價(jià)值的經(jīng)驗(yàn)規(guī)則,這些經(jīng)驗(yàn)規(guī)則涵蓋了大多數(shù)開(kāi)發(fā)人員每天所面臨的問(wèn)題的解決方案。實(shí)戰(zhàn)高并發(fā)程序設(shè)計(jì)推薦豆瓣評(píng)分,書(shū)的質(zhì)量沒(méi)的說(shuō),推薦大家好好看一下。 該文已加入開(kāi)源文檔:JavaGuide(一份涵蓋大部分Java程序員所需要掌握的核心知識(shí))。地址:https://github.com/Snailclimb... 【強(qiáng)烈推薦!非廣告!】...

    saucxs 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<