forcamla is a framework for creating mathematical formula that automatically update their value when their subformula change. In addition, formula can be used to create a powerful system of event listeners that execute based on the value a formula currently has.
We start by learning constants in OCaml as ways to have names hold values for us to use later. For instance,
let x = 2
let z = x * xsimply defines an integer constant x that has the value 2 and we later use x again to make z have the value of x * x = 2 * 2 = 4. But often in programming constants are not enough because we want to change the values these names have after a sequence of instructions. Thus, OCaml has ref types for this, observe
let x = ref 2 (* Let x be a *variable* that contains 2 *)
x := !x + 1 (* Update x so it is now x + 1 = 2 + 1 = 3 *)However, even this has it limitations. Suppose we have the following scenario
let x = ref 2
let y = ref 2
let z = ref (!x + !y) (* Make z a variable equal to 2 + 2 = 4 *)
x := 3What happens to z? Does z change to reflect the fact that x changed? Or does z stay the same? Most languages (including OCaml) do not update z when x changes. This might defy your intuition. Even worse, this behavior may be inconvenient to deal with. This is where forcamla comes into play!
forcamla introduces a formula type that behaves similarly to ref types but formula stay up-to-date when their subformulas change. Modifying the example above we have
open Formula (* To use formula *)
let x = t 2 (* Create an integer term called x *)
let y = t 2 (* Create an integer term called y *)
let z = x + y
x =:: 3 (* Set x to 3, and z now is 5 *)But it gets even better!
Suppose you're making a game. You probably have player and maybe it has type hero with a bunch of fields including health.
open Formula
type hero =
{
(* A bunuch of fields *)
health: int formula
}
let player =
{
(* Assign the fields *)
health = t 3; (* Give health a value of something, say 3 in this case. *)
}In most games, when the player.health is 0 that means "Game Over!". So, we first make a "Game Over!" function:
let game_over () = print_endline "Game Over!"Then, with the power of forcamla we can create an event listener (see the MDN documentation to understand what an event listener is) to execute game_over when the player.health is 0. Observe
let () = when_satisfied (player.health =? 0) game_over=? is used to check if an integer formula is equal to another (integer) formula. This reads as: "When player.health is 0 (i.e. when the formula is satisfied!) fire the function game_over". Therfore, if we run:
let () = player.health =:: (player.health - 1) (* Nothing happens yet! player.health is 2 now. *)
let () = player.health =:: (player.health - 1) (* Nothing happens yet! player health is 1 now. *)
let () = player.health =:: (player.health - 1) (* Now something happens! player.health is 0 and "Game Over!" is printed to the screen! *)The type for player.health =? 0 is not actually a 'a formula! We have a separate type for equations called system (short for "system of equations"). This is just a "system" of 1 equation, but as we will see we can combine equations together with (&&) and (||) to create actual systems of multiple equations!
Formula The fundamental types: formulae (combined refs), and systems (collections of equations where each equation is two formula joined by a comparison operator.).