Writing a simple symbolic computation system using R (Part 1/4)
This post is very old. Many things may be wrong due to things that happened since then, or because I did not have a good understanding of what I was writing about.
This post was originally written in portuguese.
This week I found myself thinking on how packages like dplyr use a lot of metaprogramming (that is, computation over the language itself) and functional tools to create functions having a lot of expressive power. I asked myself: can I possibly use these tools for algebraic manipulation? In particular, could I use them to build a system capable of doing symbolic differentiation?
I came up with the differentiation problem in particular because it’s clear that finding derivatives of elementary functions consists of mere formal (though tedious) manipulation. Symbolic integration, in contrast, is more an art than an algorithm.
After playing with the idea and a few days of tinkering, I figured out that doing it is not only possible, but it also can be done with relative ease. It’s an exercise that illustrates pretty well how R’s functional and metaprogramming capabilities translate to concise and expressive code.
I wanted this system, (or, being fancy, domain-specific language) to satisfy four requisites:
- It must describe concisely all elementary functions from calculus, that is, anything that’s built with polynomials, exponentials, logarithms and trigonometric functions.
- It must be able to manipulate them both symbolically and numerically.
- It must be able to compute their derivatives symbolically.
- The computed symbolic expressions must be simple, whenever possible.
Let it be clear that I’m not a symbolic computation expert, so a system for real-world use would probably need to implement a lot of optimizations to the algorithms I use.
Let’s get started, then. First, I’m going to need three packages: purrr, lazyeval, and glue, so install them if you do not have them installed yet. I’ll explain the functions as I use them.
The first tough detail is that I want functions to be callable. In other words, I want them to be evaluated using the traditional function syntax: the function name followed by a pair of parentheses enclosing its argument. At the same time, I need them to store additional data, such as a string representation, its derivative or its inverse. To do it, I’m going to need R attributes.
In R, every object can store data through attributes, setting them using attr(). To access them, you can use attr() itself or the operator %@% provided by purrr. Thus, I only need to take an ordinary mathematical function and add the data needed through attributes.
For convenience, I also define print and as.character methods for the symbolic class, so working with these functions in an interactive environment becomes more pleasurable. Finally, I’m going to create a function for computing the n-th derivative of a function symbolically.
library(purrr)
library(glue)
library(lazyeval)
symbolic = function(f, repr, df, type, params = list(), inverse = NULL) {
  class(f) = c("symbolic", "function")
  attr(f, "repr") = repr # a string representation of the function
  attr(f, "df") = lazy(df) # its derivative
  attr(f, "inverse") = lazy(inverse) # its inverse. We're not gonna use it yet.
  attr(f, "type") = type # what kind of function this is
  attr(f, "params") = params # the parameters which define a function of this type
  f
}
# auxiliary functions
is_symbolic = function(x) inherits(x, "symbolic")
as.character.symbolic = function(f) f%@%"repr"
print.symbolic = function(f) cat(glue("x -> {f}"))
# The differential operator. Takes the n-th derivative of a function recursively
D = function(f, n = 1) {
  if (n == 0) # does nothing
    f
  else if (n == 1) # for the first derivative, it only evaluates the df attribute
    lazy_eval(f%@%"df")
  else # the n-th derivative is the derivative of the (n-1)-th derivative
    D(lazy_eval(f%@%"df"), n - 1)
}
A very important tool for us, used extensively through this project, is lazy evaluation. Its usefulness will become clear when I define the first funcion with this system: the null function. We know that the derivative of the null function is the null function itself. I will define it in a seemingly circular way. Instinctively, it should result in some kind of error, but it doesn’t. Look:
Null = symbolic(
  f = function(x) 0,
  repr = "0",
  df = Null,
  type = "null"
)
is_nullf = function(x) x%@%"type" == "null"
Null
#> x -> 0
Null(5)
#> [1] 0
D(Null) # the null function itself
#> x -> 0
I’m defining Null as the result of symbolic() applied to a list of parameters, one of them being Null itself, and still it works fine? How come?
The lazy() function, used inside symbolic(), is the key. It captures the expression that defines a derivative, but does not evaluate it. In fact, I do not need to compute the derivative of a function the moment I define it, I just need to know how to compute it when I need. That’s precisely what the lazy(df) line is doing. It’s storing the expression defining the derivative so later, when I call D(), I can use lazy_eval() to obtain it.
Try running the previous code removing the references to lazy(). You’re going to get this error message:
Error in symbolic(f = function(x) 0, repr = "0", df = Null, type = "null") :
  object 'Null' not found
This happens because Null hasn’t been defined yet the moment symbolic() is called. Lazy evaluation avoids this problem. It says that, the moment I want to compute the derivative of Null, the result is going to be Null itself. But then Null will already be defined! Problem solved :)
Using the same idea, we can define a few more classes of functions:
Const = function(c) {
  if (c == 0)
    return(Null)
  symbolic(
    f = function(x) c,
    repr = as.character(c),
    df = Null,
    type = "const",
    params = list(c = c)
  )  
}
is_const = function(x) x%@%"type" == "const"
# Monomials
Mono = function(a = 1, n = 1) {
  if (a == 0)
    return(Null)
  if (n == 0)
    return(Const(a))
  symbolic(
    f = function(x) a*x^n,
    repr = glue("{a}*x^{n}"),
    df = Mono(n*a, n-1),
    type = "mono",
    params = list(a = a, n = n),
    inverse = Mono(1/a^(1/n), 1/n)
  )
}
is_mono = function(x) x%@%"type" == "mono"
Log = symbolic(
  f = log,
  repr = "log(x)",
  df = Mono(1, -1),
  type = "log",
  inverse = Exp
)
is_log = function(x) x%@%"type" == "log"
Exp = symbolic(
  f = exp,
  repr = "exp(x)",
  df = Exp,
  type = "exp",
  inverse = Log
)
is_exp = function(x) x%@%"type" == "exp"
Strictly speaking, the formula for the inverse of a monomial I gave above is valid only when the function is defined for positive numbers only. Tackling this kind of problem requires a sophistication I don’t want to engage with in this series of posts.
Now, let’s test the functions to see if everything is working fine.
f = Const(3)
f(5)
#> [1] 3
D(f)
#> x -> 0
D(f, 2)
#> x -> 0
f = Mono(0.5, 2)
f(3)
#> [1] 4.5
D(f)
#> x -> 1*x^1
D(f, 2)
#> x -> 1
f = Log
f(exp(1))
#> [1] 1
D(f)
#> x -> 1*x^-1
D(f, 2)
#> x -> -1*x^-2
f = Exp
f(1)
#> [1] 2.718282
D(f)
#> x -> exp(x)
D(f, 2)
#> x -> exp(x)
It is!
Ok, in this post we were able to define some elementary functions and compute their derivatives, but we couldn’t have much fun with them yet. Calculus becomes interesting when we can define new functions using sums, products and compositions (highlights). In the next post of this series I’m going to explain how we can add these features. See you!
- Part 2/4: Defining sums, products, and compositions
- Part 3/4: Adding algebraic simplifications
- Part 4/4: Producing smarter string representations
