Python 使用Jupyter爬取练习网站-仅供学习参考

  • 爬取简介
    • 一、环境配置
    • 二、网页分析
    • 三、网页爬取步骤
    • 四、通过小区URL爬取房源信息

爬取简介

选择广州二手房作为爬取对象,链接地址为某家网房源,作为期末作业写的,也作为第一篇在CSDN发布的文章。同时通过爬取时的问题总结,供大家参考。

一、环境配置

  1. Windows 10
  2. Python 3.8
  3. Jupyter notebook

二、网页分析

首先进入网站查看网页URL,找到其中的共同点,方便后面爬取多页信息。
可以发现爬取网页的顺序
第一页为:

https://gz.lianjia.com/ershoufang/

第二页为:

https://gz.lianjia.com/ershoufang/pg2/

不难发现,通过构造最后面的pg[i]的参数可以爬取每页信息,虽然显示了7万8千多套房源,不过这样只能爬取前100页信息,总共也就3000条房源信息。

后来发现可以通过先爬取小区网址url,再通过小区爬取房源信息,可以爬取到更多的房源信息。点击小区只会显示30个小区,需要自己点击返回全部小区列表

进去小区列表后的URL
可以看到首页为:

https://gz.lianjia.com/xiaoqu/?from=rec

第二页为:

https://gz.lianjia.com/xiaoqu/pg2/?from=rec

以此类推,可以看出格式为:

变量i为页码
https://gz.lianjia.com/xiaoqu/pg[i]/?from=rec

可以看出小区页爬取只能到第30页,但我们通过构造URL实测可以可以爬取100页,通过控制变量i即可。

三、网页爬取步骤

通过上述的了解,现在可以进入到爬取阶段了。首先需要爬取小区有多少套在售二手房,及该小区二手房的URL。实测可以通过点击套数来进去下一级页面,就可以看到小区的在售二手房源。

爬取方式,通过 右键【‘检查’】 该数字,可以查看到跳转页面的a标签下的href,然后通过爬取该标签下的信息来获取网址和套数。

这里有个问题,在我使用soup = BeatuifulSoup(data, “html.parser”),"html.parser"在爬取数据时会报错,换成”html5lib”就不会产生问题,可以正常爬取,不太清楚原理,希望看到帖子的大佬能帮忙解答一下。

爬取小区URL代码如下:

garden_id_list = []          #小区网址id列表
garden_num_list = []         #小区房源套数列表
ex_list = []                 #存储丢失页编号#爬取链家网小区,通过小区爬取房源
def xiaoqu(x):try:for a in range(x, 101):url_xiaoqu = "https://gz.lianjia.com/xiaoqu/pg" + str(a) + "/?from=rec"response = requests.get(url_xiaoqu, headers = headers)data = response.textsoup = BeautifulSoup(data, "html5lib")for i in range(1, 31):#获取小区房源详情页url,并爬取共有多少套在售二手房garden = soup.select('body > div.content > div.leftContent > ul > li:nth-child(' + str(i) + ') > div.xiaoquListItemRight > div.xiaoquListItemSellCount > a')garden_id = garden[0]['href'].replace('https://gz.lianjia.com/ershoufang/','')#将网址存进列表garden_id_list.append(garden_id)#         print(garden)#将搜索到的小区在售房源套数保存列表garden_num = garden[0].get_text().replace('套', '').strip()garden_num_list.append(garden_num)#         print(garden_num)print('第'+str(a)+'页爬取成功!')except Exception as e:print('第' +str(a)+ '页获取失败!正在重启服务获取中!')ex_list.append(a)time.sleep(3)return xiaoqu(a+1)#重构一个def获取丢失页面
def ex_xiaoqu(x,m):try:url_xiaoqu = "https://gz.lianjia.com/xiaoqu/pg" + str(x) + "/?from=rec"response = requests.get(url_xiaoqu, headers = headers)data = response.textsoup = BeautifulSoup(data, "html5lib")print('正在重新爬取第'+str(x)+'页,请耐心等待!')#重构可能爬不到30条信息,构造一个m=31尝试爬取,如果失败,则m-1,尝试爬取该页面所能爬取小区最大值。后续通过duplicates()函数去重。for i in range(1, m):#获取小区房源详情页url,并爬取共有多少套在售二手房garden = soup.select('body > div.content > div.leftContent > ul > li:nth-child(' + str(i) + ') > div.xiaoquListItemRight > div.xiaoquListItemSellCount > a')garden_id = garden[0]['href'].replace('https://gz.lianjia.com/ershoufang/','')#将网址最后的id存进列表garden_id_list.append(garden_id)
#             print(garden_id)#将搜索到的小区在售房源套数保存列表garden_num = garden[0].get_text().replace('套', '').strip()garden_num_list.append(garden_num)
#             print(garden_num)print('第'+str(x)+'页爬取成功!共爬取该页信息'+str(m-1)+'条!')#爬取到该页面后,删除该编号,防止重复输出。ex_list.remove(x)print('剩余未重新爬取列:', ex_list)return re_xq()except Exception as e:print('第' +str(x)+ '页重新获取失败!正在重启服务获取中!')time.sleep(3)m -= 1return ex_xiaoqu(x, m)#构造一个存储丢失页的列表重爬。
def re_xq():if len(ex_list) != 0:for z in ex_list:ex_xiaoqu(z, 31)else:print('全部数据爬取完毕!')xiaoqu(1)
re_xq()


将爬取的网页和数量转为DataFrame去重后并保存到csv中。

#将爬取到的信息转换为字典,并用DataFrame输出
dict_garden = {'id':garden_id_list,'num':garden_num_list}
d_garden = DataFrame(dict_garden)
d_garden

#删除重复列并重新查看信息
d_garden = d_garden.drop_duplicates()
d_garden.info()#将小区网址信息保存到csv中
import csv
d_garden.to_csv('小区网址.csv')

四、通过小区URL爬取房源信息

通过导入写入的广州二手房数据,去除没有在售二手房源的小区,然后对小区的房源信息进行爬取,获取楼盘名,楼盘地址,楼盘信息,楼盘总价,楼盘单价和区域。编写的程序进过好几轮修改,总算是达到一个还不错的效果,还算顽强,期间对代码进行了算法优化,勉勉强强爬取了6个小时才爬取到所有的数据。

比较有趣的部分在于区域爬取的优化及小区套数的算法编写,一页只能爬取30条信息,每个小区套数不同,所以需要编写小区套数的算法,以防程序报错,期间还发现比较有趣的现象,因为小区和房源是分开爬取的,期间间隔了一两天,有的房子被卖出了,所以编写程序的时候需要多一个循环,如果爬取的数据列表溢出,则减少一套房重新爬取。

希望大佬看到可以改进的地方可以教一下如何进一步优化算法及改进代码模块,增强爬取的稳定性及速度。

# 导入写入的广州二手房数据
df = pd.read_csv(r'./小区网址.csv')#查看小区网址中没有二手房源的网址,并剔除
df = df[df['num'] > 0]
df#删除多余列
df = df.drop('Unnamed: 0', axis = 1)
df

HouseName_list = []         #楼盘名列表
address_list = []           #楼盘地址列表
houseInfo_lise = []         #楼盘详细信息列表
totalPrice_list = []        #楼盘总价格列表
unitPrice_list = []         #楼盘单价(元/平方米)列表
region_list = []            #楼盘区域(区)# 获取区域,一个小区一个区域即可,可以减少重复爬取次数
def region_xq(x):url = 'https://gz.lianjia.com/ershoufang/pg1' + df['id'][x]response = requests.get(url, headers = headers)data = response.textsoup = BeautifulSoup(data, "html5lib")#获取区域#需要先进入详情页region_web = soup.select('#content > div.leftContent > ul > li:nth-child(1) > div.info.clear > div.title > a')#爬取详情页网页的<a href>region_web = region_web[0]['href']#创建一个url加载网页region_url = region_webres = requests.get(region_url, headers = headers)region_data = res.textregion_soup = BeautifulSoup(region_data, "html5lib")#爬取房源区域region = region_soup.select('body > div.overview > div.content > div.aroundInfo > div.areaName > span.info > a:nth-child(1)')region = region[0].get_text()
#     region_list.append(region)
#       print(region)return region#通过爬取每个id获取该小区房源
def house(x):try:for z in range(x, len(df)):     #如果数量大于每页所取的30个房源信息,需要查询需要多少页,最后一页有多少信息。if df['num'][z] > 30:if df['num'][z] % 30 == 0:page = df['num'][z] // 30else:page = df['num'][z] // 30 + 1else:page = 1print('第'+str(z+1)+'个小区!需要爬取信息'+str(df['num'][z])+'条!')#每个小区对应一个区域,所以可以提取出来,避免重复爬取。url = 'https://gz.lianjia.com/ershoufang/pg1' + df['id'][x]response = requests.get(url, headers = headers)data = response.textsoup = BeautifulSoup(data, "html5lib")#获取区域#需要先进入详情页region_web = soup.select('#content > div.leftContent > ul > li:nth-child(1) > div.info.clear > div.title > a')#爬取详情页网页的<a href>region_web = region_web[0]['href']#创建一个url加载网页region_url = region_webres = requests.get(region_url, headers = headers)region_data = res.textregion_soup = BeautifulSoup(region_data, "html5lib")#爬取房源区域region = region_soup.select('body > div.overview > div.content > div.aroundInfo > div.areaName > span.info > a:nth-child(1)')region = region[0].get_text()#     region_list.append(region)#       print(region)for q in range(0, page):url = 'https://gz.lianjia.com/ershoufang/pg' + str(q+1) + df['id'][z]response = requests.get(url, headers = headers)data = response.textsoup = BeautifulSoup(data, "html5lib")if df['num'][z] > 30:if df['num'][z] % 30 == 0:last_page_num = 30else:page_num = page - (q+1)if page_num == 0:last_page_num = df['num'][z] % 30else:last_page_num = 30else:last_page_num = df['num'][z]#                 print('第'+str(q+1)+'页!需要爬取信息'+str(last_page_num)+'条!')for i in range(0, last_page_num):#获取楼盘名信息HouseName = soup.select("#content > div.leftContent > ul > li:nth-child(" + str(i+1) + ") > div.info.clear > div.flood > div")HouseName = HouseName[0].get_text().replace(" ","").split('-')[0]HouseName_list.append(HouseName)
#                     print(HouseName)#获取楼盘地址address = soup.select("#content > div.leftContent > ul > li:nth-child(" + str(i+1) + ") > div.info.clear > div.flood > div")address = address[0].get_text().replace(" ","").split('-')[1]address_list.append(address)
#                     print(address)#获取楼盘具体信息houseInfo = soup.select("#content > div.leftContent > ul > li:nth-child(" + str(i+1) + ") > div.info.clear > div.address > div")houseInfo = houseInfo[0].get_text()houseInfo_lise.append(houseInfo)
#                     print(houseInfo)#获取楼盘总价格totalPrice = soup.select('#content > div.leftContent > ul > li:nth-child(' + str(i+1) +') > div.info.clear > div.priceInfo > div.totalPrice')totalPrice = totalPrice[0].get_text().replace('万','')totalPrice_list.append(totalPrice)
#                     print(totalPrice)#获取楼盘单价(元/平方米)unitPrice = soup.select('#content > div.leftContent > ul > li:nth-child(' + str(i+1) + ') > div.info.clear > div.priceInfo > div.unitPrice > span')unitPrice = unitPrice[0].get_text().replace('单价','').replace('元/平米','')unitPrice_list.append(unitPrice)
#                     print(unitPrice)#获取区域region_list.append(region)
#                     print(region)print('第'+str(z+1)+'个爬取完成!爬取信息'+str(df['num'][z])+'条!')except Exception as e:#本来可以不用try的,但是调试中发现爬一半的时候有的房子被卖了。。造成错误。#房源数量未更新,手动减一df.loc[z, 'num'] = df['num'][z] - 1print('小区房源有更新,需要重新爬取,正在重新爬取中...')if df['num'][z] == 0:print('网址被抓,已保存到第'+str(z)+'页!请更新页码重新爬取!')else:return house(z)
#         print(e)


共爬取了6w多条数据,对数据进行去重转换后保存进csv。

#将爬取到的信息转换为字典,并用DataFrame输出
from pandas import DataFrame
dict_house = {'HouseName':HouseName_list,'Address':address_list,'houseInfo':houseInfo_lise,'totalPrice':totalPrice_list,'unitPrice':unitPrice_list,'region':region_list}
d_house = DataFrame(dict_house)
d_house#删除重复列并重新查看信息
d_house = d_house.drop_duplicates()
d_house.info()#将房源信息保存到csv中
import csv
d_house.to_csv('广州二手房源信息.csv')


以上就是对于广州某家二手房的爬取操作。对于对数据处理过两天有空再继续编写,希望对需要学习这方面的朋友有所帮助。爬取时间为6月13日。希望大佬们看到可以帮忙看看哪里的不足和需要优化的点。

Python爬虫-使用Jupyter爬虫相关推荐

  1. python游戏辅助lol_Python爬虫实战,60行代码爬取英雄联盟全英雄全皮肤,找寻曾今那些被删除的绝版皮肤...

    学了一周多的爬虫课后终于按捺不住了,小编决定自己手动编写爬虫程序,刚好LJ在鼓励学员分享成果,优秀作品有奖励,就把自己用Python编程爬取各大游戏高清壁纸的过程整理了出来进行投稿,与大家一起分享. ...

  2. Python学习三: 爬虫高级技巧 与 模拟实战练习

    三大爬虫技巧 许多网站针对爬虫的访问都设置了一定的障碍,通过这三步技巧,轻松绕过部分的反爬虫限制. (1)设置程序休止时间 import time import random# 休止睡眠 1 秒 这里 ...

  3. python爬虫scrapy框架教程_Python爬虫教程-30-Scrapy 爬虫框架介绍

    从本篇开始学习 Scrapy 爬虫框架 Python爬虫教程-30-Scrapy 爬虫框架介绍 框架:框架就是对于相同的相似的部分,代码做到不出错,而我们就可以将注意力放到我们自己的部分了 常见爬虫框 ...

  4. 大佬带你详解Python反爬虫措施以及爬虫编写注意事项

    Python爬虫开发:反爬虫措施以及爬虫编写注意事项 反爬虫的几重措施 1.IP限制 如果是个人编写的爬虫,IP可能是固定的,那么发现某个IP请求过于频繁并且短时间内访问大量的页面,有爬虫的嫌疑,作为 ...

  5. 2个月精通Python爬虫——3大爬虫框架+6场实战+分布式爬虫,包教包会

    2019独角兽企业重金招聘Python工程师标准>>> 阿里云大学在线工作坊上线,原理精讲+实操演练,让你真正掌握云计算.大数据技能. 在第一批上线的课程中,有一个Python爬虫的 ...

  6. python爬虫小说代码示例-中文编程,用python编写小说网站爬虫

    原标题:中文编程,用python编写小说网站爬虫 作者:乘风龙王 原文:https://zhuanlan.zhihu.com/p/51309019 为保持源码格式, 转载时使用了截图. 原文中的源码块 ...

  7. python多久能学会爬虫-上海多久可以学会python

    上海多久可以学会python 来源:教育联展网 编辑:粉色de皮卡丘 发布时间:2018-12-29 上海Python培训班 千锋教育打造人工智能Python工程师 快速咨询 上海Python培训 上 ...

  8. python初学者web还是爬虫-还在纠结学爬虫还是数据分析,不如看看这篇文章

    原标题:还在纠结学爬虫还是数据分析,不如看看这篇文章 身为职场人,收集上万条表格数据做商业分析,裁剪上千张图片,发送数百封邮件...这些都是经常会遇到的场景.我一直期待能有个工具解放我,直到我遇到了P ...

  9. python爬虫简单实例-Python 利用Python编写简单网络爬虫实例3

    利用Python编写简单网络爬虫实例3 by:授客 QQ:1033553122 实验环境 python版本:3.3.5(2.7下报错 实验目的 获取目标网站"http://bbs.51tes ...

最新文章

  1. 印度孟买机器人餐厅_印度孟买自动化展:该国最重要、规模最大的国际自动化展...
  2. 企业文件存储服务器规划,企业文件存储服务器
  3. 高学历就一定代表着高收入么?数据分析来为你揭晓
  4. php更改html内容,请问你们怎么将html的文件的内容改变为php
  5. codeforces round div2,3周赛补题计划(从开学到期末)
  6. 赢者通吃自编码器(WTA-AE)
  7. centos7 安装openocd
  8. 蓝桥杯2015年第六届C/C++省赛A组第九题-垒骰子
  9. Echarts美国地图
  10. 关于HTTP GZIP解压问题
  11. Python常用模块大全(总结)
  12. html跳转按钮谷歌浏览器点击没反应,网页打印的按钮无效,点击打印没有任何反映!求解!...
  13. pytorch drop_last参数
  14. 【最新版】贝塔智能挪车v2.5.2+前端-已测试
  15. 刚开始参加工作的45条建议
  16. 照片如何转换成pdf?手机电脑都可以轻松转换
  17. python ui框架哪个最好用_Python UI开发最常用到的库
  18. 与大数据相关热门岗位有什么?
  19. 雷军:把UCWEB做成像GOOGLE一样伟大
  20. 浅聊WebRTC视频通话

热门文章

  1. 基于微信美食小程序系统设计与实现 开题报告
  2. Yahoo! 中国搜索使用说明(转)
  3. uploadifive上传,无需flash插件案列
  4. ios3怎么取消长按弹出菜单_iPhone一直跳出通知好烦?教你如何关闭/管理iOS通知...
  5. AppleScript学习笔记(一)初识AppleScript
  6. 增值税发票识别(调研ing)
  7. 不要随意使用批量加QQ好友软件,不然你一定会后悔的!
  8. JAVA中用流进行文件分割
  9. 废除闰秒真的有必要吗?不废除会有哪些影响?
  10. 提高国内VSCode下载速度,包含deb等格式