700字范文,内容丰富有趣,生活中的好帮手!
700字范文 > python层级抓取_070.Python聚焦爬虫数据解析

python层级抓取_070.Python聚焦爬虫数据解析

时间:2020-06-19 18:40:19

相关推荐

python层级抓取_070.Python聚焦爬虫数据解析

一 聚焦爬虫数据解析

1.1 基本介绍

聚焦爬虫的编码流程

指定url

基于requests模块发起请求

获取响应对象中的数据

数据解析

进行持久化存储

如何实现数据解析

三种数据解析方式

正则表达式

bs4

xpath

数据解析的原理

- 进行标签定位

- 获取定位好的标签里面的文本数据和属性值

1.2爬取一个网站的图片

importrequests#1. 指定url

url = "/ugc//09/01/5d6be8e4396c4.gif"headers={"User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/73.0.3683.75 Safari/537.36"}#2. 基于requests模块发起请求#3. 获取响应对象中的数据

page_content = requests.get(url=url, headers=headers).content#4. 进行持久化存储

with open('./火狐突击.gif', 'wb') as f:

f.write(page_content)

执行

就有图片文件保存

上面的功能可以使用urlib实现

#引入urllib的request的模块

from urllib importrequest#指定url

url = "/ugc//09/01/5d6be8e4396c4.gif"

#需要两个参数,一个是url.另一个是保存的文件名

request.urlretrieve(url=url,filename="./火狐突击_urllib.gif")

执行后出现

('./火狐突击_urllib.gif', )

查看目录

使用urllib模块下载图片,代码量比较少

有一个缺点,如果有些网站有UA检测机制的话,此方法就不适用

1.3 聚焦爬虫

爬取不得姐的所有图片

其中一张图片信息如下

提取上面内容,使用正则匹配出元数据部分

html ="""

"""

#ex = '

.*?'#匹配的内容:

"""

"""

#ex = '

.*?data-original="" title=".*?'#匹配的内容:

"""第一个.*?

第二个.*? 抱得动你吗?" alt="我抱得动你吗?">

"""

importre

# re.S表示单行匹配,可以将字符串中的多行拼接起来,进行统一匹配,如果不加,则会一行一行地进行匹配,最后得到的结果是[]

ex= '

.*?data-original="(.*?)" title=".*?'#获取的是一个列表后加上[0]就可以得到值

image_url=re.findall(ex,html,re.S)print(image_url[0])

执行

/ugc//04/04/5e887902d0f98.gif

完整代码

获取所有图片代码如下

#爬取不得姐网站上面的所有图片

importosimportreimportrequestsfrom urllib importrequest#1. 指定url

url = "/pic/"headers={"User-Agent": "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/80.0.3987.163 Safari/537.36"}

ex= '

.*?data-original="(.*?)" title=".*?'

#判断是否有images文件存在,如果不存在,就创建一个新的,用来存储图片,使用OS模块

if not os.path.exists('./images'):

os.mkdir('./images')#2. 基于requests模块发起请求#3. 获取响应对象中的数据

page_text = requests.get(url=url, headers=headers).text#4. 数据解析

img_url_list =re.findall(ex, page_text, re.S)#5. 进行持久化存储

for img_url inimg_url_list:#获取到图片的url之后,可以对这个如来进行切割,得到图片的名称,并进行存储,同时由于还有一层目录,就可以吧images拼接到字符串中

img_name = './images/' + img_url.split('/')[-1]#获取到图片的url之后,接下来进行图片的存储

request.urlretrieve(url=img_url, filename=img_name)

执行,就可以得到images目录,同时存有下载的图片

二 bs4数据解析

2.1 基本使用

环境安装:

它是一个解析器,如果使用bs4进行数据解析,需要依靠lxml

root@darren-virtual-machine:~# pip3 install bs4

root@darren-virtual-machine:~# pip3 install lxml

导入bs4模块

from bs4 import BeautifulSoup

BeautifulSoup解析原理:

实例化一个BeautifulSoup对象,并将需要解析的页面源码数据加载到对象中

使用该对象的相关属性和方法进行数据解析或提取

如何实例化一个BeautifulSoup对象

本地加载

f = open()

soup = BeautifulSoup(f, "lxml")

网络加载

soup = BeautifulSoup(page_text, "lxml")

创建一个本地的测试html文件

root@darren-virtual-machine:~# cd /root/Python爬虫/

root@darren-virtual-machine:~/Python爬虫# vim test_page.html

测试bs4

百里守约

李清照

王安石

苏轼

柳宗元

this is span宋朝是最强大的王朝,不是军队的强大,而是经济很强大,国民都很有钱

总为浮云能蔽日,长安不见使人愁

清明时节雨纷纷,路上行人欲断魂,借问酒家何处有,牧童遥指杏花村秦时明月汉时关,万里长征人未还,但使龙城飞将在,不教胡马度阴山岐王宅里寻常见,崔九堂前几度闻,正是江南好风景,落花时节又逢君杜甫杜牧杜小月度蜜月凤凰台上凤凰游,凤去台空江自流,吴宫花草埋幽径,晋代衣冠成古丘

实例化一个对象

from bs4 importBeautifulSoup

f= open('./test_page.html', 'r', encoding='utf-8')

soup= BeautifulSoup(f, 'lxml')#这里直接使用,就是打印的意思

soup

执行后,就会输出html的所有信息

BeautifulSoup对象相关的属性和方法

soup.tagName 获取页面源码中遇到的第一个标签

取出一个P标签

soup.p

执行

取属性: 返回的永远是一个单数

soup.tagName.attrs 取得标签里面所有的属性(返回的是一个字典)

soup.tagName.attrs[属性名] 取标签的单个属性

soup.tagName[属性名] 取标签的单个属性

取其中一个

soup.a.attrs["title"] #也可以直接soup.a["title"]

取文本:

soup.string: 只可以获取直系标签中的文本内容

soup.text: 获取标签下的所有文本内容

soup.get_text(): 获取标签下的所有文本内容

print(soup.p.string)print(soup.p.text)print(soup.p.get_text())

执行

find() 返回的永远是一个单数

find(tagName): 通过标签名进行数据解析

soup.find('a')

执行

this is span宋朝是最强大的王朝,不是军队的强大,而是经济很强大,国民都很有钱

find(tagName, 标签属性): 通过标签属性进行标签定位

soup.find('div',class_ = "song")

执行

在使用class时,必须使用class_,后要加下划线,否则会有如下错误

string text 和get_text()区别

find_all() 返回的永远是一个列表

找到所有符合要求的标签

soup.find_all("a")

执行

[

this is span宋朝是最强大的王朝,不是军队的强大,而是经济很强大,国民都很有钱,总为浮云能蔽日,长安不见使人愁,清明时节雨纷纷,路上行人欲断魂,借问酒家何处有,牧童遥指杏花村,秦时明月汉时关,万里长征人未还,但使龙城飞将在,不教胡马度阴山,岐王宅里寻常见,崔九堂前几度闻,正是江南好风景,落花时节又逢君,杜甫,杜牧,凤凰台上凤凰游,凤去台空江自流,吴宫花草埋幽径,晋代衣冠成古丘]

也可以传入列表

soup.find_all(["a","p"])

就找到所有的a和p的标签

select 选择器 返回的永远是一个列表

标签选择器, 类选择器, ID选择器, 层级选择器

层级选择器:

单层级: soup.select(".tang > ul > li")

[

清明时节雨纷纷,路上行人欲断魂,借问酒家何处有,牧童遥指杏花村,秦时明月汉时关,万里长征人未还,但使龙城飞将在,不教胡马度阴山,岐王宅里寻常见,崔九堂前几度闻,正是江南好风景,落花时节又逢君,杜甫,杜牧,杜小月,度蜜月,凤凰台上凤凰游,凤去台空江自流,吴宫花草埋幽径,晋代衣冠成古丘]

多层级: soup.select(".tang li")

[

清明时节雨纷纷,路上行人欲断魂,借问酒家何处有,牧童遥指杏花村,秦时明月汉时关,万里长征人未还,但使龙城飞将在,不教胡马度阴山,岐王宅里寻常见,崔九堂前几度闻,正是江南好风景,落花时节又逢君,杜甫,杜牧,杜小月,度蜜月,凤凰台上凤凰游,凤去台空江自流,吴宫花草埋幽径,晋代衣冠成古丘]

2.2 实际案例

使用bs4实现将诗词名句网站中三国演义小说的每一章的内容爬取到本地磁盘进行存储

importrequestsfrom bs4 importBeautifulSoup#1.指定URL(顺带加上UA伪装)

url = "/book/sanguoyanyi.html"headers={"User-Agent": "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/80.0.3987.163 Safari/537.36"}#2.使用requests发起请求

response = requests.get(url=url, headers=headers)#3.获取响应数据#获取到的是标题页的页面源码数据

page_text =response.text#4.数据解析#实例化一个BeautifulSoup对象

soup = BeautifulSoup(page_text, 'lxml')#使用该对象的相关属性和方法进行数据解析和文本的提取,下面就是文章的标题

li_list = soup.select('.book-mulu li')

f= open('./sanguo.txt', 'w', encoding='utf-8')#循环比标题

for li inli_list:#取出标题的文本

name =li.a.text#把取到的a标签和域名拼接

detail_url = "" + li.a["href"]#获取到的是详情页的页面源码数据

detail_page_text = requests.get(url=detail_url, headers=headers).text#这是因为前面获取了源码,还要重新实例化一次,我们需要重新实例化一个详情页的BeautifulSoup对象,再进行数据解析

detail_soup = BeautifulSoup(detail_page_text, "lxml")

content= detail_soup.find('div', class_="chapter_content").get_text()#5.持久化存储

f.write(name + ":" + content + "\n")

f.close()

执行后查看

查看爬取内容

三xpath解析

3.1 基本介绍

通用性比较强

环境的安装: pip install lxml

解析原理:

实例化一个etree对象,且将解析的页面源码加载到该对象中

使用该对象中的xpath方法结合着xpath表达式进行标签定位和数据解析提取

etree对象的实例化:

本地加载: tree = etree.parse("filePath")

网络加载: tree = etree.HTML(page_text)

3.2 基本使用

from lxml importetree

tree= etree.parse('./test_page.html')

tree

tree方式打印出的是一个对象

常用的xpath表达式: 基于标签的层级实现定位,返回的永远是一个列表

/: 从标签开始实现层级定位

//: 从任意位置实现标签的定位

tree.xpath('//div')

执行

属性定位: tag[@attrName="attrValue"]

tree.xpath('//div[@class="song"]')

执行

索引定位: //div[@class="tang"]/ul/li[5] 注意索引值是从1开始

tree.xpath('//div[@class="song"]/img')

执行

获取的是列表

tree.xpath('//div[@class="song"]/img/@src')

取文本:

取直系文本内容: /text()

取所有文本内容: //text()

取属性: /@attrName

#列表直接使用[]就可以提取文本

tree.xpath('//div[@class="song"]/img/@src')[0]

取class="tang"下面的杜小月文本

#需求1: 取class="tang"下面的杜小月文本

tree.xpath('//div[@class="tang"]/ul/li[6]/b/text()')[0]

执行 这里的索引是从1开始的

取"总为浮云能"这一段话

#需求2: 取"总为浮云能"这一段话

tree.xpath('//div[@class="song"]/a[2]/text()')[0]

执行结果

# 需求3: 取“”这个域名

#需求3: 取“”这个域名

tree.xpath('//a[@id="feng"]/@href')[0

执行

3.3 实例1:爬取58二手房上面的房源信息

(标题,价格,和概况)

新获取三个数据的标题 详情页url 和价格

importrequestsfrom lxml importetree#这个url是58的主页信息

url = "/ershoufang/?PGTID=0d100000-0000-43a0-55c0-fbb7aead05c1&ClickID=2"headers={"User-Agent": "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/80.0.3987.163 Safari/537.36"}#向二手房列表页发送请求,获取页面源码数据

page_text = requests.get(url=url, headers=headers).text#使用xpath进行数据解析,这里是网络加载,需要使用HTML

tree =etree.HTML(page_text)

li_list= tree.xpath('//ul[@class="house-list-wrap"]/li')

all_data_list=list()for li in li_list[0:3]:#获取的标题

title = li.xpath('./div[2]/h2/a/text()')[0]#获取详情页url,可以根据这个url请求,获取

detail_url = li.xpath('./div[2]/h2/a/@href')[0]#获取价格

price = li.xpath('./div[3]//text()')print(title)print(detail_url)print(price)

执行结果

详情页的概况定位

代码如下

importrequestsfrom lxml importetree#这个url是58的主页信息

url = "/ershoufang/?PGTID=0d100000-0000-43a0-55c0-fbb7aead05c1&ClickID=2"headers={"User-Agent": "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/80.0.3987.163 Safari/537.36"}#向二手房列表页发送请求,获取页面源码数据

page_text = requests.get(url=url, headers=headers).text#使用xpath进行数据解析,这里是网络加载,需要使用HTML

tree =etree.HTML(page_text)

li_list= tree.xpath('//ul[@class="house-list-wrap"]/li')

all_data_list=list()for li in li_list[0:2]:#获取的标题

title = li.xpath('./div[2]/h2/a/text()')[0]#获取详情页url,可以根据这个url请求,获取

detail_url = li.xpath('./div[2]/h2/a/@href')[0]#获取价格

price = li.xpath('./div[3]//text()')

detail_page_text= requests.get(url=detail_url, headers=headers).text

detail_tree=etree.HTML(detail_page_text)

detail_content= detail_tree.xpath('//div[@class="general-item-wrap"]/ul/li/span//text()')#把数据封装到一个字典里面

dic ={"title": title,"price": price,"detail_content": detail_content

}#把字典的数据放入列表

all_data_list.append(dic)print(all_data_list)

打印结果

[{'title': '光明7栋小区花园|带精装修分期1年免息|长圳地铁\xa0','price': ['\n', '126', '万', '\n', '18261元/㎡', '\n'],'detail_content': ['房屋总价', '\n 龒驋鸺万(单价龒龥驋鸺龒元/㎡)\n', '房屋户型', '2室2厅1卫', '房本面积', '69.0㎡', '房屋朝向', '南', '一手房源', '否', '所在楼层', '低层(共25层)', '装修情况', '精装修', '\n', '\n', '找装修\n', '\n', '产权年限', '70年产权', '建筑年代', '', '房屋总价', '\n 龒驋鸺万(单价龒龥驋鸺龒元/㎡)\n', '\n', '帮你凑钱\n', '\n', '房屋类型', '普通住宅', '交易权属', '经济适用房', '是否唯一', '否', '参考首付', '\n 37.8万(月供3617元/月)\n']

}, {'title': '满二唯一 红本在手 看房方便 周边配套丰富\xa0','price': ['\n', '300', '万', '\n', '33926元/㎡', '\n'],'detail_content': ['房屋总价', '\n 麣龤龤万(单价麣麣齤驋鸺元/㎡)\n', '房屋户型', '3室2厅1卫', '房本面积', '88.43㎡', '房屋朝向', '南北', '一手房源', '否', '所在楼层', '中层(共32层)', '装修情况', '精装修', '\n', '\n', '找装修\n', '\n', '产权年限', '70年产权', '建筑年代', '', '房屋总价', '\n 麣龤龤万(单价麣麣齤驋鸺元/㎡)\n', '\n', '帮你凑钱\n', '\n', '房屋类型', '普通住宅', '交易权属', '商品房', '是否唯一', '是', '参考首付', '\n 90.0万(月供8611元/月)\n

3.4 案例二:解析出所有城市名称

importrequestsfrom lxml importetree

url= "/historydata/"headers={"User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/73.0.3683.75 Safari/537.36"}#使用requests模块发送请求获取页面源码数据

page_text = requests.get(url=url, headers=headers).text

tree=etree.HTML(page_text)#使用管道符号,可以取到热门城市和城市信息

city_list = tree.xpath('//div[@class="bottom"]/ul/li/a/text() | //div[@class="bottom"]/ul/div[2]/li/a/text()')print(city_list)

执行

['北京', '上海', '广州', '深圳', '杭州', '天津', '成都', '南京', '西安', '武汉', '阿坝州', '安康', '阿克苏地区', '阿里地区', '阿拉善盟', '阿勒泰地区', '安庆', '安顺', '鞍山', '克孜勒苏州', '安阳', '蚌埠', '白城', '保定', '北海', '宝鸡', '北京', '毕节', '博州', '百色', '白沙', '白山', '保山', '保亭', '包头', '本溪', '白银', '巴彦淖尔', '滨州', '巴中', '亳州', '长春', '承德', '成都', '常德', '昌都', '赤峰', '昌江', '昌吉州', '五家渠', '澄迈', '重庆', '常熟', '长沙', '楚雄州', '朝阳', '滁州', '郴州', '潮州', '常州', '长治', '崇左', '沧州', '池州', '定安', '丹东', '东方', '东莞', '德宏州', '大连', '大理州', '大庆', '大同', '定西', '大兴安岭地区', '黔南州', '德阳', '东营', '达州', '德州', '儋州', '鄂尔多斯', '恩施州', '鄂州', '防城港', '抚顺', '佛山', '阜新', '阜阳', '富阳', '福州', '抚州', '广安', '贵港', '果洛州', '桂林', '甘南州', '贵阳', '广元', '固原', '广州', '甘孜州', '赣州', '淮安', '淮北', '鹤壁', '海北州', '河池', '邯郸', '海东地区', '哈尔滨', '合肥', '黄冈', '鹤岗', '红河州', '怀化', '黑河', '呼和浩特', '海口', '呼伦贝尔', '葫芦岛', '海门', '哈密地区', '淮南', '黄南州', '海南州', '黄山', '衡水', '黄石', '和田地区', '海西州', '衡阳', '河源', '湖州', '汉中', '杭州', '贺州', '菏泽', '惠州', '吉安', '金昌', '晋城', '景德镇', '西双版纳州', '金华', '九江', '吉林', '荆门', '江门', '即墨', '佳木斯', '济南', '济宁', '胶南', '酒泉', '句容', '湘西州', '金坛', '嘉兴', '鸡西', '济源', '揭阳', '江阴', '嘉峪关', '锦州', '荆州', '晋中', '焦作', '胶州', '库尔勒', '开封', '黔东南州', '克拉玛依', '昆明', '昆山', '喀什地区', '临安', '六安', '来宾', '聊城', '临沧', '乐东', '娄底', '廊坊', '临汾', '临高', '漯河', '丽江', '吕梁', '陇南', '六盘水', '丽水', '凉山州', '拉萨', '乐山', '陵水', '莱芜', '临夏州', '莱西', '辽源', '辽阳', '溧阳', '龙岩', '洛阳', '临沂', '连云港', '莱州', '林芝', '泸州', '柳州', '兰州', '马鞍山', '牡丹江', '茂名', '眉山', '绵阳', '梅州', '宁波', '南充', '南昌', '宁德', '南京', '内江', '怒江州', '南宁', '南平', '那曲地区', '南通', '南阳', '平度', '平顶山', '普洱', '盘锦', '蓬莱', '平凉', '莆田', '萍乡', '濮阳', '攀枝花', '青岛', '琼海', '秦皇岛', '曲靖', '齐齐哈尔', '七台河', '黔西南州', '清远', '庆阳', '钦州', '衢州', '琼中', '泉州', '荣成', '日喀则', '乳山', '日照', '寿光', '韶关', '上海', '绥化', '石河子', '石家庄', '商洛', '三明', '三门峡', '遂宁', '山南', '四平', '宿迁', '商丘', '上饶', '汕头', '汕尾', '绍兴', '松原', '沈阳', '十堰', '三亚', '邵阳', '双鸭山', '朔州', '苏州', '宿州', '随州', '深圳', '石嘴山', '泰安', '铜川', '屯昌', '太仓', '塔城地区', '通化', '天津', '铁岭', '铜陵', '通辽', '吐鲁番地区', '铜仁地区', '唐山', '天水', '太原', '台州', '泰州', '文昌', '文登', '潍坊', '瓦房店', '武汉', '乌海', '芜湖', '威海', '吴江', '乌兰察布', '乌鲁木齐', '渭南', '万宁', '文山州', '武威', '无锡', '温州', '梧州', '吴忠', '五指山', '兴安盟', '西安', '宣城', '许昌', '襄阳', '孝感', '迪庆州', '锡林郭勒盟', '厦门', '西宁', '咸宁', '湘潭', '邢台', '新乡', '咸阳', '新余', '信阳', '忻州', '徐州', '雅安', '延安', '延边州', '宜宾', '伊春', '银川', '宜春', '宜昌', '盐城', '运城', '云浮', '阳江', '营口', '玉林', '榆林', '伊犁哈萨克州', '阳泉', '玉树州', '烟台', '鹰潭', '义乌', '宜兴', '玉溪', '益阳', '岳阳', '永州', '扬州', '淄博', '自贡', '珠海', '镇江', '湛江', '诸暨', '张家港', '张家界', '张家口', '周口', '驻马店', '章丘', '肇庆', '舟山', '中山', '昭通', '中卫', '招远', '资阳', '张掖', '遵义', '郑州', '漳州', '株洲', '枣庄']

执行结果

3.5 解析图片数据(中文乱码问题)

#url:/4kmeinv/

importrequestsfrom lxml importetree

url= "/4kqiche/"headers={"User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/73.0.3683.75 Safari/537.36"}

response= requests.get(url=url, headers=headers)

page_text=response.text

tree=etree.HTML(page_text)

li_list= tree.xpath('//ul[@class="clearfix"]/li')for li inli_list:

title= li.xpath('./a/b/text()')[0]#使用ISO-8859-1这种通用编码方式可以处理大部分中文乱码的情况

title = title.encode('ISO-8859-1').decode('gbk')

detail_url= "" + li.xpath('./a/@href')[0]print(title, detail_url)

执行

《Karma SC1 Vision Con /tupian/24859.html Ford GT MK II 福特 /tupian/24849.html

福特ford gt mk ii 4k跑 /tupian/24831.html

法拉利ferrari488 pist /tupian/24830.html

奥迪audi r8 lms gt2 赛 /tupian/24829.html

法拉利Portofino跑 /tupian/23939.html

劳斯莱斯幽灵黑徽 /tupian/23824.html McLaren Senna GTR /tupian/23823.html

劳斯莱斯幽灵黑徽 /tupian/23822.html

迈凯伦McLaren 600LT Sp /tupian/23695.html

白色劳斯莱斯5k图片 /tupian/23654.html

《迈凯伦720S GT3》4k壁 /tupian/23426.html

兰博基尼Lamborghini Ur /tupian/23017.html

保时捷Porsche911 Carr /tupian/23016.html

兰博基尼LP580橙色跑车4 /tupian/22673.html

奔驰银箭Mercedes-Benz /tupian/22109.html

劳斯莱斯幻影Rolls-Royc /tupian/22107.html

McLaren 720S GT3 迈凯伦 /tupian/22041.html

迈凯伦McLaren 720S GT3 /tupian/22040.html

Mercedes-Benz Vision E /tupian/22039.html

迈凯伦720S白色跑车3440 /tupian/22036.html

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。