Let's start from scratch. From this point onward, I'll assume you're already familiar with both agent-based modelling as a method[1] as well as basic familiarity with Julia as a programming language, i.e. you have it installed and understand the fundamentals of its syntax[2]. This series on agent-based modelling in Julia will be structured somewhat analogously to the regular modelling process that every agent-based modeller knows in some form or another: We start from the description of a situation, proceed with formulating a research question, and then identify relevant aspects. Afterwards we will formalise them as agent variables and model parameters and then put them into code.
You might wonder why we don't immediately go to the coding part and just make some exemplary statements about agent behaviour and model evolution. Surely, that would be easier to just explain things, right? While certainly true at its core, I think there are two important points in favour of the approach chosen for this series:
- It's harder to remember facts than it is to remember stories. Providing yet another bunch of blogposts about the technical details of Julia will likely just lead to readers skipping to the parts that they are currently having problems with. It would be somewhat redundant with well-written documentation. Instead, telling a story over the course of this series will hopefully increase the amount of things you can remember in the long run.[3]
- Modelling is always a subjective process and depends heavily on the choices made by the modeller. It is only an opinionated representation of the real world and its interdependencies. What I will present to you is just my interpretation of the situation. If you want to do things differently, you're fully able to do so.
Model background
A day in the life
Our starting point will be a very simple example of two agents with heterogeneous properties that interact with each other on a regular basis. We will extend on this initial description over the course of the series, introducing new aspects, and making assumptions about how the world that they live in works.
Somehow we need to differentiate between our two agents, so we give them names: Sally and Richard. Their names, however, aren't their only distinctive features. Humans are very diverse, hence we will proceed with a short description of each of them.
Richard is 33 years old and loves his job as a fisherman. Very early in the morning he happily rows out with his boat for a few hours of line fishing. Some days he's lucky and catches multiple fish per hour (sometimes even up to three) while on other days it seems as if he's cursed by Glaucus and the fish just won't bite. In the end this means that Richard's daily fishing efforts can take anything between one and five hours depending on his luck. After getting back home, he proceeds with different tasks for another one to three hours. Sometimes he takes care of maintenance work on his boat or fishing rod. When he doesn't need to repair anything, he reads up on newly published reports regarding the overfishing of the seas and which types of fish can still be caught with good conscience. His fishing work is intense but he also has a decent amount of freely assignable time over the day which is less stressful and energy-sapping. In the evening he likes to knit sweaters and listen to records of his favourite Death Metal band.
Sally is 56 years old and loves her job as a baker. However, she has to deal with it all day, every day. It is a bit tiresome to get up early, prepare the dough, heat the oven, and so on but it allows her to produce a steady output of bread every day with the only limiting factor being the size of her oven which can fit up to 20 loaves at once. She has to do these steps, no matter how much bread she wants to produce, thus, her overall workload only changes slightly each day and her working times regularly end up between eight and ten hours, depending linearily on the amount of bread she bakes. When her work day is over, she likes to go trailrunning in the nearby mountains or to just enjoy having a PiΓ±a Colada in front of favourite bar (owned by Willy, a retired Barista from New York) near the oceanside.
As time goes by and they go about their work, their stocks in bread or fish increase and also decrease because they need to eat. In the short run, we may want to assume that one can live on bread and fish alone. But to be quite honest, nobody would like to eat only bread or fish every day. So both Sally and Richard have a deep desire to get some of the goods that only the other person produces.
Simply put, they have a regularly occurring need for both bread and fish that has to be satisfied. If this need is neglected over a longer period of time, their work productivity will decline. To avoid this undesirable situation, they take the opportunity to trade with each other every three days whenever they meet at the ocean side for a nightcap. When they do so, Richard trades with Sally for a few loaves of bread and she gets a few fish from Richard in return. As a starting point, let us assume that one fish is worth the same to them as one loaf of bread, i.e. they are willing to trade their goods in a 1:1 ratio, probably just because they know and like each other.
Question? Which question?
Although it is nice to just start coding and see whatever comes out of our endeavours, an agent-based model should aim to answer a specific question. We will now formulate one for the purpose of this exercise:
What is the minimum amount of work that Sally and Richard can do every day while still being able to satisfy their craving for fish and bread through mutual trade?
Given this question, we can now try to identify the aspects from the description above that are relevant to our planned ABM. Maybe first give this a try yourself before reading on. Which pieces of the provided information are interesting for us? How many agents will our model be comprised of? What are important environmental factors that have to be reflected in the model? Which information is incomplete and requires us to make assumptions?
One interpretation of the model's background described above could look as follows:
- Our model will contain two agents, Sally and Richard. What they do in their free time is unimportant for our base model for now and only their jobs and their working times are relevant.
- How many hours they work per day determines the amount of food they need to eat each day.
- Sally works 8-10 hours each day (medium exertion) and can freely decide to bake 1-20 loaves of bread each day.
- Richard goes fishing for 1-5 hours (high exertion) and does other work for 1-3 hours (low exertion). During his fishing hours, Richard has an independent chance to catch 0-3 fish each hour.
- Both agents have variables tracking their stock of fish and bread as well as their hunger for each good.
- If their aggregate hunger level stays above a certain threshold for a prolonged period of time, agents reduce their work efforts, thus producing less bread and spending less time on fishing.
- There is an opportunity at the end of every third day to trade their current stocks in fish and bread with each other.
- Most of what are about processes and do not need to be stored as agent variables. We can therefore identify five distinct agent variables to keep track of:
job, pantry_fish, pantry_bread, hunger_fish, hunger_bread
.
Wait, wasn't this supposed to be a Julia tutorial?
Phew. Those were a lot of words about plenty of things and so far not a single thought was wasted on actual code. To avoid losing some precious readers, this is probably as good a time as any to finally start talking about Julia.
A few preparations
Feel free to create a new folder for this series to code along to the examples (call it whatever you want, e.g. "julia-loves-abm"). Open your terminal, navigate to this folder and start Julia from there by running julia
. Now execute the following:
julia> using Pkg
julia> Pkg.activate(".")
Activating new project at `~/Code/julia-loves-abm`
Congratulations, you've just mastered the highly sought after skill of creating a fresh Julia project environment with the same name as the folder you've started the Julia instance from. You should activate this environment every time you continue working on this project as it will allow Julia to know which packages you have installed and which dependencies or versions should be respected.
As is so often the case in programming, there's also another way to do the same thing. If you just type ] in the julia>
prompt you will enter the built-in pkg>
mode:
(@v1.7) pkg>
This is very easy and approachable as you do not need to run using Pkg
before doing this. The pkg>
mode is just always available to you. It allows you to quickly use some commands like activate .
(to change environment where dot refers to the current working directory) and status
(to check installed packages and their versions):
(@v1.7) pkg> activate .
Activating project at `~/Code/julia-loves-abm`
(julia-loves-abm) pkg> status
Status `~/Code/julia-loves-abm/Project.toml` (empty project)
As you can see, we've switched from the base environment @v1.7
to a newly created one which is automatically assigned the name of our current working directory julia-loves-abm
. The status
command tells us that the project environment is currently empty, meaning that we haven't added any extra functionality through Julia packages. For now, you can safely ignore most of the details about environments but just keep in mind that they exist as they will be of great use to us at a later stage.
Creating agents
Let's remember our story from above. We have two people, so it seems straightforward to initialise two variables called sally
and richard
that represent them.[4] A little bit earlier we've looked at the relevant aspects of their everyday lives, telling us what to formalise as agent variables. There are a few unifying features about them, for example they each have a job which allows them to produce a certain kind of food (fish or bread respectively). Since it's important for what happens in the model, it needs to be reflected in the code which we could simply do by using a String
(a sequence of characters like letters and whitespace) describing their job:
julia> sally = "Baker"
"Baker"
julia> richard = "Fisher"
"Fisher"
Now that's already an (admittedly pretty crude) representation of what our two agents are. Whenever we call one of the variables, its evaluation will tell us the agent's job. But we also want to keep track of their stock of food and their hunger levels. This confronts us with a decision about a more appropriate data structure to use for storing all the different kinds of information about Richard and Sally.
Lined up
Maybe the simplest approach would be to use a simple collection like an Array
or a Tuple
with the values of the agent variables in it. Let's assume that Richard and Sally are both not hungry in the beginning of our simulation and that they each have a starting stock of 10 fish as well as 10 loaves of bread.
julia> sally = ("Baker", 10, 10, 0, 0)
("Baker", 10, 10, 0, 0)
julia> richard = ("Fisher", 10, 10, 0, 0)
("Fisher", 10, 10, 0, 0)
It becomes immediately obvious that this approach is not very practical. While we can reason about what "Fisher"
and "Baker"
stands for, it is pretty hard to know what exactly the numbers mean without having the verbal description from above at hand.
Give me names
Indeed, it would be nice if we could label all the entries so that it is clear what they mean. We might want to opt for a NamedTuple
instead which allows to provide names to the fields.
julia> sally = (job = "Baker", pantry_fish = 10, pantry_bread = 10, hunger_fish = 0, hunger_bread = 0)
(job = "Baker", pantry_fish = 10, pantry_bread = 10, hunger_fish = 0, hunger_bread = 0)
julia> richard = (job = "Fisher", pantry_fish = 10, pantry_bread = 10, hunger_fish = 0, hunger_bread = 0)
(job = "Fisher", pantry_fish = 10, pantry_bread = 10, hunger_fish = 0, hunger_bread = 0)
Ah, much better. Now we don't have to remember the order of the data in the Tuple
but can easily access the agent data by fieldnames:
julia> sally.job
"Baker"
julia> richard.pantry_bread
10
Again, sally.job
is a String
like in our initial approach of just assigning a string literal to each agent which described their job. However, richard.pantry_bread
is also easily accessible now and shows us that he currently owns 10
loaves of bread.
To see all the fieldnames and the types of their values, we again call the typeof
function on one of the agents:
julia> typeof(richard)
NamedTuple{(:job, :pantry_fish, :pantry_bread, :hunger_fish, :hunger_bread), Tuple{String, Int64, Int64, Int64, Int64}}
This tells us that richard
is now depicted as a NamedTuple
comprised of the fields (:job, :pantry_fish, :pantry_bread, :hunger_fish, :hunger_bread)
and their values Tuple{String, Int64, Int64, Int64, Int64}
. It is fundamentally the same as in the Tuple
case before but enhanced with the information about which value means what.
Change is inevitable
But as we know, data in ABMs regularly change over the course of the simulation. And since a NamedTuple
is immutable[5] by nature, this seems to be a really bad choice as a data structure for our agents. If we attempt to change one of the values, we get an error:
julia> sally.pantry_bread = 9
ERROR: setfield!: immutable struct of type NamedTuple cannot be changed
Stacktrace:
[1] setproperty!(x::NamedTuple{(:job, :pantry_fish, :pantry_bread, :hunger_fish, :hunger_bread), Tuple{String, Int64, Int64, Int64, Int64}}, f::Symbol, v::Int64)
@ Base ./Base.jl:43
[2] top-level scope
@ REPL[37]:1
So to change the value of any of the fields, we would need to reuse some of the old values and insert the new value in the appropriate field.
julia> sally = (sally..., pantry_bread = 9)
(job = "Baker", pantry_fish = 10, pantry_bread = 9, hunger_fish = 0, hunger_bread = 0)
While this approach does have its advantages (e.g. no accidental changes in agent-related data), it can quickly get a bit cumbersome to always overwrite the old variable with a new one. So it's probably better to completely ditch the idea of using NamedTuple
s then.
The next best idea that might come to mind would be to use a Dict
instead:
julia> sally = Dict(
:job => "Baker",
:pantry_fish => 10,
:pantry_bread => 10,
:hunger_fish => 0,
:hunger_bread => 0
)
Dict{Symbol, Any} with 5 entries:
:pantry_fish => 10
:hunger_bread => 0
:job => "Baker"
:hunger_fish => 0
:pantry_bread => 10
julia> richard = Dict(
:job => "Fisher",
:pantry_fish => 10,
:pantry_bread => 10,
:hunger_fish => 0,
:hunger_bread => 0
)
Dict{Symbol, Any} with 5 entries:
:pantry_fish => 10
:hunger_bread => 0
:job => "Fisher"
:hunger_fish => 0
:pantry_bread => 10
The keys and values of dictionaries can be comprised of just about any type that you can think of. Hence you have to take care to use Symbol
s as keys of the dictionary (or maybe String
s, if you prefer that), because unlike NamedTuple
s, Dict
s don't just automatically convert the fieldnames into Symbols (which always start with a colon :
).
To retrieve data from our agents, we use the regular syntax for dictionaries:
julia> sally[:job]
"Baker"
julia> richard[:pantry_bread]
10
Changing the value of an agent variable is now as easy as writing:
julia> sally[:pantry_bread] -= 1
9
julia> sally
Dict{Symbol, Any} with 5 entries:
:pantry_fish => 10
:hunger_bread => 0
:job => "Baker"
:hunger_fish => 0
:pantry_bread => 9
To see the current keys of a dictionary, you can just start typing the name of the dictionary followed by [:
and then press Tab twice[6]:
julia> sally[:
:hunger_bread :hunger_fish :job :pantry_bread :pantry_fish
Again, there are often multiple ways to do the same thing when coding and none of them is more or less correct than the other. Another way to retrieve the current set of keys of a dictionary is to call the keys
function on it:
julia> keys(sally)
KeySet for a Dict{Symbol, Any} with 5 entries. Keys:
:pantry_fish
:hunger_bread
:job
:hunger_fish
:pantry_bread
While Tab completion is a nice way to interactively explore the current state of your agents, having a KeySet
also allows you to go through the agent variables one after another in a programmatic way. Which of these approaches you will choose heavily depends on the current use case you are facing. Generally though, it's just good to have some options available.
Work smart, not hard
Luckily we currently only have two agents in our model, so it's not really that problematic to create them one by one. In bigger models, however, we often want to create a relatively high number of agents and that could then quickly get a bit tedious to do one by one. Here's a general word of advice about coding:
If you have to write something repeatedly, there's probably a better way to do it. :)
Indeed, we can build a custom type for our agents, allowing us to predefine the structure of the data that we want to store. So instead of having to spell out the agent variables each and every time we want to create a new agent, we can just tell Julia to use our custom struct to lay out and label the data that we provide to it. The keyword to create such a data structure (also referred to as a composite type) is struct
but we also have to prepend it with mutable
to make sure that we are able to change its values (see the problem about immutability described above).
julia> mutable struct Agent
job::String
pantry_fish::Int
pantry_bread::Int
hunger_fish::Int
hunger_bread::Int
end
As you can see, we also had to explicitly define the types for each field of the struct
as these are not automatically inferred like when creating a NamedTuple
or a Dict
. Without going into detail about the variety of types, we just use what we already know. From our previous attempt to create our agents as NamedTuple
s, we could see that the values we provided to it have been interpreted as String
and Int64
types. In the definition of the struct
above, we've simply used this knowledge and also changed Int64
for the more generalised form Int
.[7]
We can now initialise Richard and Sally as two variables of our custom-made and highly specific Agent
type:
julia> richard = Agent("Fisher", 10, 10, 0, 0)
Agent("Fisher", 10, 10, 0, 0)
julia> sally = Agent( "Baker", 10, 10, 0, 0)
Agent("Baker", 10, 10, 0, 0)
This newly created type also allows us to directly access the agent variables. We can do this just like we did it in the case of a NamedTuple
:
julia> richard.pantry_fish
10
julia> sally.pantry_bread
10
Changing the values is also possible and just as easy as it was in the case of using a Dict
:
julia> richard.pantry_fish -= 1
9
julia> richard
Agent("Fisher", 9, 10, 0, 0)
One of the major downsides of defining our own composite type is that we have to restart our Julia session to change anything about it. Say we would like to add a new field to our agent struct that describes in one convenient number how satisfied they currently are with their life:
julia> mutable struct Agent
job::String
pantry_fish::Int
pantry_bread::Int
hunger_fish::Int
hunger_bread::Int
life_satisfaction::Int
end
ERROR: invalid redefinition of constant Agent
Stacktrace:
[1] top-level scope
@ REPL[2]:1
This might seem inconvenient at first glance but in reality it doesn't happen all too often. Indeed, we have been smart modellers and took enough time to first deliberate about what we actually want to model and which agent variables are important for answering the underlying question of our model.
Great! Now that we've settled on a convenient way how to represent our agents in code, our next step will be dealing with the tasks that our agents do every day and how they affect the filling of their pantries and their hunger levels.
[1]: Explaining agent-based modelling in detail is way out of the scope of this series. If you are unfamiliar with the method but generally interested in learning more about it, you may want to start reading this text book or this introductory article or watch this video series by Complexity Explorer.
[2]: Should you find yourself wondering how to install Julia, it's really as simple as downloading it from the Julia language website and running the installer. There are also other ways to setup your Julia installation but the aforementioned method works just as fine as anything else. Just to provide an example, my preferred way is Juliaup as a platform-independent tool to manage multiple Julia versions and keep them up to date. Once you're all set up, you might want to check out the Getting Started section of the official Julia documentation or follow an introductory video course to learn about syntax and basic usage. Done that? Let's get back to ABM stuff then. :)
[3]: I sincerely believe that teaching by telling a simple story is very approachable for most people. Of course there's room for everything on the world wide web and no approach to learning is inherently better or worse than any other. It very much depends on your preferred style of learning, your previous knowledge, and maybe even your current state of mood. For example, if you already know how to work with Julia and are already very knowledgeable in the arcane arts of agent-based modelling, you might as well just skip this introductory series and go straight to the documentation of Agents.jl and read/work through the well-written tutorial and examples.
[4]: It's the Julian way to use lower case names with underscores for our variables. This is often referred to as snake_case. There are some more style recommendations that established themselves over time in the Julia community which we will try to adhere to as closely as possible. If you want to read up on the idiomatic Julia coding style, have a look at the official Julia style guide. In the end, however, it doesn't matter too much as long as you don't have to share your code with others. In the latter case it is highly recommended to try and stick to a unified coding style as it allows others (not only colleagues but maybe even strangers at some point) to more easily read and understand your code and potentially comment on it, extend it, fix it, et cetera.
[5]: Immutability means that a variable cannot be changed after its creation. This applies to both its value(s) and its composition. Here's the exemplary case of trying to modify one of the elements in a Tuple
and attempting to add a new element to it:
julia> t = (1,2,3)
(1, 2, 3)
julia> t[1] = 1
ERROR: MethodError: no method matching setindex!(::Tuple{Int64, Int64, Int64}, ::Int64, ::Int64)
Stacktrace:
[1] top-level scope
@ REPL[74]:1
julia> t[end+1] = t[end] + 1
ERROR: MethodError: no method matching setindex!(::Tuple{Int64, Int64, Int64}, ::Int64, ::Int64)
Stacktrace:
[1] top-level scope
@ REPL[73]:1
However, this does not mean that we cannot redefine the variable t
to refer to something else:
julia> t = 1
1
[6]: This Tab completion also works with a lot of other things, for example NamedTuple
s. Just write its variable name followed by a single dot:
julia> nt = (a = 1, b = 2)
(a = 1, b = 2)
julia> nt.
a b
[7]: Although definitely not necessary at this stage, you might be interested in what all these types mean. Let's dive a bit deeper. You might wonder why it is called Int64
and not just Number
, Real
or Integer
. Simply put, every Integer
is a Real
but not every Real
is an Integer
. Julia provides us with an easy way to find out about this type hierarchy:
julia> supertypes(Integer)
(Integer, Real, Number, Any)
To go up through the hierarchy, you can read this tuple of types from left to right. If you want to explore it in the opposite direction, there's also a way to do this:
julia> subtypes(Real)
4-element Vector{Any}:
AbstractFloat
AbstractIrrational
Integer
Rational
As you can see, Integer
is necessarily a subtype of Real
. We can also programmatically test this with a specific syntax in Julia by writing:
julia> Integer <: Real
true
Now an Int64
is a specific subtype of a signed Integer
number with a size of 64 bit.
julia> subtypes(Integer)
3-element Vector{Any}:
Bool
Signed
Unsigned
Being Signed
means that the Integer uses a bit of memory to store its mathematical sign. This means the resulting number can take both positive and negative values.
julia> subtypes(Signed)
6-element Vector{Any}:
BigInt
Int128
Int16
Int32
Int64
Int8
There's no general type called Int
in here but only types with predetermined size, e.g. 8 or 64 bit. If we use Int
as a type for our Agent
struct, Julia asserts that we want the possible size of the Int
to be as big as possible. Thus, it is automatically determined by the underlying architecture of our computer (most modern computers are built on 64 bit). When we create an instance of our Agent
struct, those fields typed as Int
will indeed be of type Int64
. This is nice to take into account for the hypothetical case of somebody with a 32 bit computer trying to run our model which is then possible precisely because we didn't restrict the size of the Int
too much (e.g. to always use Int64
).
And while all of this is actually very interesting, we luckily don't have to worry about it in greater detail for now. If you still want to read more about Julia's type system, have a look at the well-written section of the official docs. Let's get back to work on our ABM. :)
Top comments (8)
Hey @fbanning I have not had a chance to read through this yet but it would be really helpful if you can make the title stand alone so that when I share this out on Twitter and stuff the title gives enough context as to what this article is about
Hey Logan, thanks for your suggestion. I've used the Forem functionality of creating a series to connect the posts to each other. Do you think it's better for me to change the title instead of just mentioning that this is "part 2 of a series on agent-based modelling with Julia" ?
Edit: It's also possible to just share a link to the series itself (forem.julialang.org/fbanning/series/1), so maybe that's even better?
Yeah, if I go to share this, it will show up with just the current title which doesnβt make much sense outside the context of this series. A stand-alone title would really help get more eyes on this.
Yeah, I see where you're coming from. Will definitely keep that in mind for the future and maybe change the style of the post titles. Sadly (luckily?) the links will stay the same if I change the titles, so there's that. For now, I'll just use the link to the series itself in every promotion of a new post so people can easily see what's already available there. :)
I've changed the title to "Julia β₯ ABM #1: Starting from scratch" which I hope makes things a bit clearer.
You might want to check out Revise.jl to alleviate the problem when redefining a struct. It's sort of like a hot reload where you don't need to restart the Julia REPL if you modify things like structs.
Thank you very much for your comment. Revise.jl is indeed a great tool that should be in just about every Julia user's toolkit. :) While I'm personally aware of the package, load it automatically with my
startup.jl
file, and use it every day, it's certainly a bit too advanced of a topic for now to explain it to somebody who just started to learn using Julia for ABMs. In this series, I've deliberately opted for a slow and step-wise approach to teaching, therefore I've spared out those more intermediate level topics for now. That being said, at a later point in this series there will be an introduction to using Pkg.jl to leverage the power of Julia's great package ecosystem. Most likely that will also be the point at which I recommend a few nice helper packages from the broader community. :)