The Nature of Code introduces many concepts to simulate real-world phenomena with code. Here, I follow the book to learn about these concepts.
The notes which follow are my own. The demos are written in Lua and Love2D.
Randomness provides a first, rudimentary way to simulate real phenomena.
A random function returns a number in a range with the same probability as any number in the same range. The output isn"t truly random, but pseudo-random, whereby the function creates a series of numbers and returns one of them. The sequence repeats itself, but over a long stretch of time.
To move an entity at random, there exist several strategies:
-
with
math.random(4)
and a chain ofif
statements, move the entity in one of the four cardinal directions -
with
math.random(3)
, modify the coordinates to have the entity move or stand still -
with
math.random()
, modify the coordinates with a floating point number
Please note:
-
Lua is 1-indexed. In light of this,
math.random(4)
returns a number in the[1,4]
, andmath.random(3)
returns a number in the[1,3]
range -
math.random()
returns the same sequence of random numbers unless you first set a random seedfunction love.load() math.randomseed(os.time()) end
It is possible to use
love.math.random
, a function which is automatically seeded by Love2D, but I chose the non-seeded version to stress how the function is not truly random.
The probability of a single event is given by the number of outcomes representing the event divided by the number of possible outcomes. The probability of multiple events occurring in sequence is obtained by multiplying the single events. The concept is useful to describe the random functions, but also the distribution of other functions.
With the random function, you can obtain a certain probability with at least two methods:
-
initialize a bucket, a container with a set of options and pick at random from the set
-
ask for a random number in the
(0, 1)
range and use the value to execute an option with the given probability
Please note:
-
the method using probabilities computes the cumulative probability of each option. In the code, the for loop is a convenience alternative to a series of
if
statementsif random < 0.5 elseif random < 0.55 elseif random < 0.85 elseif random < 1 end
A normal, or Gaussian, distribution returns a number starting from two values: a mean and a standard variation. The idea of the distribution is that numbers are more likely to approach the mean than deviate from it. How often the numbers distance themselves from the mean is described by the standard deviation. With a small deviation, the numbers gather around the mean. Opposedly, and with a large deviation, the numbers scatter away from the central value.
For the normal distribution, it is useful to remember the following:
-
given a population and a normal distribution, 68% of the observations fall in the range of the standard deviation, 98% in the range of twice the standard deviation and 99.7% in the range of thrice the same value. This is often described as the 68,98,99.7 Rule
-
to compute the standard deviation with a mean and a set of observations, consider the value of each observation, subtract the mean and square the result. Sum this measure for all observations to gather the variance. The standard deviation is the square root of the variance.
-
a function
f(x)
returns a numbery
in a normal distribution with meanmu
and standard deviationsigma
with the following formulaf(x) = 1 / (sigma * (2 * math.pi) ^ 0.5) * e ^ (- 1 / 2 * ((x - mu) / sigma) ^ 2)
e
describes Euler"s number, and is approximated to2.71828
. -
the tallest point of the bell curve depends on the standard deviation, and is computed with the following formula
y = 1 / ((2 * math.pi) ^ 0.5 * sigma)
Please note:
-
in the demo the normal function is used to generate an assortment of numbers with mean
0
and standard deviation1
. These values are then used to draw a line, describing the associated bell curve, and move theMover
entity -
the demo introduces a mapping function similar to the method detailed in the processing library
To fit the needs of a simulation, you can customize a distribution in several ways. The Probability
folder already introduces two possibilities, with the methods picking a number from a given set, or using the cumulative probability. Here, a custom distribution is built with the following algorithm:
-
pick a random value
-
assign a probability to the value
-
pick a second random value
-
if the second random value is less than the probability, accept the first pick
-
else, repeat the process
This approach means that values with a greater probability are more likely to be accepted. The custom nature of the distribution comes from the way the probability is then computed.
It helps to consider the probability as the y
value for a function in an (x, y)
plane. When the probability is equal to the random value, as in the demo, the relationship is linear. When the probability is instead equal to the value squared, the relationship is squared.
Please note:
-
in the demo, the value is picked in the
(0, 1)
range, with a tendency for higher numbers. The value is finally updated to change the direction at random.if math.random() > 0.5 then x = x * -1 end -- repeat for the `y` coordinate
This is to avoid having the
Mover
entity move outside of the window too rapidly.
The Perlin noise function allows to create a sequence of numbers connected to each other, with the goal of providing smooth random values. You pick numbers from the sequence, and the distance between the numbers dictates the difference between the two. The greater the distance, the more likely the numbers will differ. The smaller the offset, the more likely the numbers will resemble one another.
While it is possible to create a function implementing the logic of Perlin noise, Love2D provides a similar functionality in love.math.noise
.
The function returns a sequence of numbers in the (0,1)
range and accepts multiple arguments, to create noise in multiple dimensions. In one dimension, each number is related to the one coming before and after it. In two, it is connected to the numbers representing the possible neighbors in the (x,y)
plane. In three, the neighbors considering a third dimension (z)
as well.
Please note:
-
the demo works similarly to
Normal distribution
, by plotting the numbers with a line and update theMover
entity to follow the line"s coordinates -
using the same offset returns the same noise value. This explains why the for loop begins at an arbitrary offset, chosen at random
local offset = math.random(OFFSET_INITIAL_MAX) for i = offset -- continues -- do end
-
OFFSET_INCREMENT
describes the distance between successive points.OFFSET_INCREMENT = 0.02
The smaller the value, the smoother the line.
-
getNoiseNumbers
creates a series of numbers to plot the line and update theMover
entity.getNoiseBackground
instead produces a grid of numbers to color the background with a makeshift texture; this last function is used to showcase howlove.math.noise
works with two arguments and dimensions
Vectors as introduced in the book are euclidean vectors, entities with a magnitude and a direction. They are introduced in the context of a plane with two dimensions, x
and y
, but fundamentally, they work in the same manner with additional dimensions.
Think of a vector with two components as an arrow. The length of the arrow describe its magnitude, while the angle relative to an axis its direction. A vector describing the position of a particle details where to position the object from the origin. A vector describing the velocity dictates where to move the same particle.
Please note:
-
Vector.lua
is used to introduce the vector entity with a table -
the demo in the
Vector
folder shows two vectors, a position vector centering a circle in the middle of the screen, and a velocity vector radiating from the starting point. The velocity vector is finally updated using the coordinates of the mouse cursor
Vectors follow specific rules to compute mathematical operations.
-
add vectors by considering the sum of the respective components
-
subtract vectors by decreasing the components of the first vector with the components of the second
-
multiply a vector with a scalar by multiplying each component with the same measure; this operation is useful to scale the vector
-
divide by a scalar by dividing each component; be sure that the scalar is different from 0
-
compute the vector"s magnitude; using pythagorean theorem,
a^2 + b^2 = c^2
, the magnitude represents the length, the distance between two points considering thex
andy
component:m = (vector.x^2 + vector.y^2)^2
-
normalize a vector by dividing the vector by its magnitud. With this operation, you obtain a vector with length 1, a unit vector; the operation is helpful to have a unit vector with the same direction as the first vector. From this point you can scale the vector to any arbitrary length, by further multiplying the unit vector with the desired magnitude
-
limit a vector to a given magnitude; the idea is to here compute the magnitude, and if greater than the input value proceed to scale down the vector to said magnitude. In practice, this operation is achieved by normalizing the vector and scaling the same entity with the input magnitude
Please note:
-
Vector.lua
is updated to compute different mathematical operations on the vector itself. Be careful that these methods (add, subtract, multiply and divide) modify the original entity -
LVector.lua
is introduced as a rudimentary version of the Processing library, and defines several methods to compute roughly the same mathematical operations. The key difference is that the methods do not modify the input vector(s), but return a new entity altogether
With two vectors describing the position and velocity, it is possible to move an object at a constant rate.
Please note:
-
in the demo, the location is updated with the vector describing the velocity. The velocity is however and first scaled to consider
dt
, delta time.self.position:add(LVector:multiply(self.velocity, dt))
This is specific to the engine Love2D, and allows to move the particles irrespective of the frame rate
With three vectors, it is possible to move an object with an accelerated rate, speeding up or slowing down. The idea is to update the position with the velocity, and the velocity with the acceleration.
self.position:add(self.velocity)
self.velocity:add(self.acceleration)
With this setup, the goal of the simulation is to then set a particular acceleration. This value can be set arbitrarily, as with a constant or random value, or following actual physics, as with of gravity or wind strength.
Please note:
- in the
Acceleration
folder you find two demos:Constant
setting a positive or negative acceleration with a particular key press, andMouse
, updating the acceleration to have the entities move toward the mouse cursor
The goal is to adapt the concept of forces and Newtonian physics to the simplified environment introduced with vectors.
In the simplified environment, a force is described as a vector which causes an object with mass to accelerate.
Newton"s first law, arguing that an object at rest stays at rest and an object in motion stays in motion, is adapted by saying that the vector describing the velocity stays constant. The only way for an object to stop, to reach an equilibrium is for the velocity to be affected by a series of forces which cancel its magnitude.
Newton"s third law, detailing an action/reaction pair for any force, is partly incorporated in the environment by occasionally including a force contrasting the original one.
Newton"s second law, providing a formula to compute a force on the basis of mass and acceleration, is finally essential to the simulation. This law states that the force is equal to mass times acceleration.
-> ->
F = m * a
In a first approximation, and assuming a mass equal to 1, an immediate way to apply the force is to therefore set the acceleration to the force itself
function Mover:applyForce(force)
self.acceleration = force
end
Please note:
-
in the demo for the
Newton"s law
folder aMover
entity is initialized without velocity, nor acceleration. The acceleration vector is then modified inlove.load
by applying a forcelocal force = LVector:new(fx, fy) mover:applyForce(force)
The previous solution works, but only when a single force is applied. With multiple forces, only the last one is incorporated in the acceleration vector. The implementation is therefore modified to consider the effect of a force as cumulative (force accumulation)
function Mover:applyForce(force)
self.acceleration:add(force)
end
Be careful that it is necessary to reset the acceleration vector. In this manner, the object considers the forces available in the specific frame, and not every force accumulated in the simulation.
self.acceleration:multiply(0)
Please note:
-
in the demo the
Mover
entity is updated to move downwards as subject to a force of gravity. When the mouse is pressed then, another force is applied to move the object horizontally as welllocal gravity = LVector:new(0, GRAVITY) mover:applyForce(gravity) if love.mouse.isDown(1) then local wind = LVector:new(WIND, 0) mover:applyForce(wind) end
In a slightly more elaborated construct, a force is weighed by the object mass.
function Mover:applyForce(force)
self.acceleration:add(LVector:divide(force, self.mass))
end
It is important to note, however, that forces like gravity already incorporate the mass in their value. For these forces, it is necessary to remove the mass"s influence.
local gravity = LVector:new(0, GRAVITY)
mover:applyForce(LVector:multiply(gravity, mover.mass))
Please note:
-
in the demo
love.load
initializes a series ofMover
entities with a random mass. The mass is also used to change the radius of the circle -
LVector:divide(force, self.mass)
is used in place offorce:divide(self.mass)
to avoid modifying the input force. In the specific demo this precaution is however not necessary, as the force is computed every frame
In a simulation, create a force with an arbitrary value, or following the guidance of an actual formula. In this last instance, the simulation needs to compute the magnitude and direction of the force vector from a given set of values.
Friction is applied on a moving object to progressively reduce its velocity. The actual formula computes the vector by considering the unit vector describing the velocity, a coefficient of friction (mu
), and the magnitude of the normal force.
-> ^
F = -1 * mu * ||N|| * velocity
In a first approximation, it is possible to simplify the force by considering its direction and magnitude. In terms of direction, friction has a direction opposite to the velocity vector. Notice how the unit vector is multiplied by -1
.
-> ^
F = -1 * velocity
In terms of magnitude, the force is scaled with a constant value.
-> ^
F = -1 * velocity * c
A more elaborate simulation would try to compute the normal vector, and its eventual magnitude, would incorporate the coefficient according to the surface creating the friction, or again the normal force and its magnitude, but in the approximation, it is enough to scale the vector with a constant. By changing the constant, the simulation is able to then describe a surface with higher/smaller friction.
Please note:
-
the
Mover
entity is simplified to have a mass of1
and move only horizontally -
the demo is simplified to only show the force of friction. There is no gravity, nor wind, but a force increasing the velocity when the mouse is being pressed
-
LVector
is updated to include a method which returns a copy of the input vector. The function is useful to normalize a copy of the velocity vector, without modifying the original velocity
A force of drag considers the density of the material through which the object is moving, rho
, the magnitude of the object"s velocity, ||v||
, the surface area subject to resistance A
, a coefficient of drag C
and the velocity"s unit vector ^v
-> ^
F = -1 / 2 * rho * ||v||^2 * A * C * velocity
Similarly to the Friction
demo, however, the force can be simplified by considering direction and magnitude. For the direction, the force is again the opposite of the velocity vector.
-> ^
F = -1 / 2 * velocity
For the magnitude, the force is scaled according to the magnitude of the velocity, and a value summarising the other constants.
-> ^
F = -1 * velocity * ||v||^2 * c
Please note:
-
the demo is updated to consider multiple entities with varying mass, as in the
Mass
demo. Notice how objects with a greater mass are subject to less drag, since the force is divided by the mass measure -
each
Mover
entity is subject to different drag forces and according to the entity"s own position. In the bottom half of the screen, the script simulates a more dense environment with a greater coefficient
The force of gravity depends on the mass of the objects involved, m1
and m2
, the distance between said objects d
, as well as a constant describing the gravitational force G
. In terms of direction, the force finally depends on the direction of the vector connecting the two objects, ^r
.
-> ^
F = G * ((m1 * m2) / d ^ 2) * r
The unit vector connecting the objects describes the direction of ths force.
-> ^
F = G * ((m1 * m2) / d ^ 2) * r
The constant, mass values and then distance finally influence the magnitude of the force.
F = G / (d ^ 2)
Please note:
-
the attraction force is computed in a method of an
Attractor
entity -
in the
Gravitational attraction
folder you find three demos-
with
Simple
the idea is to update aMover
entity toward a fixedAttractor
considering only the gravitational constant and the distance between the two objects -
in
Complex
the simulation contemplates multipleMover
entities with varying mass. The demo has also a minor interaction in the form of thepullIn
method; following a mouse click, the idea is to have the entities move rapidly toward the attractor -
in
Mouse
theMover
entities gravitate toward the mouse cursor, and away from other entities
-
To discuss oscillating motion, it is first necessary to introduce angles, polar coordinates and trigonometry. Trigonometry relates to the study of the angles and sides of right triangles, and is useful to model angles, angular velocity and angular acceleration.
To rotate an entity, modify the coordinate system with translations and rotations.
love.graphics.translate(x, y)
love.graphics.rotate(angle)
--
In the snippet, the drawing operations following the two expressions will be rotated by amount described by angle
.
Love2D, like the Processing library used in the course, works with an angle in radians, not degrees. Radians describe the angle in terms of the ratio between the length of the arc of a circle and its radius, with 1
radian being the angle at which the arc has the same length as the radius. To convert between the two, use the following formula:
radians = 2 * math.pi * (degrees / 360)
Alternatively, Lua provides the math.rad
to convert to radians, math.deg
to convert to degrees.
It is perhaps useful to note that pi is the ratio between a circle"s circumference and its diameter. It is roughly 3.14159
and is provided by Lua in the math library and math.pi
.
Please note:
-
in the demo, the script initializes a series of rectangles with varying angles. To rotate the shapes from the center, the
x
andy
coordinate offset the shape by half its width and height.love.graphics.rectangle("fill", -self.width / 2, -self.height / 2, self.width, self.height)
-
.push
and.pop
are useful to have the translation and rotation affect the singleMover
entity. Without the instruction, additional shapes would be affected by previous transformations -
Mover:render()
includes two lines to draw a circle to highlight the coordinate system; consider it a small, visual way to consider how the translation and rotation modify the render logic
With the same logic described in the update function of the Mover
entity, modify the angle with a variable describing its velocity and a variable describing its acceleration. The only difference is that the angle is represented by a single variable, and not a vector.
self.angle = self.angle + self.angularVelocity
self.angularVelocity = self.angularVelocity + self.angularAcceleration
To set the acceleration, similarly to the previous examples, use a hard-coded measure or consider the surrounding environment, the forces involved with actual formulas.
Please note:
-
in the demo the angular acceleration is set at random, but is also and further modified using the
x
coordinate of the mouse cursor. The idea is to have the rectangles rotate in the direction indicated by the mouse -
the angular velocity is limited in order to keep the value in the
(-0.1, 0.1)
range
As prefaced at the top of the section, trigonometry relates to the study of the angles and sides of right triangles. In this light, the mnemonic device sohcahtoa is useful to remember the following formulae:
math.sin(theta) = opposite / hypothenuse
math.cos(theta) = adjacent / hypothenuse
math.tan(theta) = opposite / adjacent
To find the angle, the sine, cosine and tangent are used in their inverse form. For instance and knowing the sides of the right triangle, the angle can be computed as:
theta = math.atan(opposite / adjacent)
Please note:
-
math.atan
doesn"t consider the sign of the sides involved in the formula.To fix this, either use a series of
if
statements, for the quadrants in which the sign is the opposite, ormath.atan2
. The function does consider the sign.local angle = math.atan(opposite / adjacent) local angle = math.atan2(opposite, adjacent)
Also note that, in Lua, the function accepts two arguments for the sides of the triangle, instead of one describing the ratio of the two.
-
the lines drawing a circle with an
x
offset are useful to show thatmath.atan2
works to rotate the shape in the desired directionlove.graphics.setColor(0.13, 0.86, 0.72) love.graphics.circle("fill", self.width / 4, 0, 2)
Use
math.tan
and the circle would immediately flip in the opposide end of the rectangle
While Love2D, similarly to the Processing library, renders elements in an (x, y)
plane, with cartesian coordinates, it is useful to model the simulation in polar coordinates, considering a distance and angle (r, theta)
.
The trigonometry introduced in the previous section is useful to move from one set to the other.
Please note:
- in the demo a
Mover
entity is updated increasing the angle and increasing or decreasing the distance from the center. The polar coordinates are then included in the trigonometric functions cosine and sine in order to position the entity with the cartesian counterpart
Oscillation, as the periodic movement between two points, can be defined in terms of amplitude and period.
-
amplitude: the distance between the center and either of the two points
-
period: the time it takes to complete a full cycle
Considering a sine or cosine function, the period is math.pi * 2
, and the amplitude is 1
.
Starting from the amplitude and period, it is possible to describe simple harmonic motion by updating a variable as follows.
x = amplitude * math.cos(math.pi * 2 * (frameCount / period))
The value returned by math.cos
doesn"t exceed the (-1, 1)
range, which means the variable is assigned a value in the (-amplitude, amplitude)
interval. Inside of the parenthesis, dividing the incrementing variable frameCount
by the period and multiplying the value by math.pi * 2
means the function completes a cycle as the count reaches the value of the period.
One other feature which defines oscillation is its frequency. This value describes the number of cycles per time unit, and is the inverse of the period. If an oscillation has a period of math.pi * 2
, it completes a cycle in math.pi * 2
, and has a frequency of math.pi * 2 / frame
; it covers a certain distance in the span of a single frame.
Instead of mapping a variable according to amplitude and period, the idea is to consider an incrementing variable in the trigonometric functions for the sine or cosine distribution.
x = amplitude * math.cos(angle)
In this manner the simulation re-introduces the concept of angular velocity, and ultimately angular acceleration.
angle = angle + angularVelocity
It is still possible to define the period, as the amount of time it takes for angle
to reach math.pi * 2
.
Please note:
- the demo re-introduces vectors to update the angle of the
Oscillator
entity in thex
andy
dimension
With the knowledge accumulated in the chapter, and the chapters before it, the goal is to here apply the concepts in practical simulations.
In order to create a wave, all that is necessary is to increment the angle for every entity, assigning the horizontal coordinate to a fraction of the total width and the vertical coordinate to the sine or cosine of the angle. The amplitude remains relevant, in describing the height of the line.
Please note:
-
with
getPoints()
, the script produces an horizontal coordinatex
, but also two coordinates for they
dimension,y1
andy2
. The idea is to use the opposite value returned by the cosine function to create a pattern inspired by a DNA sequence -
with
love.update()
, the initial angle modifying they
coordinates of the points is incremented by an arbitrary value. The idea is to simulate the ondulating movement of a wave -
with
CYCLES
, the script provides a way to include a shorter period (or higher frequency). The angular velocity is nevertheless scaled according tomath.pi * 2
to ensure that the first and last point have the same angle
In the simulation, a pendulum is composed of a pivot, an arm and a bob.
x <-- pivot
\
\ <-- arm
\
o <-- bob
The idea is to have the pivot function as the point around which the bob rotates, at a distance given by the inflexible arm.
With this structure, the pendulum is subject to a force of gravity, pulling the bob downwards. The fact that the bob is attached to the immovable pivot, however, means that the force of gravity doesn"t affect the round shape only in its y
dimension.
x
\
\
\
o
->/ |
F/ |
/90°|
\ |
\ |
\a|
\| gravity
Considering a right triangle whose hypothenuse describes the force of gravity, it is possible to decompose the vertical force in two different segments, and find the force F
according to the angle a
and the trigonometric functions introduced in the previous sections.
x
|a\
| \
| \
o
opposite/ |
/ |
/ |
\ | hypothenuse
\ |
\a|
\|
The angle is updated with the same logic introduced in the force chapter.
self.angle = self.angle + self.angularVelocity * dt * UPDATE_SPEED
self.angularVelocity = self.angularVelocity + self.angularAcceleration * dt * UPDATE_SPEED
Instead of a vector, here is the angle to be updated by the velocity, and the velocity by the acceleration.
Velocity and acceleration are initialized to 0
. The acceleration value is then set with the trigonometric functions prefaced in this very chapter. The relevant function is the sine function, knowing the angle and the hypothenuse, and realizing the opposite segment describes the desired force (remember the soh in sohcahtoa).
self.angularAcceleration = math.sin(self.angle) * GRAVITY * dt * -1
The value is multiplied by -1
since in Love2D, the coordinate system works left to right, top to bottom.
To create a more realistic simulation, the acceleration is also scaled according to the length of the arm.
self.angularAcceleration = math.sin(self.angle) * GRAVITY / ARM_LENGTH * dt * -1
Finally, and to have the pendulum slowly reduce its oscillation, the angular velocity is multiplied by a value slightly smaller than 1
.
self.angularVelocity = self.angularVelocity * 0.995
Please note:
-
the demo introduces multiple instances of the
Pendulum
entity create a grid of pendulums. Beside the different coordinates for the pivot, each successive instance is given a slightly different starting angle -
following a mouse press, the demo increases the angular velocity to have the pendulums resume their oscillation
Instead of an inflexible arm, like the one introduced with the simulation of the pendulum, the idea is to have an elastic arm. Instead of using trigonometric functions to then describe the position of the bob and the length of the arm, the idea is to consider the force applied by the spring. In this manner, the influence of the spring can be paired with other forces, like gravity or wind.
Starting with Hooke"s law, the force applied by a spring is directly proportional to the extension of the spring. The farther the bob stretches the spring from its rest length, the greater the force.
-> ->
F = -k * x
k
describes a constant to describe how elastic is the spring. It scales the vector describing the displacement to have a stronger or weaker force.
x
describes the displacement from the spring"s rest length.
A spring tends to a state of equilibrium, described by the rest length. As the bob stretches the arm, the displacement causes an opposite force toward the original length.
x
|
|
| restLength
|
->|
F | currentLength
o
Coming back to the formula, and with the logic introduced in the force chapter, it is necessary to evaluate the force"s magnitude and direction. In terms of magnitude, it is possible to directly use the constant k
. In terms of direction, the vector x
is obtained by comparing the current length of the arm against the rest length.
In code, the idea is to ultimately apply the force similarly to the force of gravity introduced in earlier demos.
bob:applyForce(spring)
The vector describing the force is then calculated as follows:
-
find the vector describing the difference between the bob"s position and the spring"s anchor
local direction = LVector:subtract(bob.position, self.anchor)
The magnitude of this vector provides the current length, to be compared against the rest length; the unit vector provides instead the direction at which to point the force.
-
compute the displacement,
x
, as the difference between current and rest lengthlocal currentLength = direction:getMagnitude() local displacement = currentLength - self.restLength
-
normalize the vector and describe the force as the vector, scaled by the Hooke"s formula
direction:normalize() local spring = LVector:multiply(direction, K * -1 * displacement)
Starting from a single particle, the idea is to manage multiple entities, in concert. A system is useful to simulate complex phenomena, like fire, smoke, flocks of birds.
Picking up from the logic introduced with the Mover
entity in the forces chapter, a single particle is built with three vectors: position, velocity and acceleration. The idea is to update the position with the velocity, and the velocity with the acceleration.
this = {
["position"] = position,
["velocity"] = velocity,
["acceleration"] = acceleration,
}
On top of these vectors, each particle is attributed a lifespan
. This value is useful to determine when a particle dies off. In the context of a particle system, there is usually an emitter, producing particles or a stream of particles; as new and new particles are created, it is necessary to remove existing ones.
this = {
["lifespan"] = 1
}
In a first implementation, the lifespan is connected to the alpha channel of the particle, and decreased with every iteration.
function Particle:update(dt)
self.lifespan = self.lifespan - dt * UPDATE_SPEED
end
function Particle:render()
love.graphics.setColor(0.11, 0.11, 0.11, self.lifespan)
-- draw circle
end
An additional method on the Particle
entity finally describes whether the particle is dead or not, by returning true
or false
on the basis of the lifespan value.
function Particle:isDead()
return self.lifespan == 0
end```
Please note:
-
the alpha channel in Love2D is in the
(0, 1)
range -
lifespan
is decreased to reach the minimum value of0
. This by usingmath.max
and provide a lower threshold.function Particle:update(dt) self.lifespan = math.max(0, self.lifespan - dt * UPDATE_SPEED) end
From a single particle, the idea is to produce a new entity with every iteration of the love.update
function, and remove particles when they eventually die; this last part is implemented with the :isDead
method.
When removing a particle inside of a loop considering the collection, it is important to note that the collection is updated by translating the items to the left. Iterating through the table in order, the risk is to therefore skip a particle. One immediate way to fix this is to loop through the collection backwards.
for i = #particles, 1, -1 do
particles[i]:update(dt)
if particles[i]:isDead() then
table.remove(particles, i)
end
end
Please note:
- in
love.draw
thelove.graphics.print
function is used to double check that the particles are removed from the collection. Remove the comment to see how the table has at most roughly239
items
The collection of particles is stored and managed in a ParticleSystem
entity. The system is initialized with an x
and y
coordinate, to spawn particles from a specific point of origin.
Please note::
- in the demo, the origin of the particle system is modified following the mouse cursor
Building on top of a particle system as a collection of particles, it is possible to build a collection of collections, of particle systems. This is beyond the scope of the chapter, but useful to describe a simulation in multiple layers of complexity.
Please note:
-
in the demo, the script initializes a particle system following a mouse click, and removes a collection when there are no more particles
-
always in the demo, and in order to eventually remove particle systems, the rate at which particles are produced is decreased with delta time. The rate at which the particles become fully transparent is also increased
In the scope of object oriented programming, inheritance is useful to create a system in which entities pick up and expand the logic introduced by other entities. Case in point, SquareParticle
and CircleParticle
can inherit the behavior of the Particle
entity, and render the particle with a different shape, respectively, a square and a circle.
Polymorphism, as introduced in the book, refers to how it is possible to have different classes like Dog
and Cat
are able to retain multiple types, multiple forms. A Dog
class inheriting from an Animal
class is therefore both a dog and an animal. This is useful to have a collection like an array or array list with a singular type, storing different types of animals and call the same function on every instance. What the function does, then, depends on the implementation in the different classes.
Please note:
-
the logic necessary to implement inheritance depends on the language being used. In the context of lua, there is no class system, but it is possible to recreate a mechanism similar to a prototype and have a table refer to a parent table (a concept similar to a superclass)
-
in Lua, a dynamically typed language with no concept of a class system, polymorphism is not a concern. A table stores different entities, without concern for their type
The idea is to re-introduce the concepts of earlier chapters, but in the context of a particle system. The section is also useful to describe the structure of the simulation, and how certain methods are defined on the system, as a whole, and other methods on the particles, individually.
One immediate way to influence the particle system is by applying a force to each and every particle. The logic is structured in two steps:
-
apply a force on the system
local gravity = LVector:new(0, GRAVITY) particleSystem:applyForce(gravity) particleSystem:applyRepeller(repeller)
-
apply the force on the composing particles
function ParticleSystem:applyForce(force) for i, particle in ipairs(self.particles) do particle:applyForce(force) end end
The function is defined on the Particle
entity, similar to Mover
in the dedicated chapter.
function Particle:applyForce(force)
self.acceleration:add(LVector:divide(force, self.mass))
end
Please note:
- out of convenience, the particles are attributed a mass equal to
1
Another way to affect the particles in the particle system is to create a separate structure, like a Repeller
entity.
In love.update(dt)
, the system receives the repeller as an argument.
function love.update(dt)
particleSystem:applyRepeller(repeller)
end
The dedicated function, then, loops through the set of particles to create the appropriate force of each entity.
function ParticleSystem:applyRepeller(repeller)
for i, particle in ipairs(self.particles) do
local force = repeller:repel(particle)
particle:applyForce(force)
end
end
Please note:
- to compute the force, the
repel
function considers the method developed for the Gravitational attraction demo. The logic is updated to apply a considerable force away from the repeller, and when the particle approaches an arbitrary threshold
Particles are drawn with regular shapes, like circles or rectangles as done so far, or with images.
function Particle:render()
love.graphics.setColor(1, 1, 1, self.lifespan)
love.graphics.draw(img, self.position.x, self.position.y, 0, 1, 1, img:getWidth() / 2, img:getHeight() / 2)
end
Images are also useful to introduce blend modes, modifying the default pixel value of overlapping entities. With the add
option, the rgb components are added to make the image brighter and eventually white. This is useful for instance to simulate fire particles.
function ParticleSystem:render()
love.graphics.setBlendMode("add")
-- draw particles
end
Please note:
-
in the demo the image is initialized as a drawable object in
love.load
. This is a much preferable solution to include the instruction in the particle entity, in which case the script would continuously load the filefunction love.load() img = love.graphics.newImage("particle.png") end
With vectors, forces, trigonometry it is possible to simulate a first environment with rudimentary physics. It is possible to refine the simulation considering more complex natural phenomena, but one alternative comes in the form of physics libraries. Here you find code by other developers already considering the issue of simulating life, simulating nature. A physics engine provides a level of complexity only grasped in the previous sections. The price is that you need to learn about the library, its requirements and also limitations.
Please note:
-
while the chapter is devoted to two libraries, Box2D and Verletlib, I cover the examples of the first library only
-
Love2D includes the Box2D library in the
love.physics
module
Fundamentally, a simulation with Box2D works in two steps: set up and update. In the setup phase, you initialize the world, and populate the environment with however many entities are necessary. In the update phase, Box2D considers the underlying physics to update the world as necessary; there is no need to consider the position, velocity, acceleration and forces of the individual entities.
Box2D considers all the underlying physics, but it is however necessary to set up the world with the procedure and syntax prescribed by the library.
A Box2D simulation starts with a world. This is where the simulation defines the features of the environment, like its gravity.
GRAVITY = 20
function love.load()
world = love.physics.newWorld(0, GRAVITY)
end
Box2D works with meters, kilograms and real-world units. Since Love2D works with pixels instead, it is useful to adapt the measures with the setMeter
function.
GRAVITY = 20
GRAVITY_METER = 9.81
function love.load()
love.physics.setMeter(METER)
world = love.physics.newWorld(0, GRAVITY * GRAVITY_METER)
end
Once initialized, the world is updated with the update()
function, considering every single entity included in the simulation.
function love.update(dt)
world:update(dt)
end
To populate the world, each entity needs three parts: a body, a shape and a fixture.
-
a body is but a container describing the position, velocity and other defining features of the entity. Consider it similar to the
Mover
entity introduced in previous chapters.body = love.physics.newBody(world, WINDOW_WIDTH / 2, WINDOW_HEIGHT / 4, "dynamic")
The function introduces a type between one of three options:
static
,dynamic
andkinematic
. A dynamic body is one subject to the forces of the world, and one colliding with other entities. The remaining two types are discussed in later sections. -
a shape provides the visual representation for the body, and is ultimately essential to define how a collision occurs.
shape = love.physics.newCircleShape(RADIUS)
Different functions provide different geometries, like circles, rectangles, polygons. The parameters vary according to the desired shape, considering for instance the radius for a circle.
-
a fixture is attaches a body and shape together.
fixture = love.physics.newFixture(body, shape)
The fixture is also where the entity can set other features, like density, restitution.
This is enough to have the world consider and update an object. To provide a visual then, Love2D provides different methods to retrieve the defining features of the bodies. In the context of a circle, body:getX()
. body:getY()
and shape:getRadius()
allow to find the measures for the circle"s position and radius.
The demo works to create two entities dedicated to different shapes, a circle and a rectangle, and to populate a table with multiple copies of each. Adding multiple objects is also important to stress the importance of removing entities when they are no longer necessary. Removing items involves two steps:
-
update the table so that the bodies are not rendered in
love.draw
table.remove(particles, i)
-
update the world so that the bodies are no longer considered in the simulation
particles[i].body:destroy()
Without this line the window doesn"t show the bodies, but their position, movement and collision is still computed by Box2D. It is here essential to math the data structure evaluated by Love2D with the world simulated by Box2D.
A previous section introduced how bodies have different types. By default an object is static, but it"s possible to modify this value already in the declaration of the body. This is what the previous demo achieved in love.physics.newBody()
.
love.physics.newBody(world, x, y, "dynamic")
Shortly, a body can be static, fixed in the world and not subject to its forces, dynamic, reacting to the world"s gravity, forces, and collisions, kinematic, not subject to forces, but manually moved through its velocity. Consider for instance a platform (fixed), a ball (dynamic) or a character directly controlled by the player (kinematic).
Please note:
- the demo populates the world with one platform, but allows to include more fixed objects with mouse input. Drag the cursor from point to point to generate a new platform
A curved surface is introduced with a ChainShape
. This particular shape accepts as argument a series of points, which are then connected to make up the object. Using a particular distribution or a trigonometric function, the effect is that the points produce the desired visual.
Please note:
- the demo creates a chain shape with the cosine function. In future projects, however, the
Terrain
entity is modified to experiment with other values and distributions.
There are at least two different approaches to building complex shapes:
-
use a
PolygonShape
, detailing the vertices of the desired outline; -
fix multiple shapes to the same body.
The second approach is the topic of the demo, and the reason the ComplexShape
entity actually introduces two rectangle shapes.
A joint creates a connection between multiple bodies. There are different types, each with its own usefulness and defining features. For instance:
-
a distance joint connects two bodies with a string. Attributes like frequency and damping ratio allow the string to be elastic
-
a revolute joint, anchors a body to a point and rotates the entity around the anchor
-
a mouse joint connects a body to an
x
,y
point. In the demo it is used to have a circle tied to the position of the mouse cursor, but only when said mouse is pressed in the window
Please note:
-
it is enough to initialize a joint in the world to affect the simulation.
local joint = love.physics.newDistanceJoint( circleA.body, circleB.body, circleA.body:getX(), circleA.body:getY(), circleB.body:getX(), circleB.body:getY() )
There is no need to draw a matching visual for the joint to have effect. Whatsmore, there is no need to have a reference to the joint itself; consider how in Distance joint the
PairShape
entity doesn"t include the joint in its definition (the line is commented out).local this = { ["circles"] = {circleA, circleB} -- ["joint"] = joint }
A reference is useful in the moment you need to modify the joint. Consider for instance how the revolute joint changes the speed using the horizontal coordinate of the mouse cursor, or again how the mouse joint is destroyed as the mouse is released.
Re-introducing the topic from previous chapters, it is possible to affect a Box2D simulation applying a force directly on a body.
Body:applyForce(fx, fy)
Please note:
-
in the Forces folder you find two demo, simulating a gust of wind following a mouse press or gravitational force between a fixed object and a series of dynamic particles
-
the attractor entity in the demo exploring gravitational force is included as a kinematic body, and updated manually through its velocity considering the position of the mouse cursor
Often, it is helpful to react to a collision between bodies. Box2D provides an interface to execute some code in the lifecycle of a collision, by referring to a callcback function on the world.
world:setCallbacks(beginContact)
:setCallbacks
actually accepts up to four arguments, to consider when a contact begins, ends, and when a collision is about to be resolved or has just been resolved.
world:setCallbacks(beginContact, endContact, preSolve, postSolve)
The functions must be defined in the code, and receive a series of arguments describing the collision and the objects involved. More accurately, beginContact
receives the fixtures of the objects involved, and a table describing the collision.
function beginContact(f1, f2, collision)
end
The collision occurs between two objects coming into contact with each other. It is here extremely useful to know which objects are however involved. To this end, a fixture can describe a label with a userdata
field.
fixture:setUserData("attractor")
The label is then evaluated in the body of the beginContact
function.
Please note:
- in the demo, the idea is to apply a repulsing force when a particle collides with the attractor. The data structure collecting the particles is modified to have the table use the keys with the same value as the
userdata
field. This is helpful to refer to the particular particle, but requires a small adjusment in the iterator function.ipairs
works with sequences, whilepairs
is equipped to loop through key-value pairs.
With the chapter the idea is to include entities capable of moving on their own, on the basis of a desire. These entities share three defining features:
-
a limited ability to perceive the environment
-
the ability to process the environment to formulate an action
-
the lack of a leader, of an entity detailing the agent's eventual movement
The goal is to run a simulation without a pre-ordained structure, and analyze the interaction between the independent entities.
The demos are inspired by the book which itself cites as inspiration the work of Craig Reynolds on algorithmic steering behaviors.
In the folder you find several examples to illustrate the concept of desire. An entity might desire to move toward a target, or away from an obstacle; this wanting is materialized in a function applying a force on the basis of the surrounding environment.
In the steering example, a Vehicle
entity is initilized with a structure eerily similar to the Mover
or Particle
entities introduced in previous sections. Every entity has a position, velocity and acceleration. Motion is however expressed in three layers:
-
action selection: select an action on the basis of a goal or set of goals; for instance, compute the desired velocity as the difference between the position of the vehicle and target
local desiredVelocity = LVector:subtract(target.position, self.position)
-
steering: formulate a force to materialize the action; for instance, generate a force considering the desired velocity against the current velocity
local steeringForce = LVector:subtract(desiredVelocity, self.velocity)
-
locomotion: actually move the vehicle; for instance, apply a force to modify the entity's acceleration and velocity
self:applyForce(steeringForce)
In the demo, the logic steering the vehicle toward the target is described in the steer()
method. In the body of the function, the entity computes the desired velocity and steering force, but refines the movement with two additional variables:
-
maxSpeed
describes the magnitude of the vectordesiredVelocity:normalize() desiredVelocity:multiply(self.maxSpeed)
The fixed magnitude means that the entity eventually overshoots its trajectory. In a future demo this behavior is modified to have the entity slow down as it approaches the target.
-
maxForce
reduces the force so that the entity changes direction more slowlyforce:limit(self.maxForce)
Limiting the influence of the force means that it takes some time for the vehicle to point to the target
Please note:
- in the demo the vehicle and target are position at opposite ends, but the target is updated using the mouse coordinates
The idea is to have the Vehicle
entity steer toward the target, but then slow down as it gets closer and closer to the target's position.
The function evaluates the desired velocity, but also the distance of the vector.
local desiredVelocity = LVector:subtract(target.position, self.position)
local distance = desiredVelocity:getMagnitude()
The velocity is then multiplied by a value proportional to the actual distance. The smaller the distance, the slower the force. This however, only when the vehicle is in the range of the target.
if distance < RADIUS_SLOWDOWN then
-- from [0, RADIUS_SLOWDOWN] to [0, maxSpeed]
local speed = map(distance, 0, RADIUS_SLOWDOWN, 0, self.maxSpeed)
desiredVelocity:multiply(speed)
end
Please note:
- following a mouse click the demo toggles the visibility of the circle in which the vehicle slows down
The vehicle moves toward the target considering both its position and velocity. In order to achieve this effect, the target is initialized with a velocity vector, and this vector is modified to have the entity move toward the mouse. The same vector is then incorporated in the pursue
function of the vehicle in order to modify the desired velocity.
Please note:
-
following a mouse click the demo toggles the visibility of a circle displaying the desired location for the vehicle
-
to highlight how the vehicle actually pursues the target the demo includes two precautions:
-
UPDATE_SPEED
is reduced, so that it is more evident how the entity changes direction -
VELOCITY_MULTIPLIER
scales up the velocity of the target so that the vehicle over-estimates where the target is going to be. You could explain this behavior as thinking that velocity begets velocity, and the vehicle presumes the target will continuously move in the same direction
-
The vehicle moves in the window at a constant speed, and changes this behavior in order to respect arbitrary boundaries. This is achieved by applying an force opposite to the desired velocity if the desired velocity would move the entity outside of the boundaries.
Please note:
- following a mouse click the demo toggles the visibility of the boundaries
Following a suggestion included in the book, the vehicle moves toward a point in the vicinity of the target. By choosing a point around the target, signalled by a small circle the vehicle moves unpredictably, and yet pursuing the target.
Please note:
- following a mouse click the demo toggles the visibility of the small circle around the target
With a flow field the window is divided in a certain number of columns and rows. In this grid, the cells describe a velocity, which is then picked up by the vehicle as it navigates the environment.
Each cell is attributed an angle, and there are multiple demos which differ in how this angle is computed, as well as a force vector. To compute this vector, the cosine and sine functions identify where the segment should start and end.
local x1 = math.cos(angle + math.pi)
local y1 = math.sin(angle + math.pi)
local x2 = math.cos(angle)
local y2 = math.sin(angle)
Incrementing the angle by math.pi
allows to find the origin of the segment, half a rotation from the destination. This structure is useful to draw a line without the translate and rotate functions.
From this setup, all the vehicle needs is to find a cell, and apply a force matching the vector.
As mentioned, there are multiple demos which change how the angle is computed:
-
in
Random
the value is purely randomlocal angle = math.random() * (math.pi * 2)
-
in
2D noise
the value benefits from a noise function with two argumentslocal angle = love.math.noise(offsetX, offsetY) * (math.pi * 2)
The concept was first introduced in the demo for randomness and Perlin noise, and has the effect of creating a series of angles connected to one another.
-
in
3D noise
the value is computed with a noise function with three argumentslocal angle = love.math.noise(offsetX, offsetY, time) * (math.pi * 2)
time
is initialized at0
and incremented with each iteration inField:update()
, so that the angle smoothly changes over time.
Please note:
-
it is possible to include additional vehicles by pressing the mouse with the left button
-
the visibility of the field is toggled by pressing the mouse with the right button
The Math
folder introduces a few concepts essential for future demos.
The dot product allows to compute the angle between two vectors. It is helpful to build a scalar projection, and ultimately important to introduce a simulation in which vehicles follow a given path. Defined as the multiplication of two vectors, the product is computed as follows:
-
multiply the components making up the vectors
-> A = (ax, ay) -> B = (bx, by) -> -> A * B = ax * bx + ay * by
-
multiply the magnitude of the two vectors and the cosine of the angle between the same vectors
-> A = (ax, ay) -> B = (bx, by) -> -> -> -> A * B = |A| * |B| * cos(theta)
By computing the product with the first formula, it is possible to find the angle solving the second formula for theta.
Please note:
LVector
is updated to include a function returning the dot product for two input vectors, and a function returning the angle between two vectors (using the dot product itself)
The goal is to find a point on a vector according to another vector, a projection.
->
a /|
/ |
/ |
---o---
->
b
Cast a line from the vector a
to the vector b
so that the line creates a 90
degrees angle. With this line then, the dot product allows to compute the point making up the projection through the angle.
h /|
/ |
/t |
---o
->
x
In terms of math, there are a couple of steps involved:
-
consider the angle
t
through the mnemonic sohcahtoa, as the cosine of the adjacent,x
, divided by the hypothenuse,h
cos(t) = x / h
-
consider
h
as the magnitude of the vectora
, so that the adjecent segment,x
, is computed as-> cos(t) = x / |a| -> x = cos(t) * |a|
-
notice how the formula for the adjacent segment is similar to the one for the dot product. It is actually the same in the instancethe vector
b
has a magnitude of1
(is a unit vector)-> |b| = 1 -> -> x = cos(t) * |a| * |b|
This means that ultimately, the projection is computed considering the dot product of the normalized b
vector.
Please note:
-
the demo builds from the script showcasing the dot product, but swaps the name of the vectors
b
anda
. This in line with the convention introduced in the book -
the vector
b
points to the right half of the screen, changing they
coordinate by a random amount. The value is also updated following a mouse click, to show how the projection works for any vector
The dot product and scalar projection are useful as building block for path following, another behavior studied by Craig Reynolds. The idea is to have a vehicle move in the window and follow the trajectory described by a path.
In the folder there are two demos: Straight
and Segments
, to show how a vehicle first follows a single straight line and then a series of connected segments.
A Path
entity includes two vectors for each line, describing where the line should start and end. It also describes a radius to have the vehicle move toward the line with some margin.
In the Vehicle
entity then, the follow
function receives the path and modifies the velocity of the vehicle with the following logic:
-
find
desiredLocation
, as the vector describing where the entity would be moving with its current velocitylocal velocity = LVector:copy(self.velocity) velocity:normalize() velocity:multiply(DESIRED_LOCATION_DISTANCE) local desiredLocation = LVector:add(self.position, velocity)
DESIRED_LOCATION_DISTANCE
is used to scale the unit vector so that the vehicle looks ahead by a given value. Essentially, however, the location is found by adding position and velocity -
compute the
a
andb
vectors as described in the scalar projection demo-
a
describes the vector between the desired location and the beginning of the pathx desired location -> / a x currentposition / / x------ path
local a = LVector:subtract(desiredLocation, path.start)
-
b
describes the vector for the path itself/ / / / x------x -> b
local b = LVector:subtract(path.finish, path.start)
-
-
find the projection through the dot product, again taking inspiration from the scalar projection
b:normalize() b:multiply(TARGET_MULTIPLIER) local projection = LVector:multiply(b, LVector:dot(a, b)) projection:add(path.start)
Notice that
b
is normalized, exactly as in the previous demo, but also multiplied by a factor.b:multiply(TARGET_MULTIPLIER)
The idea is to provide an offset from the projection describing the normal. Without this offset, the vehicle would move to a fixed point describing the perpendicular line connecting point to path.
Also notice that the vector is incremented by the start vector.
projection:add(path.start)
This is essential to have the projection relative to the path origin.
-
with the vector describing the projection, a point on the path, the idea is to move the entity towards the path, but only when exceeding the space given by the radius.
To compute the distance, consider the projection and the vector describing the desired location
local dir = LVector:subtract(projection, desiredLocation)
The magnitude of this vector details the distance, so that the vehicle is pushed to the projection when exceeding the desired value.
local distance = dir:getMagnitude() if distance > RADIUS_PATH then self:seek(projection) end
Please note:
-
Vehicle:seek
repeats the logic introduced in the steering demo, with the only difference that the argument of the function is a vector and not aTarget
entity -
the
Path
entity introduces two vectors, for where the line should start and end. I usefinish
instead ofend
since the latter is a reserved word in Lua -
pressing the mouse with the left button has the effect of adding a new vehicle; the right button instead changes the
y
coordinate of the path
The idea is to have a path described not by two vectors, by a series of segments, each with a start and end point.
local segments = {}
for i = 1, POINTS_PATH do
-- create segment
end
To have the segments connected to one another, Path:new
keeps a reference to the y
coordinate where each segment should end.
local previousY = math.random(HEIGHT_MIN, HEIGHT_MAX)
for i = 1, POINTS_PATH do
local x1 = xStart + xIncrement * (i - 1)
local x2 = xStart + xIncrement * i
local y1 = previousY
local y2 = math.random(HEIGHT_MIN, HEIGHT_MAX)
previousY = y2
end
end
From this setup, Path
is equipped with a segments
field, storing the desired x
and y
coordinates. The Vehicle
entity then needs to loop through the collection to evaluate the projection on each and every line.
function Vehicle:follow(path)
-- knowing desiredLocation
for i, segment in ipairs(path.segments) do
-- compute projection
end
end
The idea is to consider here the closest projection which belongs to the actual path. Both conditions are necessary to avoid moving the entity towards the wrong segment.
To check if the projection belongs to the path, it is enough to check if the x
coordinate falls between the beginning and the end of the segment.
if projection.x > segment.start.x and projection.x < segment.finish.x then
-- consider distance
end
In order to consider the closest projection, then, the distance is evaluated against a variable intialized with a large value.
local recordDistance = math.huge
local recordProjection = nil
for i, segment in ipairs(path.segments) do
-- evaluate distance
if distance > RADIUS_PATH and distance < recordDistance then
recordDistance = distance
recordProjection = projection
end
end
The projection is stored in yet another variable, in the same conditional, and is ultimately used to change the velocity of the vehicle.
if recordProjection then
self:seek(recordProjection)
end
Please note:
- the same notes for the
Straight
demos apply (seek function, add vehicles and create a new path with the mouse cursor)
With a group behavior the goal is to model the movement of Vehicle
entities relative to other entities.
Align the movement of a vehicle relative to other the entities. In the align
method, each entity receives the collection describing the instances.
function Vehicle:align(vehicles)
end
From this collection, the idea is to tally every vector describing the velocity.
local velocity = LVector:new(0, 0)
for i, vehicle in ipairs(vehicles) do
velocity:add(vehicle.velocity)
end
Every vector except the one describing the entity itself. To this end, the new
method is modified to have each entity distinguished with an id
.
local this = {
-- previous attributes
["id"] = math.random()
}
The velocity is then added only if the id do not match.
if vehicle.id ~= self.id then
velocity:add(vehicle.velocity)
end
This has the desired effect of aligning all the entities. However, as explained in the book, it is useful to limit the area of alignment to those vehicles falling in an arbitrary radius. Here, it is helpful to consider the distance between the vectors.
if distance < DISTANCE_VEHICLE and vehicle.id ~= self.id then
velocity:add(vehicle.velocity)
end
By increasing/decreasing VEHICLE_DISTANCE
, the demo shows how a vehicle aligns itself with more/less neighbors, and form a larger/smaller cluster.
Please note:
-
vehicles are added continuously as the mouse is pressed
-
LVector
is updated to have a function return the distance between two vectors. This is achieved by subtracting the vectors and returning the magnitude of the resulting vector.function LVector:distance(v1, v2) local dir = LVector:subtract(v1, v2) return dir:getMagnitude() end
-
the velocity is divided by the neighbors actually considered in the
align
functionvelocity:divide(neighbors) self:applyForce(velocity)
Instead of aligning vehicles together, the entities are separated by applying a force away from the surrounding neighbors.
The code is similar to the alignment demo, in that the function loops through the collection of vehicles and considers only those vehicles closer than a given range. However, once a vehicle is within range, the vector is computed by subtracting the entities in reverse order.
local force = LVector:subtract(self.position, vehicle.position)
The idea is to describe a force away from the neighbor. Once computed, the force is added to a vector considering the cumulative vector for every neighbor.
force:divide(distance)
separationForce:add(force)
The force is also weighed by the distance, so that the closer a neighbor is, the more influence it will have on the cumulative vector.
This is enough to consider the influence of the neighbors. However, it is finally useful to scale the vector to avoid excessive values. One way to achieve this is to normalize the vector and scale the result by an arbitrary amount.
separationForce:normalize()
separationForce:multiply(MAX_SPEED)
self:applyForce(separationForce)
Please note:
- the same notes for the alignment demo apply
The folder works to show how multiple behaviors can be paired to produce more complex simulations. The forces are computed in dedicated functions and applied in the applyBehaviors
method.
function Vehicle:applyBehaviors(vehicles, target)
local steeringForce = self:steer(target)
local separationForce = self:separate(vehicles)
self:applyForce(steeringForce)
self:applyForce(separationForce)
end
By weighing the forces with different factors, the idea is to have a particular force take precedence. For instance, and in the exercise pairing the steering and separation forces, the vector pushing the vehicles away is multiplied by three. This has the net result of creating a swarm of vehicles moving toward the intended target, but giving precedence to avoid any overlap.
The exercises include forces already developed in previous demos, in order to steer vehicles toward a target, away from each other or to follow a path. There is however a new force in the Separation, alignment and cohesion
folder, to express cohesion between the entities.
The idea of the cohesive force is to consider the surrounding neighbors and move the vehicle to the center of the group. This is achieved by considering the sum of the vectors describing the position, and computing the average.
The force is then described by the distance between the vehicle's current position and the newfound vector.
local cohesionForce = LVector:subtract(position, self.position)
With a cellular automaton the book introduces a system of rules. Such a system has three foundational ingredients:
-
a grid of cells
-
cells with a concept of state and neighborhood
-
state as a function of neighborhood
There are two particular examples in Wolfram's elementary cellular automata and Conway's game of life, but to get started, the folder includes a rudimentary system in Cellular automaton
.
The example creates a one-dimensional grid, where a cell has up to two neighbors, described by the previous or following unit.
A cell has a boolean value, initialized at random.
local isAlive = math.random() > 0.5
At each iteration, the idea is to modify the state so that a cell is alive only if one of its neighbors is alive.
local isAlive = aliveNeighbors == 1
Wolfram systems are described by one-dimensional cellular automata, where a ruleset dictates the state of the cell as a function of the state of the same cell and its surrounding neighbors.
Each cell is initialized with a binary value, so that the generation is described by a sequence of 0
s and 1
s. At each iteration then, the idea is to consider the neighboring cell to form a string, like 001
or 010
, and use the string to describe the state from a given set. For instance, a ruleset might be initialized as follows:
local ruleset = {
["111"] = 1,
["110"] = 0,
["101"] = 1,
["100"] = 0,
["011"] = 1,
["010"] = 0,
["001"] = 1,
["000"] = 1,
}
As a sequence of neighbors describes 101
, the new cell receives the matching state of 1
.
The specific sequence describes the rule, in decimal representation. For instance, ruleset 10101011
describes rule 171
.
The outcome can be described in one of four categories:
-
uniformity, where the system settles on a constant value and each generation is equal to the one before it. Consider rule 222;
-
repetition, where the system oscillates between generations, creating a predictable pattern. Consider rule 90;
-
randomness, where the system behaves unpredictably. Consider rule 30 (until the system reaches the edge of the window);
-
complexity, similar to repetition, in that it creates a series of patterns, but also to randomness, in that the patterns do not repeat in a predictable manner
Please note: the Wolfram
folder contains a series of demos, each with its own purpose:
-
Elementary cellular automata
introduces the system in its most basic version. TheAutomata
entity receives a rule as a decimal and creates a ruleset for the three neighbors -
Neighbors
expands the logic so that the system considers a variable numbers of neighbors before and after the current cell. Notice that the number of rules increases; for instance with three neighbors, and four cells, there are 16 possible configurations leading up to1111
and sixteen-bit numbers reach65535
as the greatest rule -
Wrap around
modifies the update function so that the cells at the edge of the generation consider those at the end and vice-versa as neighbors. In the previous implementation, a cell would use a default value of0
for unavailable neighbors -
Rulesets
modifies the default behavior of using the same rule over and over by re-initializing the system every time the generation reaches the bottom of the window. Notice that the size of the individual cell is reduced and the speed at which the automaton is updated is increased; this is to show more cells and rules.
The game of life provides two dimensional cellular automata. More than just an exercise, its works as a starting point for simulations inspired by nature, and specifically inspired by a system of biological reproduction. With a set of limited rules, the idea is to produce pattern and unpredictability similarly to the systems introduced in the Wolfram section. The systems ultimately settle to create uniformity or repeating the same pattern; oscillating as it were.
The two dimensional system is described by a two dimensional collection. Each iteration creates not a new generation, but a new frame of the system.
Each cell is initialized with a boolean at random.
grid[column][row] = math.random() > 0.5 and 1 or 0
With each iteration, then, the idea is to consider the available neighbors and change the state not on the basis of a ruleset, but considering the features of the neighborhood, and speficially the number of neighboring cells which are alive. The rules modify the grid as follows:
-
if the cell is alive, it dies with less than two (loneliness) or more than three neighbors (overpopulation)
-
if the cell is dead, it comes back alive with exactly three neighbors (birth)
-
outside of the previous instances, the cell maintains its previous state (stasis)
Please note: similarly to Wolfram
, the folder dedicated to the game of life includes a series of demos, each with its own purpose:
-
Game
creates the game in its simplest version, considering the available neighbors and the aforementioned rules -
Wrap around
has edge cells look at neighbors at the opposite end of the grid. The demo also includesnewGrid
to avoid creating a new collection with every update; the idea is to here update the state in the new collection and then pick up the value in the current set -
Object oriented
creates a dedicated entity for the cells, and changes the appearance of the simulation to highlight a newborn/just dead cell with a blue/red fill
Fractals are defined as geometric shapes that can be split into parts, each of which resembles the whole. These shapes share a few common features:
-
self-similarity; there is perfect self-similarity, where the part is an exact copy of the whole, and stochastic self-similarity, based on probabilities and randomness
-
a fine structure at small scale, which cannot be recreated with Euclidean geometries
-
a recursive definition
To draw a fractal it is first necessary to introduce the concept of recursion, the repeated application of a rule to successive results. In practice, this concept sees a function calling itself, but with different arguments. The exemplary use case is that of a function computing the factorial.
function factorial(n)
if n == 1 then
return 1
end
return n * factorial(n - 1)
end
Trivially, the factorial of 5
is 5!
and is computed as 5*4*3*2*1
. It can be re-written as 5*4!
, and 5*4*3!
until 1
. It is important to stress this last value because it describes the exit condition, where the recursion should stop. Without such a condition, the function would call itself indefinitely resulting in a stack overflow.
Acknowledging recursion, the demo shows how a draw
function calls itself to draw circles at different locations and with different radii.
function draw(x, y, r)
love.graphics.circle("line", x, y, r)
if r > RADIUS_MIN then
draw(x, y, r / 2)
draw(x + r, y, r / 2)
draw(x - r, y, r / 2)
draw(x, y + r, r / 2)
end
end
The function is first called with a value describing the largest circle.
draw(WINDOW_WIDTH / 2, WINDOW_HEIGHT / 2, RADIUS_MAX)
At each iteration, then, it draws more circles assuming the radius is more than an arbitrary threshold. As the draw
function calls itself with different maeasures, every circle has a smaller circles to the right, to the left and to the bottom, resulting in an intriguing pattern.
With a function drawing a shape and recursively calling itself it is possible draw Sieprinski triangle, a particular type of fractal in which a triangle is subdivided into three, smaller triangle. The structure is based on an equilateral triangle, where the sides are all the same; to find the coordinates of this shape, it is necessary to know the height of the triangle.
local height = (side * 3 ^ 0.5) / 2
Once an equilateral triangle is drawn, the idea is to position the three, smaller triangles to subdivide the shape evenly.
draw(x - side / 4, y + height / 4, side / 2)
draw(x + side / 4, y + height / 4, side / 2)
draw(x, y - height / 4, side / 2)
Notice that the code retains an exit condition, once more to have the function eventually terminate its recursive pattern.
if side > SIDE_MIN then
-- draw triangles
end
A Cantor rule is expressed by a line recursively drawn by dividing the line in three parts, and erasing the middle third. The concept is visualized in the demo showing the different steps.
Building on top of the example introduced with the Cantor rule, a Koch line divides a line in three parts, erases the middle portion and connects the remaining thirds with the sides of an equilateral triangle. It is here necessary to have a reference to the line(s), so that the new segments are created from known coordinates, and to this end, the code is modified to have the line(s) expressed with a dedicated entity and two vectors, detailing where the lines should start and end.
At first, the demo shows a single line spanning the width of the window. Following a mouse click, then, the idea is to have the line segmented following the rules mentioned above. Consider each step as a generation, much similarly to the CA described in the previous chapter.
Please note:
-
the
Vector
entity is modified to consider a rotation for the2D
vector. The functionality is essential to figure out the location of the point making up the equilateral triangle in the middle third, and modifies thex
andy
component of the vector according to the input angle and the rules described by the rotation matrix.function Vector:rotate(theta) local x1 = self.x * math.cos(theta) - self.y * math.sin(theta) local y1 = self.x * math.sin(theta) + self.y * math.cos(theta) end
-
by pressing the right button of the mouse cursor the line is animated in the
y
coordinate describing the end of each line (except the last one)
Following a suggestion from the book, the demo reiterates the concept introduced in the Koch line
to produce a regular polygon where the sides are actually Koch lines. The Snoflake
entity is initialized with a certain number of sides and generations, with default values describing a triangular shape after 5
iterations. Following a mouse press, the shape is re-initialized with a random number of sides in the [3, 10]
range.
The demo includes a bit of math, mostly to find the coordinates of the line segments, but also to have the size of the polygon change depending on the number of sides. The idea is to have a variable RADIUS
describe the radius of the circle wrapping around the polygon, and compute the length of the side and of the apothem in order to keep the shape inside the edges of the window and exactly in the center.
Trees are useful to describe fractals without perfect self-similarity. By modifying the shape with random values and probabilities the idea is to avoid a stylized, symmetric shape to find another way to emulate nature.
The folder includes multiple demos to explore the topic:
-
Transformation matrix
creates a deterministic tree with a specific production rule:-
draw a line
-
rotate to the left and to the right, each time drawing a shorter line
-
repeat step 2 until an arbitrary exit condition, for instance a line shorter than a given number
The rule is implemented with a recursive function and benefiting from the
push
andpop
functions; these functions allow to save and retrieve a particular coordinate, so that the line is drawn from a specific advantage point.Please note: following the mouse cursor, the demo maps the horizontal coordinate to the angle used to rotate the line, and in the
[0, math.pi / 2]
range -
-
Object oriented
drops the recursive draw function to create aTree
andLine
entity.The idea is to have a tree describes lines in generations, and have the
:generate()
function produce branches on the basis of the last generationPlease note: it is possible to preserve recursion by having the
:generate()
function call itself; this is exactly what is achieved in the demos which follow -
Stochastic
includes randomness and probability to have the branches change in number, angle and length. This is in line with the purpose of the tree fractal, to show stochastic, non-deterministic shapes -
Noise
animates the last generation of trees modifying thefinish
vector with a noise function
An L-system is a grammar-based sytem, a way to write strings that is adapted in the demos to map characters to particular drawing instructions. It is ultimately useful to describe production rules through strings, commanding an entity with a series of instructions.
Such a system is characterized by three defining features:
-
an alphabet describing the allowed characters; for instance
A
andB
-
an axiom detailing a string for the initial state, for generation zero; for instance
A
-
a rule, a production rule applied recursively to the string; for instance, a rule replacing every character
A
with the sequenceABA
and every characterB
with the sequenceBBB
Generation after generation, the rule makes it possible to build a sentence with a variety of characters.
The folder includes a few demos to illustrate the point:
-
with
Grammar
the idea is to show a the sentence generation after generation -
with
Cantor rule
the goal is to re-create the visual proposed in the folder bearing the same name, but with an L-system. The production rule is exactly the same as the one proposed earlier, where each characterA
creates a line, and each characterB
creates whitespace between segments -
with
Turtle
the folder contemplates more complex systems, where an entity is instructed to move in the window according to a more rich alphabet and a prescribed ruleset. The demos differ in terms of axiom, rules, but also length and angle. TheTurtle
entity and the grammar system, on the other end, are repeated in every folder
Please note:
-
Lua has no concept of a string buffer, but the table provides a similar benefit. Instead of modifying a string through concatenation and the
..
operator, each character is added to a table and the table is finally joined together through theconcat
functionlocal next = {} for i = 1, #sentence do next[#next + 1] = RULE[sentence:sub(i, i)] end sentence = table.concat(next)
The idea of the chapter is to have information passed from object to object in order to have the simulation evolve over time. The principles of biological evolution, and specifically darwinian evolution, are taken as inspiration to create a series of genetic algorithms and solve specific problems.
The first type of algorithm is useful to solve problems where the solution space is so vast to make a brute-force approach feasible. For instance, having a computer produce a certain sequence of letters matching an input string. The problem is trivial, but helps to demonstrate how a genetic algorithm works.
Inspired by actual biological evolution and specifically darwinian natural selection, genetic algorithms consider three key principles:
-
heredity, a way to pass data from generation to generation
-
variation, a way to provide different traits, be it in the initial population or as the simulation evolves
-
selection, a way to pick and choose a trait over another
The principles are put into practice in a series of steps, and are detailed in the context of a specific problem.
-
population
At the start of the algorithm, the goal is to create a population of elements with random genetic material. For instance, and for the problem at hand, the population is a collection of words as long as the desired sentence, and with random letters from the alphabet.
The larger, the more varied a population is, the easier it is to find a solution. In the instance of the sentence, the more words, and the more characters are represented in the population, the more rapidly the algorithm will find a match. The size of the population, however, affects the algorithm negatively as well, as the program needs to process a large set of values.
It is here important to distinguish between genotype, the genetic code making up the elements in the population, and phenotype, the expression of the code. In the particular example, the two happen to be one and the same, as the sequence of characters represents both the data and how the data is visualized.
-
selection
The idea is to evaluate the population to describe a fitness value for each and every element. On the basis of this value, the algorithm produces a selection, a mating pool of candidates from which the program can later pick.
The fitness function is one of the key ingredients of the algorithm, as it allows to dictate the manner in which a trait is preferred over another. For instance, in the specific problem, the fitness function evaluates the number of characters matching the desired string.
How the collection is then created is another crucial step. There are different approaches to this problem, but a first solution is to pick an element proportional to its fitness value. The probability is estimated by creating a bucket of options, and including multiple copies of the elements according to the mentioned metrics.
-
reproduction
Considering darwinian selection, the algorithm picks two elements from the mating pool and produces a new generation. This is where heredity comes into play, as the new value depends on the genetic material (the characters) of the parents (the two words). There are again different approaches, but in the traditional implementation the genetic material is influenced by:
-
crossover, where the element inherits directly from the parents; for instance a word picks the first half of one parent, and the second half of the other
-
mutation, where the element has a chance to produce a different trait; for instance, a certain probability to use a character at random
Mutation allows to cope with a population that doesn't have the traits desired in the solution; for instance, a set of words without a character used in the sentence.
Similarly to the size of the population in the setup phase, there are many variables affecting the efficiency and efficacy of the algorithm. Among these values, the probability to introduce a different trait contributes to find a solution, but if excessive makes it harder to improve the fitness score. For instance, the more random the number of characters, the more the algorithm will try to find a solution in the insurmountable way described earlier, picking characters at random.
-
The demo works to illustrate the brute force approach of finding a match for a four letter word. While ineffective, it also works to introduce the building blocks for the genetic algorithm:
-
getRandomCharacter
returns a random character betweena
andz
lowercase; the string library allows to rapidly switch between character and numerical representation, but it is important to stress the importance of the decided range.string.byte("a") -- 97 string.byte("z") -- 122
With a sentence including spaces, punctuation and other special characters, it is essential to widen the set of possible values.
string.byte(" ") -- 32
-
getRandomWord
returns a sequence of characters according to the input length.The characters are actually added to a table, and the function ultimately returns a string by concatenating the values.
return table.concat(word)
This is in line with a consideration made for the
L-system
demos, which argued for tables as a substitude for string buffers. It is more efficient for lua to add items to a table than it is to concatenate characters to a string.
With a four letter word and twenty-six possible characters, the odds of finding a match are already a measly 1
in 456 976
, motivating a different approach.
Please note: the demo differs from the logic discussed in the book in that the program works through functions instead of dedicated entities, like Population
or DNA
classes. This is on purpose to have the project comparable with Shakesperian monkey
.
The steps of the traditional genetic algorithm are included in the love.load
and love.update
functions.
-
in
love.load
, the population is initialized with a series of random words. The functions mentioned in the previous demo are useful to create the sequence of random characters -
in
love.update
, and as long as a match is not found, the script repeats a series of steps in order to modify the original population with values associated with a greater fitselection
works here as the mating pool described in the book. A collection where the words of the population are included in a number proportional to their fitness value.local selection = getSelection(population, sentence)
getSelection
loops through the population and computes the fitness for each and every word, before including a comparable number of copies in the colleciton.function getSelection(population, sentence) local selection = {} for i, word in ipairs(population) do local fitness = getFitness(word, sentence) for j = 1, fitness do table.insert(selection, word) end return selection end
getFitness
counts the number of matches between word and sentence.function getFitness(word, sentence) local fitness = 0 for i = 1, #sentence do if word:sub(i, i) == sentence:sub(i, i) then fitness = fitness + 1 end end return fitness end
With a given
selection
, the idea is to essentially replace the old population with new values, children created from parents picked from the mating pool.-- in love.update for i = 1, #population do local child = getChild(selection) population[i] = child end
getChild
implements the reproduction discussed earlier considering two parents, picked at random from the selection. Here you find both a crossover and a mutation:-
crossover: the child inherits letters from the parents alternating between the two
for i = 1, #sentence do child[#child + 1] = i % 2 == 1 and p1:sub(i, i) or p2:sub(i, i) end
-
mutation: with given odds, the child picks a character at random
for i = 1, #sentence do if math.random(MUTATION_ODDS) == 1 then child[#child + 1] = getRandomCharacter() else -- inherit end end
-
And that is essentially it for the genetic algorithm. With each iteration, population
includes fitter and fitter values, until a children is able to reproduce the input sentence.
The rest of the logic described in love.update
is useful to:
-
identify the word in the population with the best fit
-
stop the iterative process when the input is reproduced
-
store the best fit in a
words
collection, used to show the result in the window
The idea is to refactor the previous demo to implement the algorithm with DNA
and Population
entities. This allows the script to have a more general structure, one in which love.load
initializes a population and love.update
modifies the population with a series of functions.
population:select()
population:reproduce()
How the population is initialized, how the mating pool, children, parents, fitness value are calculated is then a matter delegated to the specific entities.
The exercise doesn't introduce concepts, but there are a couple of notable differences with respect to the previous demo:
-
the set of characters is described in the function initializing the population to have a table of lowercase and uppercase letters (
[A-z]
), plus the space, comma and period characters -
the number of copies included in the mating pool is proportional to the fitness of each word, but instead of adding a copy for each match, the fitness is compared to the maximum fitness value in the current generation
-
the fitness is squared to increase how likely it is to pick a sentence with a higher value
The project updates the traditional genetic algorithm in the way it selects two parents elements. Instead of populating a selection
table with a number of copies proportional to the fitness ratio, the script picks a dna at random and accepts or rejects the element according to a probability. The probability is mapped to the maximum fitness ratio, so that the greater the ratio, the more likely it will be for the element to be picked.
local parent1 = self:select(maxFitnessRatio)
local parent2 = self:select(maxFitnessRatio)
The select
function continues to pick a dna until a suitable parent is found.
while true do
local dna = self.population[math.random(#self.population)]
local fitnessRatio = dna:getFitnessRatio(self.target)
local probability = math.random() * maxFitnessRatio
if probability < fitnessRatio then
return dna
end
end
The fitness ratio is also squared to preserve the intention of the previous demo, that is increase the importance of higher values.
local fitnessRatio = dna:getFitnessRatio(self.target)
fitnessRatio = fitnessRatio ^ 2
With this demo I intend to apply the logic of the traditional genetic algorithm to the demo created in chapter 6 to moves vehicles according to a random flow field (the-nature-of-code\06 Autonomous Agents\Flow field\Random
).
The demo is modified in increments and as follows:
-
each vehicle is attributed its own field; the grid of forces described by the field is ultimately the genetic information passed from generation to generation
-
the movement of the vehicle is constrained to the edges of the window. Unlike the previous demo, where the vehicle would appear on the opposite side of the window, the entity stops by multiplying the velocity vector by
0
-
the simulation includes a target, whose position is most relevant in terms of fitness function; the position is set to have the population and target at opposite ends, but it is possible to reposition the target with the mouse cursor
-
the fitness function computes a value in the
[0, 1]
range mapping the distance between vehicle and target;MAX_DISTANCE_VEHICLE
is helpful to provide an upper threshold, whileMIN_FITNESS
is helpful to avoid having a population whose fitness value is equal to0
. In this last instance, the program would encounter an infinite loop as theselect
function would be stuck to find an impossible value -
the population includes a variable
lifespan
to terminate the current generation after a brief amount of time; the idea is to have a new generation spawn when the timer reaches this variable, after evaluating the current generation for the most fit flow field
The project is useful to stress the difference between genotype (genetic material, DNA
) and phenotype (visual representation, Vehicle
) as introduced in the beginning of the chapter. In future demos the distinction is reiterated showing the genetic material with faces or again particles.
In the exercise the genetic algorithm is tailored to have a population evolve as a result of user input. The demo shows a collection of faces, changing the size, color and position of a few prominent features. When the mouse hovers upon a specific face, the fitness value is increased by an arbitrary amount. When a new generation is created, the population consider the elements with higher fitness values so that the face becomes a mixture of the preferred patterns.
Please note:
- the mutation is increased to introduce more variety in how the population evolves
The project sets up a genetic algorithm to simulate an ecosystem evolving over time. Instead of having a clear stopping point, where the population evolves abruptly from generation to generation, the goal is to have a series of entities live on their own.
Each entity, or bloop, has a lifespan, counting down to 0
, at which point the particle dies. The lifespan is mapped to the opacity of the figure, so that the members of the population slowly fade out, but the value is increased when an entity collides with a pellet of food. Randomly and while still alive, each bloop has also a possibility to spawn a new particle, to which it passes its own genetic material in terms of size.
Variety is included in form of the size of the bloop. The size is inversely proportional to the speed, so that larger particles move slower than smaller ones. Eventually, the simulation shows a preference for a particular type of particle, preferring a smaller, faster entity than a larger, slower one.
Please note:
- the focus of the exercise is on the evolution of an ecosystem, where entities have their own lifecycle. There is no fitness value as in previous algorithms
The goal is to have a population of rockets navigate the environment toward an arbitrary target. The movement is influenced by a series of vectors describing the acceleration of the rocket at different frames. As the generation comes to an end, the fitness is computed on the basis of the distance between rocket and target, so that eventually, the population learns to follow a similar, efficient trajectory.
From this starting point, the demo is updated to have a more complex simulation, with one or more obstacles interrupting the movement of the entities. By default, the simulation includes a single obstacle, in the right half of the window, but following mouse input, it is possible to complicate the environment with additional platforms.
The fitness value takes account of the distance, but also the obstacles and target. In this light, hitting an obstacle, or the edges of the window, results in a greatly diminished fitness.
if self.hasCollided then
return fitness * 0.1
end
Hitting the target, on the other hand, increases the fitness considering the number of frames it takes the rocket to reach the target.
if self.hasReached then
return fitness * FRAMES / self.frames
end
self.frames
is a counter variable for each rocket, incrementing each time the rocket moves. Dividing by this value means the program has a preference for entities reaching the target in less frames. Multiplying for this value, however, the preference is set for slower rockets. This could be preferable in a scenario where the goal is to get closer to the target, without crashing in it.
if self.hasReached then
return fitness * self.frames / FRAMES
end
Please note:
-
the project is proposed as the last example in the chapter devoted to genetic algorithms, but in the book and conceptually, it follows the lessons learned with the evolutionary flow field
-
the code considering mouse input is taken directly from a previous demo in the chapter devoted to physics libraries and Box2D (and specifically the
Fixed
folder)
A neural network is introduced as a complex, adaptive system; a system which learns over time to solve a specific problem.
A first example of neural network is a perceptron, a system with one or more inputs, a processor and a single output.
input 1
-> processor -> output
input 2
input n
The perceptron is a type of network described as feed forward, meaning that learning happens by feeding the system a series of input values and measuring the output against a known result. As the result in known in advance, the network is also an example of supervised learning, meaning the system is trained to provide an acceptable answer on the basis of training data.
The network associates a weight to each input.
input1 x weight1
-> processor -> output
input2 x weight2
These weights are then adjusted as the network encounters a mistake, an error between guessed and known output.
In the specific demo, the structure is used to solve a classification problem, where the window is divided in two areas and the task is to categorize an input point to a specific region.
1 1/
1 / -1
/ -1
/
The perceptron receives as input an x
and y
coordinate and returns 1
or -1
according to whether the point is above or below a diagonal. The problem could be solved with linear algebra, but with a network, it is approached with a series of points describing the training data. The perceptron guesses the output starting from a random set of weights, and tunes these parameters to respect the training data.
A guess works considering the inputs, weights, and an activation function. The goal of an activation function is to produce the output in a desired form (in this instance 1
or -1
). In the specific demo, the function considers the sign of the weighted sum for all the input values (sum of the values multiplied by the respective weights).
function Perceptron:activate(n)
return n >= 0 and 1 or -1
end
It is important to note that evaluating the sign is but one of the possible activation functions. It is useful here since it provides a binary, clear way to return 1
or -1
.
Training is performed with a series of points which are already assigned the correct output value.
function Point:new()
local this = {
["label"] = x > y and 1 or -1
}
end
As it receives a point, the perceptron evaluates the error as the difference between guess and answer.
local guess = self:guess(inputs)
local error = target - guess
The combinations are described by this short table. From this error, the weights are adjusted by multiplying the error for the input value.
for i = 1, #self.weights do
self.weights[i] = self.weights[i] + error * inputs[i]
end
This is similar to the example describing the steering behavior for autonomous agents. The difference between desired and actual velocity provides the error, which is then used to steer the agent toward the desired target.
The correction is multiplied by a learning rate, to have the network assimilate the error, learn from its mistake, by moving the weights toward the desired values.
self.weights[i] = self.weights[i] + error * inputs[i] * LEARNING_RATE
This is again similar to the steering example. How much the error modifies the velocity describes how rapidly the vehicle changes its direction, and if it approaches the desired trajectory or overshoots the same value.
In the demo, the perceptron is highlighted with a series of points, colored with a fill or stroke according to the label 1
and -1
. The perceptron makes a series of guesses and highlights them with a green or red outline. At first the guesses are wrong, but by pressing the mouse cursor, the perceptron is trained with more and more points, improving the decision with each additional datum.
The first demo implementing the perceptron is modified to have the network consider a more general classification problem, where a point is assigned -1
or 1
on the basis of the equation of a line
y = slope * x + coefficient
The demo is updated to also consider a cartesian coordinate system, where the inputs are in the [-1, 1]
range and the values are mapped to the pixel coordinates with a map
function.
Given the new coordinate system, it is finally necessary to introduce a third input in a bias
function Point:new()
local this = {
["inputs"] = {x, y, 1},
}
end
The value is necessary to consider a valid guess for the (0, 0)
origin.
In order to develop a multilayer neural network, it is helpful to have a small library to manage data in matrix format, that is data in rows and columns. Matrix.lua
implements matrix and matrix math through a series of functions.
The functions can be split in two main categories: those which modify the instance matrix, and those which return a new instance altogether. In the first category, it is possible to modify the instance with:
-
add
; the elements of the matrix are increased by a scalar or the respective elements of another matrix -
multiply
; the elements are multiplied by a scalar, or again the respective elements of another matrix. This last operation is known as Hadamard product, but is not the only way with which two matrices are multiplied. If the number of rows of the first matrix matches the number of columns of the second entity, the function proceeds with the matrix product -
transpose
; the configuration of the matrix is modified to have the rows as columns and viceversa -
map
; the elements are modified with the logic described by the input function. This logic extends the way it is possible to alter the matrix's own values to any function accepting an input
In the second category, it is possible to add and multiply matrices. What changes is that the input matrixes are not modified.
In terms of object oriented programming, it is necessary to decipher if the input of the add
or multiply
function is a matrix, since the logic is different from when the input is a scalar. To achieve this with lua lua, it would be enough to check the metatable of the input.
if getmetatable(input) == self
end
The isInstance
function slightly refines this approach by returning a boolean, and considering the metatables of the input table in a loop.
function isInstance(instance, class)
while instance do
instance = getmetatable(instance)
if instance == getmetatable(class) then
return true
end
end
return false
end
Since getmetatable
returns the metatable of the instance, and nil
when a metatable is no longer available, it is possible to cycle through all the "parent metatables", and to check multiple layers.