吾爱破解 - 52pojie.cn

 找回密码
 注册[Register]

QQ登录

只需一步,快速开始

查看: 2415|回复: 35
收起左侧

[其他原创] golang 多线程爬虫

[复制链接]
rxxcy 发表于 2023-4-17 23:29
本帖最后由 rxxcy 于 2023-4-17 23:51 编辑

学到 goroutine 练练手

大概是下图这样的,10线程。菜坤写的很笨,能跑就行。如有违规请删
蓝奏 密码:esf0
解压密码:52pj
运行截图

package main

import (
        "fmt"
        "github.com/gocolly/colly"
        "xie/skk"
        "io"
        "io/ioutil"
        "log"
        "net/http"
        "os"
        "path/filepath"
        "regexp"
        "strings"
        "sync"
)

func Pwd() string {
        dir, err := filepath.Abs(filepath.Dir(os.Args[0]))
        if err != nil {
                log.Fatalln("获取基础路径失败:", err.Error())
        }
        return strings.Replace(dir, "\\", "/", -1)
}

func Mkdir(path string) bool {
        _, err := os.Stat(path)
        if err != nil {
                err = os.MkdirAll(path, 0766)
                if err != nil {
                        return false
                }
                return true
        }
        if os.IsNotExist(err) {
                err = os.MkdirAll(path, 0766)
                if err != nil {
                        return false
                }
                return true
        }
        return true
}

func Downloads(title string, url string) {
        var index = 1
        x := colly.NewCollector(
                colly.UserAgent("Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/112.0.0.0 Safari/537.36"))
        _ = x.Limit(&colly.LimitRule{DomainGlob: "*.xiezhen.*", Parallelism: 5})
        x.OnHTML(".article-content img", func(e *colly.HTMLElement) {
                src := e.Attr("src")
                path := fmt.Sprintf("%s/%s", baseDir, title)
                Mkdir(path)
                extras := regExt.FindStringSubmatch(src)
                if len(extras) < 1 {
                        skk.RedOnly("未知的文件格式")
                        return
                }
                ext := extras[1]
                f := fmt.Sprintf("%d", index)
                if index < 10 {
                        f = fmt.Sprintf("0%s", f)
                }
                fileName := fmt.Sprintf("%s/%s.%s", path, f, ext)
                fmt.Printf("\r下载第 %s 张", f)
                image := Image{Src: src, Path: fileName}
                CreateDownloadTask(image, ch)
                index++
        })

        x.OnError(func(r *colly.Response, err error) {
                log.Fatal("err: ", err.Error())
        })

        x.OnRequest(func(r *colly.Request) {
                skk.Blue("\n下载图集", title)
        })

        err := x.Visit(url)
        if err != nil {
                log.Println(err.Error())
                return
        }

}

func Download(url string, path string) {
        request, err := http.NewRequest("GET", url, nil)
        request.Header.Add("referer", "https://www.xiezhen.xyz/")
        request.Header.Add("user-agent", "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/112.0.0.0 Safari/537.36")

        response, err := client.Do(request)
        if err != nil {
                return
        }
        defer func(Body io.ReadCloser) {
                err = Body.Close()
                if err != nil {
                        log.Printf("错误: %s \n", err.Error())
                }
        }(response.Body)
        body, err := ioutil.ReadAll(response.Body)
        if err != nil {
                fmt.Println("下载文件错误: ", err.Error())
        }
        _ = ioutil.WriteFile(path, body, 0755)
}

func Bootstrap() {
        c := colly.NewCollector(
                colly.UserAgent("Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/112.0.0.0 Safari/537.36"))
        _ = c.Limit(&colly.LimitRule{DomainGlob: "*.xiezhen.*", Parallelism: 5})

        c.OnHTML(".excerpt h2", func(e *colly.HTMLElement) {
                title := e.Text
                url := e.ChildAttrs("a", "href")[0]
                Downloads(title, url)
        })

        c.OnError(func(r *colly.Response, err error) {
                log.Fatal("err: ", err.Error())
        })

        c.OnRequest(func(r *colly.Request) {
                skk.RedOnly(fmt.Sprintf("\n第 %d 页", page))
        })
        for page < 370 {
                url := fmt.Sprintf("https://www.xiezhen.xyz/page/%d", page)
                err := c.Visit(url)
                if err != nil {
                        log.Println(err.Error())
                        return
                }
                page++
        }
}

func DownloadPool(ch chan Image) {
        for image := range ch {
                Download(image.Src, image.Path)
                wg.Done()
        }
}

func CreateDownloadTask(image Image, ch chan Image) {
        wg.Add(1)
        ch <- image
}

var baseDir string
var page = 1
var regExt = regexp.MustCompile("\\.(png|jpeg|jpg|pjpg)$")
var client = &http.Client{}
var wg = sync.WaitGroup{}
var ch chan Image
var goCnt = 10

type Image struct {
        Src  string
        Path string
}

func main() {
        fmt.Printf(`

        图片下载器 🎉
        @rxxcy

`)

        tempBaseDir := Pwd()
        skk.Magenta("默认文件保存路径", tempBaseDir+"/images/")
        fmt.Printf("自定义保存路径, 回车默认: ")
        _, _ = fmt.Scanf("%s", &baseDir)

        if baseDir == "" {
                baseDir = tempBaseDir + "/images/"
        } else {
                baseDir = baseDir + "/images/"
        }
        skk.Blue("保存路径", baseDir)
        skk.Blue("默认协程数", fmt.Sprintf("%d", goCnt))
        ch = make(chan Image)
        for i := 0; i < goCnt; i++ {
                go DownloadPool(ch)
        }

        Bootstrap()
        wg.Wait()
}

免费评分

参与人数 2吾爱币 +12 热心值 +2 收起 理由
yacc + 5 + 1 欢迎分析讨论交流,吾爱破解论坛有你更精彩!
侃遍天下无二人 + 7 + 1 欢迎分析讨论交流,吾爱破解论坛有你更精彩!

查看全部评分

发帖前要善用论坛搜索功能,那里可能会有你要找的答案或者已经有人发布过相同内容了,请勿重复发帖。

yacc 发表于 2023-4-19 17:50
wang8118 发表于 2023-4-18 09:20
defer func(Body io.ReadCloser) {
                err = Body.Close()
                if err != nil ...

这是defer的机制造成的,简单说defer是一个压栈的过程(先进后出,这也是为什么反着顺序执行),每次defer都会把函数指针、参数等信息压栈。
也就是说,如果这里defer不传参,后面又把参数的内容改了,里面的结果可能就不一样,这也是为什么楼主说ide会报warning
参考https://go.dev/blog/defer-panic-and-recover
头像被屏蔽
wang8118 发表于 2023-4-18 09:20
茶叶泡咖啡 发表于 2023-4-17 23:49
 楼主| rxxcy 发表于 2023-4-17 23:51

蓝奏密码 esf0
ergouzib702 发表于 2023-4-18 05:37
有点意外
px307 发表于 2023-4-18 07:24
感谢分享
twj007 发表于 2023-4-18 08:25
没看懂,但谢谢!
ashq 发表于 2023-4-18 08:28
正是我所想要的,谢谢大大了
aa2923821a 发表于 2023-4-18 08:45
谢谢  好像看懂了些  回去研究研究
TenSir152 发表于 2023-4-18 08:46
没看明白,注释少啊
水上凌波 发表于 2023-4-18 08:47
https://bz.zzzmh.cn/index把这里的原图爬下来吧
您需要登录后才可以回帖 登录 | 注册[Register]

本版积分规则

返回列表

RSS订阅|小黑屋|处罚记录|联系我们|吾爱破解 - LCG - LSG ( 京ICP备16042023号 | 京公网安备 11010502030087号 )

GMT+8, 2025-1-11 11:07

Powered by Discuz!

Copyright © 2001-2020, Tencent Cloud.

快速回复 返回顶部 返回列表