Notes and exercises while reading through Programming Elixir by Dave Thomas.
So far, solving the exercises in the book is pretty straightforward because the prompts are written out in a way that make them easy to turn into code. The challenge when writing one's own functions is to frame the problem in a functional manner, to distill the purpose of down to its "functional" description. Once you've done that, the syntax flows more easily.
Elixir helps by lending itself to small functions that do one thing.
Error messages in Elixir are great. The error messages don't just say 'fail', but also give a recommendation on how you might resolve the unexpected error.
Here's a lengthy example from this chapter:
warning: definitions with multiple clauses and default values require a
function head. Instead of:
def foo(:first_clause, b \\ :default) do ... end
def foo(:second_clause, b) do ... end
one should write:
def foo(a, b \\ :default)
def foo(:first_clause, b) do ... end
def foo(:second_clause, b) do ... end
Named functions live inside modules.
# named_function/1
defmodule ModuleName do
def named_function(n) do
# do something / return something
end
end
iex(1)> c "times.exs"
[Times]
iex(2)> Times.double(4)
8
The name and number of parameters uniquely identify a function. function1(x)
and function1(x, y)
are two completely different and independent functions. Refer to them as function1/1 and function1/2.
Use do:
for single line functions and do..end
for multi-line functions.
Extend the Times module with a triple function that multiplies its parameter by three. Add a quadruple function. (Maybe it could call the double function....)
See times.exs.
Multiple definitions of the same function allow us to start by implementing the simplest case with a known return value. Then use that case recursively in other definitions.
See factorial1.exs.
See recursion.exs.
- Guard classes are used to decide which function to use based on a
when
clause. - Pattern matching first, then guard clause.
defmodule Guard do
def what_is(x) when is_atom(x), do: IO.puts "#{x} is an atom"
def what_is(x) when is_list(x), do: IO.puts "#{inspect(x)} is a list"
def what_is(x) when is_number(x), do: IO.puts "#{x} is a number"
end
iex(1)> Guard.what_is(:ok)
ok is an atom
:ok
- syntax:
param \\ value
defmodule Example do
def func(p1, p2 \\ 2, p3 \\ 3, p4) do
IO.inspect [p1, p2, p3, p4]
end end
iex(1)> Defaults.func("a", "b")
["a", 2, 3, "b"]
iex(2)> Defaults.func("a", "b", "c")
["a", "b", 3, "c"]
iex(3)> Defaults.func("a", "b", "c", "d")
["a", "b", "c", "d"]
If the number of provided arguments matches the number of required params, then the provided arguments are assigned, left to right, to the required params. See iex(1) above.
If the number of provided arguments is greater than the number of required params, then the number of optional params to be populated is determined and the params are assigned left to right using only the number of optional params needed. See iex(2) and iex(3) above.
This makes sense because it's all about pattern matching. The assignment process looks to match the provided arguments with the function's parameters.
An aside: Note that Ruby does the same.
> def func(p1, p2=2, p3=3, p4)
> puts [p1, p2, p3, p4].inspect()
> end
=> nil
> func("a", "b")
["a", 2, 3, "b"]
=> nil
> func("a", "b", "c")
["a", "b", 3, "c"]
> func("a", "b", "c", "d")
["a", "b", "c", "d"]
If the number of provided arguments is less than the number of required arguments or greater than the number of required plus optional arguments, then there is no match and you get an UndefinedFunctionError
error.
** (UndefinedFunctionError) function Defaults.func/1 is undefined or private.
** (UndefinedFunctionError) function Defaults.func/5 is undefined or private.
Ruby:
2.0.0-p353 :043 > func("a")
ArgumentError: wrong number of arguments (1 for 2..4)
Wrote a guesser function that finds a number by guessing halfway between the low and high of a range. Continues to adjust the number and the range until the guess matches the number.
My solution is chop.exs.
Cool!
Instead of:
people = DB.find_customers
orders = Orders.for_customers(people)
tax = sales_tax(orders, 2016)
filing = prepare_filing(tax)
filing = DB.find_customers
|> Orders.for_customers
|> sales_tax(2016)
|> prepare_filing
- provide namespaces, like so:
defmodule Mod do
def func1 do
IO.puts "in func1"
end
def func2 do
func1
IO.puts "in func2"
end
end
Mod.func1
Mod.func2
- brings a module’s functions and/or macros into the current scope
- so that you don't need to repeat the module name over and over.
import List, only: [ flatten: 1, duplicate: 2 ]
- make for shorter names
alias My.Other.Module.Parser, as: Parser
...
Parser.parse()
@name value
- top-level module-level only, functions do not have attributes
- think of them as constants used for configuration and metadata
see: attributes.exs.