当前位置: 首页 > news >正文

【Go语言】

type关键字的用法

  • 定义结构体
  • 定义接口
  • 定义类型别名
  • 类型定义
  • 类型判断

别名实际上是为了更好地理解代码/

这里要分点进行记录  使用传值的例子,当两个类型不一样需要进行类型转换

type Myint int  // 自定义类型,基于已有的类型自定义一个类型type Myint = int
// 在编译的时候,类型别名会被直接替换为int

上面两种方式定义的int是不同的,因为使用等于号是类型别名,不使用等于号的是重新定义了一个类型。

type Myint int 

如果我们即想使用int类型,又想让这种类型的变量有自己的方法,我们可以使用type自定义一种类型,这其实也算是一种结构体,我们可以为这个结构体创建一个方法,我们就可以进行调用这个方法。

type Myint=int

这个语句就是单独地给int变量起一个别名,其他用法和int一模一样。

结构体的定义和初始化

在go语言中没有面向对象的概念,但是可以使用面向对象的思想

type Person struct {name stringage intaddress stringheight float32
}// 在进行初始化的时候,我们有两种方式:
p1 := Peason{"xiaomin", 14, "佳", 1.8}p2 := Person{name:"xiaomin",age:14,
}

匿名结构体  匿名函数

只能使用一次。

address := struct {province stringcity stringaddress string
}{"北京市","通州区","xxx",
}

结构体嵌套

有两种嵌套方式

type Person struct {name stringage int
}type student struct {p Personscore int
}type student struct {Personscore int
}

匿名定义中有一些问题,如果匿名结构体中有name,但是在外边的结构体中也有name,那么我们会发现外部的结构体的name的优先级大于匿名结构体中的优先级。

结构体定义方法

接收器有两种形态,一种是值传递,一种是指针传递,当我们需要进行修改结构体中的数值或者传递的结构体的容量很大时,我们需要进行指针传递。

如果接收器是指针类型,但是我们是值类型,我们可以进行调用函数的,同理,另一种情况也是可以的。

指针:

go语言中限制了指针的运算,在C语言中,我们可以使用指针进行加减运算, 

指针的初始化

指针是需要进行初始化的

在初始化的时候,有两个关键字:make ,new  

map channel  slice  初始化推荐使用make方法

指针初始化推荐使用new函数,指针需要进行初始化,否则会出现nil pointer

map必须初始化,否则会出现panic 

nil在go中的细节

不同类型的数据的零值不一样,

bool false

numbers 0

string ""

pointer nil

slice nil

map nil

channel  interface function nil

struct 的默认值不是nil,默认值是具体字段的默认值

slice的底层介绍???????等会看一看

map也分为两种 nil map 和 empty map 在大部分场景下,nil map 和 empty map一样,但是当我们给nil map 进行赋值的时候,会进行panic

接口

什么是鸭子类型?

方法。动词,具备某些方法,go语言中强调的是动作

鸭子类型强调的是外部行为,而不是内部的结构

如何定义一个接口

type Duck interface{// 方法的申请Gaga()Walk()Swiming()
}type pskDuck struct {legs int
}func (pd *pskDuck) Gaga() {fmt.Println("this is walking...")
}func main() {var d Duck = &pskDuck{}d.Gaga()
}

多接口的实现  不是很懂

如果有两个接口中的函数出现重复,说明我们可以将函数进行复用,我们可以实现接口进行解耦。

type MyWriter interface{Write(string) error
}type MyCloser interface{Close() error
}type WriteClose struct {MyWriter  // interface 也是一种类型 
}func (wc *WriteClose) Writer(s string) error {fmt.Println("this is writer)return nil
}func (wc *WriteClose) Close() error {fmt.Println("this is closer")return nil
}

在go语言中,多接口的实现是通过组合接口来实现的,go不支持多重继承,但是可以让一个类型实现多个接口,具体来说,只要一个类型实现了某些方法,他就自动实现了相应的接口。通过这种方式,go语言允许一个类型轻松实现多个接口,促进了灵活的设计和代码复用。

通过interface解决动态类型传参

我们在实现一些

使用断言转换类型  

但是还是没有解决问题

使用switch语句,断言处理分支语句可以用到

a.(type) 可以判断是什么类型 

但是更通用的写法是泛型。

string.Split

接口嵌套

接口遇到了slice的常见错误

单元测试

Go的并发编程

go的并发编程初体验

       在php,java,python等语言中,都是有多线程编程,多进程编程,多线程和多进程存在的问题主要是耗费内存。在进行进程切换和线程切换的时候,CPU会消耗一些资源。因此在go语言中,我们推出了协程(用户级线程,绿程,轻量级线程)。

       在协程中,内存占用小(2k),切换快,在go语言的协程中,go余元诞生之后就只有协程可以使用——goroutine,非常方便。我们可以在程序中使用协程创建出10万以上的协程,但是线程在程序中只能创建出100多个。下面,我们来看一看协程的代码:

func asyncPrint() {fmt.Println("加油旭杏")
}// 主协程
func main() {go asyncPrint()
}

       上述代码是单个协程的调用,如果我们想使用多协程进行调用,我们可以使用循环来创建出多个协程。在创建协程的时候,我们有两种方式可以进行创建:一种是直接调用函数,一种是匿名函数。代码如下:

func Print() {fmt.Println("加油旭杏")
}// 匿名结构体
func main() {go func() {for {fmt.Println("加油旭杏")}}time.Sleep(10 * time.Second)
}

       这里为了使程序不退出,我们在主协程中添加了睡眠时间,但是这种方式是非常挫的,在下面我们将会学习另一种等待协程的方法。

有一个问题:在C++中,如果有多个线程去执行一个加法运算,再执行一个减法运算会出现错误的,我们需要使用锁来进行让加减操作变成原子操作。如果操作简单,我们也可以使用原子变量来进行操作。

我们来看一看下面的代码:

for i := 0; i < 100; i++ {go func() {fmt.Println(i)}()
}

       第一个原因是我们使用匿名函数,也就是闭包,只能使用一次,第二个原因是我们使用了for循环,在for循环中,每一个变量会被重新使用,所以我们在调用加操作之前的变量和调用加操作之后的变量不是同一个变量,因为这个不是原子操作,也会CPU的调度有关。

go的gmp调度原理

这个gmp调度是分层的,我们先来看一看两层结构:一个是协程,一个是线程;在Go语言中,我们会将协程放入到线程去执行,

一张图   g  gorounte  m     p

我们在协程调度的时候,我们需要将

通过waitgroup来等待协程

       子gorounte如何通知到主的gorount自己结束了,主的gorounte如何知道子的gorounte已经结束了??

       之前我们使用的例子是通过设置时间长度,但是我们不知道具体需要多少时间,所以这个代码是比较挫的,现在我们来认识一下WaitGroup。我们可以通过WaitGroup来将协程加入,如果我们事先知道协程的总数,可以直接在循环之前进行添加;如果我们不知道协程的总数,我们可以在循环的时候进行添加,每循环一次就添加一次,在添加之后我们需要将其进行done,类似于锁的机制一样,否则会造成死锁。因此,我们需要知道WaitGroup的add和done是需要配套使用的。最后我们需要进行wait方式进行等待。代码如下:

var wg sync.WaitGroup// 我们需要监控多少个gotountine执行结束
wg.Add(100)
for i := 0; i < 100; i++ {go func(i int) {fmt.Println(i)wg.Done(); // Done和Add要成对出现}()
}// 等待
wg.Wait()

       waitgroup主要用于gorountine的执行等待,add方法主要和done方法配套,最后要使用Wait方法进行等待,否则会出现死锁。

通过mutex和atomic完成全局变量的原子操作

       当我们需要进行共享资源的访问时,我们需要进行锁来对于资源竞争的保证。下面,我们来看一看代码:

func add() {defer wg.Done()for i := 0; i < 10000; i++ {lock.Lock()total += 1  // 这里会有竞争lock.Unlock()}
}func sub() {defer wg.Done()for i := 0; i < 10000; i++ {lock.Lock()total -= 1  // 这里会有竞争lock.Unlock()}
}

       当我们单独执行add函数或者单独执行sub函数,结果都是正确,但是当我们两个一起执行时,结果是错的,在Linux中,我们也出现了这种情况,因为a += 1不是原子操作,当CPU进行执行这段代码时,有可能中途退出,我们需要将这段代码进行保护起来,不要让他单独执行。在go语言中的锁是一个结构体,锁可以进行复制,但是复制过后锁就失去了自身的效果。下面是将加减操作改为原子操作,代码如下:

func add() {defer wg.Done()for i := 0; i < 10000; i++ {atomic.AddInt32(&total, 1)}
}func sub() {defer wg.Done()for i := 0; i < 10000; i++ {atomic.AddInt32(&total, -1)}
}

RWMutex 读写锁

       锁本质上就将并行的代码串行化,使用lock会影响性能。即使是设计锁,那么也应该保持并行。

       我们有两组协程,其中一组负责写数据,另一组负责读数据,web系统中绝大部分场景都是读多写少。虽然有多个gorounte,但是仔细分析我们会发现读是可以进行并发,读和写应该进行串行。我们在读的时候不能写,在写的时候不能读。

       这里我们就出现了读写锁m我们在进行读写锁的时候,我们需要将写操作的时候添加写锁:

  • 写锁会防止别的写锁获取和读锁获取,使用WaitGroup
  • 读锁不会阻止别人的锁,写锁会阻止别人的锁。
for i := 0; i < 5; ++i {go func() {defer wg.Done()for {rwlock.RLock()  // 加读锁,读锁不会阻止别人的读// TODOrwlock.RUnlock()}}
}

使用channel进行gorounte之间的通信

channel通道

在Go语言中不要通过共享内存进行通信,而是要通过通信来实现内存共享

       在C++中如何实现进程之间的通信,就是使用一个全局变量(两个线程都可以看到的)来实现通信,在其他语言中的多线程编程的时候,两个gorountine之间通信最常用的方式是channel,再加上语法糖让channel更加简单。

        channel初始化值为0,放值进去会阻塞。

func main() {var msg chan string // 定义一个通道msg = make(chan string, 1)msg <- "加油"data := <- msgfmt.Println(data)
}

       静态语言中使用channel,我们需要指明通道中的类型是什么??注意箭头的指向,如果定义的通道在箭头的左边,则是将数据放入到通道中,如果定义的通道在箭头的右边,表示从通道中取数据。

channel有缓冲和无缓冲

var msg chan stringmsg = make(chan string, 0)  // channel的初始值如果为0的话,你放值进去会进行阻塞go func(msg chan string) {// go语言中有一种 happen-before机制,可以保证线程在channel中有值才会进行data := <- msgfmt.Println(data)
}(msg)msg <- "加油旭杏" // 放值到channel中
time.Sleep(time.Second * 10)

无缓冲区channel适用于 通知,B要第一时间知道A是否完成

有缓冲区channel适用于消费者和生产者之间的通信 

go中channel的应用场景

  • 消息传递,消息过滤
  • 信号广播
  • 事件订阅和广播
  • 任务分发
  • 结果汇总
  • 并发控制
  • 同步和异步

       在默认情况下,channel是双向的,但是我们经常使用channel作为参数进行传递,希望对方是单向使用的。 双向channel可以赋值给单向channel,但是单向channel不能赋值给双向channel,代码如下:

var ch1 chan int
var ch2 chan<- float64 // 单向channel,只能写入数据
var ch3 <-chan int // 只能读取数据

我们需要进行通信,就需要进行

通过channel实现交叉打印

我们需要使用channel进行打印,我们使用channel去进行通知,好像信号量,就是我们先打印数字然后唤醒打印字符的,在去打印字符,打印完字符再去唤醒打印数字的,如此交替执行。

监控gorounte的执行

       在go语言中,select类似于switch case 语句,但是select的功能和我们操作Linux里面提供的io的select,poll,epoll不一样,select主要作用于多个channel。

       现在有一个需求,我们现在有两个gorounte正在执行,但是我的主的gorounte中,当某一个执行完成之后,这个时候我们会立马知道

相关文章:

【Go语言】

type关键字的用法 定义结构体定义接口定义类型别名类型定义类型判断 别名实际上是为了更好地理解代码/ 这里要分点进行记录 使用传值的例子&#xff0c;当两个类型不一样需要进行类型转换 type Myint int // 自定义类型&#xff0c;基于已有的类型自定义一个类型type Myin…...

【Spring Boot】元注解

元注解 1.元注解1.1 Target1.2 Retention1.3 Inherited1.4 Documented1.5 interface 2.自定义注解2.1 创建自定义注解类2.2 实现业务逻辑2.3 使用自定义注解 1.元注解 元注解就是定义注解的注解&#xff0c;是 Java 提供的用于定义注解的基本注解。 注解 说明 Retention是注解…...

基于信号分解和多种深度学习结合的上证指数预测模型

大家好&#xff0c;我是带我去滑雪&#xff01; 为了给投资者提供更准确的投资建议、帮助政府和监管部门更好地制定相关政策&#xff0c;维护市场稳定&#xff0c;本文对股民情绪和上证指数之间的关系进行更深入的研究&#xff0c;并结合信号分解、优化算法和深度学习对上证指数…...

基于Spring Boot的酒店住宿管理平台

1系统概述 1.1 研究背景 随着计算机技术的发展以及计算机网络的逐渐普及&#xff0c;互联网成为人们查找信息的重要场所&#xff0c;二十一世纪是信息的时代&#xff0c;所以信息的管理显得特别重要。因此&#xff0c;使用计算机来管理酒店客房管理系统的相关信息成为必然。开发…...

游聚对战平台 三国战纪2012CE修改器修改地址

游聚对战平台 三国战纪2012比较全的一次地址。 工具 ce修改器 自行百度下载 1袖箭 2褐色鸡蛋 3毒堂 4飞盘 5火焰弹 6绿色鸡蛋 7金珠 8毒蝎 9毒镖 10铁莲花 11张陵剑 12张角巾 13太清丹经 14黄石公 15九节杖 16隐身衣 17神仙笔 18 玉蜂术&#xff08;效果不明&#xff09;19天师…...

Qt Creator中的项目栏

shadow build: [基础]Qt Creator 的 Shadow build(影子构建)-CSDN博客 影子构建&#xff1a;将源码路径和构建路径分开&#xff08;生成的makefile文件和其他产物都不放到源码路径&#xff09;&#xff0c;以此来保证源码路径的清洁。 实验1&#xff1a; 我创建了两个项目:…...

keepalived+web 实现双机热备

环境&#xff1a;利用keeplived实现web服务器的双机热备(高可用) 注意&#xff1a; (1) 利用keeplivedweb做双击热备&#xff08;高可用&#xff09;&#xff0c;最少需要两台服务器&#xff0c;可以实现多域名对应一个VIP,并且访问不同域名&#xff0c;显示不同主页&#xf…...

关于python的import

在Python中&#xff0c;import语句用于导入其他模块或模块中的特定部分&#xff0c;以便在代码中使用它们。这就可以重用代码&#xff0c;而不是每次都从头开始编写所有的功能。 基本用法 导入整个模块&#xff1a; import module_name 例如&#xff1a; import math print(…...

帕金森后期吞咽困难:破解难题,重拾生活美味!

在这个快节奏的时代&#xff0c;健康成为了我们最宝贵的财富。然而&#xff0c;对于帕金森病患者及其家庭而言&#xff0c;随着病情的进展&#xff0c;尤其是进入后期阶段&#xff0c;吞咽困难成为了他们不得不面对的严峻挑战。今天&#xff0c;就让我们一起走进这个温暖而坚韧…...

android 添加USB网卡并配置DNS

工作需要&#xff0c;需要使用TBox分享的网络&#xff0c;Android将TBox当作一个USB网卡&#xff0c;接下来就简单了&#xff0c;配置这个网卡的信息即可。 加载默认网卡的信息在frameworks/opt/net/ethernet/java/com/android/server/ethernet/EthernetTracker.java中 Ethern…...

【面试经典150】day 8

#1024程序员节 | 征文# 作为一个未来的程序员&#xff0c;现在我要继续刷题了。 力扣时刻。 目录 1.接雨水 2.罗马数字转整数 3.最后一个单词的长度 4.最长公共前缀 5.反转字符串中的单词 1.接雨水 好好好好好好&#xff0c;一开始就接雨水。我记得接了n次了。。。 痛苦战…...

Python -- 网络爬虫

Python – 网络爬虫 流程&#xff1a; 1. 连接链接获取页面内容&#xff08;html文件&#xff09;&#xff1b; 2. 过滤获取需要信息&#xff08;正则&#xff09; [可能重复步骤1&#xff0c;2] &#xff1b; 3. 存储文件到本地。一&#xff09;网络连接获取页面内容 # 网络…...

【英特尔IA-32架构软件开发者开发手册第3卷:系统编程指南】2001年版翻译,2-5

文件下载与邀请翻译者 学习英特尔开发手册&#xff0c;最好手里这个手册文件。原版是PDF文件。点击下方链接了解下载方法。 讲解下载英特尔开发手册的文章 翻译英特尔开发手册&#xff0c;会是一件耗时费力的工作。如果有愿意和我一起来做这件事的&#xff0c;那么&#xff…...

设计模式4 适配器 (adapter)

一句话&#xff0c;适配器按照客户的需求, 适配当前已有的接口。 目标接口&#xff1a;reqeust() public interface Target {void request(); //this is client needed interface }已有接口&#xff1a;specificRequest package com.example.adapter;import android.uti…...

《分布式机器学习模式》:解锁分布式ML的实战宝典

在大数据和人工智能时代&#xff0c;机器学习已经成为推动技术进步的重要引擎。然而&#xff0c;随着数据量的爆炸性增长和模型复杂度的提升&#xff0c;单机环境下的机器学习已经难以满足实际需求。因此&#xff0c;将机器学习应用迁移到分布式系统上&#xff0c;成为了一个不…...

【项目实战】HuggingFace初步实战,使用HF做一些小型任务

Huggingface初步实战 一、前期准备工作二、学习pipline2.1.试运行代码&#xff0c;使用HuggingFace下载模型2.2. 例子1&#xff0c;情感检测分析(只有积极和消极两个状态)2.3. 例子2&#xff0c;文本生成 三、学会使用Tokenizer & Model3.1.tokenizer&#xff08;分词器&am…...

堆的应用——堆排序和TOP-K问题

1.堆排序 想法⼀&#xff1a; 基于已有数组建堆、取堆顶元素完成排序。也就是利用写好的堆数据结构&#xff08;之前的文章有讲解&#xff09;&#xff0c;去实现排序。 void HeapSort(int* a, int n){HP hp;for(int i 0; i < n; i){HPPush(&hp,a[i]);}int i 0;whi…...

探秘 MySQL 数据类型的艺术:性能与存储的精妙平衡

文章目录 前言&#x1f380;一、数据类型分类&#x1f380;二、整数类型&#xff08;举例 TINYINT 和 INT &#xff09;&#x1f3ab;2.1 TINYINT 和 INT 类型的定义2.1.1 TINYINT2.1.2 INT &#x1f3ab;2.2 表的操作示例2.2.1 创建包含 TINYINT 和 INT 类型的表2.2.2 插入数据…...

使用任意绘图软件自学并结合上课所学内容完成数据库原理图绘制

本次绘图采用亿图图示软件...

static、 静态导入、成员变量的初始化、单例模式、final 常量(Content)、嵌套类、局部类、抽象类、接口、Lambda、方法引用

static static 常用来修饰类的成员&#xff1a;成员变量、方法、嵌套类 成员变量 被static修饰&#xff1a;类变量、成员变量、静态字段 在程序中只占用一段固定的内存&#xff08;存储在方法区&#xff09;&#xff0c;所有对象共享可以通过实例、类访问 (一般用类名访问和修…...

铭豹扩展坞 USB转网口 突然无法识别解决方法

当 USB 转网口扩展坞在一台笔记本上无法识别,但在其他电脑上正常工作时,问题通常出在笔记本自身或其与扩展坞的兼容性上。以下是系统化的定位思路和排查步骤,帮助你快速找到故障原因: 背景: 一个M-pard(铭豹)扩展坞的网卡突然无法识别了,扩展出来的三个USB接口正常。…...

谷歌浏览器插件

项目中有时候会用到插件 sync-cookie-extension1.0.0&#xff1a;开发环境同步测试 cookie 至 localhost&#xff0c;便于本地请求服务携带 cookie 参考地址&#xff1a;https://juejin.cn/post/7139354571712757767 里面有源码下载下来&#xff0c;加在到扩展即可使用FeHelp…...

GitHub 趋势日报 (2025年06月08日)

&#x1f4ca; 由 TrendForge 系统生成 | &#x1f310; https://trendforge.devlive.org/ &#x1f310; 本日报中的项目描述已自动翻译为中文 &#x1f4c8; 今日获星趋势图 今日获星趋势图 884 cognee 566 dify 414 HumanSystemOptimization 414 omni-tools 321 note-gen …...

AI书签管理工具开发全记录(十九):嵌入资源处理

1.前言 &#x1f4dd; 在上一篇文章中&#xff0c;我们完成了书签的导入导出功能。本篇文章我们研究如何处理嵌入资源&#xff0c;方便后续将资源打包到一个可执行文件中。 2.embed介绍 &#x1f3af; Go 1.16 引入了革命性的 embed 包&#xff0c;彻底改变了静态资源管理的…...

MySQL 部分重点知识篇

一、数据库对象 1. 主键 定义 &#xff1a;主键是用于唯一标识表中每一行记录的字段或字段组合。它具有唯一性和非空性特点。 作用 &#xff1a;确保数据的完整性&#xff0c;便于数据的查询和管理。 示例 &#xff1a;在学生信息表中&#xff0c;学号可以作为主键&#xff…...

Python 高效图像帧提取与视频编码:实战指南

Python 高效图像帧提取与视频编码:实战指南 在音视频处理领域,图像帧提取与视频编码是基础但极具挑战性的任务。Python 结合强大的第三方库(如 OpenCV、FFmpeg、PyAV),可以高效处理视频流,实现快速帧提取、压缩编码等关键功能。本文将深入介绍如何优化这些流程,提高处理…...

在鸿蒙HarmonyOS 5中使用DevEco Studio实现指南针功能

指南针功能是许多位置服务应用的基础功能之一。下面我将详细介绍如何在HarmonyOS 5中使用DevEco Studio实现指南针功能。 1. 开发环境准备 确保已安装DevEco Studio 3.1或更高版本确保项目使用的是HarmonyOS 5.0 SDK在项目的module.json5中配置必要的权限 2. 权限配置 在mo…...

Vue3中的computer和watch

computed的写法 在页面中 <div>{{ calcNumber }}</div>script中 写法1 常用 import { computed, ref } from vue; let price ref(100);const priceAdd () > { //函数方法 price 1price.value ; }//计算属性 let calcNumber computed(() > {return ${p…...

一些实用的chrome扩展0x01

简介 浏览器扩展程序有助于自动化任务、查找隐藏的漏洞、隐藏自身痕迹。以下列出了一些必备扩展程序&#xff0c;无论是测试应用程序、搜寻漏洞还是收集情报&#xff0c;它们都能提升工作流程。 FoxyProxy 代理管理工具&#xff0c;此扩展简化了使用代理&#xff08;如 Burp…...

如何把工业通信协议转换成http websocket

1.现状 工业通信协议多数工作在边缘设备上&#xff0c;比如&#xff1a;PLC、IOT盒子等。上层业务系统需要根据不同的工业协议做对应开发&#xff0c;当设备上用的是modbus从站时&#xff0c;采集设备数据需要开发modbus主站&#xff1b;当设备上用的是西门子PN协议时&#xf…...