赞
踩
go下载地址:All releases - The Go Programming Language,windows选择下载go1.20.2.windows-amd64.msi文件。
双击go1.20.2.windows-amd64.msi,点击"Next",然后勾选同意,再点击"Next"。
选择Go的安装位置,这里我选择了"D:\0-software\0-develop\10-GO\1-go1.20.2"。再点击"Next"。
点击"Install"进行安装。然后点击"Finish"就安装完成了。
2.2、go的环境变量配置
此电脑->右键"属性"->"高级系统设置"->"环境变量"
在系统变量里面添加:
变量名:GOPATH
变量值:D:\0-software\0-develop\10-GO\1-go1.20.2
在Path里面添加:%GOPATH%\bin
然后打开cmd,输入"go version",打印下图所示,说明环境变量配置成功。
goLand的安装、配置
goLand下载地址:GoLand by JetBrains: More than just a Go IDE,安装过程省略。
新建go工程时,选择Go,不要选择Go(GOPATH),早期的GoLand比如2020版本的才选择Go(GOPATH)。
点击"File"->"new"->"Go File",编辑文件名为"Hello"
可以看到默认会有一个package包名。每一个go的源码文件都要指定包名。并且只有package是main的代码才会运行。
func表示函数的意思。
注意:Go语言的Hello World你想要运行的话要注意:
包名必须叫main
函数名也必须叫main
同一个目录下只能有一个func main()
点击绿色的三角按钮执行。
还可以通过另外一种方式运行:
进入到Hello.go所在目录文件夹
然后执行go build .\Hello.go,这样可以将Hello.go编译成可执行文件。
然后执行:.\Hello.exe,就可以执行了。
如果想省略编译过程直接执行,可以输入:
go run .\Hello.go
这样就不会编译生成Hello.exe文件,而且还会执行。
- package main
-
- import "fmt"
-
- // 全局变量和局部变量(定义在main外面的变量都是全局变量)
- //var sex = "male"
- //var ok bool = true
-
- // 简洁方式定义全局变量
- var (
- sex = "male"
- ok = true
- age = 18
- )
-
- //局部变量定义了就必须使用,不使用会报错;全局变量定义了之后可以不使用,不会报错。
-
- func main() {
- //go是静态语言,静态语言和动态语言相比,定义变量差异很大
- //1、变量必须想定义后使用 2、变量必须有类型 3、变量类型定下来之后不能改变
- //定义变量的方式
- //方式一:var variableName variableType
- var name string
- name = "旺财"
- //演示:变量类型定下来之后不能改变
- //name = 1
-
- fmt.Print(name)
-
- /*
- 方式二:var variableName = variableValue
- 可以省略variableType,因为go可以通过variableValue进行类型推断。
- */
-
- var name2 = "小强"
- fmt.Print(name2)
-
- /*
- 方式三:variableName := variableValue
- 这种平时用的比较多
- */
-
- //var age = 1
- age := 1
- fmt.Print(age)
-
- //go语言中局部变量定义了但是不使用,是不行的。
-
- //2、多变量定义
- //2、多变量定义
- //2.1、同时在一行里面定义3个string类型的变量
- var user1, user2, user3 string
- fmt.Print(user1, user2, user3)
- //2.2、同时在一行里面定义3个变量,并且初始化。
- var user4, user5, user6 = "小强", "旺财", 1
- fmt.Print(user4, user5, user6)
-
- /*
- 注意:
- 变量必须先定义再使用
- go语言是静态语言,要求变量的类型和赋值类型一致
- 局部变量名不能冲突;全局变量和局部变量的变量名可以重复,这种情况下,局部变量的优先级高。
- 简洁变量定义不能用于全局变量(方式三:variableName := variableValue)
- 变量是有零值的
- 局部变量定义了就必须使用,不使用会报错;全局变量定义了之后可以不使用,不会报错。
- */
-
- }
- package main
-
- import "fmt"
-
- func main() {
- //常量,定义的时候就指定的值,不能修改。
- //常量定义的时候全部大写。多个单词中间加下划线。
- const PI1 float32 = 3.1415926535897932384626 //显式定义
- const PI2 = 3.1415926535897932384626 //隐式定义
-
- // 同时定义多个常量
- const (
- UNKNOWN = 1
- FEMALE = 2
- MALE = 3
- )
-
- // 同时定义多个常量,并且支持类型定义。
- const (
- X int = 1
- Y
- S = "abc"
- Z
- M
- )
- fmt.Println(X, Y, S, Z, M)
-
- /*
- 1、常量类型只可以定义bool、数值(整数、浮点数和复数) 和 字符串;
- 2、不曾使用的常量,没有强制使用的要求;
- 3、显式指定类型的时候,必须确保常量 左右值类型一致。
- 4、常量在定义的时候如果没有设置类型和值的话,它就用前面的类型和值。
- */
-
- }
从打印结果可以看出,Y没有定义类型和值,但是Y可以使用前面X的类型和值。M也是同样的道理。
- package main
-
- import "fmt"
-
- func main() {
- //iota,特殊常量,可以认为是一个可以被编译器修改的常量。
- const (
- ERR1 = iota + 1
- ERR2
- ERR3
- ERR4 = "abc"
- ERR5
- ERR6 = 100
- ERR7 = iota
- )
-
- const (
- ERRNEW1 = iota
- )
-
- fmt.Println(ERR1, ERR2, ERR3, ERR4, ERR5, ERR6, ERR7)
- fmt.Println(ERRNEW1)
- /*
- 如果中断了iota那么必须显式的恢复,后续会自动递增。
- 自增类型默认是int类型
- iota能简化const类型的定义
- 每次出现const的时候,iota初始化为0
- */
-
- }
代码运行结果:
- package main
-
- func a() (int, bool) {
- return 0, false
- }
-
- func main() {
- //匿名变量:就是一个下划线
- var _ int
- //r1, ok := a()
- _, ok := a()
- /*
- 如果我接下来只想使用ok,不想使用r1,根据Go的局部变量定义规定的,定义了就必须使用,那我就必须在下面被迫的加一个多余的打印r1的代码。
- 有什么好办法吗?
- 这时,匿名变量就登场了。可以使用"_"代替r1,这样在下面可以不用也不会报错了。
- */
- if ok {
- //打印
-
- }
- }
- package main
-
- import "fmt"
-
- // 全局变量:任何一个函数都可以使用这个变量
- var globalVariable = "旺财"
-
- func main() {
- //变量的作用域
-
- //局部变量
- localVariable := "小强"
- fmt.Print(localVariable)
-
- {
- //局部变量
- localVariable2 := "张三"
- fmt.Print(localVariable2)
- }
- //在局部变量作用域外部打印该变量会报错
- //fmt.Print(localVariable2)
-
- }
- package main
-
- import "fmt"
-
- func main() {
- //基础数据类型
- //var a int8
- //var b int16
- //var c int32
- //var d int64
- //var ua uint8
- //var ub uint16
- //var uc uint32
- //var ud uint64
- 动态类型,用的时候就会知道,用起来挺麻烦的
- //var e int
- //
- 类型转换需要强转才行
- //a = int8(b)
- //
- //var f1 float32
- //var f2 float64
- //
- //f1 = 3
- //f2 = 3.14
-
- //主要是用来存放英文字符的
- var c byte
- c = 'a' + 1
- fmt.Println(c) //值是98
- fmt.Printf("c=%c", c) //打印的是 c=b
-
- c1 := 97
- fmt.Println()
- fmt.Printf("c1=%c", c1) //打印的是 c=a
-
- //也是字符,主要是用来存放中文字符的
- var c2 rune
- c2 = '旺'
- fmt.Println()
- fmt.Printf("c=%c", c2) //打印的是 c=旺
-
- //字符串
- var name string
- name = "My name is Peter Parker,I am a Super Hero. I don't like the Criminals."
- fmt.Println()
- fmt.Println(name)
- }
运行结果:
bool类型
布尔型的值只可以是常量 true 或者 false。一个简单的例子:var b bool = true
数值型
整数型
可以简单讲解一下二进制和位数的关系,以及int和uint的关系
int8 有符号 8 位整型 (-128 到 127) 长度:8bit
int16 有符号 16 位整型 (-32768 到 32767)
int32 有符号 32 位整型 (-2147483648 到 2147483647)
int64 有符号 64 位整型 (-9223372036854775808 到 9223372036854775807)
uint8 无符号 8 位整型 (0 到 255) 8位都用于表示数值:
uint16 无符号 16 位整型 (0 到 65535)
uint32 无符号 32 位整型 (0 到 4294967295)
uint64 无符号 64 位整型 (0 到 18446744073709551615)
浮点型
float32 32位浮点型数
float64 64位浮点型数
其他
byte 等于 uint8
rune 等于 int32
uint 32 或 64 位
字符
Golang中没有专门的字符类型,如果要存储单个字符(字母),一般使用byte来保存。 字符串就是一串固定长度的字符连接起来的字符序列。Go的字符串是由单个字节连接起来的。也就是说对于传统的字符串是由字符组成的,而Go的字符串不同,它是由字节组成的。
- package main
-
- import (
- "fmt"
- )
-
- func main() {
-
- var a byte
- a = 'a'
- //输出ascii对应码值 。。 这里说明一下什么是ascii码
- fmt.Println(a)
- fmt.Printf("a=%c", a)
- }
字符常量只能使用单引号括起来,例如:var a byte = ‘a’ var a int = ‘a’
- package main
-
- import (
- "fmt"
- )
-
- func main() {
-
- var a byte
- a = "a"
- //输出ascii对应码值 。。 这里说明一下什么是ascii码
- fmt.Println(a)
- fmt.Printf("a=%c", a)
- }
字符本质是一个数字, 可以进行加减乘除
- package main
-
- import (
- "fmt"
- "reflect"
- )
-
- func main() {
-
- a := 'a'
-
- //这里注意一下 1. a+1可以和数字计算 2.a+1的类型是32 3. int类型可以直接变成字符
-
- fmt.Println(reflect.TypeOf(a+1))
- fmt.Printf("a+1=%c", a+1)
- }
字符串
字符串就是一串固定长度的字符连接起来的字符序列。Go 的字符串是由单个字节连接起来的。Go 语言的字符串的字节使用 UTF-8 编码标识 Unicode 文本。
- package main
-
- import (
- "fmt"
- "strconv"
- )
-
- func main() {
- //基本类型转换
- //int和int相互转换
- var a int8 = 12
- //var b = uint8(a)
- //fmt.Print(b)
-
- //float转int
- //var f float32 = 3.14
- //var c = int32(f)
- //fmt.Println(c) //打印的是3
-
- //int转float
- //var f64 = float64(a)
- //fmt.Println(f64) //打印的是12
-
- //定义别名
- type IT int //类型要求很严格
- var c = IT(a)
- fmt.Println(c)
-
- //字符串转数字
- var istr = "12"
- myint, err := strconv.Atoi(istr)
- if err != nil {
- fmt.Println("convert erro")
- }
- fmt.Println(myint)
-
- //数字转字符串
- var myi = 32
- fmt.Println(strconv.Itoa(myi))
-
- }
type 关键字:
在 Go 语言中,type 关键字常被我们用来创建新的结构体。同样的,type 也可以用来创建其它的用户自定义类型。在创建新类型时,有类型别名和类型定义两种方式,你知道它们的区别吗?
1、type 关键字
对于使用过 Go 语言的人,相信对type
关键字都不陌生,它可以帮助我们定义结构体或接口:
- type Student struct{
- }
-
- type Man interface {
- Name()
- }
在使用 type 定义结构体时,我们可以把它看作是基于struct{}
类型定义了一个新的类型Student
。
其实,除了用于创建新类型,type
还有创建类型别名的作用。
2、类型别名 vs 类型定义
基于一个类型创建一个别名,称之为类型别名 (alias)。
基于一个类型创建一个新类型,称之为类型定义 (definition)。
- type Int1 = int // 类型别名,Int1 是 int 类型的别名
- type Int2 int // 类型定义,Int2 是新类型
它们之间的区别是什么呢?
以上述定义的Int1
和Int2
为例,我们可以用int
类型的变量初始化Int1
类型,因为Int1
只是int
类型的一个别名。
但如果我们用一个int
类型的变量初始化Int2
类型时,对Int2
类型的初始化会报错,因为Int2
是一个新的类型。
- type Int1 = int // 类型别名,Int1 是 int 类型的别名
- type Int2 int // 类型定义,Int2 是新类型
-
- var i int = 0
-
- var i1 Int1 = i
- var i2 Int2 = i // error
不过,Int1
类型和Int2
类型都可以用相应类型的字面量来初始化。
- type Int1 = int // 类型别名,Int1 是 int 类型的别名
- type Int2 int // 类型定义,Int2 是新类型
-
- var i1 Int1 = 0
- var i2 Int2 = 0
4.2.1、简单的转换操作
- valueOfTypeB = typeB(valueOfTypeA)
- 代码块1
- // 浮点数
- a := 5.0
-
- // 转换为int类型
- b := int(a)
-
- //Go允许在底层结构相同的两个类型之间互转。例如:
- // IT类型的底层是int类型
- type IT int
-
- // a的类型为IT,底层是int
- var a IT = 5
-
- // 将a(IT)转换为int,b现在是int类型
- b := int(5)
-
- // 将b(int)转换为IT,c现在是IT类型
- c := IT(b)
-
- var a int32 = 1
- var b int64 = 3
- b = int64(a)
- fmt.Println(a, b)
-
- /*
- 不是所有数据类型都能转换的,例如字母格式的string类型"abcd"转换为int肯定会失败
- 低精度转换为高精度时是安全的,高精度的值转换为低精度时会丢失精度。例如int32转换为int16,float32转换为int
- 这种简单的转换方式不能对int(float)和string进行互转,要跨大类型转换,可以使用strconv包提供的函数
- */
4.2.2、strconv
Itoa和Atoi
int转换为字符串:Itoa()
println("a" + strconv.Itoa(32)) // a32
string转换为int:Atoi()(表示 alphanumeric to integer)是把字符串转换成整型数的一个函数
- i,_ := strconv.Atoi("3")
- println(3 + i) // 6
-
- // Atoi()转换失败
- i,err := strconv.Atoi("a")
- if err != nil {
- println("converted failed")
- }
- //由于string可能无法转换为int,所以这个函数有两个返回值:第一个返回值是转换成int的值,第二个返回值判断是否转换成功。
Parse类函数
Parse类函数用于转换字符串为给定类型的值:ParseBool()、ParseFloat()、ParseInt()、ParseUint()。
- b, err := strconv.ParseBool("true")
- f, err := strconv.ParseFloat("3.1415", 64)
- i, err := strconv.ParseInt("-42", 10, 64)
- u, err := strconv.ParseUint("42", 10, 64)
ParseInt()和ParseUint()有3个参数:
- func ParseInt(s string, base int, bitSize int) (i int64, err error)
- func ParseUint(s string, base int, bitSize int) (uint64, error)
说明:
bitSize
参数表示转换为什么位的int/uint,有效值为0、8、16、32、64。当bitSize=0的时候,表示转换为int或uint类型。例如bitSize=8表示转换后的值的类型为int8或uint8。
base
参数表示以什么进制的方式去解析给定的字符串,有效值为0、2-36。当base=0的时候,表示根据string的前缀来判断以什么进制去解析:0x
开头的以16进制的方式去解析,0
开头的以8进制方式去解析,其它的以10进制方式解析。
Format类函数
将给定类型格式化为string类型:FormatBool()、FormatFloat()、FormatInt()、FormatUint()。
- s := strconv.FormatBool(true)
- s := strconv.FormatFloat(3.1415, 'E', -1, 64)
- s := strconv.FormatInt(-42, 16) //表示将-42转换为16进制数,转换的结果为-2a。
- s := strconv.FormatUint(42, 16)
第二个参数base指定将第一个参数转换为多少进制,有效值为2<=base<=36
。当指定的进制位大于10的时候,超出10的数值以a-z字母表示。例如16进制时,10-15的数字分别使用a-f表示,17进制时,10-16的数值分别使用a-g表示。
FormatFloat()参数众多:
func FormatFloat(f float64, fmt byte, prec, bitSize int) string
bitSize表示f的来源类型(32:float32、64:float64),会据此进行舍入。
fmt表示格式:‘f’(-ddd.dddd)、‘b’(-ddddp±ddd,指数为二进制)、‘e’(-d.dddde±dd,十进制指数)、‘E’(-d.ddddE±dd,十进制指数)、‘g’(指数很大时用’e’格式,否则’f’格式)、‘G’(指数很大时用’E’格式,否则’f’格式)。
prec控制精度(排除指数部分):对’f’、‘e’、‘E’,它表示小数点后的数字个数;对’g’、‘G’,它控制总的数字个数。如果prec 为-1,则代表使用最少数量的、但又必需的数字来表示f。
4.4.1、算数运算符
/ %(求余) ++ –
4.4.2、关系运算符
== != > < >= <=
4.4.3、逻辑运算符
&& | 所谓逻辑与运算符。如果两个操作数都非零,则条件变为真 |
---|---|
|| | 所谓的逻辑或操作。如果任何两个操作数是非零,则条件变为真 |
! | 所谓逻辑非运算符。使用反转操作数的逻辑状态。如果条件为真,那么逻辑非操后结果为假 |
这个和python不一样,python中使用 and or来连接
- package main
-
- import "fmt"
-
- func main() {
- var a bool = true
- var b bool = false
- if ( a && b ) {
- fmt.Printf("第一行 - 条件为 true\n" )
- }
- if ( a || b ) {
- fmt.Printf("第二行 - 条件为 true\n" )
- }
- /* 修改 a 和 b 的值 */
- a = false
- b = true
- if ( a && b ) {
- fmt.Printf("第三行 - 条件为 true\n" )
- } else {
- fmt.Printf("第三行 - 条件为 false\n" )
- }
- if ( !(a && b) ) {
- fmt.Printf("第四行 - 条件为 true\n" )
- }
- }
4.4.4、位运算符
位运算符对整数在内存中的二进制位进行操作。 下表列出了位运算符 &, |, 和 ^ 的计算:
P | Q | P & Q | P | Q | P ^ Q |
---|---|---|---|---|
0 | 0 | 0 | 0 | 0 |
0 | 1 | 0 | 1 | 1 |
1 | 1 | 1 | 1 | 0 |
1 | 0 | 0 | 1 | 1 |
假定 A = 60; B = 13; 其二进制数转换为: A = 0011 1100 B = 0000 1101
A&B = 0000 1100
A|B = 0011 1101
A^B = 0011 0001 Go 语言支持的位运算符如下表所示。假定 A 为60,B 为13:
运算符 | 描述 | 实例 |
---|---|---|
& | 按位与运算符"&"是双目运算符。 其功能是参与运算的两数各对应的二进位相与。 | (A & B) 结果为 12, 二进制为 0000 1100 |
| | 按位或运算符"|"是双目运算符。 其功能是参与运算的两数各对应的二进位相或 | (A | B) 结果为 61, 二进制为 0011 1101 |
^ | 按位异或运算符"^"是双目运算符。 其功能是参与运算的两数各对应的二进位相异或,当两对应的二进位相异时,结果为1。 | (A ^ B) 结果为 49, 二进制为 0011 0001 |
<< | 左移运算符"<<“是双目运算符。左移n位就是乘以2的n次方。 其功能把”<<“左边的运算数的各二进位全部左移若干位,由”<<"右边的数指定移动的位数,高位丢弃,低位补0。 | A << 2 结果为 240 ,二进制为 1111 0000 |
>> | 右移运算符">>"是双目运算符。右移n位就是除以2的n次方。 | |
其功能是把">>“左边的运算数的各二进位全部右移若干位,”>>"右边的数指定移动的位数。 | A >> 2 结果为 15 ,二进制为 0000 1111 |
- package main
-
- import "fmt"
-
- func main() {
-
- var a uint = 60 /* 60 = 0011 1100 */
- var b uint = 13 /* 13 = 0000 1101 */
- var c uint = 0
-
- c = a & b /* 12 = 0000 1100 */
- fmt.Printf("第一行 - c 的值为 %d\n", c )
-
- c = a | b /* 61 = 0011 1101 */
- fmt.Printf("第二行 - c 的值为 %d\n", c )
-
- c = a ^ b /* 49 = 0011 0001 */
- fmt.Printf("第三行 - c 的值为 %d\n", c )
-
- c = a << 2 /* 240 = 1111 0000 */
- fmt.Printf("第四行 - c 的值为 %d\n", c )
-
- c = a >> 2 /* 15 = 0000 1111 */
- fmt.Printf("第五行 - c 的值为 %d\n", c )
- }
4.4.5、赋值运算符
下表列出了所有Go语言的赋值运算符。
运算符 | 描述 | 实例 |
---|---|---|
= | 简单的赋值运算符,将一个表达式的值赋给一个左值 | C = A + B 将 A + B 表达式结果赋值给 C |
+= | 相加后再赋值 | C += A 等于 C = C + A |
-= | 相减后再赋值 | C -= A 等于 C = C - A |
*= | 相乘后再赋值 | C *= A 等于 C = C * A |
/= | 相除后再赋值 | C /= A 等于 C = C / A |
%= | 求余后再赋值 | C %= A 等于 C = C % A |
<<= | 左移后赋值 | C <<= 2 等于 C = C << 2 |
>>= | 右移后赋值 | C >>= 2 等于 C = C >> 2 |
&= | 按位与后赋值 | C &= 2 等于 C = C & 2 |
^= | 按位异或后赋值 | C ^= 2 等于 C = C ^ 2 |
|= | 按位或后赋值 | C |= 2 等于 C = C | 2 |
- package main
-
- import "fmt"
-
- func main() {
- var a int = 21
- var c int
-
- c = a
- fmt.Printf("第 1 行 - = 运算符实例,c 值为 = %d\n", c )
-
- c += a
- fmt.Printf("第 2 行 - += 运算符实例,c 值为 = %d\n", c )
-
- c -= a
- fmt.Printf("第 3 行 - -= 运算符实例,c 值为 = %d\n", c )
-
- c *= a
- fmt.Printf("第 4 行 - *= 运算符实例,c 值为 = %d\n", c )
-
- c /= a
- fmt.Printf("第 5 行 - /= 运算符实例,c 值为 = %d\n", c )
-
- c = 200;
-
- c <<= 2
- fmt.Printf("第 6行 - <<= 运算符实例,c 值为 = %d\n", c )
-
- c >>= 2
- fmt.Printf("第 7 行 - >>= 运算符实例,c 值为 = %d\n", c )
-
- c &= 2
- fmt.Printf("第 8 行 - &= 运算符实例,c 值为 = %d\n", c )
-
- c ^= 2
- fmt.Printf("第 9 行 - ^= 运算符实例,c 值为 = %d\n", c )
-
- c |= 2
- fmt.Printf("第 10 行 - |= 运算符实例,c 值为 = %d\n", c )
-
- }
4.4.6、其他运算符
此处讲解一下什么是地址
运算符 | 描述 | 实例 |
---|---|---|
& | 返回变量存储地址 | &a; 将给出变量的实际地址。 |
* | 指针变量。 | *a; 是一个指针变量 |
- package main
-
- import "fmt"
-
- func main() {
- var a int = 4
- var b int32
- var c float32
- var ptr *int
-
- /* 运算符实例 */
- fmt.Printf("第 1 行 - a 变量类型为 = %T\n", a );
- fmt.Printf("第 2 行 - b 变量类型为 = %T\n", b );
- fmt.Printf("第 3 行 - c 变量类型为 = %T\n", c );
-
- /* & 和 * 运算符实例 */
- ptr = &a /* 'ptr' 包含了 'a' 变量的地址 */
- fmt.Printf("a 的值为 %d\n", a);
- fmt.Printf("*ptr 为 %d\n", *ptr);
- }
4.4.7、运算符优先级
有些运算符拥有较高的优先级,二元运算符的运算方向均是从左至右。下表列出了所有运算符以及它们的优先级,由上至下代表优先级由高到低
优先级 | 分类 | 运算符 | 结合性 |
---|---|---|---|
1 | 逗号运算符 | , | 从左到右 |
2 | 赋值运算符 | =、+=、-=、*=、/=、 %=、 >=、 <<=、&=、^=、|= | 从右到左 |
3 | 逻辑或 | || | 从左到右 |
4 | 逻辑与 | && | 从左到右 |
5 | 按位或 | | | 从左到右 |
6 | 按位异或 | ^ | 从左到右 |
7 | 按位与 | & | 从左到右 |
8 | 相等/不等 | ==、!= | 从左到右 |
9 | 关系运算符 | <、<=、>、>= | 从左到右 |
10 | 位移运算符 | <<、>> | 从左到右 |
11 | 加法/减法 | +、- | 从左到右 |
12 | 乘法/除法/取余 | *(乘号)、/、% | 从左到右 |
13 | 单目运算符 | !、*(指针)、& 、++、–、+(正号)、-(负号) | 从右到左 |
14 | 后缀运算符 | ( )、[ ]、-> | 从左到右 |
当然,你可以通过使用括号来临时提升某个表达式的整体运算优先级。
- package main
-
- import "fmt"
-
- func main() {
- var a int = 20
- var b int = 10
- var c int = 15
- var d int = 5
- var e int;
-
- e = (a + b) * c / d; // ( 30 * 15 ) / 5
- fmt.Printf("(a + b) * c / d 的值为 : %d\n", e );
-
- e = ((a + b) * c) / d; // (30 * 15 ) / 5
- fmt.Printf("((a + b) * c) / d 的值为 : %d\n" , e );
-
- e = (a + b) * (c / d); // (30) * (15/5)
- fmt.Printf("(a + b) * (c / d) 的值为 : %d\n", e );
-
- e = a + (b * c) / d; // 20 + (150/5)
- fmt.Printf("a + (b * c) / d 的值为 : %d\n" , e );
- }
- package main
-
- import "fmt"
-
- func main() {
- //5.1节 rune和字符串长度
- //长度计算
- name := "lvxiaosha学go"
- fmt.Println(len(name))
-
- //将字符串类型的name转换为切片,然后就可以for循环了。
- bytes := []byte(name)
- fmt.Println(len(bytes))
- //因为 len() 传入 返回字符串所占的字节 ,而中文转 utf8 编码后占 3 字节,英文占 1 字节。所以上面两个输出结果都是:9 + 3 + 2 = 14
-
- //如果我们想得到字符长度,而不是字节长度,就需要转为 切片 后再计算长度。
- runes := []rune(name)
- fmt.Println(len(runes)) // 10
-
- }
运行结果:
5.2、转义符
- package main
-
- import "fmt"
-
- func main() {
- //5.2节转义符
- //想在字符串中打印双引号
- //方法1:使用反斜线加双引号(\")
- name := "\"吕小傻\""
- fmt.Println(name)
-
- //方法2:使用Tab键上面的那个键(`),注意不是单引号(')。
- //这个和python中的三引号("""xxx""")里面可以随便写内容类似。
- //我们大多数使用的都是这个方法。
- name2 := `"吕小傻"`
- fmt.Println(name2)
- }
转义字符 | 意义 | ASCII码值(十进制) |
---|---|---|
\n | 换行(LF) ,将当前位置移到下一行开头 | 010 |
\r | 回车(CR) ,将当前位置移到本行开头 | 013 |
\t | 水平制表(HT) (跳到下一个TAB位置) | 009 |
\\ | 代表一个反斜线字符’’\’ | 092 |
\’ | 代表一个单引号(撇号)字符 | 039 |
\" | 代表一个双引号字符 | 034 |
\? | 代表一个问号 | 063 |
5.3、格式化输出
- package main
-
- import (
- "fmt"
- "strconv"
- )
-
- func main() {
- //5.3节格式化输出
- userName := "吕小傻"
- address := "山东省青岛市"
- age := 18
- mobile := "13624558575"
-
- fmt.Println("用户名:"+userName, "地址:"+address, "年龄:"+strconv.Itoa(age), "电话:"+mobile) //这种拼接方式极难维护
- fmt.Printf("用户名:%s,地址:%s,年龄:%d,电话:%s\r\n", userName, address, age, mobile) //这种方式更常用,但是性能没有上一种好。
-
- userMessage := fmt.Sprintf("用户名:%s,地址:%s,年龄:%d,电话:%s", userName, address, age, mobile)
- fmt.Println(userMessage)
-
- var ages []int = []int{1, 2, 3}
- agesMessage := fmt.Sprintf("年龄:%#v", ages)
- fmt.Println(agesMessage)
- }
格式化后的效果 | 动词 | 描述 |
---|---|---|
[0 1] | %v | 缺省格式 |
[]int64{0, 1} | %#v | go语法打印 |
[]int64 | %T | 类型打印 |
格式化后的效果 | 动词 | 描述 |
---|---|---|
15 | %d | 十进制 |
+15 | %+d | 必须显示正负符号 |
␣␣15 | %4d | Pad空格(宽度为4,右对齐) |
15␣␣ | %-4d | Pad空格 (宽度为4,左对齐) |
1111 | %b | 二进制 |
17 | %o | 八进制 |
f | %x | 16进制,小写 |
Value: 65
(Unicode letter A)
格式化后的效果 | 动词 | 描述 |
---|---|---|
A | %c | 字符 |
‘A’ | %q | 有引号的字符 |
U+0041 | %U | Unicode |
U+0041 ‘A’ | %#U | Unicode 有引号 |
Value: 123.456
格式化后的效果 | 动词 | 描述 |
---|---|---|
1.234560e+02 | %e | 科学计数 |
123.456000 | %f | 十进制小数 |
Value: "cafe"
格式化后的效果 | 动词 | 描述 |
---|---|---|
cafe | %s | 字符串原样输出 |
␣␣cafe | %6s | 宽度为6,右对齐 |
5.4、高性能字符串拼接-string.builder
- package main
-
- import (
- "fmt"
- "strconv"
- "strings"
- )
-
- func main() {
- //5.4节 高性能字符串拼接-string.builder
- userName := "吕小傻"
- address := "山东省青岛市"
- age := 18
- mobile := "13624558575"
-
- var builder strings.Builder
- builder.WriteString("用户名:")
- builder.WriteString(userName)
- builder.WriteString(",地址:")
- builder.WriteString(address)
- builder.WriteString(",年龄:")
- builder.WriteString(strconv.Itoa(age))
- builder.WriteString(",电话:")
- builder.WriteString(mobile)
-
- response := builder.String()
- fmt.Println(response)
- }
运行结果:
5.5、字符串的比较
- package main
-
- import "fmt"
-
- func main() {
- //5.5节 字符串的比较
- a := "hello"
- b := "hello"
- fmt.Println(a == b)
- fmt.Println(a != b)
-
- //字符串的大小比较
- fmt.Println(a > b)
-
- }
5.6、字符串操作常用方法
- package main
-
- import (
- "fmt"
- "strings"
- )
-
- func main() {
- //5.6节 字符串操作常用方法
- //strings常用方法
- //是否包含
- name := "My name is Peter Parker,I am a Super Hero.I don't like the Criminals."
- fmt.Println(strings.Contains(name, "Peter Parker"))
-
- //字符串长度
- runes := []rune(name)
- fmt.Println(len(runes))
-
- //查询字符串出现的次数
- fmt.Println(strings.Count(name, "a"))
-
- //分割字符串
- fmt.Println(strings.Split(name, ","))
-
- //字符串是否包含前缀,是否包含后缀。
- fmt.Println(strings.HasPrefix(name, "My"))
- fmt.Println(strings.HasSuffix(name, "Criminals."))
-
- //查询子串出现的位置
- fmt.Println(strings.Index(name, "Peter Parker"))
- fmt.Println(strings.IndexRune(name, []rune(name)[17]))
-
- //字符串替换
- fmt.Println(strings.Replace(name, "Peter Parker", "Super Man", 1)) // 1表示只替换第一个
- fmt.Println(strings.Replace(name, "Peter Parker", "Iron Man", -1)) // -1表示全部替换
-
- //大小写转换
- fmt.Println(strings.ToLower("Peter Parker"))
- fmt.Println(strings.ToUpper("Peter Parker"))
-
- //去掉字符串左右两边的特殊字符,下面的方法表示,只要字符串两边出现了"#"或"$",就去掉。
- //除了strings.Trim,还有strings.TrimLeft和strings.TrimRight方法
- fmt.Println(strings.Trim("$#Peter #Parker#", "#$"))
-
- }
运行结果:
6.1、if条件判断
- package main
-
- import "fmt"
-
- /*
- *
-
- if 布尔表达式 {
- 逻辑
- }
- */
- func main() {
- //6.1节 if条件判断
- age := 18
- country := "中国"
- //简单的if条件判断,布尔表达式的括号省略
- if age < 18 {
- fmt.Println("未成年")
- }
-
- //复杂的if条件判断,布尔表达式的括号不能省略
- if (age < 18) && (country == "中国") {
- fmt.Println("未成年")
- } else if age == 18 {
- fmt.Println("刚刚成年")
- } else {
- fmt.Println("成年了")
- }
-
- }
6.2、for循环基础用法
- package main
-
- import (
- "fmt"
- )
-
- /*
- * go语言中只有for循环,没有while循环。
- * for init; condition; post {
- * //do something
- * }
- */
- func main() {
- //6.1节 for循环基础用法
- //标准写法
- for i := 0; i < 3; i++ {
- fmt.Println(i)
- }
-
- //初始变量在循环外面定义
- var j int
- for ; j < 3; j++ {
- fmt.Println(j)
- }
-
- // init; condition; post 都没有,就等效与while(true)
- //var k int
- //for {
- // //睡眠2秒
- // time.Sleep(2 * time.Second)
- // fmt.Println(k)
- // k++
- //}
-
- // 可以将post部分放到循环体里面
- var m int
- for m < 3 {
- fmt.Println(m)
- m++
- }
-
- }
6.3、for循环打印九九乘法表
- package main
-
- import "fmt"
-
- /*
- * 6.3节 for循环打印九九乘法表
- * for循环打印九九乘法表
- */
- func main() {
- //遍历,处理第几行
- for y := 1; y <= 9; y++ {
- //遍历,处理第几列。
- for x := 1; x <= y; x++ {
- fmt.Printf("%d * %d = %d\t", x, y, x*y)
- }
- fmt.Println()
- }
-
- }
运行结果:
6.4、for range的循环用法
- package main
-
- import (
- "fmt"
- )
-
- /*
- * 6.4节 for range的循环用法,主要用于 字符串、数组、切片、map、channel
- */
- func main() {
- name := "My name is Peter Parker."
- //for index, value := range name {
- // fmt.Printf("%d, %c\r\n", index, value)
- //}
-
- for _, value := range name {
- fmt.Printf("%c\r\n", value)
- }
-
- /*
- * 字符串 字符串的索引(key) 字符串对应的索引的字符值的拷贝(index),因为是字符串值的拷贝,所以在字符串循环时是改不了原字符串的 如果不写key,那么index返回的是索引
- * for index := range name {
- * fmt.Printf("%c\r\n", index)
- * }
- *
- * 数组 数组的索引 索引的对应值的拷贝 如果不写key,那么value返回的是索引
- * 切片 切片的索引 索引的对应值的拷贝 如果不写key,那么value返回的是索引
- * map 字符串的索引(key) value 返回的是 key 对应值的拷贝 如果不写key,那么value返回的是map的值
- * channel value 返回的是 channel 接受的数据
- */
-
- //下面这种方式循环字符串时会出现中文乱码
- name2 := "哈哈哈"
- for index := range name2 {
- fmt.Printf("%c\r\n", name2[index])
- }
-
- //下面这种方式循环字符串时,就不会出现中文乱码
- name3 := "哈哈哈"
- for _, value := range name3 {
- fmt.Printf("%c\r\n", value)
- }
-
- }
6.5、for循环的continue和break语句
- package main
-
- import (
- "fmt"
- "time"
- )
-
- func main() {
- index := 0
- for {
- time.Sleep(1 * time.Second)
- index++
-
- //如果index等于5,就直接进入下一次循环,不执行下面的代码。
- if index == 5 {
- continue
- }
-
- fmt.Println(index)
- if index > 10 {
- //退出循环
- break
- }
- }
- }
6.6、goto语句的基本用法
- package main
-
- import "fmt"
-
- /*
- * 6.6节 goto语句的基本用法
- * goto语句可以让我们的代码跳到指定的代码块中运行,灵活性很强。但是开发中不建议使用。
- *
- * goto语句可以实现程序的跳转,goto语句使用场景最多的是程序的错误处理,也就是说当程序出现错误的时候统一跳转到相应的标签处,统一处理。
- */
- func main() {
- for i := 0; i < 5; i++ {
- for j := 0; j < 5; j++ {
- if j == 2 {
- goto over
- }
- fmt.Println(i, j)
- }
-
- }
-
- over:
- fmt.Println("over")
-
- }
运行结果:
6.7、switch语法
- package main
-
- import "fmt"
-
- /*
- * 6.7节 switch语句 比if语句执行性能更高一些,而且代码更整洁。
- * switch var1 {
- * case val1:
- * ...
- * case val2:
- * ...
- * case val3:
- * ...
- * default:
- * ...
- *
- * var1变量可以是任意的变量类型
- */
- func main() {
- //中文的星期几,输出对应的英文
- day := "星期三"
- switch day {
- case "星期一":
- fmt.Println("Monday")
- case "星期二":
- fmt.Println("Tuesday")
- case "星期三":
- fmt.Println("Wednesday")
- case "星期四":
- fmt.Println("Thursday")
- case "星期五":
- fmt.Println("Friday")
- case "星期六":
- fmt.Println("Saturday")
- case "星期日":
- fmt.Println("Sunday")
- default:
- fmt.Println("Unknown")
- }
-
- //还有更灵活的switch用法
- score := 95
- switch {
- case score < 60:
- fmt.Println("E")
- case score >= 60 && score < 70:
- fmt.Println("D")
- case score >= 70 && score < 80:
- fmt.Println("C")
- case score >= 80 && score < 90:
- fmt.Println("B")
- case score >= 90 && score < 100:
- fmt.Println("A")
- default:
- fmt.Println("Unknown")
- }
-
- //还有可以这样用switch
- count := 90
- switch count {
- case 60, 70, 80:
- fmt.Println("Ordinary")
- case 90:
- fmt.Println("Excellent")
- default:
- fmt.Println("Unknown")
- }
-
- }
运行结果:
跳转链接:
下一篇:二、容器、go编程思想
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。