我用了30行代碼,爬了知乎好多妹子

寫一個爬蟲有多簡單?答案是:不到30行代碼

我用了不到30行代碼,爬了知乎好多妹子圖?。。?/p>

跟著我一步一步來,你也可以簡簡單單創(chuàng)建一個爬蟲。

目標(biāo)

爬蟲的第一步就是決定要爬些什么。作為一個屌的不能再屌的屌絲,當(dāng)然要爬妹子?。?!

每當(dāng)這個時候,就要拿出我的儲備了。 欲罷不能的大美妞 ,要說妹子圖的質(zhì)量,還是得知乎啊。

分析

目標(biāo)有了,我們就要先分析一下。頁面結(jié)構(gòu)。在頁面中找任一個爬取目標(biāo),點擊鼠標(biāo)右鍵,選擇【檢查】選項。便會打開瀏覽器的控制臺,并定位到我們的目標(biāo)圖片的節(jié)點上。如下圖:

圖中的img標(biāo)簽就是我們的爬取目標(biāo),我們可以明確的看到 data-original 元素的內(nèi)容與 src 元素的內(nèi)容都是圖片的地址。經(jīng)過驗證, src 元素的內(nèi)容可能會是縮略圖地址,所以我們放棄 src 選擇 data-original。

以上就是我們要分析的部分,接下來就是代碼。

編碼

新建項目

這里我們使用 maven 作為項目的依賴管理工具。

新建一個 maven 項目,并在 pom.xml 中引入依賴。

<dependencies>
        <dependency>
            <groupId>com.github.zhangyingwei</groupId>
            <artifactId>cockroach</artifactId>
            <version>1.0-Alpha</version>
        </dependency>
        <dependency>
            <groupId>log4j</groupId>
            <artifactId>log4j</artifactId>
            <version>1.2.17</version>
        </dependency>
        <dependency>
            <groupId>com.squareup.okhttp3</groupId>
            <artifactId>okhttp</artifactId>
            <version>3.8.1</version>
        </dependency>
        <!--json-lib-->
        <dependency>
            <groupId>net.sf.json-lib</groupId>
            <artifactId>json-lib</artifactId>
            <version>2.4</version>
            <classifier>jdk15</classifier>
        </dependency>
        <!-- https://mvnrepository.com/artifact/org.jsoup/jsoup -->
        <dependency>
            <groupId>org.jsoup</groupId>
            <artifactId>jsoup</artifactId>
            <version>1.8.3</version>
        </dependency>
    </dependencies>

這里我們使用了爬蟲框架 cockroach 。并引入了 log4j 作為日志框架,okhttp3 作為 http 客戶端,json-lib 作為 json 解析工具,以及 jsoup 作為 html 解析工具。

建立包結(jié)構(gòu)如下

  • store 主要存放頁面解析以及結(jié)果存儲相關(guān)類
  • utils 主要存放項目中用到的相關(guān)工具類
  • App.java 項目入口

編碼

程序主入口 App.java

public class App {
    public static void main(String[] args) throws InstantiationException, IllegalAccessException, InterruptedException {
        CockroachConfig config = new CockroachConfig().setAppName("知乎上的妹子們").setThread(10).setStore(ZhihuGirlsStore.class);
        TaskQueue queue = TaskQueue.of();
        getPageFrom(queue);
        CockroachContext context = new CockroachContext(config);
        context.start(queue);
    }

    private static void getPageFrom(TaskQueue queue) throws InterruptedException {
        String basePath = "https://www.zhihu.com/collection/72114548?page=";
        for (int i = 1; i <= 68; i++) {
            queue.push(new Task(basePath + i));
        }
    }
}

在上邊的代碼中,我們一共分為5步。

  1. 我們創(chuàng)建了一個名稱為 知乎上的妹子們 的爬蟲,使用了 10 個線程來爬取內(nèi)容,并指定了頁面解析以及結(jié)果存儲的處理類為 ZhihuGirlsStore.class
  2. 創(chuàng)建了一個默認(rèn)長度的任務(wù)隊列
  3. 初始化任務(wù)到任務(wù)隊列中
  4. 創(chuàng)建了一個 Cockroach 爬蟲上下文對象
  5. 啟動爬蟲

頁面解析以及存儲 ZhihuGirlsStore.java

public class ZhihuGirlsStore implements IStore {
    public void store(TaskResponse taskResponse) throws Exception {
        if (taskResponse.getTask().getGroup().equals("img")) {
            byte[] bytes = taskResponse.getResponse().body().bytes();
            ImageUtils.save(bytes);
        } else {
            Elements imgs = taskResponse.select(".zm-item-answer").select("img");
            imgs.stream().map(element -> element.attr("data-original")).forEach(url -> {
                try {
                    taskResponse.getQueue().push(new Task(url, "img"));
                } catch (Exception e) {
                    e.printStackTrace();
                }
            });
        }
    }
}

Cockroach 的 task 中為我們提供了 group 字段來標(biāo)識每一個 任務(wù),如果不設(shè)置,任務(wù)默認(rèn)的 group 為 default 。
在本爬蟲程序中,我們的 task 一共可分為兩種,一種是解析頁面得到頁面中的圖片地址,另外一種就是爬取圖片內(nèi)容。
這里我們通過 group 字段來區(qū)分兩種任務(wù),具體操作就是在解析到圖片地址并添加到隊列中的時候,給 task 設(shè)置 group 為 img,這樣我們在收到一個結(jié)果的時候,就可以通過 task 中的 group 字段來區(qū)分我們要做何種操作(解析頁面 / 保存圖片)。

圖片存儲 ImageUtils.java

public class ImageUtils {
    public static void save(byte[] bytes) throws IOException {
        String fileName = UUID.randomUUID().toString();
        String dirpath = "meizhi2";
        File dir = new File(dirpath);
        if(!dir.exists()){
            dir.mkdirs();
        }
        FileOutputStream outputStream = new FileOutputStream(dirpath + "/" + fileName + ".jpg");
        outputStream.write(bytes);
        outputStream.close();
        System.out.println("save image:" + fileName);
    }
}

以上代碼就是把接收到的二進(jìn)制圖片內(nèi)容保存為圖片。圖片的名稱使用一個隨機的 UUID 值。

沒錯老鐵們,以上就是我們的全部的代碼。包括方法聲明在內(nèi)的有效代碼不到30行!?。?/p>

文章來源

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 179,034評論 25 709
  • 沒想到一個學(xué)校的體育測試竟然水成這樣。如果你不能幾個,那可以拿個及格分;及格以后,就看你跳多遠(yuǎn)了。讓我驚訝的是,在...
    瀟瀟公主駕到閱讀 191評論 0 0
  • 今天中午原本打算搜索一些英語學(xué)習(xí)視頻,可是一不小心搜索出了銷售彥論,結(jié)果發(fā)現(xiàn)根本停不下來! 從中午12:59分,一...
    余保林閱讀 188評論 0 0
  • 如果,我說愛。 文/阿語花 十一月十二日晚,我夢見一個很溫柔很溫柔的男生。 身材比例完美,做事多為人著想,會做飯,...
    阿語花閱讀 502評論 0 0
  • 夕陽西下 我牽著老馬 它說腳有些疲乏 漫漫的水面難免有些眼花 深深淺淺地試探 晃晃悠悠地出發(fā) 暖黃的太陽就是方向 ...
    七彩城主閱讀 404評論 0 0

友情鏈接更多精彩內(nèi)容