# 【Go】Go 语言与 Java 语言对比
# Go 与 Java
# 零.GoApi 文档和中文社区网址
Go 的中文 api 文档:https://studygolang.com/pkgdoc
Go 中文社区网站:https://studygolang.com/
# 一。关于 Java
# 1.Java 的用途
1 2 3 4 5 6 7 首先我们来回顾下Java的主要用途和应用场景: 1 用途一:服务器后端系统开发(web后端、微服务后端支付系统、业务系统、管理后台,各种后台交互的接口服务)。 用途二:大数据框架的底层实现和Java的API支持。(Hadoop)。 用途三:其它中间件的底层开发。(Tomcat、RocketMq、Hbase、Kafka(部分)、SpringCloud,Dubbo...)。
# 2.Java 的优势和特点
1 2 3 4 5 6 7 8 9 那我们看Java语言有什么优势和特点呢? 1 *.做服务端系统性能高。 *.有虚拟机,跨平台。 *.功能强大,支持的类库多,生态圈类库多,开发框架和工具更易找。 *.市场占有率高,约60%的中国程序员都是做Java相关的工作。
# 二。关于 Go
# 1.Go 的出生原因
1 2 3 Go语言是Google内部公司大佬开发的,主要起因于Google公司有大量的C程序项目,但是开发起来效率太低,维护成本高,于是就开发了Go语言来提高效率,而且性能只是差一点。 (Go是2007年开始研发,2009推出发布)
# 2. 宏观看 Go 与 Java 的差异
1 2 3 4 5 6 7 8 9 10 11 接着,我们来看一下Go语言与Java的差异之处: 1 *.无虚拟机,不跨平台(这里的平台指操作系统)(可以运行多个平台,每个平台打不同的二进制程序包),需要打包编译成对应服务器操作系统版本(windows/linux)的可执行程序(比如windows是exe)。(注:说go跨平台的是指32位和64位相同操作系统之间的跨平台) *.因为Go程序直接打包成操作系统可执行的文件,没有虚拟机在中间转换的一层,所以理论上执行效率会更高(理论上更高,实际情况需具体分析)。 *.相比Java的语言和代码编写风格,Go更简洁,可以用更少的代码实现同样的功能。 *.Go语言底层也是C实现的,又做了高并发的设计(Java出生时(1995)还没有多核cpu,所以他的并发支持后来添加上去的,Go(2009)出生时已经有了多核cpu的电脑,它在设计语言时就考虑了充分利用多核cpu(英特尔2005首次推出多核)的性能),所以性能高,高并发的支持(高并发支持其中指的一个就是充分利用多核cpu的性能资源,比如go程序默认使用所有cpu(除非自己设置使用多少))也好。 *.天然的适用一些特定系统的开发,比如区块链类系统(如以太坊底层系统、以太坊上层应用程序),云计算和容器(Docker,K8s底层都是go开发的)开发的(大公司自研运维管理项目也大多是用go做底层的开发),网络编程(类似于java的Netty)。
# 3.Go 和 Java 的语言类型区别
计算机编程语言按照运行的方式可以分为编译型编程语言和解释型编译语言。
我来举一个例子,你要教别人一门沟通交流的语言,比如英语。
编译型的教的方式就是录 (这里的录相当于计算机中把程序编译成二进制可执行文件) 一个视频课程,语音课程,把每一句英语发音录下来,这样学生学的时候只要播放你的录音,然后跟着读就行,你只需要录制一次,学生就可以无数次听。
解释性的教的方式就是你亲自到学生家里给他补习,你当面教他,你读 (读相当于每次执行都重新用解释器解释一遍) 一句他学一句,
这样的话,你想要教他一句你必须就得先读一句,每次教都得重新一遍一遍的读。
这两种教学方式还有一个差别,你录 (编译) 视频语音教他,你录的英语他就只能学英语,空间环境一变,他现在要去日本,要学日语,你的视频语音教程因为已经录好了,是英语类型 (英语类型类比操作系统类型) 的,所以,你就得再录一套日语的语音教程。
而现场教他,你也会日语的话,你只需要读 (读相当于解释器解释) 日语给他听,让他学就行了,是不用考虑语言环境 (操作系统类型环境) 不同的问题的。
现在我们再来看编程语言,我们的程序执行有两种方式,一种是编译成操作系统可执行的二进制可执行程序,这样相当于编译一次,之后每次执行都不用再编译了,但是因为不同操作系统对于二进制文件的执行规范不同,不同的操作系统你要编译成不同的可执行文件。
解释型语言就是多了一个解释器,解释器我们可以类比为一个老师,你执行一行代码我们类比为学一句话的读音,解释器解释一句,就是老师先读一句,你跟着才能读一句,也就是解释器每解释一行代码为可运行的代码,操作系统执行一行代码,这样的话每次执行都需要解释器重新解释一遍,执行几次就得解释几次。
Go 是编译型的语言,运行在不同的平台需要打包成不同操作系统类型下的可执行文件。
Java 是半编译,半解释型语言。编译是指他的代码都会编译成 class 类型的文件,class 类型的文件只需要编译一次,可以在不同的操作系统的 Java 虚拟机上执行 ,半解释是指在 Java 虚拟机中,他还是需要一句一句的将 class 的二进制代码解释成对应操作系统可执行的代码。
# 4.Go 语言目前的主要应用场景
1 2 3 4 5 6 7 8 9 10 *.和Java一样,Go语言最多的应用场景就是服务器后端系统的开发,包括Web后端,微服务后端接口。 *.Go非常适用需要高性能高并发的网络编程,这里的网络编程是指不需要界面,底层只是用Socket相互传输数据的系统,类似于Java中Netty的用途。 *.一些云计算容器,比如Docker,K8s,底层就是Go语言开发的,也可以用做底层自研运维项目的开发。 *.一些游戏系统的开发,可以用Go语言。 *.区块链的一些底层软件和一些应用软件。(区块链程序的第一开发语言)
# 5. 现在市场上都有哪些公司在使用 Go 语言?
我们不讲虚的,直接 BOSS 直聘看哪些公司招,招的是干什么系统开发的。
这是腾讯的一个岗位。
看看岗位描述,是做互联网保险 产品的业务系统开发,业务系统是啥意思,和 JAVA 后端业务系统一样啊,说明腾讯的一部分项目已经用 Go 来开发业务系统了, 至少他这个保险团队是这样的。
再看小米也是:
也是后端,这是要和 JAVA 抢饭碗。。。
再看一个常见的,Go 非常适合开发运维管理系统,这个估计是开发维护阿里内部的自动化运维项目的,也就是说他们的运维支持可能是他们自己用 Go 语言写的项目。(实在不理解你就想下他们自己自研开发了一个类似于 Jenkins 和 Docker 之类的环境和代码流程发布的项目)
再来看一个字节跳动的,也是开发内部流程自动部署自动运维程序的
再看华为的,好像 Java 架构师的要求啊,微服务,缓存,消息中间件,数据库。。。
这里不多看,自己看看去吧,大多数你能知道的大公司都有用 go 语言尝试的新部门,新项目,市场占有率虽然比 Java 少,但是岗位实际上蛮多的。自己可以去 BOSS 上详细查查。
# 三.Go 和 Java 微观对比
# 1.GoPath 和 Java 的 ClassPath
我们先来看看关于 Java 的 classpath:
在我们的开发环境中,一个 web 程序 (war 包) 有一个 classpath, 这个 classpath 在 IDEA 的开发工具中目录体现为 src 目录和 resource 目录,实际上在真正的 war 包中他定位的是指 WEB-INF 下的 classes 文件夹下的资源 (比如 class 文件)。
我们编译后的文件都放在 classpath (类路径) 下。我们多个项目程序会有多个 classpath 目录。
在 Go 语言中,GoPath 在同一系统上的同一用户,一般规定只有一个,无论这个用户创建多少个 go 项目,都只有一个 GoPath, 并且这些项目都放在 GoPath 下的 src 目录下。
GoPath 下有三个目录:
1.bin (用于存放项目编译后的可执行文件)
2.pkg (用于存放类库文件,比如.a 结尾的包模块)
3.src (用于存放项目代码源文件)
注意:当我们在 windows 上开发 Go 程序时,需要新建一个文件夹 (文件夹名任意) 作为 GOPATH 的文件目录,在其中新建三个文件夹分别是:bin,pkg,src。如果是在集成开发工具上开发的话,需要在设置中把 GOPATH 路径设置为你自定义的那个文件夹,之后产生的文件和相关内容都会在其中。
如果是在 linux 上想跑测试开发执行 go 程序,需要在 /etc/profile 添加名为 GOPATH 的环境变量,目录设置好自己新建的。
例如:全局用户设置 GOPATH 环境变量
1 2 3 4 5 vi /etc/profile export GOPATH=/pub/go/gopath//立即刷新环境变量生效 source /etc/profile
单用户设置 GOPATH 环境变量
1 2 3 4 5 6 vi ~/.bash_profile export GOPATH=/home/user/local/soft/go/gopath//立即刷新环境变量生效 source vi ~/.bash_profile
注意:这是在 linux 上开发 go 程序才需要的,如果只是生产运行程序的话是不需要任何东西的,直接运行二进制可执行程序包即可,他所有的依赖全部打进包中了。
如果是在 windows 下的 cmd,dos 窗口运行相关的 go 命令和程序,则需要在 windows 的【此电脑】–>【右键】–>【属性】–>【高级系统设置】–>【环境变量】-【新建一个系统变量】–>【变量名为 GOPATH,路径为你自己指定的自定义文件夹】(如果是在 IDEA 中开发,不需要在此配置环境变量,只需要在 IDEA 中配置好 GOPATH 的目录设置即可)
# 2.Go 的开发环境搭建
(配置环境变量 GOPATH 参考上一节内容)
我们要开发 Go 的程序,需要如下两样东西:
1.Go SDK
GO 中文社区 SDK 下载地址:https://studygolang.com/dl
go1.14 (最新的)
我们用 1.14 版就可以,因为 1.13 后才完全支持 Module 功能。
有两种安装模式,一种是压缩包解压的方式,一种是图形化安装。
推荐使用 windows 图形安装傻瓜式安装,windows 图形安装下载这个
https://studygolang.com/dl/golang/go1.14.6.windows-amd64.msi
Go 的集成软件开发环境
参考三 (4) 中的 go 集成开发环境选择。
# 3.Go 与 Java 的文件结构对比
# 1).go 文件结构模板
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 package mainimport "fmt" const num = 10 var name string = "li_ming" type P struct {} func init () {} func main () { fmt.Printf("Hello World!!!" ); }
# 2).Java 文件结构
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 package my_package; import java.io.*;public Class MainTest{ public void static main (String[] args) { } } Class People { public String name; public int age; public void doSomething () { } }
# 4.Go 与 Java 的集成开发环境
# 1).Go 的集成开发环境
1 2 3 4 5 6 7 最常用的有三种: Visual Studio Code(VS Code) 微软开发的一款Go语言开发工具。 LiteIDE 是国人开发的Go语言开发工具。 GoLand 这个非常好用,和Java中的IDEA是一家公司。(推荐使用)
# 2).Java 的集成开发环境
1 2 3 MyEclipse,Eclipse(已过时)。 IntelliJ IDEA(大多数用这个)。
# 5.Go 和 Java 常用包的对比
1 2 3 4 5 6 7 8 9 10 11 12 Go中文API文档地址: https://studygolang.com/pkgdoc 12 Go Java IO流操作: bufio/os java.lang.io 字符串操作: strings java.lang.String 容器 container(heap/list/ring) java.lang.Collection 锁 sync juc 时间 time java.time/java.lang.Date 算数操作 math java.math 底层Unsafe unsafe unsafe类
# 6.Go 的常用基础数据类型和 Java 的基础数据类型对比
# 1).go 中的常用基础数据类型有:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 1.布尔型:关键字【bool】: true false 2.有符号整形:头一位是代表正负 int 默认整形 4或8字节 32位或64位 int8 1字节 8位 int16 2字节 16位 int32 4字节 32位 in64 8字节 64位 【int是32还是64位取决于操作系统的位数,现在电脑一般都是64位的了,所以一般都是64位】 3.无符号整形 uint 4或8字节 32位或64位 uint8 1字节 8位 uint16 2字节 16位 uint32 4字节 32位 uint64 8字节 64位 4.浮点型 注:go语言没有float类型,只有float32和float64。 float32 32位浮点数 float64 64位浮点数 5.字符串 string 6. byte 等同uint8,只是类似于一个别名的东西 rune 等同int32 只是一个别名,强调表示编码概念对应的数字
# 2).go 中派生数据类型有:
1 2 3 4 5 6 7 8 9 10 注:这里简单列举一下 指针 Pointer 数组 Array[] 结构体 struct 进程管道: channel 函数 func 切片 slice 接口 interface 哈希 map 123456789
# 3).Java 中的基础数据类型
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 byte short int long float double boolean char
# 7.Go 和 Java 的变量对比
# 1).go 的变量
1 2 3 4 5 6 7 8 9 10 11 12 13 14 package mainimport ( "fmt" ) func main () { var name string = "li_ming" name2:="xiao_hong" fmt.Println("name = " ,name) fmt.Println("name2 = " ,name2) }
# 2).Java 的变量
1 2 3 4 5 6 7 8 9 public class MyTest { public static void main (String[] args) { String name = "li_ming" ; int i = 10 ; System.out.println("name =" +name); System.out.println("i =" +i); } }
# 8.Go 和 Java 的常量对比
# 1).go 的常量
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 go中的常量和java中的常量含义有一个本质的区别: go中的常量是指在编译期间就能确定的量(数据), 而java中的常量是指被赋值一次后就不能修改的量(数据)。 所以两者不一样,因为Java中的常量也是JVM跑起来后赋值的,只不过不允许更改; go的常量在编译后就确实是什么数值了。 12345 package main import( //包含print函数 "fmt" ) func main() { //const 常量名 常量类型 = 常量值 显示推断类型 const name string = "const_li_ming" //隐式推断类型 const name2 ="const_xiao_hong" fmt.Println("name = ",name) fmt.Println("name2 = ",name2) }
# 2).Java 的常量
1 2 3 4 5 6 7 public class MyTest { public static final String TAG = "A" ; public static void main (String[] args) { System.out.println("tag= " +TAG); } }
# 9.Go 与 Java 的赋值对比
# 1).go 的赋值
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 Go方法内的赋值符号可以用 := ,也可以用 =,方法外只能用 = 。 例如: 12 package main import( //包含print函数 "fmt" ) //方法外只能用 = 赋值 var my_name string = "my_name" var my_name2 = "my_name2" //my_name3:="my_name3" 不在方法内,错误 func main() { fmt.Println("name = ",my_name) fmt.Println("name2 = ",my_name2) } 12345678910111213141516 Go支持多变量同时赋值: 1 package main import( //包含print函数 "fmt" ) func main() { //多变量同时赋值 var name,name2 = "li_ming","xiao_hong" fmt.Println("name = ",name) fmt.Println("name2 = ",name2) } 12345678910111213 Go的丢弃赋值 1 package main import( //包含print函数 "fmt" ) func main() { //丢弃赋值 把 1和2丢弃 只取3 //在必须一次取两个以上的值的场景下,又不想要其中一个值的时候使用,比如从map中取key,value var _,_,num = 1,2,3 fmt.Println("num = ",num) }
# 2).java 的赋值
1 2 3 4 5 6 7 8 9 public class MyTest { public static void main (String[] args) { String name = "li_ming" ; int i = 10 ; System.out.println("name =" +name); System.out.println("i =" +i); } }
# 10.Go 与 Java 的注释
1 2 3 4 5 6 7 8 9 Go中的注释写法和Java中的基本一样。 例如: //单行注释,两者相同 /* Go的多行注释 */ /** Java多行注释 */
# 11.Go 和 Java 的访问权限设置区别
首先我们来回忆一下,Java 的权限访问修饰符有哪些?
public 全局可见
protected 继承相关的类可见
default 同包可见
private 私有的,本类可见
关于 Java 中的访问权限修饰符,是用于修饰变量,方法,类的,被修饰的对象被不同的访问权限修饰符修饰后,其它程序代码要想访问它,必须在规定的访问范围内才可以,比如同包,同类,父子类,全局均可访问。
那么,Go 中的访问权限设置又有什么区别呢?
要理解这个问题,首先我们要来看一下一个 Go 程序的程序文件组织结构是什么样子的?
一个可运行的编译后的 Go 程序,必须有一个入口,程序从入口开始执行,这个入口必须是 main 包,并且从 main 包的 main 函数开始执行。
但是,为了开发的效率和管理开发任务的协调简单化,对于代码质量的可复用,可扩展等特性的要求,我们一般采用面向对象的,文件分模块式的开发。
比如,我是一个游戏程序,我的 main 函数启动后,首先要启动 UI 界面,那么关于 UI 界面相关的代码我们一般会专门分出一个模块去开发,然后这个模块有很多个程序文件,这里关于 UI 模块比如有 3 个文件,a.go,b.go,c.go,那么我们在实际当中会建一个以 ui 为名的包文件夹,然后把 a.go,b.go,c.go 全部放到 ui 这个包文件夹下,然后因为这个包没有 main 包,没有 main 函数,所以它打出来的一个程序文件就是以.a 结尾的工具包,类似于 Java 中的 jar 包,工具包文件名为 ui.a。
参考如下:
----com.add.mygame.ui
------------------------------------a.go
------------------------------------b.go
------------------------------------c.go
a.go 文件如下示例:
1 2 3 4 5 6 7 8 9 10 11 12 //这里的ui,也就是package后面的名称尽量和包文件夹的名称一致,不一致也可以 package ui //相关方法和业务 func main() { } //启动游戏UI func StartGameUI() { }
这里需要注意一个点,在程序中的 package 后面的 ui 包名可以和文件夹 com.mashibing.mygame.ui 中最后一层的 ui 文件夹名称不一致,
我们一般按规范写是要求写一致的,不一致时的区别如下:
我们把 ui.a 打包完毕后,我们就可以在别的程序中用 import 导入这个包模块 ,然后使用其中的内容了。
上面两个 ui 不同之处在于,在我们 import 的代码后面,需要写的模块名称是在 ${gopath}/src/ 下的文件夹名,也就是 com.mashibing.mygame.ui 中的 ui。
例如:
1 2 3 4 5 6 7 8 9 10 11 12 //游戏主程序 package main //这里的ui是com.mashibing.mygame.ui的最后一层文件夹名 import "ui" //相关方法和业务 func main() { //这里的ui不是文件夹名,而是之前a.go程序中package后面写的包名 ui.StartGameUI() }
接下来进入主题,我们的 go 语言关于访问修饰符的是指的限制什么权限,以及如何实现?
我们之前可以看出来,实战中的 go 程序是有一个 main 程序 import 很多其它包模块,每个模块实现对应的功能,最后统一在 main 程序中组合来完成整个软件程序,那么有一些其它模块的函数和变量,我只想在本程序文件中调用,不想被其它程序 import 能调用到,如何实现?
import 后是否能调用对应包中的对象 (变量,结构体,函数之类的) 就是 go 关于访问权限的定义,import 后,可以访问,说明是开启了访问权限,不可以访问,是说明关闭了其它程序访问的权限。
在 go 中,为了遵循实现简洁,快速的原则,用默认的规范来规定访问权限设置。
默认规范是:某种类型(包括变量,结构体,函数,类型等)的名称定义首字母大写就是在其它包可以访问,首字母非大写,就是只能在自己的程序中访问。
这样我们就能理解为什么导入 fmt 包后,他的 PrintF 函数的首字母 P 是大写的。
参照如下代码:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 package ui import "fmt" func main() { //这里的P是大写 //所有调用别的包下的函数,都是首字母大写 fmt.Printf("aa") } //这里的Person的首字母P也是表示外部程序导入该包后可以使用此Person类 type Person struct{ } //这里的D同上 var Data string = "li_ming"
# 12.Go 与 Java 程序文件的后缀名对比
1 2 3 4 5 6 7 8 9 10 11 Java的编译文件是.class结尾,多个.class打成的一个可执行文件是.jar结尾,.jar不能直接在windows和linux上执行,得用java命令在JVM中执行。 Go语言的程序文件后缀为.go,有main包main函数的,.go文件打包成二进制对应操作系统的可执行程序,如windows上的.exe结尾的可执行程序。 Java的类库会以.jar结尾,Go语言非main包没有main函数的程序编译打包会打成一个类库,以.a结尾,也就是说Go语言的类库以.a结尾。 Go的类库如下: 包名.a my_util.a 注:my_util是最顶层文件夹名,里面包含着一个个程序文件。 12345678910
# 13.Go 与 Java 选择结构的对比
# 1).if
1 Go中的if和Java中的if使用相同,只不过是把小括号给去掉了。
示例 1:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 package mainimport ( "fmt" ) func main () { var num int fmt.Printf("请输入数字" ) fmt.Scan(&num) if num > 10 { fmt.Println("您输入的数字大于10" ) } }
示例 2:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 package mainimport ( "fmt" ) func main () { var num int fmt.Printf("请输入数字" ) fmt.Scan(&num) if num > 10 { fmt.Println("您输入的数字大于10" ) } else { fmt.Println("您输入的数字不大于10" ) } }
示例 3:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 package mainimport ( "fmt" ) func main () { var num int fmt.Printf("请输入数字" ) fmt.Scan(&num) if num > 10 { fmt.Println("您输入的数字大于10" ) } else if num == 10 { fmt.Println("您输入的数字等于10" ) } else { fmt.Println("您输入的数字小于10" ) } }
# 2).switch
示例 1:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 package mainimport ( "fmt" ) func main () { var a = "li_ming" switch a { case "li_ming" : fmt.Println("Hello!LiMing" ) case "xiao_hong" : fmt.Println("Hello!XiaoHong" ) default : fmt.Println("No!" ) } }
示例 2:一分支多值
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 package mainimport ( "fmt" ) func main () { var name = "li_ming" var name2 = "xiao_hong" switch name { case "li_ming" , "xiao_hong" : fmt.Println("li_ming and xiao_hong" ) } switch name2 { case "li_ming" , "xiao_hong" : fmt.Println("li_ming and xiao_hong" ) } }g
示例 3:switch 表达式
1 2 3 4 5 6 7 8 9 10 11 12 package mainimport ( "fmt" ) func main () { var num int = 11 switch { case num > 10 && num < 20 : fmt.Println(num) } }
示例 4:fallthrough 下面的 case 全部执行
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 package mainimport ( "fmt" ) func main () { var num = 11 switch { case num == 11 : fmt.Println("==11" ) fallthrough case num < 10 : fmt.Println("<12" ) } }
# 14.Go 与 Java 循环结构的对比
# 1).for 循环
示例 1:省略小括号
1 2 3 4 5 6 7 8 9 10 11 package mainimport ("fmt" ) func main () { for i := 1 ; i < 10 ; i++ { fmt.Println(i) } }
示例 2:和 while 相同,break,continue 同 java
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 package mainimport ( "fmt" ) func main () { i := 0 for i < 3 { i++ } for i == 3 { fmt.Println(i) break } }
示例 3:死循环,三项均省略
1 2 3 4 5 6 7 8 9 10 11 12 package mainfunc main () { for { } for true { } }
示例 4:嵌套循环和 java 也一样,不演示了
示例 5: range 循环
1 2 3 4 5 6 7 8 9 10 package mainimport "fmt" func main () { var data [10 ]int = [10 ]int {1 ,2 ,3 ,4 ,5 ,6 ,7 ,8 ,9 ,10 } for i, num := range data { fmt.Println(i,num) } }
# 2).goto
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 package mainimport "fmt" func main () { fmt.Println("start" ) goto my_location fmt.Println("over" ) my_location: fmt.Println("location" )
# 15.Go 与 Java 的数组对比
1)go 的一维数组
1 var 数组名 [数组长度]数组类型 = [数组长度]数组类型{元素1,元素2...}
示例 1:
1 2 3 4 5 6 7 8 9 10 11 12 13 package mainimport "fmt" var my_arr [6 ]int var my_arr_1 [3 ]int = [3 ]int {1 ,2 ,3 }func main () { this_arr := [2 ]int {1 , 2 } fmt.Println(my_arr) fmt.Println(my_arr_1) fmt.Println(this_arr) }
2)二维数组
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 package mainimport "fmt" var my_arr [4 ][6 ]int var my_arr_1 [2 ][3 ]int = [...][3 ]int {{1 , 2 , 3 }, {4 , 5 , 6 }}func main () { this_arr := [2 ][3 ]int {{1 , 2 , 3 }, {8 , 8 , 8 }} this_arr2 := [...][2 ]int {{1 , 1 }, {2 , 2 }, {3 , 3 }} fmt.Println(my_arr) fmt.Println(my_arr_1) fmt.Println(this_arr) fmt.Println(this_arr2) }
# 16.Go 有指针概念,Java 没有指针概念
1 2 3 4 5 6 7 8 9 10 11 12 Go中有指针的概念,Java中没有指针的概念。 指针简单的说就是存储一个【变量地址】的【变量】。 12 Go中使用指针的方法 //*+变量类型 = 对应变量类型的指针类型,&+变量名 = 获取变量引用地址 var 指针变量名 *指针变量类型 = &变量名 例如: var my_point *int = &num //通过&+指针变量 = 修改原来的变量真实值 &指针变量名 = 修改的变量值 例如: &my_point = 100;
示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 package mainimport "fmt" func main () { var name string ="li_ming" var name_point *string name_point = &name fmt.Println("name 变量的地址是:" , &name ) fmt.Println("name_point变量储存的指针地址:" , name_point ) fmt.Println("*name_point 变量的值:" , *name_point ) }
输出结果:
1 2 3 name 变量的地址是: 0x10ae40f0 name_point变量储存的指针地址: 0x10ae40f0 *name_point 变量的值: li_ming
# 17.Go 语言的中 new,make 和 Java 中的 new 对象有什么区别?
首先,Java 中的 new 关键字代表创建关于某一个类的一个新的对象。
如:
1 List list = new ArrayList ();
Go 中的创建一个 struct 结构体的对象,是不需要用 new 关键字的,参考【20】中有代码示例讲解如何创建结构体对象。
Go 中 new 的概念是和内存相关的,我们可以通过 new 来为基础数据类型申请一块内存地址空间,然后把这个把这个内存地址空间赋值给
一个指针变量上。(new 主要就是为基础数据类型申请内存空间的,当我们需要一个基础数据类型的指针变量,并且在初始化这个基础指针变量时,不能确定他的初始值,此时我们才需要用 new 去内存中申请一块空间,并把这空间绑定到对应的指针上,之后可以用该指针为这块内存空间写值。new 关键字在实际开发中很少使用,和 java 很多处用 new 的情况大不相同)
参考如下示例代码:
1 2 3 4 5 6 7 8 9 10 11 12 package mainimport "fmt" func main () { var num *int fmt.Println(num) *num = 1 fmt.Println(*num) }
改为如下代码即可:
1 2 3 4 5 6 7 8 9 10 11 12 13 package mainimport "fmt" func main () { var num = new (int ) fmt.Println(num) *num = 1 fmt.Println(*num) }
接下来我们来看一个 go 中的 make 是做什么用的?
go 中的 make 是用来创建 slice (切片),map (映射表),chan (线程通信管道) 这三个类型的对象的,返回的就是对应类型的对象,他的作用就相当于 Java 中 new 一个 ArrayList,new 一个 HashMap 时候的 new 的作用,只不过是 go 语法规定用 make 来创建 slice (切片),map (映射表),chan (线程通信管道)。
示例代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 package mainimport "fmt" func main () { myMap := make (map [string ]int ) myMap["li_ming" ] = 20 myChan := make (chan int ) fmt.Println(myChan) mychan2 := make (chan int ,2 ) fmt.Println(mychan2) myArr := []int {1 ,2 ,3 ,4 ,5 } mySlice1 := myArr[2 :4 ] fmt.Println(mySlice1) mySliceEmpty := make ([]int ,2 ,4 ) fmt.Println(mySliceEmpty) mySliceEmpty2 := make ([]int ,5 ) fmt.Println(mySliceEmpty2) }
# 18.Go 相关的数据容器和 Java 的集合框架对比
1 2 3 4 5 6 7 8 Go中有的数据结构:数组,切片,map,双向链表,环形链表,堆 Go自己的类库中没有set,没有集合(List),但是第三方库有实现。 Java中有: Map,Set,List,Queue,Stack,数组 Java中没有切片的概念。 Go中的数组打印格式是[1,2,3,4,5] Go中的切片打印格式是[[1,2,3]] Go中切片的概念:切片是数组的一个子集,就是数组截取某一段。 Go的map和Java的map大致相同
# 19.Go 中的函数,Go 的方法和 Java 中的方法对比
# 1).Go 中的函数定义
1 2 3 4 Go中返回值可以有多个,不像Java中多个值得封装到实体或map返回 //注:【】内的返回值可不写,无返回值直接把返回值部分全部去掉即可。 func 函数名(变量1 变量类型,变量2 变量2类型...)【(返回值1 类型1,返回值2 类型2...)】 { //注意:这个方法的右中括号必须和func写在同一行才行,否则报错,不能按c语言中的换行写 123
示例 1:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 package mainimport "fmt" func main () { var a int = 100 var b int = 200 var result int result = max(a, b) fmt.Println( "最大值是 :" , result ) } func max (num1, num2 int ) int { var result int if (num1 > num2) { result = num1 } else { result = num2 } return result }
示例 2:返回多个值
1 2 3 4 5 6 7 8 9 10 11 12 13 package mainimport "fmt" func main () { a, b := swap("li_ming" , "xiao_hong" ) fmt.Println(a, b) } func swap (x, y string ) (string , string ) { return y, x }
注意点:函数的参数:基础类型是按值传递,复杂类型是按引用传递
示例 3: 函数的参数:变长参数传递
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 package mainimport "fmt" func main () { manyArgs(1 ,2 ,"2" ,"3" ,"4" ) manyArgs(1 ,2 ,"5" ,"5" ,"5" ) dataStr := []string {"11" ,"11" ,"11" } manyArgs(1 ,2 ,dataStr...) } func manyArgs (a int ,b int ,str ...string ) { for i,s := range str { fmt.Println(i,s) } }
注意点:函数的返回值:如果有返回值,返回值的类型必须写,返回值得变量名根据情况可写可不写。
示例 4: defer:推迟执行 (类似于 java 中的 finally)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 package mainimport "fmt" func main () { testMyFunc(); } func testDefer1 () { fmt.Println("print defer1" ) } func testDefer2 () { fmt.Println("print defer2" ) } func testMyFunc () { defer testDefer2() defer testDefer1() fmt.Println("print my func" ) }
示例 5 :丢弃返回值
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 package mainimport "fmt" func main () { _,num2,_:= testFun(1 ,2 ,"3" ); fmt.Println(num2) _,num3,_:= testFun(1 ,3 ,"4" ); fmt.Println(num3) } func testFun (num1,num2 int ,str string ) (n1 int ,n2 int ,s1 string ){ n1 = num1 n2 = num2 s1 = str return } func testFun2 (num1,num2 int ,str string ) (int ,int ,string ){ return num1,num2,str }
# 2).Java 中的方法定义
1 2 3 4 访问修饰符 返回值类型 方法名(参数1类型 参数1,参数2类型 参数2...) { return 返回值; }
示例:
1 2 3 4 public Integer doSomething (String name,Integer age) { return 20 ; }
# 20.Go 的内置函数和 Java 的默认导入包 java.lang.*
为了在 Java 中快速开发,Java 语言的创造者把一些常用的类和接口都放到到 java.lang 包下,lang 包下的特点就是不用写 import 语句导入包就可以用里面的程序代码。
Go 中也有类似的功能,叫做 Go 的内置函数,Go 的内置函数是指不用导入任何包,直接就可以通过函数名进行调用的函数。
Go 中的内置函数有:
1 2 3 4 5 6 7 8 9 10 11 close 关闭channel len 求长度 make 创建slice,map,chan对象 append 追加元素到切片(slice)中 panic 抛出异常,终止程序 recover 尝试恢复异常,必须写在defer相关的代码块中
参考示例代码 1:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 package mainimport "fmt" func main () { array := [5 ]int {1 ,2 ,3 ,4 ,5 } str := "myName" fmt.Println(len (str)) fmt.Println(len (array)) fmt.Println(len (array[1 :])) intChan := make (chan int ,1 ) myMap := make (map [string ]interface {}) mySlice := make ([]int ,5 ,10 ) fmt.Println(intChan) fmt.Println(myMap) fmt.Println(mySlice) close (intChan) array2 := append (array[:],6 ) fmt.Println(array2) num := new (int ) fmt.Println(num) }
参考示例代码 2:panic 和 recover 的使用
他们用于抛出异常和尝试捕获恢复异常
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 func func1 () { fmt.Println("1" ) } func func2 () { defer func () { err := recover () fmt.Println(err) fmt.Println("释放资源.." ) }() panic ("抛出异常" ) fmt.Println(2 ") } func func3() { fmt.Println(" 3 ") } func main() { func1() func2() func3() }
Java 中的 java.lang 包下具体有什么在这里就不赘述了,请参考 JavaAPI 文档:
1 JavaAPI文档导航:https://www.oracle.com/cn/java/technologies/java-se-api-doc.html
# 21.Go 的标准格式化输出库 fmt 和 java 的输出打印库对比
Java 的标准输出流工具类是 java.lang 包下的 System 类,具体是其静态成员变量 PrintStream 类。
他有静态三个成员变量:
分别是 PrintStream 类型的 out,in,err
我们常见 System.out.println (), 实际上调用的就是 PrintStream 类对象的 println 方法。
Go 中的格式化输出输入库是 fmt 模块。
fmt 在 Go 中提供了输入和输出的功能,类型 Java 中的 Scanner 和 PrintStream (println)。
它的使用方法如下:
1 2 3 4 5 Print: 原样输出到控制台,不做格式控制。 Println: 输出到控制台并换行 Printf : 格式化输出(按特定标识符指定格式替换) Sprintf:格式化字符串并把字符串返回,不输出,有点类似于Java中的拼接字符串然后返回。 Fprintf:来格式化并输出到 io.Writers 而不是 os.Stdout
详细占位符号如下:
代码示例如下:
# 22.Go 的面向对象相关知识
# 1. 封装属性 (结构体)
Go 中有一个数据类型是 Struct, 它在面向对象的概念中相当于 Java 的类,可以封装属性和封装方法,首先看封装属性如下示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 package mainimport "fmt" type People struct { name string age int sex bool } func main () { var l1 People l1.name = "li_ming" l1.age = 22 l1.sex = false fmt.Println(l1.name) var l2 *People = new (People) l2.name = "xiao_hong" l2.age = 33 l2.sex = true fmt.Println(l2.name,(*l2).name) var l3 *People = &People{ name:"li_Ming" ,age:25 ,sex:true } fmt.Println(l3.name,(*l3).name) }
# 2. 封装方法 (方法接收器)
如果想为某个 Struct 类型添加一个方法,参考如下说明和代码:
go 的方法和 Java 中的方法对比,go 的函数和 go 方法的不同
Go 中的函数是不需要用结构体的对象来调用的,可以直接调用
Go 中的方法是必须用一个具体的结构体对象来调用的,有点像 Java 的某个类的对象调用其方法
我们可以把指定的函数绑定到对应的结构体上,使该函数成为这个结构体的方法,然后这个结构体的对象就可以通过。来调用这个方法了
绑定的形式是:在 func 和方法名之间写一个 (当前对象变量名 当前结构体类型),这个叫方法的接受器,其中当前对象的变量名就是当前结构体调用该方法的对象的引用,相当于 java 中的 this 对象。
参考如下示例为 Student 学生添加一个 learn 学习的方法
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 package mainimport "fmt" type Student struct { num int name string class int sex bool } func (stu Student) learn() { fmt.Printf("%s学生正在学习" ,stu.name) } func main () { stu := Student{1 ,"li_ming" ,10 ,true } stu.learn() }
方法的接收器也可以是指针类型的
参考如下案例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 package mainimport "fmt" type Student struct { num int name string class int sex bool } func (stu *Student) learn() { fmt.Printf("%s学生正在学习" ,stu.name) } func main () { stu := &Student{1 ,"li_ming" ,10 ,true } stu.learn() }
注意有一种情况,当一个对象为 nil 空时,它调用方法时,接收器接受的对于自身的引用也是 nil,需要我们做一些健壮性的不为 nil 才做的判断处理。
# 3.Go 的继承 (结构体嵌入)
Go 中可以用嵌入结构体实现类似于继承的功能:
参考如下代码示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 package mainimport "fmt" type Computer struct { screen string keyboard string } func (cp Computer) compute(num1,num2 int ) int { return num1+num2; } type NoteBookComputer struct { Computer wireless_network_adapter string } func main () { var cp1 NoteBookComputer = NoteBookComputer{} cp1.screen = "高清屏" cp1.keyboard = "防水键盘" cp1.wireless_network_adapter = "新一代无线网卡" fmt.Println(cp1) fmt.Println(cp1.compute(1 ,2 )) }
需要注意的是,Go 中可以嵌入多个结构体,但是多个结构体不能有相同的方法,如果有参数和方法名完全相同的方法,在编译的时候就会报错。所以 Go 不存在嵌入多个结构体后,被嵌入的几个结构体有相同的方法,最后不知道选择执行哪个方法的情况,多个结构体方法相同时,直接编译就会报错。
参考如下示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 package mainimport "fmt" func main () { man := Man{} fmt.Println(man) } type Man struct { FatherA FatherB } func (p FatherA) doEat() { fmt.Printf("FatherA eat" ) } func (t FatherB) doEat() { fmt.Printf("FatherB eat" ) } type FatherB struct {} type FatherA struct {}
# 4.Go 的多态 (接口)
接下来我们讲 Go 中如何通过父类接口指向具体实现类对象,实现多态:
go 语言中的接口是非侵入式接口。
java 语言中的接口是侵入式接口。
侵入式接口是指需要显示的在类中写明实现哪些接口。
非侵入式接口是指不要显示的在类中写明要实现哪些接口,只需要方法名同名,参数一致即可。
参考如下代码示例:接口与多态
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 package mainimport "fmt" type Animal interface { eat() sleep() } type Cat struct {} type Dog struct {} func (cat Cat) eat() { fmt.Println("小猫在吃饭" ) } func (cat Cat) sleep(){ fmt.Println("小猫在睡觉" ) } func (dog Dog) eat(){ fmt.Println("小狗在吃饭" ) } func (dog Dog) sleep(){ fmt.Println("小狗在睡觉" ) } func main () { var cat Animal = Cat{} var dog Animal = Dog{} cat.eat() cat.sleep() dog.eat() dog.sleep() }
接口可以内嵌接口
参考如下代码示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 package maintype Learn interface { LearnByHear LearnByLook } type LearnByHear interface { hear() } type LearnByLook interface { look() }
# 23.Go 语言中线程的实现和 Java 语言中线程的实现
go 中的线程相关的概念是 Goroutines (并发),是使用 go 关键字开启。
Java 中的线程是通过 Thread 类开启的。
在 go 语言中,一个线程就是一个 Goroutines,主函数就是(主) main Goroutines。
使用 go 语句来开启一个新的 Goroutines
比如:
普通方法执行
myFunction()
开启一个 Goroutines 来执行方法
go myFunction()
java 中是
new Thread(()->{
// 新线程逻辑代码
}).start();
参考下面的代码示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 package mainimport ( "fmt" ) func myFunction () { fmt.Println("Hello!!!" ) } func goroutineTestFunc () { fmt.Println("Hello!!! Start Goroutine!!!" ) } func main () { }
线程间的通信:
java 线程间通信有很多种方式:
比如最原始的 wait/notify
到使用 juc 下高并发线程同步容器,Go 和 Java 关于 Socket 编程的对比同步队列
到 CountDownLatch 等一系列工具类
…
甚至是分布式系统不同机器之间的消息中间件,单机的 disruptor 等等。
Go 语言不同,线程间主要的通信方式是 Channel。
Channel 是实现 go 语言多个线程(goroutines)之间通信的一个机制。
Channel 是一个线程间传输数据的管道,创建 Channel 必须声明管道内的数据类型是什么
下面我们创建一个传输 int 类型数据的 Channel
代码示例:
1 2 3 4 5 6 7 8 package mainimport "fmt" func main () { ch := make (chan int ) fmt.Println(ch) }
channel 是引用类型,函数传参数时是引用传递而不是值拷贝的传递。
channel 的空值和别的应用类型一样是 nil。
== 可以比较两个 Channel 之间传输的数据类型是否相等。
channel 是一个管道,他可以收数据和发数据。
具体参照下面代码示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 package mainimport ( "fmt" "time" ) var chanStr = make (chan string )func main () { fmt.Println("main goroutine print Hello " ) go startNewGoroutineOne() go startNewGoroutineTwo() time.Sleep(100 *time.Second) } func startNewGoroutineOne () { fmt.Println("send channel print Hello " ) chanStr <- "Hello!!!" } func startNewGoroutineTwo () { fmt.Println("receive channel print Hello " ) strVar := <-chanStr fmt.Println(strVar) }
无缓存的 channel 可以起到一个多线程间线程数据同步锁安全的作用。
缓存的 channel 创建方式是
make (chan string, 缓存个数)
缓存个数是指直到多个数据没有消费或者接受后才进行阻塞。
类似于 java 中的 synchronized 和 lock
可以保证多线程并发下的数据一致性问题。
首先我们看一个线程不安全的代码示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 package mainimport ( "fmt" "time" ) var moneyA int =1000 func subtractMoney (subMoney int ) { time.Sleep(3 *time.Second) moneyA-=subMoney } func getMoney () int { return moneyA; } func main () { go func () { if (getMoney()>200 ) { subtractMoney(200 ) fmt.Printf("200元扣款成功,剩下:%d元\n" ,getMoney()) } }() go func () { if (getMoney()>900 ) { subtractMoney(900 ) fmt.Printf("900元扣款成功,剩下:%d元\n" ,getMoney()) } }() time.Sleep(5 *time.Second) fmt.Println(getMoney()) }
缓存为 1 的 channel 可以作为锁使用:
示例代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 package mainimport ( "fmt" "time" ) var moneyA = 1000 var synchLock = make (chan int ,1 )func subtractMoney (subMoney int ) { time.Sleep(3 *time.Second) moneyA-=subMoney } func getMoney () int { return moneyA; } func main () { go func () { synchLock<-10 if (getMoney()>200 ) { subtractMoney(200 ) fmt.Printf("200元扣款成功,剩下:%d元\n" ,getMoney()) } <-synchLock }() go func () { synchLock<-10 if (getMoney()>900 ) { subtractMoney(900 ) fmt.Printf("900元扣款成功,剩下:%d元\n" ,getMoney()) } synchLock<-10 }() time.Sleep(5 *time.Second) fmt.Println(getMoney()) }
go 也有互斥锁
类似于 java 中的 Lock 接口
参考如下示例代码:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 package mainimport ( "fmt" "sync" "time" ) var moneyA = 1000 var lock sync.Mutex;func subtractMoney (subMoney int ) { lock.Lock() time.Sleep(3 *time.Second) moneyA-=subMoney lock.Unlock() } func getMoney () int { lock.Lock() result := moneyA lock.Unlock() return result; } func main () { go func () { if (getMoney()>200 ) { subtractMoney(200 ) fmt.Printf("200元扣款成功,剩下:%d元\n" ,getMoney()) }else { fmt.Println("余额不足,无法扣款" ) } }() go func () { if (getMoney()>900 ) { subtractMoney(900 ) fmt.Printf("900元扣款成功,剩下:%d元\n" ,getMoney()) }else { fmt.Println("余额不足,无法扣款" ) } }() time.Sleep(5 *time.Second) fmt.Println(getMoney()) }
# 24.Go 中的反射与 Java 中的反射对比
整体概述:反射是一个通用的概念,是指在程序运行期间获取到变量或者对象,结构体的元信息,比如类型信息,并且能够取出其中变量的值,调用对应的方法。
首先我们先来回顾一下 Java 语言用到反射的场景有哪些?
1. 比如说我们的方法参数不能确定是什么类型,是 Object 类型,我们就可以通过反射在运行期间获取其真实的类型,然后做对应的逻辑处理。
2. 比如动态代理,我们需要在程序运行时,动态的加载一个类,创建一个类,使用一个类。
3. 比如在想要强行破解获取程序中被 private 的成员。
4.Java 的各种框架中用的非常多,框架中用反射来判断用户自定义的类是什么类型,然后做区别处理。
Go 中的反射大概也是相同的,比如,go 中有一个类型 interface,interface 类型相当于 Java 中的 Object 类,当以 interface 作为参数类型时,可以给这个参数传递任意类型的变量。
例如示例 1:
1 2 3 4 5 6 7 8 9 10 11 12 13 package mainimport "fmt" func main () { testAllType(1 ); testAllType("Go" ); } func testAllType (data interface {}) { fmt.Println(data) }
那么第一种应用场景就出现了,当我们在 go 中想实现一个函数 / 方法,这个函数 / 方法的参数类型在编写程序的时候不能确认,在运行时会有各种不同的类型传入这个通用的函数 / 方法中,我们需要对不同类型的参数做不同的处理,那么我们就得能获取到参数是什么类型的,然后根据这个类型信息做业务逻辑判断。
反射我们需要调用 reflect 包模块,使用 reflect.typeOf () 可以获取参数的类型信息对象,再根据类型信息对象的 kind 方法,获取到具体类型,详细参考下面代码。
例如示例 2:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 package mainimport ( "fmt" "reflect" ) func main () { handleType(1 ) handleType(true ) } func handleType (data interface {}) { d := reflect.TypeOf(data) fmt.Println(d.Kind()) switch d.Kind() { case reflect.Invalid: fmt.Println("无效类型" ) case reflect.Int,reflect.Int8,reflect.Int16,reflect.Int32,reflect.Int64: fmt.Println("整形" ) case reflect.Bool: fmt.Println("bool类型" ) } } 因为传入进来的都是interface 类型,所以我们需要用的时候要区分类型,然后取出其中真正类型的值。
反射取出值得方法就是先通过 reflect.ValueOf () 获取参数值对象,然后再通过不同的具体方法获取到值对象,比如 int 和 bool
示例 3:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 package mainimport ( "fmt" "reflect" ) func main () { handleValue(1 ) handleValue(true ) } func handleValue (data interface {}) { d := reflect.ValueOf(data) fmt.Println(d.Kind()) switch d.Kind() { case reflect.Invalid: fmt.Println("无效类型" ) case reflect.Int,reflect.Int8,reflect.Int16,reflect.Int32,reflect.Int64: var myNum = d.Int() fmt.Println(myNum) case reflect.Bool: var myBool = d.Bool() fmt.Println(myBool) } }
结构体中的属性和方法怎么获取呢?
获取结构体属性的个数是先 ValueOf 获取结构体值对象 v 后,用 v.NumField () 获取该结构体有几个属性,通过 v.Field (i) 来获取对应位置的属性的元类型。
示例代码 4:
后续反射还有几个 api 和代码示例和具体应用场景,正在补。。。
# 25. 变量作用域的区别
Go 语言的变量作用域和 Java 中的一样,遵循最近原则,逐渐往外层找。
这个比较简单,就不做过多赘述了。
# 26.Go 语言和 Java 语言字符串操作的区别
# 27.Go 语言和 Java 语言 IO 操作的区别
# 28.Go 语言中有匿名函数,有闭包,Java 中没有 (高阶函数用法)
函数也是一种类型,它可以作为一个参数进行传递,也可以作为一个返回值传递。
Go 中可以定义一个匿名函数,并把这个函数赋值给一个变量
示例 1: 匿名函数赋值给变量
1 2 3 4 5 6 7 8 9 10 11 12 13 package mainimport "fmt" var myFun = func (x,y int ) int { return x+y } func main () { fmt.Println(myFun(1 ,2 )) }
输出结果:
Go 的函数内部是无法再声明一个有名字的函数的,Go 的函数内部只能声明匿名函数。
示例 2:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 package mainimport "fmt" func main () { myFunc3() } func myFun1 () { } func myFunc3 () { var f = func () { fmt.Println("Hi,boy!" ) } f() func () { fmt.Println("Hello,girl!" ) }() }
输出:
Go 中有闭包的功能。(闭包是一个通用的编程概念,一些语言有,一些没有,javascript 中就有这个概念,Java 中没有)
闭包,通俗易懂的讲,就是你有一个 A 函数,A 函数有一个 a 参数,然后在 A 函数内部再定义或者调用或者写一个 B 函数,这个 B 函数叫做闭包函数。B 函数内部的代码可以访问它外部的 A 函数的 a 参数,正常 A 函数调用返回完毕,a 参数就不能用了,可是闭包函数 B 函数仍然可以访问这个 a 参数,B 函数能不受 A 函数的调用生命周期限制可以随时访问其中的 a 参数,这个能访问的状态叫做已经做了闭包,闭包闭的是把 a 参数封闭到了 B 函数中,不受 A 函数的限制。
也就是说,我们用程序实现一个闭包的功能,实质上就是写一个让外层的函数参数或者函数内变量封闭绑定到内层函数的功能。
接下来我们看代码示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 package mainimport "fmt" func main () { var f = f1(100 ) f(100 ) f(100 ) f(100 ) } func f1 (x int ) func (int ) { return func (y int ) { x+=y fmt.Printf("x=%d\n" ,x) } }
输出:
做下闭包的总结,如何实现一个闭包:
1. 定义一个 A 函数,此函数返回一个匿名函数。(定义一个返回匿名函数的 A 函数)
2. 把在 A 函数的 b 参数或 A 函数代码块中的 b 变量,放入匿名函数中,进行操作。
3. 这样我们调用 A 函数返回一个函数,这个函数不断的调用就可以一直使用之前 b 参数,b 变量,并且 b 值不会刷新,有点像在匿名函数外部自定义了一个 b 的成员变量(成员变量取自 Java 中类的相关概念)
# 29.Go 中的 map 和 Java 中的 HashMap
Go 中的 map 也是一个存储 key-value,键值对的这么一种数据结构。
我们来看下如何使用:
如何创建一个 map?(map 是引用类型,默认值是 nil,必须用 make 为其创建才能使用)
创建一个 map 必须要用 make,否则会是 nil
格式为: make (map [key 类型] value 类型) (下面有代码示例)
往 Go 中的 map 赋值添加元素用 【 map 变量名称 [key] = value 】 的方式
示例 1:创建 map 以及添加元素
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 package mainimport "fmt" func main () { myMap := make (map [string ]int ) myMap["li_age" ] = 20 myMap["hong_age" ] = 30 fmt.Println(myMap) }
我们从 map 中取值得格式为: 【 mapValue := map 变量名 [key]】
当我们填写的 key 在 map 中找不到时返回对应的 value 默认值,int 是 0,引用类型是 nil
当我们的 key 取不到对应的值,而 value 的类型是一个 int 类型,我们如何判断这个 0 是实际值还是默认值呢
此时我们需要同时取两个值
通过 map 的 key 取出两个值,第二个参数为 bool 类型,false 为该值不存在,true 为成功取到值
参考下面:
示例 2:从 map 中取值
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 package mainimport "fmt" func main () { myMap := make (map [string ]int ) myMap["li_age" ] = 20 myMap["hong_age" ] = 30 fmt.Println(myMap) fmt.Println(myMap["no" ]) value,existsValue := myMap["no" ] if !existsValue { fmt.Println("此值不存在" ) } else { fmt.Printf("value = %d" ,value) } }
Go 中因为返回值可以是两个,所以的 map 遍历很简单,不像 java 还得弄一个 Iterator 对象再逐个获取,它一次两个都能取出来,用 for 搭配 range 即可实现。
示例 3:遍历
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 package mainimport "fmt" func main () { myMap := make (map [string ]int ) myMap["num1" ] = 1 myMap["num2" ] = 2 myMap["num3" ] = 3 myMap["num4" ] = 4 myMap["num5" ] = 5 myMap["num6" ] = 6 for key,value := range myMap { fmt.Println(key,value) } for key := range myMap { fmt.Println(key) } for _,value := range myMap { fmt.Println(value) } }
删除函数:用内置函数 delete 删除
示例 4:删除 map 元素
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 package mainimport "fmt" func main () { myMap := make (map [string ]int ) myMap["num1" ] = 1 myMap["num2" ] = 2 myMap["num3" ] = 3 myMap["num4" ] = 4 myMap["num5" ] = 5 myMap["num6" ] = 6 delete (myMap,"num6" ) fmt.Println(myMap["num6" ]) }
在 Java 中有一些复杂的 Map 类型,比如:
1 2 Map<String,Map<String,Object>> data = new HashMap<>(); 1
实际上,在 Go 语言中,也有复杂的类型,我们举几个代码示例
示例 5:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 package mainimport "fmt" func main () { slice := make ([]map [string ]int ,5 ,10 ) slice0 := make ([]map [string ]int ,0 ,10 ) fmt.Println("slice=" ,slice) fmt.Println("slice=0" ,slice0) slice[0 ] = make (map [string ]int ,10 ) slice[0 ]["age" ] = 19 fmt.Println(slice[0 ]["age" ]) }
输出结果:
1 2 3 4 slice= [map[] map[] map[] map[] map[]] slice=0 [] 19 123
接下来继续看代码:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 package mainimport "fmt" func main () { slice := make ([]map [string ]int ,5 ,10 ) slice0 := make ([]map [string ]int ,0 ,10 ) fmt.Println("slice=" ,slice) fmt.Println("slice=0" ,slice0) }
输出:
1 panic: assignment to entry in nil map
看下面这个报错:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 package mainimport "fmt" func main () { slice := make ([]map [string ]int ,5 ,10 ) slice0 := make ([]map [string ]int ,0 ,10 ) fmt.Println("slice=" ,slice) fmt.Println("slice=0" ,slice0) slice0[0 ]["age" ] = 10 ; }
输出:
1 slice= [mappanic: runtime error: index out of range
接下来我们看一个:类似于 Java 中常用的 map 类型
1 2 3 4 5 6 7 8 9 10 11 12 13 14 package mainimport "fmt" func main () { var myMap = make (map [string ]map [string ]interface {},10 ) fmt.Println(myMap) myMap["li_ming_id_123" ] = make (map [string ]interface {},5 ) myMap["li_ming_id_123" ]["school" ] = "清华大学" fmt.Println(myMap) }
输出:
1 2 map[] map[li_ming_id_123:map[school:清华大学]]
# 30.Go 中的 time 时间包模块和 Java 中的时间 API 使用区别
Go 中关于时间处理的操作在 time 包中
1. 基本获取时间信息
参考如下代码示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 package mainimport ( "fmt" "time" ) func main () { now := time.Now() year := now.Year() month := now.Month() day := now.Day() hour := now.Hour() min := now.Minute() second :=now.Second() timestamp := now.Unix() ntimestamp := now.UnixNano() fmt.Println("year=" ,year) fmt.Println("month=" ,month) fmt.Println("day=" ,day) fmt.Println("hour=" ,hour) fmt.Println("min=" ,min) fmt.Println("second=" ,second) fmt.Println("timestamp=" ,timestamp) fmt.Println("ntimestamp=" ,ntimestamp) }
2. 格式化时间
Go 的时间格式化和其它语言不太一样,它比较特殊,取了 go 的出生日期作为参数标准
参考如下代码示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 package mainimport ( "fmt" "time" ) func main () { now := time.Now() fmt.Println(now.Format("2006-01-02 15:04:05" )) fmt.Println(now.Format("2006/01/02 15:04:05" )) fmt.Println(now.Format("2006/01/02" )) fmt.Println(now.Format("15:04:05" )) }
# 31.Go 和 Java 关于 Socket 编程的对比
# 32. 聊聊 Go 语言如何连接 Mysql 数据库
# 33. 聊聊 Go 语言如何使用 Redis
# 34.Go 中的依赖管理–Module, 对比 Java 的 maven
# 35.Go 的协程高并发支持与 Java 的区别
# 36.Go 的性能调优和 Java 的性能调优
# 37.Go 的测试 API 与 Java 的单元测试
# 38. 自定义类型 Type
# 39.Go 的参数值传递与引用传递
接下来我们讲一下 Go 中的参数传递原理。
关于参数传递是一个什么概念呢,参数传递相关的知识是在研究当调用一个函数时,把外部的一个变量传入函数内,在函数内修改这个参数是否会对外部的参数变量的值有影响。参数传递用在的一个地方是函数的参数传递。(还有方法的接收器参数传递)
比如李明今天没有写作业,到了学校后匆匆忙忙的找小红要作业本 (小红的作业本为方法调用处传入的参数),想要抄一抄补上,所以李明有一个抄作业的任务 (抄作业的任务为函数),那么他有两个选择可以完成抄作业的任务。
第一个是直接拿过来小红的作业本开始抄,这在函数中叫做引用传递,因为如果小明抄的时候不小心桌子上的水打翻了,弄湿了小红的作业本,小红的作业本就真湿了,没法交了。
第二个是用打印机把小红的作业打印一份,然后拿着打印的那份抄,这叫做值传递,也就是说我拷贝一份值来用,那么我在抄作业 (任务函数内) 无论怎么弄湿小红的作业本,小红真正的自己的作业本也不受到影响。
在编程语言的函数中,如果是值传递,则是一个拷贝,在方法内部修改该参数值无法对其本身造成影响,如果是引用传递的概念,则可以改变其对象本身的值。
在 Go 语言中只有值传递,也是是说,无论如何 Go 的参数传递的都是一个拷贝。
重点来了:
Go 中的值传递有两种类型,如下:
1. 第一种值传递是具体的类型对象值传递,可能是 int,string,struct 之类的。
1 在此时,如果我们要自定义一个struct类型,传入参数中,可能遇到一个坑,因为是值传递,所以会拷贝一个struct对象,如果这个对象占内存比较大,而且这个函数调用频繁,会大量的拷贝消耗性能资源。
2. 第二种传递是叫指针参数类型的值传递,此时参数是一个指针类型,到具体的方法中,我们的参数也要用指针类型的参数接受,但是此时 Go 语言的内部做了一个黑箱操作。
举例 (下面还有完整可执行代码示例,先文字和伪代码举例):
我们有一个类型为 Boy 的结构体,还有一个方法 Mod
这个 Mod 方法的参数是一个指针类型的 Boy 对象,
我们要调用的时候应该这样传参数:
1 2 3 var boy = Boy{} //用&取boy对象的指针地址,然后传入Mod方法 Mod(&boy)
我们看看下面的代码示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 package mainimport "fmt" type Boy struct { name string age int } func Mod (b *Boy) { fmt.Printf("b的值(之前boy的地址)是%p\n" ,b) fmt.Printf("b这个指针自己的地址是=%p\n" ,&b) fmt.Println(b.name,b.age) } func main () { boy := Boy{"li_ming" ,20 } fmt.Printf("main函数中的boy地址是:%p\n" ,&boy) Mod(&boy) }
所以,Go 中的参数传递所有的都是值传递,
只不过值传递中,值可以是指针类型,是创建了一个新的指针存储原来参数 (这个参数是原对象的地址) 的值。
所以你用原对象的地址改它的属性,是有点类似于引用类型传递的效果的。
为啥说指针类型也是值传递,因为他还是创建了一个新的指针对象,值传递就是拷贝,拷贝就得创建对象,只不过这个新的指针变量存储的值是原来的参数对象的地址。
最后总结一下:
1.Go 的参数传递都是值传递。
2. 指针类型的值传递可以改变原来对象的值。
3.make 和 new 从底层原理上创建的所有对象都是指针对象,所以 make 和 new 创建出来的 slice,map,chan 或者其它任何对象都是指针传递,改变值后都可以使原来的对象属性发生变化。
# 40. 结构体转 JSON
# 41.Go 如何搭建 HTTP-Server
# 42.Go 如何搭建 HTTP-Client
# 43.Go 如何设置使用的 CPU 个数
Go 语言天生支持高并发,其中一个体现就是如果你的 Go 程序不设置并发时使用的最大 cpu 核数的话,在高并发情况下 Go 会自动把所有 CPU 都用上,跑满。
1 2 3 4 5 6 拓展阅读: 我们简单理解一下cpu(懂得可以跳过) 举例:比如有一个专门做财务的公司(计算机),他们的赚钱业务很简单(计算机工作),就是帮别人做算术题(计算机工作的具体任务),加减乘除之类的算术题,现在公司有4个员工(物理意义上的4个cpu核数),有4本算数书(4个进程),每本书有10道题(线程),一共有40道算术题要算(40个线程任务),于是4个人一起干活,在同一时间,有4道算术题被计算,最后大致上每个人算了10道算数题。 第二天,有8本算术书(8个进程),他们为了快速完成任务,规定一人(每个人是一个物理cpu核数)管2本算数书(单物理cpu内部实际上是管理的两个不同的算数书,也就是相当于有两个不同的逻辑cpu),为的就是如果第一本做烦了可以换着做第2本,混合着做,最后都做完就可以。 cpu是进行最终二进制计算的核心计算器,cpu核数是有两个概念,一个是真实世界的物理硬件核数,比如4核cpu,就是有4个物理硬件内核,然而我们在生产环境的linux服务器上top的时候,出现的cpu个数实际上是逻辑cpu数,有可能linux服务器只有4核物理cpu,可是每个物理cpu分为两个逻辑cpu,这个时候我们在linux上top看的时候就是有8个cpu信息行数据。
我们回顾一下 Java,Java 运行时我们一般管理的都是线程数,而所有的 java 线程均在 JVM 这个虚拟机进程中,于是在高并发情况下,当 cpu 资源充足时,我们需要根据 cpu 的逻辑核数来确定我们的线程池线程数 (在高并发环境下一定要设置优化线程数啊!!!线程池就能设置线程数!!!),比如我们是 4 个物理 cpu, 每个双核逻辑,一共逻辑八核 cpu, 此时,比如我们要做并发定时任务,这台服务器没有其它程序,8 个 cpu 全都给我们自己用,那么我们的线程数最少也要设置成 8,再细化,我们得根据程序执行的任务分别在 cpu 计算 (正常处理程序业务逻辑) 的耗时和 cpuIO 耗时(IO 耗时比如查 mysql 数据库数据),假如我们定时跑批任务一个任务计算用时 0.2 秒,查数据库 0.8 秒 (自己可以写程序监测),那么可以参考如下公式:
总任务耗时 /cpu 耗时 = 多少个线程 (每个逻辑 cpu)
我们算出每个逻辑 cpu 要跑多少个线程后再乘以逻辑 cpu 的个数,就能算出来了
如下:
(0.2+0.8)/0.2=5 个线程 (每个逻辑 cpu)
5*8 = 40
于是我们在线程池的时候应该这么写:
1 ExecutorService fixedThreadPool = Executors.newFixedThreadPool(40);
至于公式为什么要这么写,是因为 IO 操作的时候,cpu 是空闲的,也就是说,0.8 秒数据库操作的时候,cpu 都是空闲的,那么我们就多开几个线程让 cpu 在这 0.8 秒的时候工作,开几个呢,要等 0.8 秒,一个任务 cpu 要计算 0.2 秒,0.8/0.2=4 (个),可是这个逻辑 cpu 还有一个主线程在那 0.8 秒上等着结果呢,所以是 4+1=5(个)线程。
上述我们回顾了 Java 中的线程数和 CPU 核数相关,接下来我们来看 Go 语言。
我们下面来仔细讲讲 Go 中的 goroutine (实际是协程),是如何天然的支持高并发的,它与 Java 中的线程 Thread 又有什么区别,为什么它比线程能更好的支持高并发。
# 44. 初始化结构体,匿名结构体,结构体指针 (再讲)
# 45. 方法中的值接收和指针接收的区别 (方法进阶细节讲解)
我们之前讲了如果给一个类型绑定上一个接受者,就可以为这个类型添加一个这个类型独有的函数,只有这个类型对象自己能调用的函数,这个特殊的函数叫方法。
现在,我们讲一下方法关于传递接受者(自身引用)的进阶玩法。
Go 语言中的参数传递
# 46. 基于包模块的 Init 函数
# 47.Go 语言中的初始化依赖项
# 48.slice 相关知识点
slice 的中文意思是切片。
要想理解切片,我们首先要理解数组。
数组是一个长度不能变化的容器,存储同一数据类型的数据。
比如:int 数组
切片是对数组中一截,一小段,一个子集的地址的截取,切片存储的是它指向的底层数组中的一小截数据的地址,切片中不存数据,创建切片也不会把数组中的数据 copy 一份,切片只是存储着数组中一部分连续的数据的地址,切片的每一个元素实际上都指向具体的数组的中一个元素。
切片内部包含三个元素:
1. 底层数组(它指向的是哪一个数组)
我们要理解底层数组是什么,先举例:
[1,2,3] 这是一个 int 数组,其中元素 1 的地址是 0x0001, 元素 2 的地址是 0x0002,元素 3 的地址是 0x0003。
那么如果我们创建一个通过数组 [1,2,3] 创建一个切片 x。
这个 x 里面存储的并不是拷贝的另外一份新的 [1,2,3]。
切片 x 实际上是这样子的:
[0x0001,0x0002,0x0003]
当我们取出 x [0] 的时候,它操作的实际上是 0x0001 这个地址的元素,而这个地址实际上就是数组 [1,2,3] 中的 1 的地址。
也就是说,当我们修改了数组 [1,2,3] 中的 1 后,比如 0x0001 = 5 , 切片 x 中的 0 元素的取值自动也不一样了,因为 0x0001 地址上存储的 值已经被改成 5 了,而 x [0] 实际上还是 0x0001, 所以此时取出 x [0], 得到的就是 5。
切片存储的每一个元素实际上是它指向的底层数组的每一个元素的地址。
也就是说切片是一个引用类型,它不存储元素,不拷贝元素,它存储数组元素的引用,通过修改切片会修改原来数组的值。
2. 切片的长度
这个切片中有有几个元素,指向了数组中的几个连续的元素。
3. 切片的容量
从切片在底层数组的起始下标 (切片的首个元素) 到底层数组的最后一个元素,一共有几个元素,切片的容量就是几。
例如:(下面先用伪代码示例,后面有具体可执行代码)
原数组:a = [1,2,3,4,5,6,7,8]
切片: b = a [2:5] 从数组 a 的下标为 2 的开始,也就是具体数值是 3 开始,截取到下标为 5,下标为 5 的是 6,因为切片截取是左开右闭,所以切片中包括下标为 2 的数值 3,不包含下标为 5 的数值 6。
切片存储的地址指向的数据是:[3,4,5]
因为 3,4,5 有三个数,所以切片的长度是 3。
因为从切片的起始元素 3 到底层数组的末尾元素 8 之间有 6 个元素,所以切片的容量是 6。
修改切片实际上是修改切片指向的底层数组中的值。
# 49.Go 中类似于函数指针的功能
Go 中要实现函数指针非常简单。
因为 Go 中的函数也是一种类型。
所以我们只要声明一个变量,把某一个函数赋值给这个变量,就能实现函数指针的效果。
如下代码示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 package mainimport "fmt" func myAddFun (x,y int ) int { return x+y } func mySubFun (x,y int ) int { return x-y; } var myPointFun func (x,y int ) int func main () { myPointFun = myAddFun fmt.Printf("a+b = %d\n" ,myPointFun(10 ,20 )) myPointFun = mySubFun fmt.Printf("a-b = %d\n" ,myPointFun(100 ,50 )) }
输出:
# 50.Go 有没有注解
原生的 Go 语言的 SDK 是不支持注解功能的,但是有一些其它的第三方机构为了实现自己的某些功能需求编写了一些自定义的注解。
# 51.Go 不能做大数据相关的开发
因为大数据的一些底层都是 Java 开发的,用 Java 实现接口开发功能非常方便快捷,对于 Go 语言的支持包比较少,另外就是一些算法库像 numpy,pandas 和一些机器学习,深度学习算法库 Python 支持的比较好,对于 Go 的支持很不好。
# 52.Go 没有泛型
Go 中不支持泛型 (明确)
# 53.Go 如何产生随机数 (随机数和种子)
# 54.Go 如何打类似于 (java jar 那种依赖包).a 的工具依赖包 (有了 Module 后不用这个了)
Go 中也有很多通过命令来完成辅助开发的工具,就像 Java 中 jdk 中的 java javac javap 等指令那种命令工具。
比如有 go build xxx 命令,go clean xxx 命令, go run xxx 命令…
Java 中打 jar 包可以通过 IDEA 集成开发环境图形界面化直接打包,也可以使用 jar 命令在命令行操作中 (使用不同的参数) 进行打包。
与 java jar 命令打包对应的 Go 的命令是 go install, 这个 go install 也类似于 maven 中的 install, 它会把打成的.a 后缀名结尾的工具包文件
放入 ${GOPATH} /pkg 下。
具体使用如下示例:
注意:使用 go install 之前必须在操作系统的环境变量中定义 ${GOPATH} 这个环境变量
1. 查看我们当前的操作系统中环境变量有没有定义 GOPATH。
2. 查看 ${GOPATH} 目录下是否有 src,pkg,bin 目录,并且保证我们的代码是在 src 下的。
3. 打开一个命令行窗口,比如 windows 是 cmd 打开一个 dos 窗口。
4. 我们在最开始之前已经把 go 的安装包下的包含 Go 操作指令的 bin 目录配置在了 PATH 环境变量中,所以此时我们可以不用管目录直接使用 go install 命令。
例如目录结构如下:
-------------------------mygopackge
MyUtil.go
1 记住一点,此时MyUtil中不能是main包,也不能有main函数,否则打不出来.a结尾的依赖包。
此时编写执行命令:
1 go install com/mashibing/gotest/mygopackge
此指令运行时,首先会去找 ${GOPATH} 目录
然后把后面的 com/mashibing/gotest/mygopackge 拼接上去
也就是 ${GOPATH}/com/mashibing/gotest/mygopackge
然后会把 ${GOPATH}/com/mashibing/gotest/mygopackge 下的所有.go 文件,比如 MyUtil.go 全部打包压缩进 mygopackge.a 文件
最后会把 mygopackage.a 放入 G O P A T H /p k g / {GOPATH}/pkg/GOPA T**H /pk g /{标示操作系统的一个名字 (这个不重要)}/com/mashibing/gotest/ 下。
最终.a 文件存储的结构是这样的:
${GOPATH}/pkg/com/mashibing/gotest/mygopackge.a
# 55.Go 中的依赖管理 Module 使用
# 1. 什么是 GoModule?(Go 中 Module 和包的区别?)
首先我们要理解一下 Go 的 Module 是一个什么概念?
我先简单的说一下,Go 中的 Module 是 GoSDK1.11 后提出的一个新的类似于包的机制,叫做模块,在 1.13 版本后成熟使用,GoSDK 中 Module 功能是和相当于一个包的增强版,一个模块类型的项目在根目录下必须有一个 go.mod 文件,这个模块项目内部可以有很多个不同的包,每个包下有不同的代码,我们下载依赖的时候是把这个模块下载下来 (模块以压缩包 (比如 zip) 的形式存储在 G O P A T H /p k g /m o d /c a c h e / 下,源码文件也会在 {GOPATH}/pkg/mod/cache/ 下,源码文件也会在 GOPA T**H /pk g /mo d /ca ch e / 下,源码文件也会在 {GOPATH}/pkg/mod/ 下)。
我们导入模块的时候只需要引入一次,使用模块中不同的包的时候可以通过 import 模块下不同的包名,来引入不同包的功能。
比如下面的结构
-----------com.mashibing.module
-----------------------package1
--------------test1.go
------------------------package2
-------------test2.go
然后我们只需要在 go.mod 中引入这一个模块,就能在 import 的时候任意引入 package1 或 package2。
# 2. 为什么要使用 GoModule?
# 1). 团队协作开发中对于包的版本号的管理
在没有 Module 之前,我们都是把自己写的 Go 程序打成包,然后别的程序引用的话引入这个包。
可是在开发中这些包的版本有个明显的不能管理的问题。
比如我怎么知道这个包是今天开发的最新版还是明天开发的,我在团队协同开发中怎么把别人写的最新版本的包更新到我的项目中。
# 2)便于开发中的依赖包管理
其次,我们在开发中下载了别人的项目,怎么快速的观察有哪些依赖包,如何快速的把所有依赖包从仓库中下载下来,都是一个问题,
这两个问题就可以通过观察项目根目录下的 go.mod 文件的依赖模块列表和执行 go mod download 命令快速从第三方模块仓库中下载依赖包来完成。
# 3). 隔离管理不同类别的项目
有了 Module 后,我们可以把我们自己的项目和系统的项目隔离管理,我们的项目不用必须放在 ${GOPATH}/src 下了
# 3. 哪些项目能使用 GoModule?
一个 GoModule 项目要想引入其它依赖模块,需要在根目录下的 go.mod 中添加对应的依赖模块地址。
注意:!!!重点来了!!!
GoModule 只能引用同样是 Module 类型的项目,经常用于引用内部自己的项目。
像 maven 仓库一样引用开源模块的依赖也是一个特别常用的场景。
不过我们需要修改代理地址访问国内的第三方 GoModule 提供商。
https://goproxy.cn/ 是一个国内的可访问的 GoModule 依赖仓库,类似于 Java 中 maven 中央仓库的概念。
# 4.GoModule 的版本问题?
我们使用 Go module 之前,首先要保证我们的 Go SDK 是在 1.13 以及以上版本。(Go1.11 以上就可以使用 Module, 但是需要设置一些开启等,1.13 后默认开启)
因为在 1.13 版本上官方正式推荐使用,成为稳定版了。
Go 也有代码仓库,比如可以使用 github 作为 go 项目的代码仓库,Go 语言本身就提供了一个指令 go get 来从指定的仓库中
拉取依赖包和代码,不过 go get 这个指令在开启模块功能和关闭模块功能下用法不一样,下面有开启模块下的用法。
# 5.GoModule 和 Java 中 Maven 的区别?
Go 中的 Module 和 Java 中的 Maven 不同:
首先,Module 是官方的 SDK 包自带的,它并非像 maven 一样还得安装 maven 插件之类的。
关于中央依赖仓库,Go 和 Java 中的概念是类似的,都是国内的第三方提供的。
# 6. 如何开启 GoModule?(GO111MODULE)
具体我们如何使用 Module 呢?
我们首先要检查我们的 GoSDK 版本是 1.11 还是 1.13 之上。
如果是 1.11 的话我们需要设置一个操作系统的中的环境变量,用于开启 Module 功能,这个是否开启的环境变量名是 GO111MODULE,
他有三种状态:
第一个是 on 开启状态,在此状态开启下项目不会再去 ${GOPATH} 下寻找依赖包。
第二个是 off 不开启 Module 功能,在此状态下项目会去 ${GOPATH} 下寻找依赖包。
第三个是 auto 自动检测状态,在此状态下会判断项目是否在 G O P A T H /s r c 外,如果在外面,会判断项目根目录下是否有 g o . m o d 文件,如果均有则开启 M o d u l e 功能,如果缺任何一个则会从 {GOPATH}/src 外,如果在外面,会判断项目根目录下是否有 go.mod 文件,如果均有则开启 Module 功能,如果缺任何一个则会从 GOPA T**H /src 外,如果在外面,会判断项目根目录下是否有 g**o .mo d 文件,如果均有则开启 Mo du l**e 功能,如果缺任何一个则会从 {GOPATH} 下寻找依赖包。
GoSDK1.13 版本后 GO111MODULE 的默认值是 auto,所以 1.13 版本后不用修改该变量。
注意:在使用模块的时候, GOPATH
是无意义的,不过它还是会把下载的依赖储存在 ${GOPATH}/src/mod
中,也会把 go install
的结果放在 ${GOPATH}/bin
中。
windows
linux
# 7.GoModule 的真实使用场景 1:
接下来我们代入具体的使用场景:
今天,小明要接手一个新的 Go 项目,他通过 GoLand 中的 git 工具,从公司的 git 仓库中下载了一个 Go 的项目。(下载到他电脑的非 ${GOPATH}/src 目录,比如下载到他电脑的任意一个自己的工作空间)
此时他要做的是:
1). 先打开项目根目录下的 go.mod 文件看看里面依赖了什么工具包。(这个就是随便了解一下项目)
2).Go 的中央模块仓库是 Go 的官网提供的,在国外是 https://proxy.golang.org 这个地址,可是在国内无法访问。
我们在国内需要使用如下的中央模块仓库地址:https://goproxy.cn
我们 Go 中的 SDK 默认是去找国外的中央模块仓库的,如何修改成国内的呢?
我们知道,所有的下载拉取行为脚本实际上是从 go download 这个脚本代码中实现的,而在这个脚本中的源码实现里,肯定有一个代码是写的是取出操作系统中的一个环境变量,这个环境变量存储着一个地址,这个地址代表了去哪个中央模块仓库拉取。
在 GoSDK 中的默认实现里,这个操作系统的环境变量叫做 GOPROXY,在脚本中为其赋予了一个默认值,就是国外的 proxy.golang.org 这个值。
我们要想修改,只需要在当前电脑修改该环境变量的值即可:
(注意,这个变量值不带 https, 这只是一个变量,程序会自动拼接 https)
windows
linux
1 export GOPROXY=goproxy.cn
3). 切换到项目的根目录,也就是有 go.mod 的那层目录,打开命令行窗口。
执行 download 指令 (下载模块项目到 ${GOPATH}/pkg/mod 下)
4). 如果不报错,代表已经下载好了,可以使用了,此时在项目根目录会生成一个 go.sum 文件。
一会再讲解 sum 文件。
5). 此时可以进行开发了。
# 8.GoModule 的真实使用场景 2:
场景 2:我们如何用命令创建一个 Module 的项目,(开发工具也能手动创建)。
切换到项目根目录,执行如下指令:
然后会在根目录下生成一个 go.mod 文件
我们看看这个 go.mod 文件长啥样?
1 2 3 4 // 刚才init指令后的模块名参数被写在module后了 module 模块名 //表示使用GoSDK的哪个版本 go 1.14
修改 go.mod 文件中的依赖即可。
我们有两种方式下载和更新依赖:
1. 修改 go.mod 文件,然后执行 go mod down 把模块依赖下载到自己 ${GOPATH}/pkg/mod 下,这里面装的是所有下载的 module 缓存依赖文件,其中有 zip 的包,也有源码,在一个项目文件夹下的不同文件夹下放着,还有版本号文件夹区分,每个版本都是一个文件夹。
2. 直接在命令行使用 go get package@version 更新或者下载依赖模块,升级或者降级模块的版本。(这里是开启模块后的 go get 指令用法)
例如:
1 go get github.com/gin-contrib/sessions@v0.0.1
这个指令执行过后,会自动修改 go.mod 中的文件内容,不需要我们手动修改 go.mod 文件中的内容。
# 9.go.mod 文件详解
接下来我们讲讲核心配置文件 go.mod
go.mod 内容如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 //表示本项目的module模块名称是什么,别的模块依赖此模块的时候写这个名字 module test //表示使用GoSDK的哪个版本 go 1.14 //require中声明的是需要依赖的包和包版本号 require ( //格式如下: 需要import导入的模块名 版本号 // 需要import导入的模块名2 版本号2 // ... ... github.com/gin-contrib/sessions v0.0.1 github.com/gin-contrib/sse v0.1.0 // indirect github.com/gin-gonic/gin v1.4.0 github.com/go-redis/redis v6.15.6+incompatible github.com/go-sql-driver/mysql v1.4.1 github.com/golang/protobuf v1.3.2 // indirect github.com/jinzhu/gorm v1.9.11 github.com/json-iterator/go v1.1.7 // indirect github.com/kr/pretty v0.1.0 // indirect github.com/mattn/go-isatty v0.0.10 // indirect github.com/sirupsen/logrus v1.2.0 github.com/ugorji/go v1.1.7 // indirect ) //replace写法如下,表示如果项目中有引入前面的依赖模块,改为引用=>后面的依赖模块, //可以用于golang的国外地址访问改为指向国内的github地址,当然你在上面require直接写github就不用在这里repalce了 replace ( golang.org/x/crypto v0.0.0-20190313024323-a1f597ede03a => github.com/golang/crypto v0.0.0-20190313024323-a1f597ede03a ) //忽略依赖模块,表示在该项目中无论如何都使用不了该依赖模块,可以用于限制使用某个有bug版本的模块 exclude( github.com/ugorji/go v1.1.7 )
注:go.mod 提供了 module, require、replace 和 exclude 四个命令
module 语句指定包的名字(路径)
require 语句指定的依赖项模块
replace 语句可以替换依赖项模块
exclude 语句可以忽略依赖项模块
上面 github.com/ugorji/go v1.1.7 //indirect 有 indirect 和非 indirect
indirect 代表此模块是间接引用的,中间隔了几个项目
这个不用特殊写,可以注释写便于识别和开发
# 10.GoModule 有哪些命令?如何使用?
Go 有如下关于 Module 的命令:
1 2 3 4 5 6 7 8 9 //go mod 命令: download //下载依赖模块到${GOPATH}/pkg/mod edit //一系列参数指令用于操作go.mod文件,参数太多,具体下面有例子 graph //输出显示每一个模块依赖了哪些模块 init //在一个非module项目的根目录下创建一个go.mod文件使其变为一个module管理的项目 tidy //根据项目实际使用的依赖包修改(删除和添加)go.mod中的文本内容 vendor //在项目根目录创建一个vender文件夹 然后把${GOPATH}/pkg/mod下载的本项目需要的依赖模块拷贝到本项目的vender目录下 verify //校验${GOPATH}/pkg/mod中的依赖模块下载到本地后是否被修改或者篡改过 why //一个说明文档的功能,用于说明一些包之间的为什么要这么依赖。(没啥用)
# 0). init 和 download
我们之前在案例中讲了 init,download 指令,这里不再赘述
# 1).go mod edit
是指在命令行用指令通过不同的参数修改 go.mod 文件,这个指令必须得写参数才能正确执行,不能空执行 go mod edit
参数 1 :-fmt
格式化 go.mod 文件,只是格式规范一下,不做其它任何内容上的修改。
其它任何 edit 指令执行完毕后都会自动执行 - fmt 格式化操作。
这个使用场景就是我们如果不想做任何操作,就想试试 edit 指令,就只需要跟上 - fmt 就行,因为单独不加任何参数
只有 go mod edit 后面不跟参数是无法执行的。
我们如何升级降级依赖模块的版本,或者说添加新的依赖和移除旧的依赖呢
参数 2: -require=path@version /-droprequire=path flags
添加一个依赖
1 go mod edit -require=github.com/gin-contrib/sessions@v0.0.1
删除一个依赖
1 go mod edit -droprequire=github.com/gin-contrib/sessions@v0.0.1
这两个和 go get package@version 功能差不多,但是官方文档更推荐使用 go get 来完成添加和修改依赖(go get 后的 package 和上面的 path 一个含义,都是模块全路径名)
参数 3:-exclude=path@version and -dropexclude=path@version
排除某个版本某个模块的使用,必须有该模块才可以写这个进行排除。
1 go mod edit -exclude=github.com/gin-contrib/sessions@v0.0.1
删除排除
1 go mod edit -dropexclude=github.com/gin-contrib/sessions@v0.0.1
简单来说,执行这两个是为了我们在开发中避免使用到不应该使用的包
… 还有好几个,基本很少用,省略了
# 2).go mod graph
命令用法: 输出每一个模块依赖了哪些模块 无参数,直接使用 ,在项目根目录下命令行执行
比如:
模块 1 依赖了模块 a
模块 1 依赖了模块 b
模块 1 依赖了模块 c
模块 2 依赖了模块 x
模块 2 依赖了模块 z
如下是具体例子:
1 2 3 4 5 6 7 8 9 10 11 12 13 C:\${GOPAHT}\file\project>go mod graph file\project github.com/edgexfoundry/go-mod-bootstrap@v0.0.35 github.com/edgexfoundry/go-mod-bootstrap@v0.0.35 github.com/BurntSushi/toml@v0.3.1 github.com/edgexfoundry/go-mod-bootstrap@v0.0.35 github.com/edgexfoundry/go-mod-configuration@v0.0.3 github.com/edgexfoundry/go-mod-bootstrap@v0.0.35 github.com/edgexfoundry/go-mod-core-contracts@v0.1.34 github.com/edgexfoundry/go-mod-bootstrap@v0.0.35 github.com/edgexfoundry/go-mod-registry@v0.1.17 github.com/edgexfoundry/go-mod-bootstrap@v0.0.35 github.com/edgexfoundry/go-mod-secrets@v0.0.17 github.com/edgexfoundry/go-mod-bootstrap@v0.0.35 github.com/gorilla/mux@v1.7.1 github.com/edgexfoundry/go-mod-bootstrap@v0.0.35 github.com/pelletier/go-toml@v1.2.0 github.com/edgexfoundry/go-mod-bootstrap@v0.0.35 github.com/stretchr/testify@v1.5.1 github.com/edgexfoundry/go-mod-bootstrap@v0.0.35 gopkg.in/yaml.v2@v2.2.8 github.com/edgexfoundry/go-mod-configuration@v0.0.3 github.com/cenkalti/backoff@v2.2.1+incompatible github.com/edgexfoundry/go-mod-configuration@v0.0.3 github.com/hashicorp/consul/api@v1.1.0
# 3).go mod tidy
根据实际项目使用到的依赖模块,在 go.mod 中添加或者删除文本引用
有一个参数可选项 -v 输出在 go.mod 文件中删除的引用模块信息
比如我们项目用到一个模块,go.mod 中没写,执行后 go.mod 中就会添加上该模块的文本引用。
如果我们在 go.mod 中引用了一个模块,检测在真实项目中并没有使用,则会在 go.mod 中删除该文本引用。
使用如下:
输出:
1 unused github.com/edgexfoundry/go-mod-bootstrap
输出表示检测项目没有使用到该模块,然后从 go.mod 中把该包的引用文字给删除了。
# 4).go mod vender
该指令会在项目中建立一个 vender 目录,然后把 ${GOPATG}/pkg/mod 中下载的依赖拷贝到项目的 vender 目录中,方便管理和方便在 idea 中引用依赖。 -v 参数可以在控制台输出相关的结果信息
# 5).go mod verify
验证下载到 ${GOPATH}/pkg/mod 中的依赖模块有没有被修改或者篡改。
结果会输出是否被修改过
比如输出:
这个是所有模块已经验证,代表没有被修改,如果被修改,会提示哪些被修改。
# 6).go mod why
这个没啥用,说白了就是一个解释文档,输入参数和依赖他说明哪些包为啥要依赖这些包,不用看它,用处不大。
# 11.go.sum 详细讲解
# 1).go.sum 什么时候会更新或者新建生成?
当我们通过 go mod download 下载完依赖模块或者 go get package@version 更新了依赖包的时候
,会检查根目录下有没有一个叫 go.sum 的文件,没有的话则创建一个并写入内容,有的话会更新 go.sum 中的内容。
# 2).go.sum 是用来做什么的?
go.sum 的作用是用来校验你下载的依赖模块是否是官方仓库提供的,对应的正确的版本的,并且中途没有被黑客篡改的。
go.sum 主要是起安全作用和保证依赖的版本肯定是官方的提供的那个版本,版本确认具体是确认你下载的那个模块版本里面的代码的和官方提供的模块的那个版本的代码完全相同,一字不差。
通过 go.sum 保证安全性是很有必要的,因为如果你的电脑被黑客攻击了,黑客可以截取你对外发送的文件,也可以修改发送给你的文件,那么就会产生一个问题:
本来的路径应该是这样的: 第三方模块依赖库 ------------> 你的电脑
结果中间有黑客会变成这样:
第三方模块依赖库 --------> 黑客修改了依赖库中的代码,植入病毒代码,并重新打成模块发送给你 ---------> 你以为是官方的版本
结果黑客就把病毒代码植入到了你的项目中,你的项目就不安全了,面临着数据全部泄露的风险。
# 3).go.sum 是如何实现校验机制的?它包含什么内容?
说到校验安全机制,有一种常规的玩法就是使用不可逆加密算法,不可逆加密算法是指将 a 文本通过算法加密成 b 文本后,b 文本永远也不能反着计算出 a 文本。
不可加密算法的具体是怎么应用的呢?它是如何起作用的?
我们在这里先讲一个不可逆的加密算法 SHA-256 算法。
SHA-256 算法的功能就是将一个任意长度的字符串转换成一个固定长度为 64 的字符串,比如:
4e07408562bedb8b60ce05c1decfe3ad16b72230967de01f640b7e4729b49fce
这里从 4e07 代表四个字符串,按此算,这个加密后的字符串为 64 个。
为什么是 64 个呢?
因为 64 个字符串每两个字符为一组,比如 4e 是一组,07 是一组,也就是说有 32 组,每一组是一个十六进制的数值,一个十六进制的数值也就是两个字符用计算机中的 8 个字节内存空间存储,也就是一个十六进制的数字,有两个字符串,占 8 个字节,一个字节等同 8 位 (bit)(位只能存储 0 和 1 两个值),也就是说:
32(32 个十六进制数,每个十六进制数用两个字符表示)*8 字节 = 256 位。
仔细看名字,SHA 代表是算法的加密方式类型,256 代表的是他这个是 256 位的版本。
具体原理实现是 SHA 内部定义了一系列固定数值的表,然后加密的时候无论是需要加密多少文字,它都按照一定的规则从需要加密的文字中按一定规则抽取其中的缩略一部分,然后拿缩略的一部分和 SHA 内部的固定数值表进行固定的 hash 映射和算术操作,这个 hash 映射和算术操作的顺序是固定写死的,公共数据表是写死的,这个写死的顺序和公共数据表就是这个算法的具体内容本质。
这样的话,因为抽取的是缩略的内容,所以我们可以把输出结果固定在 64 个字符,256 位。
因为是缩略的内容, 所以我们不可能通过缩略的内容反推出完整的结果。
但是,相同的文本按照这个算法加密出来的 64 个字符肯定是相同的,同时,只要改变原需要加密文本的一个字符,也会造成加密出来的 64 个字符大不相同。
我们用 SHA-256 通常是这么用的:
A 方 要 发送信息给 B 方
B 方 要确定信息是 A 方发送的,没有经过篡改
此时 A 和 B 同时约定一个密码字符串,比如 abc。
这个 abc 只有 A 方和 B 方知道。
A 方把 需要传输的文本拼接上 abc,然后通过 SHA-256 加密算出一个值,把原文本和算出的值全部发送给 B。
B 方 拿出原文本,拼接上 abc,进行 SHA256 计算,看看结果是否和传输过来的 A 传输的值一样,如果一样,代表中间没有被篡改。
为什么呢?
因为如果有一个黑客 C 想要篡改,他就得同时篡改原文本和算出的签名值。
可是 C 不知道密码是 abc,它也就不能把 abc 拼接到原文后,所以它算出来的签名和 B 算出来的签名肯定不一致。
所以 B 如果自己算出的签名值与接收到的签名值不一致,B 就知道不是 A 发过来的,就能校验发送端的源头是否是官方安全的了。
接下来我们讲一下 go.sum 的验证机制。
首先说下 go.sum 中存储的内容,这个文件存储的每一行都是如下格式
模块名 版本号 hash 签名值
示例:
1 2 3 github.com/google/uuid v1.1.1 h1:Gkbcsh/GbpXz7lPftLA3P6TYMwjCLYm83jiFQZF/3gY= github.com/google/uuid v1.1.1/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
这里的 hash 签名值是拿当前模块当前版本号内的所有代码字符串计算出来的一个值,就是通过上面讲解的 SHA-256 计算的。
所以哪怕是这个模块中的代码有一个字变了,计算出来的 hash 值也不相同。
第三方模块库在每发布一个新的模块版本后,会按照 SHA-256 计算出对应版本的 hash 值,然后提供给外部获取用于检验安全性。
当我们 go mod download 和 go get package@version 后 会更新 go.mod 中的模块路径和版本。
然后会更新或者创建根目录下 go.sum 文件中的模块名 版本号 和 hash 值。
在 go.sum 中的 hash 值是在下载和更新依赖包的时候,同时获取官方提供的版本号得来的。
也就是说,基本上 go.sum 中的文件都是从官网(外国)(中国是第三方模块仓库)上获得的正品版本号,这个版本号是仓库方自己算的,你只是获取到了存储到你自己的 go.sum 中。
具体如何获取版本号有个小知识点:
1 go module机制在下载和更新依赖的时候会取出操作系统中名为`GOSUMDB`的环境变量中的值,这个服务器地址值代表了从哪个第三方仓库获取对应的正品版本号。
重点来了,当你在 go build 打包创建 go 项目的时候,go build 的内部指令会去拿你本地的模块文件进行 SHA-256 计算,然后拿到一个计算出来的结果值,之后它会拿此值和 go.sum 中的正确的从官网拉取的值进行对比,如果不一样,说明这个模块包不是官方发布的,也就是你本地的模块包和官方发布的模块包中的代码肯定有差异。
# 四。专门详解 Go 并发编程相关知识
# 1.Go 为什么天然支持高并发,纤程比线程的优势是什么?
Go 语言在设计的时候就考虑了充分利用计算机的多核处理器,具体表现为,Go 中开启一个并发的任务以操作系统的线程资源调度为单位的,而是 Go 的创造者们自己写了一套管理多个任务的机制,在这个机制下,每一个并发的任务线程叫做纤程,这个纤程的作用等同一个线程,也是并发执行的,只不过纤程是在应用程序管理的,懂底层的可以讲是在用户态的一个线程,而 Java 中调度的线程是属于操作系统,也就是操作系统内核态的线程。
用户态的纤程归属于用户编写的软件管理和调度,优点是可以根据情况灵活实现堆栈的内存分配,最优化其中的运行资源配置。
内核态的线程归属于操作系统调度和管理,他底层是有 windows 或者 linux 操作系统底层的代码管理的,那么他就不灵活,每个线程分配的资源可能造成浪费,创建的线程数肯定也有一定的限制。
Go 的创造可以为自己的语言和任务灵活配置资源,Linux 和 windows 操作系统的代码是通用的,总不能为你这个语言修改源代码把。
在实际程序运行中,一个操作系统的内核态线程可能管理着好几个甚至数十个纤程 (根据实际情况和设置不同而不同),所以省去了线程时间片上下文切换的时间。
同时因为内部机制灵活,所以执行效率高,占用内存也少。
这就是 Go 语言的并发优势的核心所在。
# 2. 并发和并行的区别?
并发是指的一个角色在一段时间内通过来回切换处理了多个任务。
并行是指两个或者多个角色同时处理自己的任务。
举例:
并发:在一个小时内,你写了 10 分钟语文作业,又写了 10 分钟数学,之后又写了 10 分中英语作业,然后再从语文 10 分钟,数学 10 分钟,英文 10 分钟又来一次。
这个叫做你并发的写语文数学英语作业。
你一个一段时间(一个小时内)通过切换(一会写数学,一会写语文。。。),处理了多个任务(写了三门课的作业)
并行:你和小明同时写自己的作业。你们俩同时运行的状态叫做并行运作状态,强调的是你们两个人同时在处理任务 (做作业)。
你和小明 (两个以上的角色) 同时写作业 (处理自己的任务)。
在计算机中,比如有 4 个 cpu,4 个 cpu 同时工作,叫做这 4 个 cpu 并行执行任务,每个 cpu 通过时间片机制上下文切换处理 100 个小任务,叫做每个 cpu 并发的处理 100 个任务。
# 3.Go 是如何用 Channel 进行协程间数据通信数据同步的?
go 中的线程相关的概念是 Goroutines (并发),是使用 go 关键字开启。
Java 中的线程是通过 Thread 类开启的。
在 go 语言中,一个线程就是一个 Goroutines,主函数就是(主) main Goroutines。
使用 go 语句来开启一个新的 Goroutines
比如:
普通方法执行
myFunction()
开启一个 Goroutines 来执行方法
go myFunction()
java 中是
new Thread(()->{
// 新线程逻辑代码
}).start();
参考下面的代码示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 package mainimport ( "fmt" ) func myFunction () { fmt.Println("Hello!!!" ) } func goroutineTestFunc () { fmt.Println("Hello!!! Start Goroutine!!!" ) } func main () { }
线程间的通信:
java 线程间通信有很多种方式:
比如最原始的 wait/notify
到使用 juc 下高并发线程同步容器,同步队列
到 CountDownLatch 等一系列工具类
…
甚至是分布式系统不同机器之间的消息中间件,单机的 disruptor 等等。
Go 语言不同,线程间主要的通信方式是 Channel。
Channel 是实现 go 语言多个线程(goroutines)之间通信的一个机制。
Channel 是一个线程间传输数据的管道,创建 Channel 必须声明管道内的数据类型是什么
下面我们创建一个传输 int 类型数据的 Channel
代码示例:
1 2 3 4 5 6 7 8 package mainimport "fmt" func main () { ch := make (chan int ) fmt.Println(ch) }
channel 是引用类型,函数传参数时是引用传递而不是值拷贝的传递。
channel 的空值和别的应用类型一样是 nil。
== 可以比较两个 Channel 之间传输的数据类型是否相等。
channel 是一个管道,他可以收数据和发数据。
具体参照下面代码示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 package mainimport ( "fmt" "time" ) var chanStr = make (chan string )func main () { fmt.Println("main goroutine print Hello " ) go startNewGoroutineOne() go startNewGoroutineTwo() time.Sleep(100 *time.Second) } func startNewGoroutineOne () { fmt.Println("send channel print Hello " ) chanStr <- "Hello!!!" } func startNewGoroutineTwo () { fmt.Println("receive channel print Hello " ) strVar := <-chanStr fmt.Println(strVar) }
无缓存的 channel 可以起到一个多线程间线程数据同步锁安全的作用。
缓存的 channel 创建方式是
make (chan string, 缓存个数)
缓存个数是指直到多个数据没有消费或者接受后才进行阻塞。
类似于 java 中的 synchronized 和 lock
可以保证多线程并发下的数据一致性问题。
首先我们看一个线程不安全的代码示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 package mainimport ( "fmt" "time" ) var moneyA int =1000 func subtractMoney (subMoney int ) { time.Sleep(3 *time.Second) moneyA-=subMoney } func getMoney () int { return moneyA; } func main () { go func () { if (getMoney()>200 ) { subtractMoney(200 ) fmt.Printf("200元扣款成功,剩下:%d元\n" ,getMoney()) } }() go func () { if (getMoney()>900 ) { subtractMoney(900 ) fmt.Printf("900元扣款成功,剩下:%d元\n" ,getMoney()) } }() time.Sleep(5 *time.Second) fmt.Println(getMoney()) }
缓存为 1 的 channel 可以作为锁使用:
示例代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 package mainimport ( "fmt" "time" ) var moneyA = 1000 var synchLock = make (chan int ,1 )func subtractMoney (subMoney int ) { time.Sleep(3 *time.Second) moneyA-=subMoney } func getMoney () int { return moneyA; } func main () { go func () { synchLock<-10 if (getMoney()>200 ) { subtractMoney(200 ) fmt.Printf("200元扣款成功,剩下:%d元\n" ,getMoney()) } <-synchLock }() go func () { synchLock<-10 if (getMoney()>900 ) { subtractMoney(900 ) fmt.Printf("900元扣款成功,剩下:%d元\n" ,getMoney()) } synchLock<-10 }() time.Sleep(5 *time.Second) fmt.Println(getMoney()) }
# 4.Go 中的 Goroutine 使用和 GMP 模型?
Go 中的线程 (实际是纤程) goroutine 的底层管理和调度是在 runtime 包中自己实现的,其中遵循了 GMP 模型。
G 就是一个 goroutine,包括它自身的一些元信息。
M 是指操作系统内核态的线程的一个虚拟表示,一个 M 就是操作系统内核态的一个线程。
P 是一个组列表,P 管理着多个 goroutines,P 还有一些用于组管理的元数据信息。
# 5.Go 的 select 怎么用?
Go 中的 select 是专门用于支持更好的使用管道 (channel) 的。
我们之前虽然讲了能从管道中读取数据,但是这有一个缺陷,就是我们在一个 Goroutine 中不能同时处理读取多个 channel,因为在一个 Goroutine 中,一个 channel 阻塞后就无法继续运行了,所以无法在一个 Goroutine 处理多个 channel, 而 select 很好的解决了这个问题。
select 相当于 Java 中 Netty 框架的多路复用器的功能。
举例代码示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 package mainimport "fmt" func main () { myChan := make (chan int ,1 ) for i:=1 ;i<=100 ;i++{ select { case data := <-myChan: fmt.Println(data) case myChan<-i: default : fmt.Println("default !!!" ) } } }
# 6.Go 中的互斥锁 (类似于 Java 中的 ReentrantLock)
先按线程不安全的数据错误的代码示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 package mainimport ( "fmt" "sync" ) var num int var wait sync.WaitGroupfunc main () { wait.Add(5 ) go myAdd() go myAdd() go myAdd() go myAdd() go myAdd() wait.Wait() fmt.Printf("num = %d\n" ,num) } func myAdd () { defer wait.Done() for i:=0 ;i<10000 ;i++ { num+=1 } }
打印输出结果:
互斥锁示例代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 package mainimport ( "fmt" "sync" ) var num int var wait sync.WaitGroupvar lock sync.Mutexfunc main () { wait.Add(5 ) go myAdd() go myAdd() go myAdd() go myAdd() go myAdd() wait.Wait() fmt.Printf("num = %d\n" ,num) } func myAdd () { defer wait.Done() for i:=0 ;i<10000 ;i++ { lock.Lock() num+=1 lock.Unlock() } }
# 7.Go 中的读写锁 (类似于 Java 中的 ReentrantReadWriteLock)
读写锁用于读多写少的情况,多个线程并发读不上锁,写的时候才上锁互斥
读写锁示例代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 package mainimport ( "fmt" "sync" "time" ) var moneyA = 1000 var rwLock sync.RWMutex;var wait sync.WaitGroupfunc subtractMoney (subMoney int ) { rwLock.Lock() time.Sleep(3 *time.Second) moneyA-=subMoney rwLock.Unlock() } func getMoney () int { rwLock.RLock() result := moneyA rwLock.RUnlock() return result; } func main () { wait.Add(2 ) go func () { defer wait.Done() if (getMoney()>200 ) { subtractMoney(200 ) fmt.Printf("200元扣款成功,剩下:%d元\n" ,getMoney()) }else { fmt.Println("余额不足,无法扣款" ) } }() go func () { defer wait.Done() if (getMoney()>900 ) { subtractMoney(900 ) fmt.Printf("900元扣款成功,剩下:%d元\n" ,getMoney()) }else { fmt.Println("余额不足,无法扣款" ) } }() wait.Wait() fmt.Println(getMoney()) }
# 8.Go 中的并发安全 Map (类似于 CurrentHashMap)
Go 中自己通过 make 创建的 map 不是线程安全的,具体体现在多线程添加值和修改值下会报如下错误:
1 2 fatal error : concurrent map writes 1
这个错类似于 java 中多线程读写线程不安全的容器时报的错。
Go 为了解决这个问题,专门给我们提供了一个并发安全的 map,这个并发安全的 map 不用通过 make 创建,拿来即可用,并且他提供了一些不同于普通 map 的操作方法。
参考如下代码示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 package mainimport ( "fmt" "sync" ) var myConcurrentMap = sync.Map{}var myRangeMap = sync.Map{}func main () { myConcurrentMap.Store(1 ,"li_ming" ) name,ok := myConcurrentMap.Load(1 ) if (!ok) { fmt.Println("不存在" ) return } fmt.Println(name) name2, ok2 := myConcurrentMap.LoadOrStore(1 ,"xiao_hong" ) fmt.Println(name2,ok2) name3, ok3 := myConcurrentMap.LoadOrStore(2 ,"xiao_hong" ) fmt.Println(name3,ok3) myConcurrentMap.Delete(1 ) rangeFunc() } func rangeFunc () { myRangeMap.Store(1 ,"xiao_ming" ) myRangeMap.Store(2 ,"xiao_li" ) myRangeMap.Store(3 ,"xiao_ke" ) myRangeMap.Store(4 ,"xiao_lei" ) myRangeMap.Range(func (k, v interface {}) bool { fmt.Println("data_key_value = :" ,k,v) return true }) }
# 9.Go 中的 AtomicXXX 原子操作类 (类似于 Java 中的 AtocmicInteger 之类的)
Go 中的 atomic 包里面的功能和 Java 中的 Atomic 一样,原子操作类,原理也是 cas, 甚至提供了 cas 的 api 函数,这里不做过多讲解,
简单举一个代码示例,因为方法太多,详细的请参考 api 文档中的 atomic 包:
1 2 3 4 5 6 7 8 9 package mainimport "sync/atomic" func main () { var num int64 = 20 atomic.AddInt64(&num,1 ) }
# 10.Go 中的 WaitGroup (类似于 Java 中的 CountDownLatch)
现在让我们看一个需求,比如我们开启三个并发任务,然后三个并发任务执行处理完毕后我们才让主线程继续往下面走。
这时候肯定不能用睡眠了,因为不知道睡眠多长时间。
这是 Go 中的 sync 包提供了一个 WaitGroup 的工具,他基本上和 Java 中的 CountDownLatch 的功能一致。
接下来让我们看代码示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 package mainimport ( "fmt" "sync" "time" ) var wait sync.WaitGroupfunc main () { wait.Add(3 ) go myFun1() go myFun2() go myFun3() wait.Wait() } func myFun1 () { defer wait.Done() time.Sleep(time.Second*5 ) fmt.Println("fun1执行完毕" ) } func myFun2 () { defer wait.Done() time.Sleep(time.Second*5 ) fmt.Println("fun2执行完毕" ) } func myFun3 () { defer wait.Done() time.Sleep(time.Second*5 ) fmt.Println("fun3执行完毕" ) } 码如下: `` `go package main import ( "fmt" "sync" ) //全局变量 var num int var wait sync.WaitGroup var lock sync.Mutex func main() { wait.Add(5) go myAdd() go myAdd() go myAdd() go myAdd() go myAdd() wait.Wait() //预期值等于5万,可是因为线程不安全错误,小于5万 fmt.Printf("num = %d\n",num) } func myAdd() { defer wait.Done() for i:=0 ;i<10000;i++ { lock.Lock() num+=1 lock.Unlock() } }
# 7.Go 中的读写锁 (类似于 Java 中的 ReentrantReadWriteLock)
读写锁用于读多写少的情况,多个线程并发读不上锁,写的时候才上锁互斥
读写锁示例代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 package mainimport ( "fmt" "sync" "time" ) var moneyA = 1000 var rwLock sync.RWMutex;var wait sync.WaitGroupfunc subtractMoney (subMoney int ) { rwLock.Lock() time.Sleep(3 *time.Second) moneyA-=subMoney rwLock.Unlock() } func getMoney () int { rwLock.RLock() result := moneyA rwLock.RUnlock() return result; } func main () { wait.Add(2 ) go func () { defer wait.Done() if (getMoney()>200 ) { subtractMoney(200 ) fmt.Printf("200元扣款成功,剩下:%d元\n" ,getMoney()) }else { fmt.Println("余额不足,无法扣款" ) } }() go func () { defer wait.Done() if (getMoney()>900 ) { subtractMoney(900 ) fmt.Printf("900元扣款成功,剩下:%d元\n" ,getMoney()) }else { fmt.Println("余额不足,无法扣款" ) } }() wait.Wait() fmt.Println(getMoney()) }
# 8.Go 中的并发安全 Map (类似于 CurrentHashMap)
Go 中自己通过 make 创建的 map 不是线程安全的,具体体现在多线程添加值和修改值下会报如下错误:
1 fatal error : concurrent map writes
这个错类似于 java 中多线程读写线程不安全的容器时报的错。
Go 为了解决这个问题,专门给我们提供了一个并发安全的 map,这个并发安全的 map 不用通过 make 创建,拿来即可用,并且他提供了一些不同于普通 map 的操作方法。
参考如下代码示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 package mainimport ( "fmt" "sync" ) var myConcurrentMap = sync.Map{}var myRangeMap = sync.Map{}func main () { myConcurrentMap.Store(1 ,"li_ming" ) name,ok := myConcurrentMap.Load(1 ) if (!ok) { fmt.Println("不存在" ) return } fmt.Println(name) name2, ok2 := myConcurrentMap.LoadOrStore(1 ,"xiao_hong" ) fmt.Println(name2,ok2) name3, ok3 := myConcurrentMap.LoadOrStore(2 ,"xiao_hong" ) fmt.Println(name3,ok3) myConcurrentMap.Delete(1 ) rangeFunc() } func rangeFunc () { myRangeMap.Store(1 ,"xiao_ming" ) myRangeMap.Store(2 ,"xiao_li" ) myRangeMap.Store(3 ,"xiao_ke" ) myRangeMap.Store(4 ,"xiao_lei" ) myRangeMap.Range(func (k, v interface {}) bool { fmt.Println("data_key_value = :" ,k,v) return true }) }
# 9.Go 中的 AtomicXXX 原子操作类 (类似于 Java 中的 AtocmicInteger 之类的)
Go 中的 atomic 包里面的功能和 Java 中的 Atomic 一样,原子操作类,原理也是 cas, 甚至提供了 cas 的 api 函数,这里不做过多讲解,
简单举一个代码示例,因为方法太多,详细的请参考 api 文档中的 atomic 包:
1 2 3 4 5 6 7 8 9 package mainimport "sync/atomic" func main () { var num int64 = 20 atomic.AddInt64(&num,1 ) }
# 10.Go 中的 WaitGroup (类似于 Java 中的 CountDownLatch)
现在让我们看一个需求,比如我们开启三个并发任务,然后三个并发任务执行处理完毕后我们才让主线程继续往下面走。
这时候肯定不能用睡眠了,因为不知道睡眠多长时间。
这是 Go 中的 sync 包提供了一个 WaitGroup 的工具,他基本上和 Java 中的 CountDownLatch 的功能一致。
接下来让我们看代码示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 package mainimport ( "fmt" "sync" "time" ) var wait sync.WaitGroupfunc main () { wait.Add(3 ) go myFun1() go myFun2() go myFun3() wait.Wait() } func myFun1 () { defer wait.Done() time.Sleep(time.Second*5 ) fmt.Println("fun1执行完毕" ) } func myFun2 () { defer wait.Done() time.Sleep(time.Second*5 ) fmt.Println("fun2执行完毕" ) } func myFun3 () { defer wait.Done() time.Sleep(time.Second*5 ) fmt.Println("fun3执行完毕" ) }
# 关于我
Brath 是一个热爱技术的 Java 程序猿,公众号「InterviewCoder」定期分享有趣有料的精品原创文章!
非常感谢各位人才能看到这里,原创不易,文章如果有帮助可以关注、点赞、分享或评论,这都是对我的莫大支持!