流程控制
if条件语句
module main
fn main() {
a := 10
b := 20
if a < b {
println('$a < $b')
} else if a > b {
println('$a > $b')
} else {
println('$a == $b')
}
}
条件赋值(if表达式)
num := 777
// 简单的条件赋值
s := if num % 2 == 0 { 'even' } else { 'odd' }
println(s)
// "odd"
// 多条件赋值
a, b, c := if true { 1, 'awesome', 13 } else { 0, 'bad', 0 }
println(a)
println(b)
println(c)
likely/unlikely
_likely_和_unlikely_这两个内置函数实现的是跟C的likely一样的效果,可以实现给条件分支的性能做优化,一般的代码来说,使用的场景不多
module main
fn main() {
x := 1
if _likely_(x == 1) { //告诉编译器,大部分情况的分支是这个,做代码优化
println('a')
} else {
println('b')
}
if _unlikely_(x == 1) { //告诉编译器,大部分情况的分支不是这个,做代码优化
println('a')
} else {
println('b')
}
}
match分支语句
match要求穷尽所有可能,所以基本都要带上else语句,else语句只能有一个。
fn main() {
os := 'macos'
match os {
'windows' { println('windows') }
'linux' { println('linux') }
'macos' { println('macos') }
else { println('unknow') }
}
}
匹配的值也可以多个,用逗号分隔:
fn main() {
os := 'macos'
match os {
'windows' { println('windows') }
'macos', 'linux' { println('macos or linux') }
else { println('unknow') }
}
}
match赋值(match表达式)
os := 'macos'
price := match os {
'windows' { 100 }
'linux' { 120 }
'macos' { 150 }
else { 0 }
}
println(price)
//输出150
//多变量match赋值
a, b, c := match false {
true { 1, 2, 3 }
false { 4, 5, 6 }
else { 7, 8, 9 }
}
println(a)
println(b)
println(c)
match的同时,加上mut ,可以修改匹配变量,通常是配合for in 语句结合使用
//参考代码
for stmt in file.stmts {
match mut stmt {
ast.ConstDecl {
c.stmt(*it)
}
else {}
}
}
使用match判断联合类型的具体类型
module main
struct User {
name string
age int
}
pub fn (m &User) str() string {
return 'name:$m.name,age:$m.age'
}
type MySum = User | int | string //联合类型声明
pub fn (ms MySum) str() string {
match ms { //如果函数的参数或者接收者是联合类型,可以使用match进一步判断类型
int {
return ms.str()
}
string {
return ms // ms的类型是string
}
User {
return ms.str() // ms的类型是User
}
// else { //如果之前的分支已经穷尽了所有可能,else语句不需要,如果没有穷尽所有可能,则else语句是必须的
// return 'unknown'
// }
}
}
for 循环语句
for的四种形式:
传统的:for i=0;i<100;i++ {}
module main
fn main() {
for i := 0; i < 10; i++ {
// 跳过6
if i == 6 {
continue
}
println(i)
}
mut a := 0
for i := 0; i < 10; i++, a++ { //递增可以包含多个变量
println(a)
}
}
为了简洁的目的,for里面的i默认就是mut可变的,不需要特别声明为mut,如果声明了编译器会报错
替代while:for i<100 {}
mut sum := 0
mut i := 0
for i <= 100 {
sum += i
i++
}
println(sum)
// 输出"5050"
无限循环:for {}
mut num := 0
for {
num++
if num >= 10 {
break
}
}
println(num)
// "10"
遍历:for i in xxx {}
for in可以用来遍历字符串,数组,区间,字典这四种类型
遍历字符串:
str := 'abcdef'
// 遍历value
for s in str {
println(s.str())
}
// 遍历index和value
for i, s in str {
println('index:$i,value:$s.str()')
}
遍历数组:
numbers := [1, 2, 3, 4, 5]
for num in numbers { //遍历动态数组
println('num:$num')
}
for i, num in numbers {
println('i:$i,num:$num')
}
// 或者这种区间的写法也可以
for i in 0 .. numbers.len {
println('num:${numbers[i]}')
}
// 遍历固定长度数组
arr = [0, 1, 2, 3, 4, 5, 6, 7]!
for a in arr { //使用for in遍历固定长度数组
println(a)
}
遍历区间:
mut sum := 0
for i in 1 .. 11 { // 左闭右开,遍历区间
sum += i
}
println(sum) // 55
遍历字典:
m := {
'name': 'jack'
'age': '20'
'desc': 'good man'
}
for key, value in m {
println('key:$key,value:$value')
}
跟其他语言一样continue用来重新继续当前循环,break用来跳出当前循环
如果存在多层嵌套的循环,也可以使用continue label和break label来控制重新/跳出标签那一层的循环
fn main() {
mut i := 4
goto L1
L1: for { //在for前加标签
i++
for {
if i < 7 {continue L1} //从顶层循环继续
else {break L1} //直接跳出顶层循环
}
}
println(i)
goto L2
L2: for ;; i++ {
for {
if i < 17 {continue L2}
else {break L2}
}
}
println(i)
goto L3
L3: for e in [1,2,3,4] {
i = e
for {
if i < 3 {continue L3}
else {break L3}
}
}
println(i)
}
for is语句
用于联合类型的类型循环判断(感觉没啥用,就是一个语法糖而已)
module main
struct Milk {
mut:
name string
}
struct Eggs {
mut:
name string
}
type Food = Eggs | Milk
fn main() {
mut f := Food(Eggs{'test'})
//不带mut
for f is Eggs {
println(typeof(f).name)
break
}
//等价于
for {
if f is Eggs {
println(typeof(f).name)
break
}
}
//带mut
for mut f is Eggs {
f.name = 'eggs'
println(f.name)
break
}
//等价于
for {
if mut f is Eggs {
f.name = 'eggs'
println(f.name)
break
}
}
}
for循环结合or代码块
module main
import os
fn main() {
//for循环结合or代码块,更简洁一些
for line in os.read_lines(@FILE) or { panic('文件不存在') } {
// 报错
// for line in os.read_lines('不存在的文件') or { panic('文件不存在') } {
println(line)
}
for select语句
for select语句主要在并发中使用,用来循环监听多个chanel,更多内容可以参考并发章节
fn main() {
ch1 := chan int{}
ch2 := chan f64{}
go do_send(ch1, ch2)
mut a := 0
mut b := 0
for select { // 循环监听channel的写入,写入后执行for代码块,直到所有监听的channel都已关闭
x := <-ch1 {
a += x
}
y := <-ch2 {
a += int(y)
}
} { // for代码块
b++ // 写入3次
println('${b}. event')
}
println(a)
println(b)
}
fn do_send(ch1 chan int, ch2 chan f64) {
ch1 <- 3
ch2 <- 5.0
ch2.close()
ch1 <- 2
ch1.close()
}
goto语句
goto语句只能在函数内部跳转
fn main() {
mut i := 0
a: //定义跳转标签
i++
// a: i++ //标签和语句在同一行也可以正常运行
if i < 3 {
goto a //跳转到a标签
}
println(i)
}
最后更新于