1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313
|
<!--
SPDX-License-Identifier: Apache-2.0
SPDX-FileCopyrightText: 2021 The Elixir Team
-->
# Basic types
In this chapter we will learn more about Elixir basic types: integers, floats, booleans, atoms, and strings. Other data types, such as lists and tuples, will be explored in the next chapter.
```elixir
iex> 1 # integer
iex> 0x1F # integer
iex> 1.0 # float
iex> true # boolean
iex> :atom # atom / symbol
iex> "elixir" # string
iex> [1, 2, 3] # list
iex> {1, 2, 3} # tuple
```
## Basic arithmetic
Open up `iex` and type the following expressions:
```elixir
iex> 1 + 2
3
iex> 5 * 5
25
iex> 10 / 2
5.0
```
Notice that `10 / 2` returned a float `5.0` instead of an integer `5`. This is expected. In Elixir, the operator [`/`](`//2`) always returns a float. If you want to do integer division or get the division remainder, you can invoke the [`div`](`div/2`) and [`rem`](`rem/2`) functions:
```elixir
iex> div(10, 2)
5
iex> div 10, 2
5
iex> rem 10, 3
1
```
Notice that Elixir allows you to drop the parentheses when invoking functions that expect one or more arguments. This feature gives a cleaner syntax when writing declarations and control-flow constructs. However, Elixir developers generally prefer to use parentheses.
Elixir also supports shortcut notations for entering binary, octal, and hexadecimal numbers:
```elixir
iex> 0b1010
10
iex> 0o777
511
iex> 0x1F
31
```
Float numbers require a dot followed by at least one digit and also support `e` for scientific notation:
```elixir
iex> 1.0
1.0
iex> 1.0e-10
1.0e-10
```
Floats in Elixir are 64-bit precision.
You can invoke the [`round`](`round/1`) function to get the closest integer to a given float, or the [`trunc`](`trunc/1`) function to get the integer part of a float.
```elixir
iex> round(3.58)
4
iex> trunc(3.58)
3
```
Finally, we work with different data types, we will learn Elixir provides several predicate functions to check for the type of a value. For example, [`is_integer`](`is_integer/1`) can be used to check if a value is an integer or not:
```elixir
iex> is_integer(1)
true
iex> is_integer(2.0)
false
```
You can also use [`is_float`](`is_float/1`) or [`is_number`](`is_number/1`) to check, respectively, if an argument is a float, or either an integer or float.
## Booleans and `nil`
Elixir supports `true` and `false` as booleans:
```elixir
iex> true
true
iex> true == false
false
```
Elixir also provides three boolean operators: [`or`](`or/2`), [`and`](`and/2`), and [`not`](`not/1`). These operators are strict in the sense that they expect something that evaluates to a boolean (`true` or `false`) as their first argument:
```elixir
iex> true and true
true
iex> false or is_boolean(true)
true
```
Providing a non-boolean will raise an exception:
```elixir
iex> 1 and true
** (BadBooleanError) expected a boolean on left-side of "and", got: 1
```
`or` and `and` are short-circuit operators. They only execute the right side if the left side is not enough to determine the result:
```elixir
iex> false and raise("This error will never be raised")
false
iex> true or raise("This error will never be raised")
true
```
Elixir also provides the concept of `nil`, to indicate the absence of a value, and a set of logical operators that also manipulate `nil`: `||/2`, `&&/2`, and `!/1`. For these operators, `false` and `nil` are considered "falsy", all other values are considered "truthy":
```elixir
# or
iex> 1 || true
1
iex> false || 11
11
# and
iex> nil && 13
nil
iex> true && 17
17
# not
iex> !true
false
iex> !1
false
iex> !nil
true
```
Similarly, values like `0` and `""`, which some other programming languages consider to be "falsy", are also "truthy" in Elixir.
As a rule of thumb, use `and`, `or` and `not` when you are expecting booleans. If any of the arguments are non-boolean, use `&&`, `||` and `!`.
## Atoms
An atom is a constant whose value is its own name. Some other languages call these symbols. They are often useful to enumerate over distinct values, such as:
```elixir
iex> :apple
:apple
iex> :orange
:orange
iex> :watermelon
:watermelon
```
Atoms are equal if their names are equal.
```elixir
iex> :apple == :apple
true
iex> :apple == :orange
false
```
Often they are used to express the state of an operation, by using values such as `:ok` and `:error`.
The booleans `true` and `false` are also atoms:
```elixir
iex> true == :true
true
iex> is_atom(false)
true
iex> is_boolean(:false)
true
```
Elixir allows you to skip the leading `:` for the atoms `false`, `true` and `nil`.
## Strings
Strings in Elixir are delimited by double quotes, and they are encoded in UTF-8:
```elixir
iex> "hellö"
"hellö"
```
> Note: if you are running on Windows, there is a chance your terminal does not use UTF-8 by default. You can change the encoding of your current session by running `chcp 65001` before entering IEx.
You can concatenate two strings with the [`<>`](`<>/2`) operator:
```elixir
iex> "hello " <> "world!"
"hello world!"
```
Elixir also supports string interpolation:
```elixir
iex> string = "world"
iex> "hello #{string}!"
"hello world!"
```
String concatenation requires both sides to be strings but interpolation supports any data type that may be converted to a string:
```elixir
iex> number = 42
iex> "i am #{number} years old!"
"i am 42 years old!"
```
Strings can have line breaks in them. You can introduce them using escape sequences:
```elixir
iex> "hello
...> world"
"hello\nworld"
iex> "hello\nworld"
"hello\nworld"
```
You can print a string using the [`IO.puts`](`IO.puts/1`) function from the `IO` module:
```elixir
iex> IO.puts("hello\nworld")
hello
world
:ok
```
Notice that the [`IO.puts`](`IO.puts/1`) function returns the atom `:ok` after printing.
Strings in Elixir are represented internally by contiguous sequences of bytes known as binaries:
```elixir
iex> is_binary("hellö")
true
```
We can also get the number of bytes in a string:
```elixir
iex> byte_size("hellö")
6
```
Notice that the number of bytes in that string is 6, even though it has 5 graphemes. That's because the grapheme "ö" takes 2 bytes to be represented in UTF-8. We can get the actual length of the string, based on the number of graphemes, by using the [`String.length`](`String.length/1`) function:
```elixir
iex> String.length("hellö")
5
```
The `String` module contains a bunch of functions that operate on strings as defined in the Unicode standard:
```elixir
iex> String.upcase("hellö")
"HELLÖ"
```
## Structural comparison
Elixir also provides [`==`](`==/2`), [`!=`](`!=/2`), [`<=`](`<=/2`), [`>=`](`>=/2`), [`<`](`</2`) and [`>`](`>/2`) as comparison operators. We can compare numbers:
```elixir
iex> 1 == 1
true
iex> 1 != 2
true
iex> 1 < 2
true
```
But also atoms, strings, booleans, etc:
```elixir
iex> "foo" == "foo"
true
iex> "foo" == "bar"
false
```
Integers and floats compare the same if they have the same value:
```elixir
iex> 1 == 1.0
true
iex> 1 == 2.0
false
```
However, you can use the strict comparison operator [`===`](`===/2`) and [`!==`](`!==/2`) if you want to distinguish between integers and floats:
```elixir
iex> 1 === 1.0
false
```
The comparison operators in Elixir can compare across any data type. We say these operators perform _structural comparison_. For more information, you can read our documentation on [Structural vs Semantic comparisons](`Kernel#module-structural-comparison`).
Elixir also provides data-types for expressing collections, such as lists and tuples, which we learn next. When we talk about concurrency and fault-tolerance via processes, we will also discuss ports, pids, and references, but that will come on later chapters. Let's move forward.
|