235 lines
5.1 KiB
Zig
235 lines
5.1 KiB
Zig
const assert = @import("std").debug.assert;
|
|
|
|
test "switch with numbers" {
|
|
testSwitchWithNumbers(13);
|
|
}
|
|
|
|
fn testSwitchWithNumbers(x: u32) void {
|
|
const result = switch (x) {
|
|
1, 2, 3, 4...8 => false,
|
|
13 => true,
|
|
else => false,
|
|
};
|
|
assert(result);
|
|
}
|
|
|
|
test "switch with all ranges" {
|
|
assert(testSwitchWithAllRanges(50, 3) == 1);
|
|
assert(testSwitchWithAllRanges(101, 0) == 2);
|
|
assert(testSwitchWithAllRanges(300, 5) == 3);
|
|
assert(testSwitchWithAllRanges(301, 6) == 6);
|
|
}
|
|
|
|
fn testSwitchWithAllRanges(x: u32, y: u32) u32 {
|
|
return switch (x) {
|
|
0...100 => 1,
|
|
101...200 => 2,
|
|
201...300 => 3,
|
|
else => y,
|
|
};
|
|
}
|
|
|
|
test "implicit comptime switch" {
|
|
const x = 3 + 4;
|
|
const result = switch (x) {
|
|
3 => 10,
|
|
4 => 11,
|
|
5, 6 => 12,
|
|
7, 8 => 13,
|
|
else => 14,
|
|
};
|
|
|
|
comptime {
|
|
assert(result + 1 == 14);
|
|
}
|
|
}
|
|
|
|
test "switch on enum" {
|
|
const fruit = Fruit.Orange;
|
|
nonConstSwitchOnEnum(fruit);
|
|
}
|
|
const Fruit = enum {
|
|
Apple,
|
|
Orange,
|
|
Banana,
|
|
};
|
|
fn nonConstSwitchOnEnum(fruit: Fruit) void {
|
|
switch (fruit) {
|
|
Fruit.Apple => unreachable,
|
|
Fruit.Orange => {},
|
|
Fruit.Banana => unreachable,
|
|
}
|
|
}
|
|
|
|
test "switch statement" {
|
|
nonConstSwitch(SwitchStatmentFoo.C);
|
|
}
|
|
fn nonConstSwitch(foo: SwitchStatmentFoo) void {
|
|
const val = switch (foo) {
|
|
SwitchStatmentFoo.A => i32(1),
|
|
SwitchStatmentFoo.B => 2,
|
|
SwitchStatmentFoo.C => 3,
|
|
SwitchStatmentFoo.D => 4,
|
|
};
|
|
assert(val == 3);
|
|
}
|
|
const SwitchStatmentFoo = enum {
|
|
A,
|
|
B,
|
|
C,
|
|
D,
|
|
};
|
|
|
|
test "switch prong with variable" {
|
|
switchProngWithVarFn(SwitchProngWithVarEnum{ .One = 13 });
|
|
switchProngWithVarFn(SwitchProngWithVarEnum{ .Two = 13.0 });
|
|
switchProngWithVarFn(SwitchProngWithVarEnum{ .Meh = {} });
|
|
}
|
|
const SwitchProngWithVarEnum = union(enum) {
|
|
One: i32,
|
|
Two: f32,
|
|
Meh: void,
|
|
};
|
|
fn switchProngWithVarFn(a: *const SwitchProngWithVarEnum) void {
|
|
switch (a.*) {
|
|
SwitchProngWithVarEnum.One => |x| {
|
|
assert(x == 13);
|
|
},
|
|
SwitchProngWithVarEnum.Two => |x| {
|
|
assert(x == 13.0);
|
|
},
|
|
SwitchProngWithVarEnum.Meh => |x| {
|
|
const v: void = x;
|
|
},
|
|
}
|
|
}
|
|
|
|
test "switch on enum using pointer capture" {
|
|
testSwitchEnumPtrCapture();
|
|
comptime testSwitchEnumPtrCapture();
|
|
}
|
|
|
|
fn testSwitchEnumPtrCapture() void {
|
|
var value = SwitchProngWithVarEnum{ .One = 1234 };
|
|
switch (value) {
|
|
SwitchProngWithVarEnum.One => |*x| x.* += 1,
|
|
else => unreachable,
|
|
}
|
|
switch (value) {
|
|
SwitchProngWithVarEnum.One => |x| assert(x == 1235),
|
|
else => unreachable,
|
|
}
|
|
}
|
|
|
|
test "switch with multiple expressions" {
|
|
const x = switch (returnsFive()) {
|
|
1, 2, 3 => 1,
|
|
4, 5, 6 => 2,
|
|
else => i32(3),
|
|
};
|
|
assert(x == 2);
|
|
}
|
|
fn returnsFive() i32 {
|
|
return 5;
|
|
}
|
|
|
|
const Number = union(enum) {
|
|
One: u64,
|
|
Two: u8,
|
|
Three: f32,
|
|
};
|
|
|
|
const number = Number{ .Three = 1.23 };
|
|
|
|
fn returnsFalse() bool {
|
|
switch (number) {
|
|
Number.One => |x| return x > 1234,
|
|
Number.Two => |x| return x == 'a',
|
|
Number.Three => |x| return x > 12.34,
|
|
}
|
|
}
|
|
test "switch on const enum with var" {
|
|
assert(!returnsFalse());
|
|
}
|
|
|
|
test "switch on type" {
|
|
assert(trueIfBoolFalseOtherwise(bool));
|
|
assert(!trueIfBoolFalseOtherwise(i32));
|
|
}
|
|
|
|
fn trueIfBoolFalseOtherwise(comptime T: type) bool {
|
|
return switch (T) {
|
|
bool => true,
|
|
else => false,
|
|
};
|
|
}
|
|
|
|
test "switch handles all cases of number" {
|
|
testSwitchHandleAllCases();
|
|
comptime testSwitchHandleAllCases();
|
|
}
|
|
|
|
fn testSwitchHandleAllCases() void {
|
|
assert(testSwitchHandleAllCasesExhaustive(0) == 3);
|
|
assert(testSwitchHandleAllCasesExhaustive(1) == 2);
|
|
assert(testSwitchHandleAllCasesExhaustive(2) == 1);
|
|
assert(testSwitchHandleAllCasesExhaustive(3) == 0);
|
|
|
|
assert(testSwitchHandleAllCasesRange(100) == 0);
|
|
assert(testSwitchHandleAllCasesRange(200) == 1);
|
|
assert(testSwitchHandleAllCasesRange(201) == 2);
|
|
assert(testSwitchHandleAllCasesRange(202) == 4);
|
|
assert(testSwitchHandleAllCasesRange(230) == 3);
|
|
}
|
|
|
|
fn testSwitchHandleAllCasesExhaustive(x: u2) u2 {
|
|
return switch (x) {
|
|
0 => u2(3),
|
|
1 => 2,
|
|
2 => 1,
|
|
3 => 0,
|
|
};
|
|
}
|
|
|
|
fn testSwitchHandleAllCasesRange(x: u8) u8 {
|
|
return switch (x) {
|
|
0...100 => u8(0),
|
|
101...200 => 1,
|
|
201, 203 => 2,
|
|
202 => 4,
|
|
204...255 => 3,
|
|
};
|
|
}
|
|
|
|
test "switch all prongs unreachable" {
|
|
testAllProngsUnreachable();
|
|
comptime testAllProngsUnreachable();
|
|
}
|
|
|
|
fn testAllProngsUnreachable() void {
|
|
assert(switchWithUnreachable(1) == 2);
|
|
assert(switchWithUnreachable(2) == 10);
|
|
}
|
|
|
|
fn switchWithUnreachable(x: i32) i32 {
|
|
while (true) {
|
|
switch (x) {
|
|
1 => return 2,
|
|
2 => break,
|
|
else => continue,
|
|
}
|
|
}
|
|
return 10;
|
|
}
|
|
|
|
fn return_a_number() error!i32 {
|
|
return 1;
|
|
}
|
|
|
|
test "capture value of switch with all unreachable prongs" {
|
|
const x = return_a_number() catch |err| switch (err) {
|
|
else => unreachable,
|
|
};
|
|
assert(x == 1);
|
|
}
|