爬取簡(jiǎn)書(shū)全站文章并生成 API(二)

簡(jiǎn)書(shū)

第一節(jié)已經(jīng)介紹了簡(jiǎn)書(shū)網(wǎng)站的結(jié)構(gòu)青瀑,爬取文章前對(duì)網(wǎng)頁(yè)源碼進(jìn)行必要的分析,以及整個(gè)項(xiàng)目的步驟萧诫,這一節(jié)開(kāi)始介紹如何爬取簡(jiǎn)書(shū)分類(lèi)目錄下的文章斥难,如有不明白的,請(qǐng)務(wù)必看完前一節(jié)的介紹:

爬取“新上榜”目錄下的文章


說(shuō)明: 所有代碼都在 python2.7 環(huán)境下運(yùn)行帘饶。

首先創(chuàng)建一個(gè) Django 項(xiàng)目:

# django-admin startproject jianshu_api

# cd jianshu_api

# django-admin startapp jianshu

在 jianshu_api/settings.py 下配置數(shù)據(jù)庫(kù):

DATABASES = {
    'default': {
        'ENGINE': 'django.db.backends.mysql',
        'NAME': 'jianshu',
        'HOST': '127.0.0.1',
        'USER': 'root',
        'PASSWORD': 'tianfeiyu',
        'PORT': 3306
    }
}
開(kāi)始寫(xiě) models:

此 API 的設(shè)計(jì)是模仿知乎日?qǐng)?bào) API 的形式哑诊,models 分兩層,第一層是概要信息及刻,第二層是詳細(xì)內(nèi)容镀裤,以概要信息作為外鍵竞阐。

class ArticleList(models.Model):
    """
        文章概要信息
    """
    article_id = models.CharField('ID', primary_key=True, max_length=100)
    article_title = models.CharField('文章標(biāo)題', max_length=100)
    article_url = models.URLField('文章URL')
    article_user = models.CharField('作者', max_length=100)
    article_user_url = models.URLField('作者URL')
    created = models.DateTimeField(auto_now_add=True)

    class Meta:
        ordering = ['-created', ]

    def __unicode__(self):
        return self.article_title
    def __str__(self):
        return self.article_title    

class ArticleDetail(models.Model):
    """
        文章詳細(xì)信息
    """
    image = models.URLField('圖片URL' )
    title = models.CharField('文章標(biāo)題', max_length=100)
    body = models.TextField('文章內(nèi)容', null=True)
    time = models.CharField('發(fā)表時(shí)間' , max_length=100, null=True)
    views_count = models.CharField('閱讀數(shù)', max_length=100)
    public_comments_count = models.CharField('評(píng)論數(shù)', max_length=100)
    likes_count = models.CharField('喜歡', max_length=100)
    total_rewards_count = models.CharField('打賞', max_length=100)
    created = models.DateTimeField('創(chuàng)建時(shí)間', auto_now_add=True)
    article_abstract = models.ForeignKey('ArticleList', verbose_name='文章摘要')
     
    class Meta:
        ordering = ['-created', ]
    
    def __unicode__(self):
        return self.title
    def __str__(self):
        return self.title

然后進(jìn)入 MySQL 中,設(shè)置使用的編碼:

root@127.0.0.1 : (none) 09:41:33> set character_set_client=utf8 ;
root@127.0.0.1 : (none) 09:41:33> set character_set_connection=utf8 ;
root@127.0.0.1 : (none) 09:41:33> set character_set_database=utf8 ;
root@127.0.0.1 : (none) 09:41:33> set character_set_results=utf8 ;
root@127.0.0.1 : (none) 09:41:33> set character_set_server=utf8 ;
root@127.0.0.1 : (none) 09:41:33> set character_set_system=utf8 ;

最后驗(yàn)證下是否正確:

設(shè)定 mysql 字符集

開(kāi)始創(chuàng)建數(shù)據(jù)庫(kù):

# python manage.py makemigrations    
# python manage.py migrate
爬取“新上榜”的文章:

初始化代碼:

if __name__ == '__main__':
    
    # 從 jianshu_api/settings.py 文件中導(dǎo)入數(shù)據(jù)庫(kù)的信息
    host = DATABASES['default']['HOST']
    user = DATABASES['default']['USER']
    passwd = DATABASES['default']['PASSWORD']
    db = DATABASES['default']['NAME']
    port = DATABASES['default']['PORT']

    # 保存文章所用到的表
    article_list_table = 'jianshu_articlelist' 
    article_detail_table = 'jianshu_articledetail'
    
    # 使用 colorama 模塊進(jìn)行顏色控制暑劝,通過(guò)使用 autoreset 參數(shù)可以讓變色效果只對(duì)當(dāng)前輸出起作用骆莹,輸出完成后顏色恢復(fù)默認(rèn)設(shè)置
    init(autoreset=True)
    domain_name = 'http://www.reibang.com'
    
    # “新上榜”目錄的 URL
    base_url = 'http://www.reibang.com/recommendations/notes'

    # 初始化一個(gè) mysql 實(shí)例    
    mysql = Mysql(host, user, passwd, db, port)
    
    # 解析 dom,獲取文章的詳細(xì)信息
    get_details(mysql, base_url, domain_name, article_list_table, article_detail_table)

爬取文章詳細(xì)信息代碼:

def get_details(mysql, base_url, domain_name, article_list_table, article_detail_table):
    """
        爬取文章并獲取詳細(xì)信息
    """
    
    # OrderedDict() 是一個(gè)有序的字典担猛,將文章的信息保存在 dict 中
    article_list = OrderedDict()
    article_detail = OrderedDict()
    html = requests.get(base_url).content  

    # html 是網(wǎng)頁(yè)的源碼幕垦,soup 是獲得一個(gè)文檔的對(duì)象
    soup = BeautifulSoup(html, 'html.parser', from_encoding='utf-8')
    tags = soup.find_all('li', class_="have-img")
    print Fore.YELLOW + "---------------all-------------------:",len(tags)
    ct = 1
    
    # 獲取文章的信息
    for tag in tags:
        image = tag.img['src'].split('?')[0]
        article_user = tag.p.a.get_text()
        
        # 將 URL 補(bǔ)全
        article_user_url = tag.p.a['href']
        if article_user_url.startswith('/users/'):
            article_user_url = domain_name + article_user_url
        created = tag.p.span['data-shared-at']
        article_title = tag.h4.get_text(strip=True)
        article_title = article_title.replace('"', '\\"')
        article_url = tag.h4.a['href']
        article_id = article_url.split('/')[2]
        if article_url.startswith('/p/'):
            article_url = domain_name + article_url
        tag_a = tag.div.div.find_all('a')
        views = tag_a[0].get_text(strip=True)
        
        # 提取其中的數(shù)字 
        views = filter(str.isdigit, str(views))
        
        comments = tag_a[1].get_text(strip=True)
        comments = filter(str.isdigit, str(comments))    
        tag_span = tag.div.div.find_all('span')
        likes = tag_span[0].get_text(strip=True)
        likes = filter(str.isdigit, str(likes))

        # 閱讀,評(píng)論傅联,喜歡一定存在先改,打賞不一定有
        try:
            tip = tag_span[1].get_text()
            tip = filter(str.isdigit, str(tip))
        except Exception as e:
            tip = 0

獲取文章的內(nèi)容:

def get_body(article_url):
    """
        獲取文章內(nèi)容
    """
    # 解析文章對(duì)應(yīng)的 URL
    html = requests.get(article_url).content
    soup = BeautifulSoup(html, 'html.parser', from_encoding='utf-8')
    tags = soup('div', class_="show-content")
    body = str(tags[0])

    # 將 body 中 " 轉(zhuǎn)換為 \", ' 轉(zhuǎn)換為 \'
    body = body.replace('"','\\"')
    body = body.replace("'","\\'")
    return body

將數(shù)據(jù)保存在 mysql 中:

def insert_data(self, table, my_dict):
        try:    
            # 從 dict 中分別取出 key,value 
            cols = ','.join(my_dict.keys())
            values = '","'.join(my_dict.values())
            values = '"' + values + '"'
            try:
                sql = "insert into %s (%s) values(%s)" %(table, cols, values)
                result = self.cur.execute(sql) 
                self.db.commit()
                if result:
                    return 1
                else:
                    return 0 
            except MySQLdb.Error as e:
                self.db.rollback()
                if "key 'PRIMARY'" in e.args[1]:
                     print Fore.RED + self.get_current_time(), "數(shù)據(jù)已存在蒸走,未插入數(shù)據(jù)"
                else:
                    print Fore.RED + self.get_current_time(), "插入數(shù)據(jù)失敗仇奶,原因 %d: %s" % (e.args[0], e.args[1])
        except MySQLdb.Error as e:
            print Fore.RED + self.get_current_time(), "數(shù)據(jù)庫(kù)錯(cuò)誤,原因%d: %s" % (e.args[0], e.args[1])

代碼運(yùn)行結(jié)果:

運(yùn)行結(jié)果

上一節(jié)已經(jīng)提到過(guò)比驻,“熱門(mén)”目錄和“新上榜”目錄下代碼的結(jié)構(gòu)相同猜嘱,所以以上代碼稍作修改完全可以爬取“熱門(mén)”下的文章,但我將“熱門(mén)”下爬取到的文章放在了一張表里嫁艇,大家可以自行嘗試朗伶。

完整代碼請(qǐng)查看項(xiàng)目中的 popular_articles_jianshu.py 文件!

下一節(jié)將介紹 API 的生成。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末步咪,一起剝皮案震驚了整個(gè)濱河市论皆,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌猾漫,老刑警劉巖点晴,帶你破解...
    沈念sama閱讀 211,290評(píng)論 6 491
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異悯周,居然都是意外死亡粒督,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,107評(píng)論 2 385
  • 文/潘曉璐 我一進(jìn)店門(mén)禽翼,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)屠橄,“玉大人,你說(shuō)我怎么就攤上這事闰挡∪袂剑” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 156,872評(píng)論 0 347
  • 文/不壞的土叔 我叫張陵长酗,是天一觀的道長(zhǎng)溪北。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么之拨? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,415評(píng)論 1 283
  • 正文 為了忘掉前任茉继,我火速辦了婚禮,結(jié)果婚禮上蚀乔,老公的妹妹穿的比我還像新娘馒疹。我一直安慰自己,他們只是感情好乙墙,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,453評(píng)論 6 385
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著生均,像睡著了一般听想。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上马胧,一...
    開(kāi)封第一講書(shū)人閱讀 49,784評(píng)論 1 290
  • 那天汉买,我揣著相機(jī)與錄音,去河邊找鬼佩脊。 笑死蛙粘,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的威彰。 我是一名探鬼主播出牧,決...
    沈念sama閱讀 38,927評(píng)論 3 406
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼歇盼!你這毒婦竟也來(lái)了舔痕?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 37,691評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤豹缀,失蹤者是張志新(化名)和其女友劉穎伯复,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體邢笙,經(jīng)...
    沈念sama閱讀 44,137評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡啸如,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,472評(píng)論 2 326
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了氮惯。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片叮雳。...
    茶點(diǎn)故事閱讀 38,622評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖妇汗,靈堂內(nèi)的尸體忽然破棺而出债鸡,到底是詐尸還是另有隱情,我是刑警寧澤铛纬,帶...
    沈念sama閱讀 34,289評(píng)論 4 329
  • 正文 年R本政府宣布厌均,位于F島的核電站,受9級(jí)特大地震影響告唆,放射性物質(zhì)發(fā)生泄漏棺弊。R本人自食惡果不足惜晶密,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,887評(píng)論 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望模她。 院中可真熱鬧稻艰,春花似錦、人聲如沸侈净。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,741評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)畜侦。三九已至元扔,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間旋膳,已是汗流浹背澎语。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,977評(píng)論 1 265
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留验懊,地道東北人擅羞。 一個(gè)月前我還...
    沈念sama閱讀 46,316評(píng)論 2 360
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像义图,于是被迫代替她去往敵國(guó)和親减俏。 傳聞我的和親對(duì)象是個(gè)殘疾皇子硫惕,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,490評(píng)論 2 348

推薦閱讀更多精彩內(nèi)容