《Go 简易速速上手小册》第5章:并发编程(2024 最新版)

news2025/1/3 11:52:11

在这里插入图片描述

文章目录

  • 5.1 Goroutines 的基础 - Go 语言中的轻盈舞者
    • 5.1.1 基础知识讲解
    • 5.1.2 重点案例:并发下载器
      • 功能描述
      • 实现代码
      • 扩展功能
    • 5.1.3 拓展案例 1:网站健康检查
      • 功能描述
      • 实现代码
      • 扩展功能
    • 5.1.4 拓展案例 2:并发日志处理器
        • 拓展案例 2:并发日志处理器
          • 功能描述
          • 实现代码
          • 扩展功能
  • 5.2 Channels 的使用 - Go 语言中的通信艺术
    • 5.2.1 基础知识讲解
    • 5.2.2 重点案例:任务分发系统
      • 功能描述
      • 实现代码
    • 5.2.3 拓展案例 1:数据流处理
        • 拓展案例 1:数据流处理
          • 功能描述
          • 实现代码
          • 扩展功能
    • 5.2.4 拓展案例 2:实时消息系统
      • 功能描述
      • 实现代码
      • 扩展功能
  • 5.3 并发模式与同步 - 编织 Go 语言中的并发之网
    • 5.3.1 基础知识讲解
    • 5.3.2 重点案例:简易聊天服务器
      • 功能描述
      • 实现代码
      • 扩展功能
    • 5.3.3 拓展案例 1:实时数据监控
      • 功能描述
      • 实现代码
      • 扩展功能
    • 5.3.4 拓展案例 2:并发 Web 爬虫
      • 功能描述
      • 实现代码
      • 扩展功能

5.1 Goroutines 的基础 - Go 语言中的轻盈舞者

Ahoy, 并发编程的舞者们!让我们一起深入探索 Go 语言中的 Goroutines —— 这些轻盈的并发执行单位,它们就像是在 CPU 的舞台上轻盈跳跃的舞者。通过 Goroutines,Go 让并发编程变得异常简单和高效,就像是为我们的应用程序注入了一剂速效的能量药剂。

5.1.1 基础知识讲解

Goroutines 的定义

Goroutines 是 Go 语言中实现并发的核心。你可以把它们想象成轻量级的线程,由 Go 运行时管理。与操作系统的线程相比,Goroutines 的启动和销毁成本更低,内存占用也更小,这使得你可以轻松地创建成千上万的 Goroutines。

go function() {
    // 这里是你的代码
}

只需在函数调用前加上 go 关键字,这个函数就会在新的 Goroutine 中异步执行。是的,就是这么简单!

Goroutines 的特点

  • 轻量级:每个 Goroutine 在堆栈上只占用几 KB 的内存。
  • 动态增长的堆栈:Goroutines 的堆栈大小不是固定的,可以根据需要动态增长和缩小。
  • 简单的创建和销毁:创建和销毁 Goroutines 的成本远低于重量级线程。

5.1.2 重点案例:并发下载器

在这个快速发展的互联网时代,下载多个文件是一项常见的任务。利用 Go 语言的 Goroutines,我们可以轻松实现一个并发下载器,这样可以大大加快下载速度,提升用户体验。让我们一起来扩展这个并发下载器的案例,使其更加实用和高效。

功能描述

  1. 并发下载:使用 Goroutines 并发下载多个文件。
  2. 错误处理:捕获下载过程中的错误,并报告。
  3. 进度反馈:实时显示每个文件的下载进度和状态。
  4. 同步等待:使用sync.WaitGroup确保所有下载任务完成后程序才退出。

实现代码

首先,我们模拟一个下载函数,它接收文件名和一个用于报告下载进度的通道:

package main

import (
    "fmt"
    "math/rand"
    "sync"
    "time"
)

// downloadFile 模拟文件下载
func downloadFile(file string, progress chan<- string, wg *sync.WaitGroup) {
    defer wg.Done()

    for i := 0; i <= 100; i += rand.Intn(25) {
        progress <- fmt.Sprintf("%s 下载进度: %d%%", file, i)
        time.Sleep(time.Duration(rand.Intn(300)) * time.Millisecond)
    }
    progress <- fmt.Sprintf("%s 下载完成", file)
}

然后,我们创建一个 Goroutine 来处理每个文件的下载,并使用sync.WaitGroup来同步等待所有下载任务完成:

func main() {
    files := []string{"file1.zip", "file2.zip", "file3.zip"}
    var wg sync.WaitGroup

    // 创建一个通道来报告下载进度
    progress := make(chan string)
    // 计数器设置为需要下载的文件数
    wg.Add(len(files))

    for _, file := range files {
        go downloadFile(file, progress, &wg)
    }

    // 启动一个 Goroutine 来打印进度信息
    go func() {
        for p := range progress {
            fmt.Println(p)
        }
    }()

    // 等待所有下载任务完成
    wg.Wait()
    close(progress) // 关闭通道,停止打印进度信息
}

扩展功能

  • 错误处理:我们可以修改downloadFile函数,让它有一定概率模拟下载失败的情况,并通过另一个通道报告错误。
  • 限制并发数:为避免同时启动过多的 Goroutines,我们可以使用带缓冲的通道作为并发限制的信号量。

通过这个扩展案例,我们构建了一个更加健壮和实用的并发下载器,它不仅可以并发下载多个文件,还能处理错误、报告下载进度,并且保证所有任务完成后才退出程序。这个案例展示了 Goroutines 和通道在实际应用中的强大能力,为我们解决并发任务提供了简单有效的工具。现在,就让我们利用这些工具,去构建更多令人激动的并发应用吧!

5.1.3 拓展案例 1:网站健康检查

在维护任何在线服务时,定期检查网站的健康状况是至关重要的。通过并发执行网站健康检查,我们可以在最短的时间内获得多个网站的状态,从而迅速响应可能出现的问题。利用 Go 语言的 Goroutines 和 Channels,我们可以构建一个高效的网站健康检查工具。

功能描述

  1. 并发执行网站健康检查:使用 Goroutines 并发地向多个网站发送请求。
  2. 收集并报告结果:收集每个网站的健康检查结果,并汇总报告。

实现代码

首先,定义一个简单的函数来检查单个网站的健康状况:

package main

import (
    "fmt"
    "net/http"
    "sync"
    "time"
)

// checkWebsite 检查网站健康状况
func checkWebsite(url string, wg *sync.WaitGroup, results chan<- string) {
    defer wg.Done()
    start := time.Now()
    resp, err := http.Get(url)
    duration := time.Since(start)

    if err != nil || resp.StatusCode != 200 {
        results <- fmt.Sprintf("[失败] %s 耗时 %s", url, duration)
        return
    }

    results <- fmt.Sprintf("[成功] %s 状态码 %d 耗时 %s", url, resp.StatusCode, duration)
}

然后,使用 Goroutines 并发执行多个网站的健康检查,并使用sync.WaitGroup同步等待所有检查任务完成:

func main() {
    websites := []string{
        "https://www.google.com",
        "https://www.github.com",
        "https://www.stackoverflow.com",
        "https://golang.org",
        "https://www.example.com",
    }

    var wg sync.WaitGroup
    results := make(chan string, len(websites))
    wg.Add(len(websites))

    for _, url := range websites {
        go checkWebsite(url, &wg, results)
    }

    go func() {
        wg.Wait()
        close(results)
    }()

    // 打印检查结果
    for result := range results {
        fmt.Println(result)
    }
}

扩展功能

  • 超时控制:为http.Get请求添加超时控制,防止某些网站响应过慢影响整体检查进程。
  • 重试机制:对于检查失败的网站,可以实现重试机制,以确保偶发的网络问题不会导致误报。

通过这个扩展案例,我们构建了一个可以并发执行网站健康检查的工具,它能够快速收集和报告多个网站的状态。利用 Go 语言的并发特性,我们的工具不仅执行效率高,而且代码结构清晰简洁。这种并发模式的应用,在开发高效且可靠的网络服务和工具时非常有价值。现在,就让我们继续探索 Go 语言的并发世界,开发更多强大的应用吧!

5.1.4 拓展案例 2:并发日志处理器

拓展案例 2:并发日志处理器

在大型系统中,日志是监控系统健康、诊断问题的重要手段。随着系统规模的扩大,日志量也会急剧增加。使用并发日志处理器,我们可以高效地从多个来源并发地收集、处理日志,提高日志处理的速度和效率。

功能描述
  1. 并发收集日志:使用 Goroutines 并发地从多个日志来源(如文件、网络等)收集日志。
  2. 日志处理:对收集到的日志执行一系列处理操作,如过滤、格式化。
  3. 日志聚合:将处理后的日志聚合到一个中心位置,以便分析和存储。
实现代码

首先,定义一个模拟的日志收集函数,假设日志来自不同的文件:

package main

import (
    "fmt"
    "sync"
    "time"
)

// collectLogs 从指定的日志来源收集日志
func collectLogs(source string, wg *sync.WaitGroup, logChan chan<- string) {
    defer wg.Done()
    // 模拟从不同来源收集日志的时间消耗
    time.Sleep(time.Duration(1+rand.Intn(5)) * time.Second)
    logMsg := fmt.Sprintf("日志来自 %s: 日志内容", source)
    logChan <- logMsg
}

接着,实现并发的日志收集和处理逻辑:

func main() {
    logSources := []string{"文件1", "文件2", "网络流", "数据库"}
    var wg sync.WaitGroup
    logChan := make(chan string, len(logSources))

    // 并发从各个日志来源收集日志
    wg.Add(len(logSources))
    for _, source := range logSources {
        go collectLogs(source, &wg, logChan)
    }

    // 启动一个 Goroutine 来处理日志
    go func() {
        for logMsg := range logChan {
            fmt.Println("处理日志:", logMsg)
            // 这里可以添加更复杂的日志处理逻辑
        }
    }()

    // 等待所有日志收集任务完成
    wg.Wait()
    close(logChan) // 关闭通道,结束日志处理 Goroutine
}
扩展功能
  • 日志过滤:可以在处理日志的 Goroutine 中加入过滤逻辑,只保留符合特定条件的日志。
  • 日志格式化:对日志进行格式化处理,例如转换为 JSON 格式,以便于后续处理和存储。
  • 错误处理:增加错误处理逻辑,确保日志收集和处理过程中的错误能够被妥善处理。

通过这个扩展案例,我们构建了一个能够高效处理大量日志的并发日志处理器。利用 Go 语言的并发特性,我们的处理器可以轻松应对来自不同来源的日志,提高了日志处理的速度和灵活性。这种并发处理模式对于构建高性能的日志系统来说是非常有价值的。现在,让我们继续探索 Go 语言的并发特性,开发更多强大且高效的系统吧!

5.2 Channels 的使用 - Go 语言中的通信艺术

Ahoy,并发航海者们!进入 Go 的并发世界后,我们已经学会了如何让多个 Goroutines 舞动起来。现在,是时候让这些舞者学会如何交流了。在 Go 语言中,Channels 是 Goroutines 之间沟通的红绸带,让并发的执行流可以优雅地传递消息。

5.2.1 基础知识讲解

Channels 的定义

Channels 是 Go 语言中的一种类型,用于在 Goroutines 之间进行通信和数据的传递。你可以将 Channel 想象为一条河流,数据就像是河流中的水,可以从一个地方流向另一个地方。

ch := make(chan int)

上面的代码创建了一个传递int类型数据的 Channel。

Channels 的发送和接收

向 Channel 发送数据和从 Channel 接收数据,都使用<-运算符。

ch <- 42 // 向 Channel 发送数据
v := <-ch // 从 Channel 接收数据并赋值给 v

关闭 Channels

当你完成了 Channel 的使用,可以关闭它来防止发生更多的数据发送。接收操作可以继续进行,直到 Channel 中的现有数据都被接收完毕。

close(ch)

5.2.2 重点案例:任务分发系统

在许多应用场景中,我们需要将大量任务分发给不同的工作单元进行并发处理,然后收集和汇总处理结果。这不仅可以显著提高任务处理的效率,还能优化资源的利用。通过使用 Go 语言的 Goroutines 和 Channels,我们可以构建一个高效的任务分发系统。

功能描述

  1. 并发任务处理:创建多个工作 Goroutines 并发处理任务。
  2. 任务队列:使用 Channel 作为任务队列,分发任务给工作 Goroutines。
  3. 结果收集:工作 Goroutines 处理完成后,通过另一个 Channel 将结果返回。

实现代码

首先,定义TaskResult的结构体,以及一个模拟的任务处理函数:

package main

import (
    "fmt"
    "sync"
    "time"
)

type Task struct {
    ID   int
    Data string
}

type Result struct {
    TaskID int
    Output string
}

// 模拟任务处理函数
func processTask(data string) string {
    // 模拟处理时间
    time.Sleep(time.Second)
    return data + " processed"
}

实现工作 Goroutines,从任务 Channel 接收任务,处理任务,并将结果发送到结果 Channel:

func worker(taskChan <-chan Task, resultChan chan<- Result, wg *sync.WaitGroup) {
    defer wg.Done()
    for task := range taskChan {
        // 处理任务
        output := processTask(task.Data)
        resultChan <- Result{TaskID: task.ID, Output: output}
    }
}

构建任务分发和结果收集的主逻辑:

func main() {
    // 创建任务和结果的 Channels
    taskChan := make(chan Task, 10)
    resultChan := make(chan Result, 10)

    // 使用 WaitGroup 等待所有工作 Goroutines 完成
    var wg sync.WaitGroup

    // 启动工作 Goroutines
    for w := 1; w <= 3; w++ {
        wg.Add(1)
        go worker(taskChan, resultChan, &wg)
    }

    // 分发任务
    for i := 1; i <= 5; i++ {
        taskChan <- Task{ID: i, Data: fmt.Sprintf("Task %d", i)}
    }
    close(taskChan)

    // 启动一个 Goroutine 等待所有工作完成后关闭结果 Channel
    go func() {
        wg.Wait()
        close(resultChan)
    }()

    // 收集并打印处理结果
    for result := range resultChan {
        fmt.Printf("Task %d: %s\n", result.TaskID, result.Output)
    }
}

通过这个扩展案例,我们构建了一个灵活且高效的任务分发系统。它展示了如何利用 Go 语言的并发特性来并行处理任务,并通过 Channels 安全地在 Goroutines 之间传递数据。这种模式非常适合于处理那些可以并行化的独立任务,极大地提高了任务处理的速度和效率。现在,就让我们继续探索 Go 语言的并发世界,发现更多的可能性吧!

5.2.3 拓展案例 1:数据流处理

拓展案例 1:数据流处理

数据流处理是一种常见的编程模式,特别适用于需要对数据进行一系列转换或计算的场景。在 Go 语言中,我们可以利用 Channels 和 Goroutines 构建一个高效的数据流处理管道(pipeline),这样可以并发地对数据进行处理,提高处理效率。

功能描述
  1. 创建处理管道:使用 Channels 将一系列的数据处理步骤连接起来,形成一个处理管道。
  2. 并发数据处理:每个处理步骤都运行在独立的 Goroutine 中,以实现并发处理。
  3. 灵活的数据传递:通过 Channels 在管道的各个阶段之间传递数据。
实现代码

首先,定义几个简单的数据处理函数,每个函数代表管道中的一个处理阶段:

package main

import (
    "fmt"
    "strings"
    "time"
)

// stage1:将字符串转换为大写
func stage1(input <-chan string) <-chan string {
    output := make(chan string)
    go func() {
        for s := range input {
            output <- strings.ToUpper(s)
        }
        close(output)
    }()
    return output
}

// stage2:在字符串后添加特定后缀
func stage2(input <-chan string) <-chan string {
    output := make(chan string)
    go func() {
        for s := range input {
            output <- s + " PROCESSED"
        }
        close(output)
    }()
    return output
}

// stage3:模拟耗时操作,如写入数据库
func stage3(input <-chan string) <-chan string {
    output := make(chan string)
    go func() {
        for s := range input {
            // 模拟耗时操作
            time.Sleep(1 * time.Second)
            output <- s + " -> SAVED"
        }
        close(output)
    }()
    return output
}

接着,构建并运行数据流处理管道:

func main() {
    // 初始数据源
    input := make(chan string)
    go func() {
        for _, s := range []string{"data1", "data2", "data3"} {
            input <- s
        }
        close(input)
    }()

    // 构建处理管道
    stage1Output := stage1(input)
    stage2Output := stage2(stage1Output)
    stage3Output := stage3(stage2Output)

    // 收集并打印处理结果
    for result := range stage3Output {
        fmt.Println(result)
    }
}
扩展功能
  • 错误处理:可以在管道的每个阶段添加错误处理逻辑,确保处理过程的健壮性。
  • 动态管道构建:根据实际需求动态地添加或移除处理阶段,使管道更加灵活。

通过这个扩展案例,我们构建了一个并发的数据流处理管道,它展示了如何使用 Go 语言的 Channels 和 Goroutines 来实现数据的并发处理。这种模式非常适合处理大量数据或进行复杂的数据转换和计算任务,能够显著提高处理效率。利用这种模式,我们可以轻松地构建出灵活、高效的数据处理应用。现在,让我们继续探索 Go 语言并发编程的强大功能,开发更多高效的应用吧!

5.2.4 拓展案例 2:实时消息系统

实时消息系统是现代应用中常见的需求,无论是聊天应用、实时数据处理系统还是监控告警系统,都需要快速有效地处理和分发消息。利用 Go 语言的 Channels 和 Goroutines,我们可以构建一个高效且响应迅速的实时消息系统。

功能描述

  1. 消息接收:并发接收来自不同来源的消息。
  2. 消息分发:将接收到的消息分发给多个消费者 Goroutines,以并发方式处理。
  3. 动态消费者管理:能够动态添加或移除消费者 Goroutines。

实现代码

首先,定义消息结构和消费者处理函数:

package main

import (
    "fmt"
    "sync"
    "time"
)

// Message 定义消息结构
type Message struct {
    ID      int
    Content string
}

// consumer 消费者处理函数
func consumer(id int, messages <-chan Message) {
    for msg := range messages {
        fmt.Printf("消费者 %d 处理消息: %v\n", id, msg)
        time.Sleep(time.Second) // 模拟消息处理时间
    }
    fmt.Printf("消费者 %d 结束\n", id)
}

接着,实现消息接收和分发逻辑:

func main() {
    messages := make(chan Message, 10)

    // 启动多个消费者 Goroutines
    var wg sync.WaitGroup
    for i := 1; i <= 3; i++ {
        wg.Add(1)
        go func(id int) {
            defer wg.Done()
            consumer(id, messages)
        }(i)
    }

    // 模拟消息生产
    go func() {
        for i := 1; i <= 5; i++ {
            messages <- Message{ID: i, Content: fmt.Sprintf("消息内容 %d", i)}
        }
        close(messages) // 关闭 Channel,通知消费者结束
    }()

    wg.Wait() // 等待所有消费者 Goroutines 完成
}

扩展功能

  • 消息过滤:在消息分发前添加过滤逻辑,只将符合特定条件的消息分发给消费者。
  • 消费者负载均衡:实现更复杂的分发逻辑,根据消费者的处理能力动态调整其接收的消息量,实现负载均衡。
  • 消息确认机制:引入消息确认机制,确保每条消息都被正确处理,增强系统的可靠性。

通过这个扩展案例,我们构建了一个基本的实时消息系统,它展示了如何使用 Go 语言的并发特性来实现消息的接收、分发和处理。这种模式在需要快速响应的系统中特别有用,能够保证消息在最短时间内被处理。利用 Go 的 Channels 和 Goroutines,我们可以轻松扩展和维护这个系统,以满足不断增长的需求。现在,让我们继续探索 Go 语言,并发编程的可能性,开发出更多功能丰富、响应迅速的应用吧!

5.3 并发模式与同步 - 编织 Go 语言中的并发之网

Ahoy,并发编程的舵手们!在 Go 语言的海洋中,我们不仅需要让 Goroutines 如舞者般自由舞动,还需要确保他们能够和谐地在同一舞台上表演,不发生踩脚或错位的尴尬情况。这就引出了并发模式与同步的主题,它们像是指挥家的手杖,确保每个动作都准确无误地完成。

5.3.1 基础知识讲解

并发模式

并发模式是一组解决并发问题的模板或策略。在 Go 中,常见的并发模式包括但不限于:

  • 管道(Pipeline):通过一系列处理阶段的 Channels 传递数据,每个阶段由 Goroutines 处理。
  • 工作池(Worker Pool):创建一组 Goroutines 来处理任务,可以有效控制并发量,避免资源耗尽。
  • 发布/订阅(Pub/Sub):允许一个或多个生产者发布消息,一个或多个消费者订阅并处理消息。

同步机制

在并发执行时,同步是确保数据一致性和避免竞态条件的关键。Go 语言提供了多种同步机制:

  • WaitGroup:等待一组 Goroutines 完成。
  • Mutex(互斥锁):防止多个 Goroutines 同时访问共享资源。
  • Channel:用于在 Goroutines 之间安全地传递数据。

5.3.2 重点案例:简易聊天服务器

在这个案例中,我们将构建一个简易的聊天服务器,该服务器能够处理多个客户端的连接请求,并实现消息的实时广播功能。通过使用 Go 语言的并发特性,我们可以让服务器同时接受多个客户端连接,并且当任一客户端发送消息时,服务器能够将该消息广播给所有连接的客户端。

功能描述

  1. 客户端连接处理:服务器并发接受来自多个客户端的连接请求。
  2. 实时消息广播:服务器接收到来自任一客户端的消息后,实时将其广播给所有已连接的客户端。
  3. 并发控制:通过同步机制确保服务器在处理客户端连接和消息广播时的线程安全。

实现代码

首先,我们定义聊天服务器的基本结构和构造函数:

package main

import (
    "bufio"
    "fmt"
    "net"
    "sync"
)

// ChatServer 定义聊天服务器的结构
type ChatServer struct {
    clients   map[net.Conn]bool
    broadcast chan string
    lock      sync.Mutex
}

// NewChatServer 创建新的聊天服务器实例
func NewChatServer() *ChatServer {
    return &ChatServer{
        clients:   make(map[net.Conn]bool),
        broadcast: make(chan string),
    }
}

接下来,实现处理客户端连接的方法:

// handleConnection 处理新的客户端连接
func (cs *ChatServer) handleConnection(conn net.Conn) {
    defer conn.Close()

    // 将新客户端添加到 clients 集合中
    cs.lock.Lock()
    cs.clients[conn] = true
    cs.lock.Unlock()

    // 监听客户端发送的消息
    scanner := bufio.NewScanner(conn)
    for scanner.Scan() {
        msg := scanner.Text()
        cs.broadcast <- msg
    }

    // 客户端断开连接后,从 clients 集合中移除
    cs.lock.Lock()
    delete(cs.clients, conn)
    cs.lock.Unlock()
}

实现消息广播的方法:

// startBroadcasting 监听广播频道并向所有客户端广播消息
func (cs *ChatServer) startBroadcasting() {
    for msg := range cs.broadcast {
        cs.lock.Lock()
        for client := range cs.clients {
            fmt.Fprintln(client, msg)
        }
        cs.lock.Unlock()
    }
}

最后,启动聊天服务器,监听端口并接受客户端连接:

// Start 启动聊天服务器
func (cs *ChatServer) Start(port string) {
    listener, err := net.Listen("tcp", "localhost:"+port)
    if err != nil {
        fmt.Println("Error starting server:", err)
        return
    }
    defer listener.Close()

    go cs.startBroadcasting()

    fmt.Println("Chat server started on port", port)
    for {
        conn, err := listener.Accept()
        if err != nil {
            fmt.Println("Error accepting connection:", err)
            continue
        }
        go cs.handleConnection(conn)
    }
}

func main() {
    chatServer := NewChatServer()
    chatServer.Start("8080")
}

扩展功能

  • 昵称支持:允许客户端在连接时设置昵称,将昵称包含在广播的消息中。
  • 私聊功能:实现客户端之间的私聊功能,允许消息只发送给特定的客户端。
  • 客户端退出通知:当客户端断开连接时,服务器向所有客户端广播一条退出通知消息。

通过这个扩展案例,我们展示了如何使用 Go 语言构建一个简易的聊天服务器,它能够处理多个客户端的并发连接并实现实时消息广播。这个案例体现了 Go 语言在并发编程方面的强大能力,通过 Goroutines 和 Channels 轻松管理并发任务和数据通信。现在,让我们继续探索 Go 并发编程的更多可能性,开发出更多功能丰富、响应迅速的应用吧!

5.3.3 拓展案例 1:实时数据监控

在许多现代应用场景中,实时数据监控对于确保系统的稳定性和性能至关重要。通过构建一个实时数据监控系统,我们可以并发地收集、处理和分析来自不同数据源的监控数据,实时反馈系统的运行状况。

功能描述

  1. 并发数据收集:从多个数据源并发收集监控数据。
  2. 数据处理和分析:对收集到的数据进行实时处理和分析,提取有价值的监控指标。
  3. 实时反馈:将处理和分析结果实时展示给用户或触发告警。

实现代码

首先,定义一个模拟的数据收集函数,表示从一个数据源收集数据:

package main

import (
    "fmt"
    "math/rand"
    "sync"
    "time"
)

// collectData 模拟从数据源收集数据
func collectData(source string, dataChan chan<- int) {
    for {
        data := rand.Intn(100) // 模拟生成监控数据
        fmt.Printf("数据源 %s 收集到数据: %d\n", source, data)
        dataChan <- data
        time.Sleep(time.Second * time.Duration(rand.Intn(5))) // 模拟数据收集的间隔
    }
}

接下来,实现数据处理和分析的逻辑,这里简单地模拟数据的处理过程:

// processData 模拟数据处理和分析
func processData(dataChan <-chan int, resultChan chan<- string) {
    for data := range dataChan {
        // 模拟数据处理逻辑
        result := fmt.Sprintf("处理后的数据: %d", data*2)
        resultChan <- result
    }
}

构建主程序逻辑,包括并发的数据收集、处理和实时反馈:

func main() {
    dataSources := []string{"数据源1", "数据源2", "数据源3"}
    dataChan := make(chan int, 10)
    resultChan := make(chan string, 10)

    // 并发收集数据
    for _, source := range dataSources {
        go collectData(source, dataChan)
    }

    // 启动数据处理 Goroutine
    go processData(dataChan, resultChan)

    // 实时展示处理结果
    go func() {
        for result := range resultChan {
            fmt.Println(result)
        }
    }()

    // 模拟主程序运行一段时间后退出
    time.Sleep(30 * time.Second)
    fmt.Println("监控程序结束运行")
}

扩展功能

  • 数据过滤和聚合:在数据处理阶段,可以引入更复杂的逻辑,如对数据进行过滤、聚合等,以提取更有价值的监控指标。
  • 动态数据源管理:实现动态添加或移除数据源的功能,以适应监控需求的变化。
  • 告警机制:根据处理和分析的结果,实现实时告警机制,当监控指标超出预设阈值时触发告警。

通过这个扩展案例,我们构建了一个基本的实时数据监控系统,它展示了如何利用 Go 语言的并发特性来实现数据的实时收集、处理和分析。这种模式适用于需要快速响应和处理大量实时数据的场景,能够帮助我们及时了解和优化系统的运行状况。现在,让我们继续利用 Go 的并发编程能力,开发出更多高效、可靠的实时处理系统吧!

5.3.4 拓展案例 2:并发 Web 爬虫

构建一个并发 Web 爬虫可以显著提高数据抓取的效率,特别适合处理大规模的网页数据收集任务。通过使用 Go 语言的并发特性,我们可以同时对多个网页进行爬取和分析,大大缩短整个抓取过程的时间。

功能描述

  1. 并发爬取网页:使用 Goroutines 并发地对多个网页进行爬取。
  2. 数据提取:从爬取的网页中提取有价值的信息。
  3. 结果汇总:将所有爬取的结果汇总并存储或进行进一步的处理。

实现代码

首先,定义一个模拟的网页爬取函数,表示对单个网页的爬取过程:

package main

import (
    "fmt"
    "math/rand"
    "sync"
    "time"
)

// fetchURL 模拟爬取单个网页,返回模拟的网页内容
func fetchURL(url string) string {
    // 模拟网络延迟
    time.Sleep(time.Millisecond * time.Duration(rand.Intn(500)))
    return fmt.Sprintf("网页内容: [%s]", url) // 模拟返回网页内容
}

接下来,实现并发爬取网页的逻辑,并提取数据:

// crawlWebsite 并发爬取多个网页,并提取数据
func crawlWebsite(urls []string) {
    var wg sync.WaitGroup
    resultChan := make(chan string, len(urls))

    for _, url := range urls {
        wg.Add(1)
        go func(u string) {
            defer wg.Done()
            content := fetchURL(u)
            resultChan <- content // 将抓取结果发送到结果 Channel
        }(url)
    }

    // 等待所有爬取任务完成
    go func() {
        wg.Wait()
        close(resultChan) // 所有任务完成后关闭 Channel
    }()

    // 收集并打印爬取结果
    for result := range resultChan {
        fmt.Println(result)
    }
}

最后,定义主函数,启动并发 Web 爬虫:

func main() {
    urls := []string{
        "http://example.com/page1",
        "http://example.com/page2",
        "http://example.com/page3",
    }

    fmt.Println("开始并发爬取网页...")
    crawlWebsite(urls)
    fmt.Println("所有网页爬取完成。")
}

扩展功能

  • 错误处理:在爬取过程中,增加错误处理逻辑,确保单个任务的失败不会影响整体进程。
  • 限速控制:实现限速控制,防止因请求过快而被目标网站封禁。
  • 动态任务分配:根据任务的完成速度动态调整 Goroutines 的数量,以达到最优的资源利用和爬取效率。

通过这个扩展案例,我们演示了如何构建一个基本的并发 Web 爬虫,它能够有效地提高数据爬取的速度和效率。利用 Go 语言的并发特性,我们可以轻松地扩展爬虫的规模,处理大量的网页爬取任务。这种并发爬虫的模式非常适合进行网页数据的大规模收集和分析。现在,让我们继续探索 Go 并发编程的强大能力,开发出更多高效的应用吧!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1452396.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

异质结太阳能电池中氢化本征非晶硅的设计

在硅异质结太阳能电池&#xff08;SHJ&#xff09;中&#xff0c;pn结由两种不同形貌的硅形成&#xff0c;即一种是n型晶体硅&#xff08;c-Si&#xff09;&#xff0c;另一种是p掺杂&#xff08;III族&#xff09;元素掺杂&#xff09;非晶硅&#xff08;a-Si&#xff09;。许…

静态时序分析:SDC约束命令set_clock_transition详解

相关阅读 静态时序分析https://blog.csdn.net/weixin_45791458/category_12567571.html?spm1001.2014.3001.5482 在静态时序分析&#xff1a;SDC约束命令create_clock详解一文的最后&#xff0c;我们谈到了针对理想(ideal)时钟&#xff0c;可以使用set_clock_transition命令直…

java数据结构与算法刷题-----LeetCode696. 计数二进制子串

java数据结构与算法刷题目录&#xff08;剑指Offer、LeetCode、ACM&#xff09;-----主目录-----持续更新(进不去说明我没写完)&#xff1a;https://blog.csdn.net/grd_java/article/details/123063846 文章目录 1. 法一&#xff1a;时间复杂度O(n).空间复杂度O(n)2. 法二&…

【C++ QT项目2】——高仿安信可串口调试助手

【C QT项目2】——高仿安信可串口调试助手 1. 项目概述2. 项目UI设计3. 串口通信核心代码开发3.1 QSerialPort介绍及示例3.2 扫描系统串口3.3 数据的收发3.4 定时发送&#xff08;QT定时器&#xff09;3.5 HEX显示与发送 4. 串口调试助手功能的优化4.1 串口的实时扫描4.2 获取系…

skimage库简介

scikit-image 是专注于图像处理的Python包&#xff0c;全称是scikit-image SciKit。该包由python语言编写&#xff0c;由scipy 社区开发和维护&#xff0c;使用原生的Numpy数组作为图像对象。 一、skimage简介 skimage&#xff08;scikit-Image&#xff09;是基于python开发的…

petalinux安装

跟着正点原子文档安装的&#xff0c;记录一下 1. 安装包下载 xilinx官网&#xff08;没有注册需要注册&#xff0c;注册比较慢&#xff0c;嫌弃耽搁时间直接用正点原子网盘里下好的&#xff09; https://china.xilinx.com/support/download/index.html/content/xilinx/zh/dow…

人工智能学习与实训笔记(十四):Langchain Agent

0、概要 Agent是干什么的&#xff1f; Agent的核心思想是使用语言模型&#xff08;LLM&#xff09;作为推理的大脑&#xff0c;以制定解决问题的计划、借助工具实施动作。在agents中几个关键组件如下&#xff1a; Agent&#xff1a;制定计划和思考下一步需要采取的行动。Tools…

拿捏单链表

目录 引言 一&#xff1a;链表的定义 二&#xff1a;单链表的定义 三&#xff1a;单链表的增删查改 1.单链表增删查改及遍历的声明 注&#xff1a;在测试中创建指向头结点的指针plist 2.二级指针应用的说明 3.单链表的遍历 4.创建节点 5.单链表的插入 (1)头插 …

【深度学习:DICOM 注释工具】在 DICOM 注释工具中寻找的 7 个功能

【深度学习&#xff1a;DICOM 注释工具】在 DICOM 注释工具中寻找的 7 个功能 原生 DICOM 支持原生 3D 注释易于使用的界面DICOM 图像的自动注释质量控制功能审计跟踪SOC2 和 HIPAA 合规性 如果您尝试为医疗 AI 模型创建训练数据&#xff0c;您可能已经使用了免费的开源工具&am…

html从零开始9:javaScript简介,语句、标识符,变量,JavaScript引入到文件【搬代码】

javaScript简介 javaScript语句、标识符 变量 var num 10; var就是固定声明,num就是变量名&#xff0c;10就是变量&#xff1b;<!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta http-equiv"X-UA-Comp…

【pyopenGL编程手册- 01/20】pyopenGL安装和简要说明

目录 一、说明二、测试系统安装的健康性三、安装64位的openGL四、写给程序员的4. 1 函数库介绍4.2 库内函数的命名 五、常见库的函数介绍5.1 OpenGL 核心库 GL5.2 OpenGL 实用库 GLU5.3 OpenGL 工具库 GLUT5.4 Windows 专用库 WGL 六、错误引发点和异常追踪6.1 错误检查开关6.…

人工智能学习与实训笔记(五):神经网络之推荐系统处理

目录 ​​​​​​​七、智能推荐系统处理 7.1 常用的推荐系统算法 7.2 如何实现推荐​​​​​​​ 7.3 基于飞桨实现的电影推荐模型 7.3.1 电影数据类型 7.3.2 数据处理 7.3.4 数据读取器 7.3.4 网络构建 7.3.4.1用户特征提取 7.3.4.2 电影特征提取 7.3.4.3 相似度…

TenorFlow多层感知机识别手写体

文章目录 数据准备建立模型建立输入层 x建立隐藏层h1建立隐藏层h2建立输出层 定义训练方式建立训练数据label真实值 placeholder定义loss function选择optimizer 定义评估模型的准确率计算每一项数据是否正确预测将计算预测正确结果&#xff0c;加总平均 开始训练画出误差执行结…

C++初阶(十一) list

一、list的介绍及使用 1.1 list的介绍 list的文档介绍 1. list是可以在常数范围内在任意位置进行插入和删除的序列式容器&#xff0c;并且该容器可以前后双向迭代。 2. list的底层是双向链表结构&#xff0c;双向链表中每个元素存储在互不相关的独立节点中&#xff0c;在节点…

爱上JVM——常见问题(一):JVM组成

1 JVM组成 1.1 JVM由那些部分组成&#xff0c;运行流程是什么&#xff1f; 难易程度&#xff1a;☆☆☆ 出现频率&#xff1a;☆☆☆☆ JVM是什么 Java Virtual Machine Java程序的运行环境&#xff08;java二进制字节码的运行环境&#xff09; 好处&#xff1a; 一次编写&…

巨抽象的前端vue3

根据实践证明&#xff0c;越是简单的问题&#xff0c;越容易造成大bug 一个自定义组件的路径就废了我老半天了 各种查询&#xff0c;各种百度&#xff0c;各种问&#xff0c;结果规规矩矩去导入组件路径&#xff0c;成了&#xff01; 错误代码&#xff1a; <script setu…

canal监听binlog记录业务数据的变更;canalAdmin对instance做web配置

概述 平时在开发中会通过logback打印一些开发日志&#xff0c;有时也会需要记录一些业务日志&#xff0c;简单的就直接用log记录一下&#xff0c;但是系统中需要记录日志的地方越来越多时&#xff0c;不能每个地方都写一套log记录&#xff1b; 由于平常用的大多都是mysql&…

c语言遍历文件夹中的文件

文件目录如下&#xff0c;文件夹里还有一些txt文件未展示出来。 使用递归实现&#xff0c;深度优先遍历文件夹中的文件。 代码如下&#xff0c;用了一点C的语法。 #include <io.h> #include <iostream> using namespace std;#define MAX_PATH_LENGTH 100int Tr…

创新技巧|迁移到 Google Analytics 4 时如何保存历史 Universal Analytics 数据

Google Universal Analytics 从 2023 年 7 月起停止收集数据&#xff08;除了付费 GA360 之外&#xff09;。它被Google Analytics 4取代。为此&#xff0c;不少用户疑惑&#xff1a;是否可以将累积&#xff08;历史&#xff09;数据从 Google Analytics Universal 传输到 Goog…

@ControllerAdvice 的介绍及三种用法

ControllerAdvice 的介绍及三种用法 浅析ControllerAdvice 首先&#xff0c;ControllerAdvice本质上是一个Component&#xff0c;因此也会被当成组建扫描&#xff0c;一视同仁&#xff0c;扫扫扫。 然后&#xff0c;我们来看一下此类的注释&#xff1a; 这个类是为那些声明了&…