【python爬虫02】使用Scrapy框架爬取拉勾网招聘信息

使用Scrapy框架爬取拉勾网招聘信息


最近接触了Scrapy爬虫框架,简单写了个爬虫爬取拉钩网的招聘信息,加深对Scrapy框架的理解,不得不说Scrapy框架其实还是蛮方便的,就像爬虫流水线一样,如果是大项目的话使用Scrapy会变得更加容易管理,废话不多说,下面就看看如何使用Scrapy爬取拉勾网招聘消息吧。

 【python爬虫02】使用Scrapy框架爬取拉勾网招聘信息_第1张图片

【python爬虫02】使用Scrapy框架爬取拉勾网招聘信息_第2张图片

 

【python爬虫02】使用Scrapy框架爬取拉勾网招聘信息_第3张图片


我们发现由于数据是分页显示的,如果想要获取每一页的内容,自然需要获取每一页的链接。

我们点击下一页,发现页面回到了顶端,并且url栏没有任何变化!为什么呢?当前页的参数去哪里了?我们需要这里又用到了chrone浏览器强大的开发者工具了。在点击“3”之后,出现了下面的请求。

 【python爬虫02】使用Scrapy框架爬取拉勾网招聘信息_第4张图片


可以看到,传到后台的页码参数正安静地躺在那里呢,也就是图中的pn,而kd就是我们传入的关键字keyword。在这里我们也可以得到请求的url。但是,如何爬取ajax请求返回的数据呢?在scrapy中,我们可以通过start_requests这个函数返回一个请求列表,框架会按照这个请求列表去请求,然后将得到的response交给我们写好的回调函数,json数据就包含在response中。代码如下:

def start_requests(self):
    #修改city参数更换城市
    url= "https://www.lagou.com/jobs/positionAjax.json?needAddtionalResult=false&isSchoolJob=0&city=广州"
    requests = []
#range是查询的页码范围
    for i in range(1, 60):
        #修改kd参数更换关键字
        formdata = {'first':'false', 'pn':str(i),'kd':'java'}
        request = FormRequest(url, callback=self.parse_model,formdata=formdata)
        requests.append(request)
        print(request)
    return requests

这里我们同样需要构造我们的请求头,scrapy框架会自动去读取的。

custom_settings = {
        "DEFAULT_REQUEST_HEADERS": {
            'Accept': 'application/json, text/javascript, */*; q=0.01',
            'Accept-Encoding': 'gzip, deflate, br',
            'Accept-Language': 'zh-CN,zh;q=0.8',
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8',            'Host': 'www.lagou.com',
            'Origin': 'https://www.lagou.com',
            'Referer': 'https://www.lagou.com/jobs/list_java?',
            'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3112.113 Safari/537.36',
            'X-Anit-Forge-Code': '0',
            'X-Anit-Forge-Token': 'None',
            'X-Requested-With': 'XMLHttpRequest'
        },
        "ITEM_PIPELINES": {
            'lagou.pipelines.LagouPipeline': 300
        }
    }

分析传回来的json数据的结构,可以得到我们的处理函数如下:

def parse_model(self, response):
    print(response.body.decode())
    jsonBody =json.loads(response.body.decode())
    results = jsonBody['content']['positionResult']['result']
    items=[]
    for result in results:
        item=LagouItem()
        item['name']=result['positionName']
        item['workLocation']=result['city']
        if result['district']:
            item['workLocation']+=result['district']
        if result['businessZones']:
            for zone in result['businessZones']:
                item['workLocation'] +=zone
        #item['catalog']
        item['money']=result['salary']
        item['demand']=result['workYear']+"/"+result['education']
        item['skillLabel']=",".join(result['positionLables'])
        item['positionAdvantage']=result['positionAdvantage']
        item['publishTime']=result['formatCreateTime']
        item['company']=result['companyFullName']
        item['companyField']=result['industryField']
        item['companyLabelList']=",".join(result['companyLabelList'])
        item['detailLink']="https://www.lagou.com/jobs/"+str(result['positionId'])+".html"
        item['detailCompany']="https://www.lagou.com/gongsi/"+str(+result['companyId'])+".html"
        items.append(item)
    return items

Item具体如下:

class LagouItem(Item):
    name = Field()                # 职位名称
    workLocation = Field()        # 工作地点
    catalog = Field()             # 职位类别
    money= Field()                # 薪资水平
    demand=Field()                # 要求
    skillLabel=Field()            # 技能标签
    publishTime = Field()         # 发布时间
    company = Field()             # 公司名称
    companyField=Field()          # 公司服务领域
    companyLabelList=Field()      # 公司简介
    positionAdvantage=Field()     # 职位福利
    detailLink = Field()          # 职位详情页链接
    detailCompany=Field()         # 公司详情页链接

数据提取完之后,由Pine来处理并进行后续操作,如储存在excel/数据库中。这里我将爬到的数据存到xlsx中。代码如下:

class LagouPipeline(object):
   def __init__(self):
        self.workbook = Workbook()
        self.ws = self.workbook.active
        self.ws.append(['职位名称', '工作地点', '薪资水平', '要求', '技能标签', '发布时间','公司名称','公司服务领域','公司简介','职位福利','职位详情页链接','公司详情页链接'])  # 设置表头
        #self.file = codecs.open('tencent.json','w', encoding='utf-8')
    def process_item(self, item, spider):
        line = [item['name'], item['workLocation'],item['money'], item['demand'], item['skillLabel'],
                item['publishTime'],item['company'],item['companyField'],item['companyLabelList'],item['positionAdvantage'],item['detailLink'],item['detailCompany']]  # 把数据中每一项整理出来
        self.ws.append(line)
        self.workbook.save('lagou.xlsx')  # 保存xlsx文件
        #line = json.dumps(dict(item), ensure_ascii=False)+ "\n"
        #self.file.write(line)
        return item
    def spider_closed(self, spider):
        self.file.close()

最后为了防止拉勾网的反爬虫机制,在setting里设置拒绝遵守robot协议,并且设置爬取时延,防止被认为是爬虫。

ROBOTSTXT_OBEY = False 
DOWNLOAD_DELAY = 10


最终效果如下:

【python爬虫02】使用Scrapy框架爬取拉勾网招聘信息_第5张图片

你可能感兴趣的:(python爬虫)