GO_04:GO语言基础条件、跳转、Array和Slice
1. 判断语句if
1. 条件表达式没有括号(这点其他语言转过来的需要注意)
2. 支持一个初始化表达式(可以是并行方式,即:a, b, c := 1, 2, 3)
3. 左大括号必须和条件语句或 else 在同一行
4. 支持单行模式
5. 初始化语句中的变量为 block 级别,同时隐藏外部同名变量
有关 if 语句示例代码如下:
package main import "fmt" func main() {
a := true
if a, b, c := , , ; a + b + c > {
fmt.Println("大于6")
} else {
fmt.Println("小于等于6")
fmt.Println(a)
}
fmt.Println(a)
if % == {
fmt.Println("7 is even")
} else {
fmt.Println("7 is odd")
}
}
运行打印结果如下所示:
小于等于6
1
true
7 is odd
if 判断语句比较简单,只要掌握了相关的语法结构以及上一节讲到的相关基础知识,应用 if 语句也就没有什么问题了。
2. 循环语句 for
1. Go 只有 for 一个循环语句关键字,但是它支持3中形式
2. 初始化和步进表达式可以是多个值
3. 条件语句每次循环都会被重新检查,因此不建议在条件语句中使用函数,尽量提前计算好条件并以变量或常量代替
4. 左大括号必须和条件语句在同一行
package main import (
std "fmt"
) func main() {
/**
以下演示for的3种表现形式
*/
i :=
for i <= { // 类似于java中的:while(i <= 3){}
std.Println(i)
i = i +
} for { // 类似于java中的:while(true){}
std.Println("while loop")
break
} for n := ; n <= ; n++ { // 类似于java中的:for (int n = 0; n <= 5; n++) {}
if n % == {
continue
}
std.Println(n)
}
}
以上代码打印结果如下所示:
1
2
3
while loop
1
3
5
3. 选择语句 switch
1. 可以使用任何类型或表达式作为条件语句
2. 不需要写break,一旦条件符合自动终止
3. 如果希望继续执行下一个case,需使用 fallthrough 语句
4. 支持一个初始化表达式(可以是并行方式),右侧需要跟分号,和 if 条件表达式一样
5. 左大括号必须和条件语句在同一行
package main import (
std "fmt"
"time"
) func main() {
i :=
std.Println("Write number ", i, " as ")
switch i { // 基本switch使用
case :
std.Println("one")
case :
std.Println("two")
case :
std.Println("three")
default:
std.Println("ohter")
} switch time.Now().Weekday() { // 条件表达式作为判断条件
case time.Saturday, time.Sunday:
std.Println("It's the weekend")
default:
std.Println("It's a weekday")
} t := time.Now()
switch { // 没有判断语句则和 if/else 效果一样
case t.Hour() < :
std.Println("It's before noon")
default:
std.Println("It's after noon")
} a :=
switch {
case a >= :
std.Println("a >= 0")
fallthrough // 这里不加的话,只会打印 a >= 0,不会向下执行了
case a >= :
std.Println("a >= 1")
}
}
以上代码运行结果如下:
Write number 2 as
two
It's a weekday
It's after noon
a >= 0
a >= 1
4. 跳转语句 goto、break、continue
1. 三个语法都可以配合标签使用
2. 标签名区分大小写,若不使用会造成编译错误
3. break 与 continue 配合标签可用于多层循环的跳出
4. goto 是调整执行位置,与其他 2 个语句配合标签的结果并不相同
package main import std "fmt" func main() {
LABEL_BREAK:
for {
for i := ; i < ; i++ {
if i > {
break LABEL_BREAK // 结束循环到指定目标位置
} else {
std.Println("break number is : ", i)
}
}
} LABEL_CONTINUE:
for i := ; i < ; i++ {
for {
std.Println(i)
continue LABEL_CONTINUE // 结束当前循环层到指定目标位置重新开始
}
} // LABEL_GOTO 如果goto跳转目标放在这里,则形成了死循环
for {
for i := ; i < ; i++ {
if i > {
goto LABEL_GOTO // 跳转到指定目标,从指定目标后开始继续执行
} else {
std.Println("goto number is : ", i)
}
}
}
LABEL_GOTO:
}
以上代码打印结果如下:
break number is : 0
break number is : 1
break number is : 2
0
1
2
3
4
5
6
7
8
9
goto number is : 0
goto number is : 1
goto number is : 2
5. 数组 Array
1. 定义数组的格式:var name [num]type --> var a [10]int (定义1个长度为10类型为int的数组,数组名为 a)
2. 数组长度也是类型的一部分,因此具有不同长度的数组为不同的类型,这点很重要
3. 数组再 Go 中为值类型,故数组之间可以使用 ==、!= 进行比较,但不可以使用 <、> 进行判断
4. 可以使用 new 来创建数组,此方法返回一个指向数组的指针
5. Go 支持多维数组,但是就和其他语言一样,不推荐使用
针对数组的基本应用案例如下:
package main import "fmt" func main() {
var a []int // 以为数组的基本应用
fmt.Println("emp:", a) a[] =
fmt.Println("set:", a)
fmt.Println("get:", a[]) fmt.Println("len:", len(a)) b := []int{, , , , } // 数组初始化并赋值
fmt.Println("arrayB:", b) var twoD [][]int // 二维数组的应用
for i := ; i < ; i++ {
for j := ; j < ; j++ {
twoD[i][j] = i + j
}
}
fmt.Println("twoD:", twoD)
}
以上代码打印结果如下所示:
emp: [0 0 0 0 0]
set: [0 0 0 0 100]
get: 100
len: 5
arrayB: [1 2 3 4 5]
twoD: [[0 1 2] [1 2 3]]
那么,这里使用数组采用冒泡排序方式实现对一组数排序功能,代码如下:
package main import (
"fmt"
) func main() {
// 未排序数组,此处使用的变长数组其长度决定于你传入数据的多少
sort := [...]int{, , , , }
fmt.Println(sort) // 冒泡排序,由大到小
num := len(sort)
for i := ; i < num; i++ {
for j := i + ; j < num; j++ {
// 比较大小
if sort[i] < sort[j] {
temp := sort[i]
sort[i] = sort[j]
sort[j] = temp
}
}
} fmt.Println(sort)
}
以上代码运行结果如下所示:
[9 7 4 3 2]
6. 切片 Slice
1. 其本身并不是数组,它指向底层的数组
2. 作为变长数组的替代方案,可以关联底层数组的局部或全部
3. slice 类型为引用类型
4. 可以直接创建或从底层数组获取生存
5. 使用 len() 获取元素个数,cap() 获取容量
6. 一般使用 make() 创建
7. 如果多个 slice 指向相同底层数组,其中一个的值改变会影响全部
8. make([]T, len, cap) 其中 cap 可以省略,则和 len 的值相同,len 表示元素的个数,cap 表示当前最大可以存放的容量
package main import "fmt" func main() {
// 创建长度为3,容量为3的切片,容量不设置默认为与len相同
s := make([]string, )
fmt.Println(s) // 通过操作切片对数组进行赋值
s[] = "a"
s[] = "b"
s[] = "c"
fmt.Println("set:", s)
fmt.Println("get:", s[]) fmt.Println("len:", len(s), ", cap:", cap(s)) // 创建长度为3容量为10类型为int对切片,当放置元素超过10底层会自动翻倍到容量为20,再超再翻倍为49,如此类推
s1 := make([]int, , )
fmt.Println("len:", len(s1), ", cap:", cap(s1)) s1 = append(s1, , , )
fmt.Println("append:", s1)
fmt.Println("len:", len(s1), ", cap:", cap(s1)) // 实现切片等拷贝
c := make([]int, len(s1))
copy(c, s1)
fmt.Println("copy:", c) // 实现切片上等再切片,类似与python中等截取
l := s1[:] // 范围为:[2, 5) 即前闭后开
fmt.Println("slice[2:5]:", l) l = s1[:] // 范围为:[0, 5)
fmt.Println("slice[:5]:", l) l = s1[:] // 范围为:[2, len(s1))
fmt.Println("slice[2:]:", l) // 也可以初始化切片时直接赋初始值
t := []string{"a", "b", "g", "h", "l"}
fmt.Println("init slice:", t) // 切片还可以存放不同长度等组合
twoD := make([][]int, )
for i := ; i < ; i++ {
innerLen := i +
twoD[i] = make([]int, innerLen)
// 为内部数据进行赋值
for j := ; j < innerLen; j++ {
twoD[i][j] = i + j
}
}
fmt.Println("towD:", twoD)
}
以上代码运行结果如下所示:
[ ]
set: [a b c]
get: c
len: 3 , cap: 3
len: 3 , cap: 10
append: [0 0 0 9 7 5]
len: 6 , cap: 10
copy: [0 0 0 9 7 5]
slice[2:5]: [0 9 7]
slice[:5]: [0 0 0 9 7]
slice[2:]: [0 9 7 5]
init slice: [a b g h l]
towD: [[0] [1 2] [2 3 4]]
以上切片代码,其实可以用下面这张图片解释 Slice 与底层数组的对应关系:
切片概念很重要,对后续很多程序都有很大的用处,这里真对切片中几个重要的概念再着重提一下:
reslice(如上图中的 Slice_a 和 Slice_b)
1. reslice 时索引以被 slice 的切片为准
2. 索引不可以超过被 slice 的切片的容量 cap() 值
3. 索引越界不会导致底层数组的重新分配而是引发错误
说明:这里说的 reslice 代表当前切片是基于数组或 slice 截取生成的,即是它们其中的一部分
append
1. 可以在 slice 尾部增加元素
2. 可以将一个 slice 追加在另外一个 slice 尾部
3. 如果最终长度未超过追加到 slice 的容量则返回原是 slice
4. 如果超过追加到的 slice 的容量则将重新分配数组并拷贝原始数据
copy
1. coyp(dest, source) 两个参数代表将第二个参数代表的切片拷贝给第一个参数代表的切片
小总结:Array 与 Slice 之间的区别
数组Array所有使用案例如下:
var m [3]int // 定义了一个长度为3的int数组,元素值都为0
m:= [3]int{} // 声明了一个长度为3的int数组,元素值都为0
a := [3]int{1, 2, 3} // 声明了一个长度为3的int数组
b := [10]int{1, 2, 3} // 声明了一个长度为10的int数组,其中前三个元素初始化为1、2、3,其它默认为0
c := [...]int{4, 5, 6} // 可以省略长度而采用 "..." 的方式,Go会自动根据元素个数来计算长度
切片Slice所有使用案例如下:
var m []int // 声明了一个指向底层数组的切片
m:= []int{} // 声明了一个指向底层数组的切片
a := array[2:5] // a指向数组的第3个元素开始,并到第五个元素结束的切片
b := array[:5] // b指向数组的第0个元素开始,并到第五个元素结束的切片
c := array[2:] // c指向数组的第2个元素开始,并到最后一个元素结束的切片
d := b[3:] // d指向切片的第3个元素开始,并到最后一个元素结束的切片(即可以切片嵌套)
由上面的结果可以看出,声明数组时,方括号内写明了数组的长度或使用...自动计算长度,而声明 slice时,方括号内没有任何字符。
GO_04:GO语言基础条件、跳转、Array和Slice的更多相关文章
- GO语言基础条件、跳转、Array和Slice
1. 判断语句if 1. 条件表达式没有括号(这点其他语言转过来的需要注意) 2. 支持一个初始化表达式(可以是并行方式,即:a, b, c := 1, 2, 3) 3. 左大括号必须和条件语句或 e ...
- day04<Java语言基础+>
Java语言基础(循环结构概述和for语句的格式及其使用) Java语言基础(循环结构for语句的练习之获取数据) Java语言基础(循环结构for语句的练习之求和思想) Java语言基础(循环结构f ...
- C语言基础(转载自大海笔记)
# C语言基础2015年03月26日10:04:411. 语言排行榜C——java——objective-C2. 进制:进制:进位机制.用普通的话讲,应该为人为的定义一种度量来标识一样东西 ...
- JavaScript 语言基础
js语言基础 一 基本知识 UniCode编码 区分大小写(HTML不区分/XHTML区分) Unicode转义序列 \uxxxx (\u加4位16进制表示) 注释 单行注释:// 多行注释:/* * ...
- Go语言基础知识总结(持续中)
Go基础知识总结 变量声明 Go语言中的变量需要声明以后才可以使用(需要提前定义变量)并且声明后必须使用(不适用会报错) 标准声明 var 变量名 变量类型 example: var name str ...
- Object Pascal 语言基础
Delphi 是以Object Pascal 语言为基础的可视化开发工具,所以要学好Delphi,首先要掌握的就是Object Pascal 语言.Object Pascal语言是Pascal之父在1 ...
- 01_C语言基础
内容提要: 1. C语言概述2. 数据类型.运算符与表达式3. C语言程序结构 4. VC6.0使用练习 知识详解01:C语言的历史 1. C语言与其它语言比较 汇编语言: (1).可直接对硬件进行操 ...
- 2008技术内幕:T-SQL语言基础
2008技术内幕:T-SQL语言基础 单表查询摘记 这里的摘抄来自<Microsoft SQL Server 2008技术内幕:T-SQL语言基础>,书中用到的案例数据库是这个 TSQLF ...
- C语言基础复习总结
C语言基础复习总结 大一学的C++,不过后来一直没用,大多还给老师了,最近看传智李明杰老师的ios课程的C语言入门部分,用了一周,每晚上看大概两小时左右,效果真是顶一学期的课,也许是因为有开发经验吧, ...
随机推荐
- Teamproject Week7 --Scrum Meeting #1 2014.10.28
这是团队的第一次会议,具体议题如下: 1)我们明确了团队成员的职责所需: PM职责:根据项目范围.质量.时间与成本的综合因素的考虑,进行项目的总体规划与阶段计划. 控制项目组各成员的工作进度,即时了 ...
- Alpha版本测试文档
概述 本次测试主要是为了测试是否有导致崩溃的bug,验证是否符合软件基本需求. 测试环境 硬件测试:安卓系统手机,安卓平板. 测试人员 赖彦谕,金哉仁. 实际进度 2015/11/6 – 2015/1 ...
- 炸弹人 之 N A B C D
团队开发之个人——NABCD理解 项目名称:炸弹人(app)N(need): 随着移动终端的发展,各类软件的需求必然会有长期的需求,而游戏类软件是不同年龄阶段的人共同的需求,我们将要开发的这款游 ...
- NABCD模型分析
1.N——need需求 目前,学习英语是所有学生会面临的问题.提高词汇量对学习英语是十分必要的,尤其是对大学生来说对手机的使用特别频繁,我们提高英语词汇量也应该把手机更好的利用起来,利用自己对手机的使 ...
- static和final
是静态修饰符,什么叫静态修饰符呢?大家都知道,在程序中任何变量或者代码都是在编译时由系统自动分配内存来存储的,而所谓静态就是指在编译后所分配的内存会一直存在,直到程序退出内存才会释放这个空间,也就是只 ...
- OSG学习:计算纹理坐标
在很多时候,直接指定纹理坐标是非常不方便的,如曲面纹理坐标,只有少数的曲面(如圆锥.圆柱等)可以在不产生扭曲的情况下映射到平面上,其他的曲面在映射到表面时都会产生一定程度的扭曲.一般而言,曲面表面的曲 ...
- Linux里的稀疏文件
今天发现一个有意思的现象,文件系统大小只有37GB,上面却有一个900GB的文件!查了下,这个叫“稀疏文件”,我理解类似于VMWare里的瘦硬盘模式吧,先预先划出一块空间,然后往里填数据. [root ...
- SQL SERVER ENTERPRISE EDITION-CORE VS SERVER+CAL – DEMO ON DIFFERENCES
SQL SERVER ENTERPRISE EDITION-CORE VS SERVER+CAL – DEMO ON DIFFERENCES DHARMENDRA KESHARIAugust 03, ...
- 微信小程序组件 自定义弹出框
<!-- 点击立即抢拼弹出框 --> <view class='add-rob' bindtap="setModalStatus" data-status=&qu ...
- Java 使用 dom4j 读取 xml文档 demo
<?xml version="1.0" encoding="UTF-8"?> <web-app xmlns="http://www. ...