【后端开发】Go语言编程实践,Goroutines和Channels,基于共享变量的并发,反射与底层编程

【后端开发】Go语言编程实践,Goroutines和Channels,基于共享变量的并发,反射与底层编程
【后端开发】Go语言高级编程,CGO、Go汇编语言、RPC实现、Web框架实现、分布式系统

文章目录

      • 1、并发基础, Goroutines和Channels
      • 2、基于共享变量的并发, sync.WaitGroup和sync.Mutex
      • 3、反射与底层编程

参考资料:

  • 1、框架
    go语言源码-124k,
    go精选框架-133k,
    go-gin-HTTP Web框架-80k,
    go-Llama 3.2框架-90k,
    rclone-云存储挂载-50k,
    go-zero-原生微服务框架-20k

  • 2、教程
    The Go programming language - 124k
    Go语言圣经 《The Go Programming Language》 中文版-4.5k
    Go语言高级编程-20k,
    go成长路线-6k,
    go学习指南-3k
    go基础语法-菜鸟

  • 3、应用
    7天实现web-gee和数据库orm框架-16k
    构建goweb应用-43k

1、并发基础, Goroutines和Channels

Goroutine 是 Go 中实现并发的基本单位。它是一种轻量级线程,使用 go 关键字启动。

  • 如果你使用过操作系统或者其它语言提供的线程,那么你可以简单地把goroutine类比作一个线程,这样你就可以写出一些正确的程序了。
  • 当一个程序启动时,其主函数即在一个单独的goroutine中运行,我们叫它main goroutine。新的goroutine会用go语句来创建。
  • 在语法上,go语句是一个普通的函数或方法调用前加上关键字go。go语句会使其语句中的函数在一个新创建的goroutine中运行。而go语句本身会迅速地完成。
go func() {// 这里是并发运行的代码
}()func aaa(str string){}
go aaa("aa")f()    // call f(); wait for it to return
go f() // create a new goroutine that calls f(); don't wait

Channels

  • 如果说goroutine是Go语言程序的并发体的话,那么channels则是它们之间的通信机制。一个channel是一个通信机制,它可以让一个goroutine通过它给另一个goroutine发送值信息。
  • 每个channel都有一个特殊的类型,也就是channels可发送数据的类型。一个可以发送int类型数据的channel一般写为chan int。
  • 和map类似,channel也对应一个make创建的底层数据结构的引用。当我们复制一个channel或用于函数参数传递时,我们只是拷贝了一个channel引用,因此调用者和被调用者将引用同一个channel对象。和其它的引用类型一样,channel的零值也是nil。
ch := make(chan int) ch <- x  // a send statement
x = <-ch // a receive expression in an assignment statement
<-ch     // a receive statement; result is discardedclose(ch) // 随后对基于该channel的任何发送操作都将导致panic异常。ch = make(chan int)    // unbuffered channel
ch = make(chan int, 0) // unbuffered channel
ch = make(chan int, 3) // buffered channel with capacity 3

操作不同状态的chan会引发三种行为:

panic

  • 向已经关闭的通道写数据
  • 重复关闭通道

阻塞

  • 向未初始化的通道写/读数据
  • 向缓冲区已满的通道写入数据
  • 通道中没有数据,读取该通道

非阻塞

  • 读取已经关闭的通道,这个操作会返回通道元素类型的零值(可用comma, ok语法)
  • 向有缓冲且没有满的通道读/写
package mainimport ("testing"
)func TestChanOperateStatus(t *testing.T) {t.Run("向已经关闭的通道写数据", func(t *testing.T) {ch := make(chan int)close(ch) // 关闭通道ch <- 1   // 这里会引发panic,因为向已关闭的通道发送数据// panic: send on closed channel [recovered]})t.Run("重复关闭通道", func(t *testing.T) {ch := make(chan int)close(ch) // 第一次关闭通道close(ch) // 再次关闭通道会引发panic// panic: close of closed channel [recovered]})t.Run("向未初始化的通道写/读数据", func(t *testing.T) {var ch chan intgo func() {ch <- 1// x := <-ch}()_ = <-ch// fatal error: all goroutines are asleep - deadlock!})t.Run("向缓冲区已满的通道写入数据", func(t *testing.T) {ch := make(chan int, 1)ch <- 1 // 第一次写入,缓冲区未满ch <- 2 // 这里会阻塞,因为缓冲区已满,没有goroutine读取数据// fatal error: all goroutines are asleep - deadlock!})t.Run("通道中没有数据,读取该通道", func(t *testing.T) {ch := make(chan int)_ = <-ch // 这里会阻塞,因为没有goroutine发送数据到通道// fatal error: all goroutines are asleep - deadlock!})t.Run("读取已经关闭的通道,这个操作会返回通道元素类型的零值(可用comma, ok语法)", func(t *testing.T) {ch := make(chan int)close(ch)     // 关闭通道x, ok := <-ch // x 将会是int类型的零值,ok 将会是falseexpectx, expectok := 0, falseif ok != expectok && x != expectx {t.Errorf("expect 0, false, get %d, %t\n", x, ok)}})t.Run("向有缓冲且没有满的通道写,向有缓冲且不为空的通道读", func(t *testing.T) {ch := make(chan int, 2) // 1 也不会堵塞ch <- 1                 // 写入数据,不会阻塞_ = <-ch                // 读取数据,不会阻塞})
}

2、基于共享变量的并发, sync.WaitGroup和sync.Mutex

sync.WaitGroup 计数器,等待并发完成
sync.Mutex 互斥锁,保护共享资源
闭包,捕获外部变量的值

sync.WaitGroup 是一个计数器,用于等待一组 goroutine 完成。使用它的步骤如下:

  • 添加计数:使用 Add(n int) 方法增加计数,通常在启动 goroutine 之前调用。
  • 完成计数:在 goroutine 内部,使用 Done() 方法来减少计数。
  • 等待完成:使用 Wait() 方法阻塞当前 goroutine,直到计数器变为零。
var wg sync.WaitGroupfor i := 0; i < n; i++ {wg.Add(1) // 增加计数go func(i int) {defer wg.Done() // 在完成时减少计数// 某些操作}(i)
}wg.Wait() // 等待所有 goroutine 完成

sync.Mutex 是一种互斥锁,用于保护共享资源,确保同一时间只有一个 goroutine 访问它。使用方法如下:

  • 锁定:使用 Lock() 方法加锁,确保线程安全。
  • 解锁:使用 Unlock() 方法解锁。通常推荐使用 defer 来确保在函数退出时解锁。
var mtx sync.Mutexmtx.Lock() // 上锁
// 对共享资源的访问
mtx.Unlock() // 解锁

在并发操作中,通过收集错误并处理它们也是很常见的做法。可以使用切片来存储可能发生的错误,并且在访问这个切片时,需要使用 Mutex 来保证线程安全。

var errs []error
var mtx sync.Mutexif err != nil {mtx.Lock()errs = append(errs, err)mtx.Unlock()
}

闭包
在 goroutine 内部,可以使用闭包来捕获外部变量的值。这对于确保在并发执行时每个 goroutine 使用到的是正确的变量非常重要。

for _, value := range values {go func(v string) {// 使用 v,确保 v 是当前循环中的值}(value)
}

结合以上组件,可以实现并发的操作,例如:

var wg sync.WaitGroup
var mtx sync.Mutex
var errs []errorfor _, id := range ids {wg.Add(1)go func(id string) {defer wg.Done()// 假设这里是某个并发操作if err := doSomething(id); err != nil {mtx.Lock()errs = append(errs, err)mtx.Unlock()}}(id)
}wg.Wait()// 处理错误
if len(errs) > 0 {// 处理错误逻辑
}

3、反射与底层编程

反射:遍历不确定的结构体的每个字段,可以用反射来获取结构体的字段的值 。以及判断字段数据类型,在调用适当的函数,做神奇操作。
底层编程:cgo

反射是由 reflect 包提供的。它定义了两个重要的类型,Type 和 Value。一个 Type 表示一个Go类型。

  • 反射是程序在运行时能够“观察”并且修改自己的行为的能力。在Go语言中,反射是通过reflect包实现的,它提供了两个核心功能:Type和Value
  • 获取Type和Value:使用reflect.TypeOf()和reflect.ValueOf()可以获取变量的动态类型和值。TypeOf返回的是Type接口,而ValueOf返回的是Value接口
  • 类型和值的查询:通过Type和Value接口的方法,可以查询变量的类型信息和值。例如,Kind()方法可以返回一个常量,表示底层数据类型,如Uint、Float64、Slice等
  • 修改值:使用Value的Set()方法可以修改值,但需要注意的是,只有当值是可设置的(settable)时才能修改。可设置性意味着值必须通过指针传递,并且使用Elem()方法获取指针指向的值进行修改
  • 动态方法调用和字段访问:反射不仅可以用于基础类型和结构体,还可以用于动态地调用方法和访问字段。例如,可以通过MethodByName方法动态调用对象的方法
package mainimport ("fmt""reflect"
)// 定义一个示例结构体
type Employee struct {Name     stringAge      intSalary   float64Active   bool
}// 一个通用函数,使用反射打印并修改结构体字段
func inspectAndModify(v interface{}, newValue interface{}) {// 获取传入值的反射值val := reflect.ValueOf(v)// 检查是否是指针类型,如果是,获取其元素if val.Kind() == reflect.Ptr {val = val.Elem()}// 打印结构体的字段fmt.Printf("结构体类型: %s\n", val.Type())fmt.Println("字段:")for i := 0; i < val.NumField(); i++ {field := val.Field(i)fieldType := val.Type().Field(i)// 打印字段名称和类型fmt.Printf("  %s (%s): %v\n", fieldType.Name, fieldType.Type, field)// 这里我们简单示例,用传入的值替换第一个可设置的字段if i == 0 && field.CanSet() {value := reflect.ValueOf(newValue)if value.Type().AssignableTo(field.Type()) {field.Set(value)fmt.Printf("  %s字段已更新为: %v\n", fieldType.Name, field)} else {fmt.Printf("  无法将值%v赋给字段%s,类型不匹配。\n", newValue, fieldType.Name)}}}
}func main() {// 创建一个 Employee 实例emp := &Employee{Name: "Alice", Age: 30, Salary: 65000.0, Active: true}// 使用 inspectAndModify 函数打印结构体信息并修改字段inspectAndModify(emp, "Bob")// 打印修改后的结构体fmt.Println("修改后的 Employee:", emp)
}// 结构体类型: main.Employee
// 字段:
//   Name (string): Alice
//   Name字段已更新为: Bob
//   Age (int): 30
//   Salary (float64): 65000
//   Active (bool): true
// 修改后的 Employee: &{Bob 30 65000 true}
package mainimport ("fmt""reflect"
)type Person struct {Name    stringAge     intAddress string
}func main() {// 创建一个Person实例p := Person{Name:    "John Doe",Age:     30,Address: "123 Main St",}// 获取Person实例的反射值v := reflect.ValueOf(p)// 确保v是一个结构体if v.Kind() == reflect.Struct {// 遍历结构体的所有字段for i := 0; i < v.NumField(); i++ {field := v.Field(i)// 获取字段的名称fieldName := v.Type().Field(i).Name// 获取字段的值fieldValue := field.Interface()fmt.Printf("Field: %s, Value: %v\n", fieldName, fieldValue)}}
}

Go的底层编程

  • Go的底层编程涉及到更接近硬件和操作系统的细节,包括内存管理、指针操作等。
  • unsafe包:Go提供了unsafe包,它允许程序员绕过Go的类型系统,进行指针操作和内存对齐等操作。unsafe包中的Sizeof、Alignof和Offsetof可以用于获取类型的存储大小、内存对齐和字段偏移量
  • unsafe.Pointer:unsafe.Pointer是一个特殊的类型,它可以存储任何类型的指针,并允许进行指针转换和算术操作
  • 调用C代码:通过cgo,Go程序可以调用C语言代码,这涉及到底层的内存管理和类型转换
  • 性能考虑:底层编程和反射操作通常比直接的Go操作要慢,因为它们涉及到额外的动态查询和类型转换。因此,在性能敏感的应用中需要谨慎使用
  • 要编译和运行这个示例,你需要确保你的Go环境可以使用cgo。通常,对于大多数操作系统,cgo是默认启用的。

调用c语言代码

// main.go
package main/*
#include <stdio.h>
#include <stdlib.h>// 定义转置函数
void transpose(int* src, int* dest, int rows, int cols) {for (int i = 0; i < rows; i++) {for (int j = 0; j < cols; j++) {dest[j * rows + i] = src[i * cols + j];}}
}
*/
import "C"
import ("fmt""unsafe"
)func main() {// 定义一个3x3的矩阵rows, cols := 3, 3src := []int{1, 2, 3, 4, 5, 6, 7, 8, 9}dest := make([]int, rows*cols)// 打印原始矩阵fmt.Println("原始矩阵:")printMatrix(src, rows, cols)// 调用C代码进行转置C.transpose((*C.int)(unsafe.Pointer(&src[0])), (*C.int)(unsafe.Pointer(&dest[0])), C.int(rows), C.int(cols))// 打印转置后的矩阵fmt.Println("转置后的矩阵:")printMatrix(dest, cols, rows)
}func printMatrix(matrix []int, rows, cols int) {for i := 0; i < rows; i++ {for j := 0; j < cols; j++ {fmt.Printf("%d ", matrix[i*cols+j])}fmt.Println()}
}

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

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

相关文章

人机交互革命,为智能座舱市场激战注入一针「催化剂」

从AIGC到AGI赋能&#xff0c;智能座舱人机交互体验迎来新范式。 不断训练、迭代的大模型&#xff0c;为智能座舱带来了更全面的感知能力、更准确的认知理解&#xff0c;以及更丰富的交互模态&#xff0c;显著提升了其智能化水平。 “AI大模型的快速应用与迭代&#xff0c;推动…

计算机视觉硬件知识点整理六:工业相机选型

文章目录 前言一、工业数字相机的分类二、相机的主要参数三、工业数字摄像机主要接口类型四、选择工业相机的考量因素六、实例分析 前言 随着科技的不断进步&#xff0c;工业自动化领域正经历着前所未有的变革。作为工业自动化的重要组成部分&#xff0c;工业相机在工业检测、…

如何使用brew安装phpredis扩展?

如何使用brew安装phpredis扩展&#xff1f; phpredis扩展是一个用于PHP语言的Redis客户端扩展&#xff0c;它提供了一组PHP函数&#xff0c;用于与Redis服务器进行交互。 1、cd到php某一版本的bin下 /usr/local/opt/php8.1/bin 2、下载 phpredis git clone https://githu…

硬件看门狗工作原理

硬件看门狗是什么&#xff1f; 硬件看门狗&#xff08;Hardware Watchdog&#xff09;是一种用于监控系统运行状态的硬件设备或电路。它的主要功能是检测系统是否正常运行&#xff0c;并在系统出现故障或无响应时自动重启或采取其他恢复措施。 工作原理与引脚 硬件看门狗一般…

神经网络中的优化方法(一)

目录 1. 与纯优化的区别1.1 经验风险最小化1.2 代理损失函数1.3 批量算法和小批量算法 2. 神经网络中优化的挑战2.1 病态2.2 局部极小值2.3 高原、鞍点和其他平坦区域2.4 悬崖和梯度爆炸2.5 长期依赖2.6 非精确梯度2.7 局部和全局结构间的弱对应 3. 基本算法3.1 随机梯度下降(小…

海康gige工业相机无驱动取像突破(c#实现,最后更新,你也可以移植到linux下去用)

买了3个海康的相机&#xff0c;最初测试成功的是500万相机。 然后写了一个通用版&#xff0c;害怕有问题&#xff0c;又买了600万的相机&#xff0c;测试果然不及格&#xff0c;花了九牛二虎之力找到一个小问题&#xff0c;就这个 if (changdu > 1000)&#xff1b; 最后又…

Linux -初识 与基础指令1

博客主页&#xff1a;【夜泉_ly】 本文专栏&#xff1a;【Linux】 欢迎点赞&#x1f44d;收藏⭐关注❤️ 文章目录 &#x1f4da; 前言&#x1f5a5;️ 初识&#x1f510; 登录 root用户&#x1f465; 两种用户➕ 添加用户&#x1f9d1;‍&#x1f4bb; 登录 普通用户⚙️ 常见…

Elasticsearch在liunx 中单机部署

下载配置 1、下载 官网下载地址 2、上传解压 tar -zxvf elasticsearch-XXX.tar.gz 3、新建组和用户 &#xff08;elasticsearch 默认不允许root账户&#xff09; #创建组 es groupadd es #新建用户 useradd ryzhang -g es 4、更改文件夹的用户权限 chown -R ryzhang …

Refit 使用详解

Git官网&#xff1a;https://github.com/reactiveui/refit Refit 是一个针对 .NET 应用程序的 REST API 客户端库&#xff0c;它通过接口定义 API 调用&#xff0c;从而简化与 RESTful 服务的交互。其核心理念是利用声明性编程的方式来创建 HttpClient 客户端&#xff0c;使得…

《山海经》:北山

《山海经》&#xff1a;北山 北山一经单狐山求如山&#xff08;水马&#xff1a;形状与马相似&#xff0c;滑鱼&#xff1a;背部红色&#xff09;带山&#xff08;䑏疏&#xff1a;似马&#xff0c;一只角&#xff0c;鵸鵌&#xff1a;状乌鸦五彩斑斓&#xff0c;儵鱼&#xff…

使用gemini-1.5-pro-002做视频检测

使用Google Cloud Video Intelligence API做视频检测最大的缺陷是无法自定义规则&#xff0c;若使用gemini-1.5-pro-002多模拟模型&#xff0c;则可以自定义检测的规则&#xff0c;具有更好的灵活性。 安装SDK pip install --upgrade google-cloud-aiplatform gcloud auth ap…

动态规划——子序列问题

文章目录 目录 文章目录 前言 一、动态规划思路简介 二、具体实现 1. 字符串问题 1.1 最长公共字符串 1.2.0 最长回文子串 1.2.1 最长回文子序列 2.编辑距离问题 2.1 判断子序列 2.2 编辑距离 总结 前言 上文提到动态规划的背包问题&#xff0c;本文继续介绍动态…

Ubuntu24.04配置DINO-Tracker

一、引言 记录 Ubuntu 配置的第一个代码过程 二、更改conda虚拟环境的默认安装路径 鉴于不久前由于磁盘空间不足引发的重装系统的惨痛经历&#xff0c;在新系统装好后当然要先更改虚拟环境的默认安装路径。 输入指令&#xff1a; conda info可能因为我原本就没有把 Anacod…

vulnhub靶场【哈利波特】三部曲之Aragog

前言 使用virtual box虚拟机 靶机&#xff1a;Aragog : 192.168.1.101 攻击&#xff1a;kali : 192.168.1.16 主机发现 使用arp-scan -l扫描&#xff0c;在同一虚拟网卡下 信息收集 使用nmap扫描 发现22端口SSH服务&#xff0c;openssh 80端口HTTP服务&#xff0c;Apach…

顶刊算法 | 鱼鹰算法OOA-BiTCN-BiGRU-Attention多输入单输出回归预测(Maltab)

顶刊算法 | 鱼鹰算法OOA-BiTCN-BiGRU-Attention多输入单输出回归预测&#xff08;Maltab&#xff09; 目录 顶刊算法 | 鱼鹰算法OOA-BiTCN-BiGRU-Attention多输入单输出回归预测&#xff08;Maltab&#xff09;效果一览基本介绍程序设计参考资料 效果一览 基本介绍 1.Matlab实…

getchar()

getchar():从计算机终端&#xff08;一般是键盘&#xff09;输入一个字符 1、getchar返回的是字符的ASCII码值&#xff08;整数&#xff09;。 2、getchar在读取结束或者失败的时候&#xff0c;会返回EOF 输入密码并确认&#xff1a; scanf读取\n之前的内容即12345678 回车符…

动态规划-----路径问题

动态规划-----路径问题 下降最小路径和1&#xff1a;状态表示2&#xff1a;状态转移方程3 初始化4 填表顺序5 返回值6 代码实现 总结&#xff1a; 下降最小路径和 1&#xff1a;状态表示 假设&#xff1a;用dp[i][j]表示&#xff1a;到达[i,j]的最小路径 2&#xff1a;状态转…

网络安全防护指南

网络安全防护指南 网络安全是指保护网络系统中的硬件、软件及数据不受偶然或恶意原因而遭到破坏、更改或泄露&#xff0c;确保网络系统连续可靠地正常运行。随着互联网的普及和技术的发展&#xff0c;网络安全问题日益严峻&#xff0c;对个人、企业和国家都构成了巨大威胁。因…

实现PDF文档加密,访问需要密码

01. 背景 今天下午老板神秘兮兮的来问我&#xff0c;能不能做个文档加密功能&#xff0c;就是那种用户下载打开需要密码才能打开的那种效果。boss都发话了&#xff0c;那必须可以。 需求&#xff1a;将 pdf 文档经过加密处理&#xff0c;客户下载pdf文档&#xff0c;打开文档需…

HarmonyOS Next 模拟器安装与探索

HarmonyOS 5 也发布了有一段时间了&#xff0c;不知道大家实际使用的时候有没有发现一些惊喜。当然随着HarmonyOS 5的更新也带来了很多新特性&#xff0c;尤其是 HarmonyOS Next 模拟器。今天&#xff0c;我们就来探索一下这个模拟器&#xff0c;看看它能给我们的开发过程带来什…