Go的switch支持任意可比较类型表达式,如整数、字符串、布尔值、指针、struct(字段全可比较)等;不支持切片、map、func及含它们的类型;case默认自动break,fallthrough需显式声明且仅限末尾;无表达式时等价于if-else链。

switch 后面能跟哪些表达式
Go 的 switch 不限于常量或整数,支持任意可比较类型的表达式:整数、字符串、布尔值、接口(当底层值可比较时)、甚至自定义类型(只要实现了 == 语义,比如 struct 字段全为可比较类型)。但不能是切片、map、func 或包含它们的结构体——这些在运行时会直接报错 invalid operation: cannot compare。
常见误用是试图 switch 一个 []string 或 map[string]int,Go 编译器会立刻拒绝,不给运行机会。
- 安全:字符串、int、rune、bool、指针、struct(字段全可比较)
- 不安全:[]int、map[int]string、func()、interface{}(若动态值是 slice/map)
- 注意:
nil可以参与比较,但仅限于指针、map、slice、chan、func、interface —— 这些类型的nil是合法比较值
case 分支默认带 break,fallthrough 需显式写出
Go 的每个 case 自动终止,不会隐式“穿透”到下一个 case。这和 C/Java 完全不同,是刻意设计的安全机制。如果真需要 fallthrough(极少场景),必须手动写 fallthrough 语句,且它只能出现在 case 最后一行(后面不能有其他语句)。
典型陷阱:从其他语言转过来的人常忘记这点,以为多个 case 共享逻辑,结果只有第一个匹配的 case 执行,其余被跳过。
立即学习“go语言免费学习笔记(深入)”;
switch mode {
case "dev":
log.Println("dev mode")
fallthrough // 必须显式写,且只能在末尾
case "test":
log.Println("test mode") // 这行会被执行
case "prod":
log.Println("prod mode")
}
switch 无表达式时等价于 if-else 链
省略 switch 后的表达式(即 switch {),每个 case 后跟的是布尔表达式,整个结构就变成多分支条件判断。这种写法比嵌套 if 更清晰,尤其适合处理多个边界条件或组合判断。
优势在于可读性提升,且 Go 编译器仍能做变量作用域隔离(每个 case 是独立作用域)。
switch {
case x < 0:
fmt.Println("negative")
case x == 0:
fmt.Println("zero")
case x > 0 && x < 100:
fmt.Println("small positive")
default:
fmt.Println("large positive")
}
- case 表达式按顺序求值,遇到第一个 true 就执行对应分支,其余跳过
- 所有 case 表达式必须是布尔类型;混入非布尔值(如
case "hello")会编译失败 - 没有
default也没关系,只是意味着“都不满足时什么也不做”
type switch 判断接口底层类型
对 interface{} 或其他接口类型做类型断言时,switch 是最简洁、最惯用的方式。它比一连串 if v, ok := x.(T); ok { ... } 更易读、更高效(编译器可优化)。
注意:case nil: 是合法的,用于检测接口值是否为 nil(即动态值和动态类型都为 nil);而 case (*T)(nil): 这类写法无效,type switch 的 case 只接受类型字面量或 nil。
var x interface{} = "hello"
switch v := x.(type) {
case string:
fmt.Printf("string: %s\n", v)
case int:
fmt.Printf("int: %d\n", v)
case nil:
fmt.Println("x is nil")
default:
fmt.Printf("unknown type: %T\n", v)
}
-
v := x.(type)是唯一合法语法;不能写成x.(type) == string或漏掉v := - 每个 case 中的
v类型由当前分支决定(string 分支里v是 string 类型) - 如果接口值是 nil,只有
case nil:匹配;其他任何类型 case 都不触发
nil 的特殊地位,以及无表达式 switch 里 case 必须为布尔表达式这两点。










