如何爬取汽车之家的车标

目标地址:https://car.m.autohome.com.cn/

image.png

爬虫的意义在于将目标网页的源码下载到本地后分析该源码并提取想要的数据。所以要实现一个爬虫就只需要两步。第一步是下载源码,第二步分析源码并提取其中的数据。
1、首先看看能不能直接获取源码数据。

 采用python3中的requests库去获取目标网页的源码
import requests
response=requests.get("https://car.m.autohome.com.cn/")
response.encoding="utf-8" #设置编码,以防中文乱码
print(response.text)

结果如下:


image.png

可以直接获取目标网页中的数据,这样就方便多了。 个人感觉对于爬虫来说如果能获取到数据,那这个爬虫就成功了一大半了。

2、提取数据
通过分析目标网页中的源码发现,所有车标都位于

<div class="listing">
      <div class="item">
        车标
     </div>
</div >
image.png

所以只需要定位到车标位置便可以从img 标签中提取src中的图片地址和车标文字了。对于提取数据可以使用python3中的pyquery。pyquery可以使用jquery中的语法去标记元素。提取到图片地址和文字后,便可以使用python中的requests库去下载图片了。

python源代码如下:

import requests
from pyquery import PyQuery as pq
import  uuid
def downLoadImage(fileName,downLoadUrl):
    import requests
    r = requests.get(downLoadUrl)
    fileName=fileName+".jpg";
    print("正在下载 "+fileName+"\n")
    with open("c:\\img1\\"+fileName, 'wb') as f:
        f.write(r.content)
response=requests.get("https://car.m.autohome.com.cn/")
response.encoding="utf-8"
info=pq(pq(pq(response.text)(".listing"))("div"))(".item")
for i in info:
    carBrand=pq(i).text()
    imgageCarBrand=pq(i)("img").attr("src")
    if(imgageCarBrand==None):
        imgageCarBrand =pq(i)("img").attr("data-src")
    imgageCarBrand="https:"+imgageCarBrand
    downLoadImage(carBrand,imgageCarBrand)

运行结果如下:


image.png

代码很简单,也很容易理解。不过我感觉下载速度有点慢,所以想写一个多线程爬虫去下载车标。python中的多线程我还不是很熟悉,正好学习了java中的多线程。所以决定采用java实现多线程爬虫下载车标。爬虫的原理和分析过程和上述一致。这里只略谈下我是如何实现多线程的。

3、java多线程实现
在上面的python爬虫中我感觉比较耗费时间的就是车标的下载,所以我这里将用单线程去提取车标的地址,然后再使用多线程去下载车标。我将提取到的车标信息放入了一个车标类的静态数组中,以便多线程可以按区间去访问车标地址并下载图片。

车标封装类(1、车标名 2、车标地址)

class CarBrand {
    private String carBrand;
    private String carBrandImageUrl;

    public CarBrand(String carBrand, String carBrandImageUrl) {
        this.carBrand = carBrand;
        this.carBrandImageUrl = carBrandImageUrl;
    }

    public String getCarBrand() {
        return carBrand;
    }

    public void setCarBrand(String carBrand) {
        this.carBrand = carBrand;
    }

    public String getCarBrandImageUrl() {
        return carBrandImageUrl;
    }

    public void setCarBrandImageUrl(String carBrandImageUrl) {
        this.carBrandImageUrl = carBrandImageUrl;
    }
}

从目标站点中得到车标地址

 public static CarBrand[] getCarBrandscarBrands() throws Exception {
        Document carDocument = Jsoup.connect("https://car.m.autohome.com.cn/").get();
        Elements elements = carDocument.select(".listing").select("div").select(".item");
        for (Element element : elements) {
            String carBrandImageUrl = element.select("img").attr("src").equals("") ?
                    element.select("img").attr("data-src") :
                    element.select("img").attr("src");
            String carBrandName = element.text();
            carBrands[length++] = new CarBrand(carBrandName, "https:" + carBrandImageUrl);
        }
        return carBrands;
    }

下载车标图片

  private void downLoadImage(String fileName, String carBrandImageUrl) {
        try {
            System.out.println(Thread.currentThread().getName() + " 正在下载 " + fileName + " " + carBrandImageUrl);
            FileUtils.copyURLToFile(new URL(carBrandImageUrl), new File("c:\\img\\" + fileName + ".jpg"));
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

整个代码如下:

package sicau;

import org.apache.commons.io.FileUtils;
import org.jsoup.Jsoup;
import org.jsoup.nodes.Document;
import org.jsoup.nodes.Element;
import org.jsoup.select.Elements;

import java.io.*;
import java.net.URL;

class CarBrand {
    private String carBrand;
    private String carBrandImageUrl;

    public CarBrand(String carBrand, String carBrandImageUrl) {
        this.carBrand = carBrand;
        this.carBrandImageUrl = carBrandImageUrl;
    }

    public String getCarBrand() {
        return carBrand;
    }

    public void setCarBrand(String carBrand) {
        this.carBrand = carBrand;
    }

    public String getCarBrandImageUrl() {
        return carBrandImageUrl;
    }

    public void setCarBrandImageUrl(String carBrandImageUrl) {
        this.carBrandImageUrl = carBrandImageUrl;
    }
}

public class Spider implements Runnable {
    private Integer startIndex;
    private Integer endIndex;
    private static CarBrand[] carBrands = new CarBrand[500];
    private static Integer length = 0;

    public Spider(Integer startIndex, Integer endIndex) {
        this.startIndex = startIndex;
        this.endIndex = endIndex;
    }

    private void downLoadImage(String fileName, String carBrandImageUrl) {
        try {
            System.out.println(Thread.currentThread().getName() + " 正在下载 " + fileName + " " + carBrandImageUrl);
            FileUtils.copyURLToFile(new URL(carBrandImageUrl), new File("c:\\img\\" + fileName + ".jpg"));
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    @Override
    public void run() {
        for (int i = startIndex; i <= endIndex; i++) {
            downLoadImage(carBrands[i].getCarBrand(), carBrands[i].getCarBrandImageUrl());
        }
    }

    public static CarBrand[] getCarBrandscarBrands() throws Exception {
        Document carDocument = Jsoup.connect("https://car.m.autohome.com.cn/").get();
        Elements elements = carDocument.select(".listing").select("div").select(".item");
        for (Element element : elements) {
            String carBrandImageUrl = element.select("img").attr("src").equals("") ?
                    element.select("img").attr("data-src") :
                    element.select("img").attr("src");
            String carBrandName = element.text();
            carBrands[length++] = new CarBrand(carBrandName, "https:" + carBrandImageUrl);
        }
        return carBrands;
    }



    public static void main(String[] args) throws Exception {
        getCarBrandscarBrands();
        new Thread(new Spider(0, 50)).start();
        new Thread(new Spider(51, 100)).start();
        new Thread(new Spider(101, 150)).start();
        new Thread(new Spider(151, 202)).start();

    }
}

运行结果如下:


总结

本爬虫程序并不难实现,只有找对了库。下面总结下本次使用的库。
python3: requests、pyquery

java: maven、jsoup、commons-io

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,391评论 25 707
  • 用两张图告诉你,为什么你的 App 会卡顿? - Android - 掘金 Cover 有什么料? 从这篇文章中你...
    hw1212阅读 12,680评论 2 59
  • 目录 | 第一章 古道长亭 第二章 总角之交 此时尚不到日中,离午饭时间还早,铁珩的房中却传出阵阵香气。 还伴着男...
    青色百合99阅读 2,363评论 53 86
  • 今天逛微博,发现一师姐的微博,即将毕业的她独自一人去北京面试,超级佩服她的勇气~~~不管结果如何,这段经历注定是难...
    isvivi阅读 194评论 0 0
  • 孤独是生命的佛性。唯经苦难才懂孤独,唯具智慧才觉孤独。孤独自有孤独的品位。孤独不是...
    冰夫阅读 502评论 0 0