Values

values.zig

  1. // Top-level declarations are order-independent:
  2. const print = std.debug.print;
  3. const std = @import("std");
  4. const os = std.os;
  5. const assert = std.debug.assert;
  6. pub fn main() void {
  7. // integers
  8. const one_plus_one: i32 = 1 + 1;
  9. print("1 + 1 = {}\n", .{one_plus_one});
  10. // floats
  11. const seven_div_three: f32 = 7.0 / 3.0;
  12. print("7.0 / 3.0 = {}\n", .{seven_div_three});
  13. // boolean
  14. print("{}\n{}\n{}\n", .{
  15. true and false,
  16. true or false,
  17. !true,
  18. });
  19. // optional
  20. var optional_value: ?[]const u8 = null;
  21. assert(optional_value == null);
  22. print("\noptional 1\ntype: {}\nvalue: {?s}\n", .{
  23. @TypeOf(optional_value), optional_value,
  24. });
  25. optional_value = "hi";
  26. assert(optional_value != null);
  27. print("\noptional 2\ntype: {}\nvalue: {?s}\n", .{
  28. @TypeOf(optional_value), optional_value,
  29. });
  30. // error union
  31. var number_or_error: anyerror!i32 = error.ArgNotFound;
  32. print("\nerror union 1\ntype: {}\nvalue: {!}\n", .{
  33. @TypeOf(number_or_error), number_or_error, });
  34. number_or_error = 1234;
  35. print("\nerror union 2\ntype: {}\nvalue: {!}\n", .{
  36. @TypeOf(number_or_error), number_or_error,
  37. });
  38. }

Shell

  1. $ zig build-exe values.zig
  2. $ ./values
  3. 1 + 1 = 2
  4. 7.0 / 3.0 = 2.3333333e0
  5. false
  6. true
  7. false
  8. optional 1
  9. type: ?[]const u8
  10. value: null
  11. optional 2
  12. type: ?[]const u8
  13. value: hi
  14. error union 1
  15. type: anyerror!i32
  16. value: error.ArgNotFound
  17. error union 2
  18. type: anyerror!i32
  19. value: 1234

Primitive Types

Primitive Types
TypeC EquivalentDescription
i8int8_tsigned 8-bit integer
u8uint8_tunsigned 8-bit integer
i16int16_tsigned 16-bit integer
u16uint16_tunsigned 16-bit integer
i32int32_tsigned 32-bit integer
u32uint32_tunsigned 32-bit integer
i64int64_tsigned 64-bit integer
u64uint64_tunsigned 64-bit integer
i128int128signed 128-bit integer
u128unsigned int128unsigned 128-bit integer
isizeintptr_tsigned pointer sized integer
usizeuintptr_t, size_tunsigned pointer sized integer. Also see #5185
c_charcharfor ABI compatibility with C
c_shortshortfor ABI compatibility with C
c_ushortunsigned shortfor ABI compatibility with C
c_intintfor ABI compatibility with C
c_uintunsigned intfor ABI compatibility with C
c_longlongfor ABI compatibility with C
c_ulongunsigned longfor ABI compatibility with C
c_longlonglong longfor ABI compatibility with C
c_ulonglongunsigned long longfor ABI compatibility with C
c_longdoublelong doublefor ABI compatibility with C
f16_Float1616-bit floating point (10-bit mantissa) IEEE-754-2008 binary16
f32float32-bit floating point (23-bit mantissa) IEEE-754-2008 binary32
f64double64-bit floating point (52-bit mantissa) IEEE-754-2008 binary64
f80double80-bit floating point (64-bit mantissa) IEEE-754-2008 80-bit extended precision
f128_Float128128-bit floating point (112-bit mantissa) IEEE-754-2008 binary128
boolbooltrue or false
anyopaquevoidUsed for type-erased pointers.
void(none)Always the value void{}
noreturn(none)the type of break, continue, return, unreachable, and while (true) {}
type(none)the type of types
anyerror(none)an error code
comptime_int(none)Only allowed for comptime-known values. The type of integer literals.
comptime_float(none)Only allowed for comptime-known values. The type of float literals.

In addition to the integer types above, arbitrary bit-width integers can be referenced by using an identifier of i or u followed by digits. For example, the identifier i7 refers to a signed 7-bit integer. The maximum allowed bit-width of an integer type is 65535.

See also:

Primitive Values

Primitive Values
NameDescription
true and falsebool values
nullused to set an optional type to null
undefinedused to leave a value unspecified

See also:

String Literals and Unicode Code Point Literals

String literals are constant single-item Pointers to null-terminated byte arrays. The type of string literals encodes both the length, and the fact that they are null-terminated, and thus they can be coerced to both Slices and Null-Terminated Pointers. Dereferencing string literals converts them to Arrays.

Because Zig source code is UTF-8 encoded, any non-ASCII bytes appearing within a string literal in source code carry their UTF-8 meaning into the content of the string in the Zig program; the bytes are not modified by the compiler. It is possible to embed non-UTF-8 bytes into a string literal using \xNN notation.

Indexing into a string containing non-ASCII bytes returns individual bytes, whether valid UTF-8 or not.

Unicode code point literals have type comptime_int, the same as Integer Literals. All Escape Sequences are valid in both string literals and Unicode code point literals.

string_literals.zig

  1. const print = @import("std").debug.print;
  2. const mem = @import("std").mem; // will be used to compare bytes
  3. pub fn main() void {
  4. const bytes = "hello";
  5. print("{}\n", .{@TypeOf(bytes)}); // *const [5:0]u8
  6. print("{d}\n", .{bytes.len}); // 5
  7. print("{c}\n", .{bytes[1]}); // 'e'
  8. print("{d}\n", .{bytes[5]}); // 0
  9. print("{}\n", .{'e' == '\x65'}); // true
  10. print("{d}\n", .{'\u{1f4a9}'}); // 128169
  11. print("{d}\n", .{'💯'}); // 128175
  12. print("{u}\n", .{'⚡'});
  13. print("{}\n", .{mem.eql(u8, "hello", "h\x65llo")}); // true
  14. print("{}\n", .{mem.eql(u8, "💯", "\xf0\x9f\x92\xaf")}); // also true
  15. const invalid_utf8 = "\xff\xfe"; // non-UTF-8 strings are possible with \xNN notation.
  16. print("0x{x}\n", .{invalid_utf8[1]}); // indexing them returns individual bytes...
  17. print("0x{x}\n", .{"💯"[1]}); // ...as does indexing part-way through non-ASCII characters
  18. }

Shell

  1. $ zig build-exe string_literals.zig
  2. $ ./string_literals
  3. *const [5:0]u8
  4. 5
  5. e
  6. 0
  7. true
  8. 128169
  9. 128175
  10. true
  11. true
  12. 0xfe
  13. 0x9f

See also:

Escape Sequences

Escape Sequences
Escape SequenceName
\nNewline
\rCarriage Return
\tTab
\Backslash
\’Single Quote
\”Double Quote
\xNNhexadecimal 8-bit byte value (2 digits)
\u{NNNNNN}hexadecimal Unicode code point UTF-8 encoded (1 or more digits)

Note that the maximum valid Unicode point is 0x10ffff.

Multiline String Literals

Multiline string literals have no escapes and can span across multiple lines. To start a multiline string literal, use the \\ token. Just like a comment, the string literal goes until the end of the line. The end of the line is not included in the string literal. However, if the next line begins with \\ then a newline is appended and the string literal continues.

multiline_string_literals.zig

  1. const hello_world_in_c =
  2. \\#include <stdio.h>
  3. \\
  4. \\int main(int argc, char **argv) {
  5. \\ printf("hello world\n");
  6. \\ return 0;
  7. \\}
  8. ;

See also:

Assignment

Use the const keyword to assign a value to an identifier:

constant_identifier_cannot_change.zig

  1. const x = 1234;
  2. fn foo() void {
  3. // It works at file scope as well as inside functions.
  4. const y = 5678;
  5. // Once assigned, an identifier cannot be changed.
  6. y += 1;
  7. }
  8. pub fn main() void {
  9. foo();
  10. }

Shell

  1. $ zig build-exe constant_identifier_cannot_change.zig
  2. constant_identifier_cannot_change.zig:8:7: error: cannot assign to constant
  3. y += 1;
  4. ~~^~~~
  5. referenced by:
  6. main: constant_identifier_cannot_change.zig:12:5
  7. callMain: /home/ci/actions-runner/_work/zig-bootstrap/out/host/lib/zig/std/start.zig:501:17
  8. remaining reference traces hidden; use '-freference-trace' to see all reference traces

const applies to all of the bytes that the identifier immediately addresses. Pointers have their own const-ness.

If you need a variable that you can modify, use the var keyword:

mutable_var.zig

  1. const print = @import("std").debug.print;
  2. pub fn main() void {
  3. var y: i32 = 5678;
  4. y += 1;
  5. print("{d}", .{y});
  6. }

Shell

  1. $ zig build-exe mutable_var.zig
  2. $ ./mutable_var
  3. 5679

Variables must be initialized:

var_must_be_initialized.zig

  1. pub fn main() void {
  2. var x: i32;
  3. x = 1;
  4. }

Shell

  1. $ zig build-exe var_must_be_initialized.zig
  2. var_must_be_initialized.zig:2:15: error: expected '=', found ';'
  3. var x: i32;
  4. ^

undefined

Use undefined to leave variables uninitialized:

assign_undefined.zig

  1. const print = @import("std").debug.print;
  2. pub fn main() void {
  3. var x: i32 = undefined;
  4. x = 1;
  5. print("{d}", .{x});
  6. }

Shell

  1. $ zig build-exe assign_undefined.zig
  2. $ ./assign_undefined
  3. 1

undefined can be coerced to any type. Once this happens, it is no longer possible to detect that the value is undefined. undefined means the value could be anything, even something that is nonsense according to the type. Translated into English, undefined means “Not a meaningful value. Using this value would be a bug. The value will be unused, or overwritten before being used.”

In Debug mode, Zig writes 0xaa bytes to undefined memory. This is to catch bugs early, and to help detect use of undefined memory in a debugger. However, this behavior is only an implementation feature, not a language semantic, so it is not guaranteed to be observable to code.