Racket features
Racket has been under active development as a vehicle for programming language research since the mid-1990s, and has accumulated many features over the years. This article describes and demonstrates some of these features. Note that one of Racket's main design goals is to accommodate creating new languages, both domain-specific languages and completely new languages.
Therefore, some of the following examples are in different languages, but they are all implemented in Racket. Please refer to the main article for more information.
The core Racket implementation is highly flexible. Even without using dialects, it can function as a full-featured scripting language, capable of running both with and without windows-native GUI, and capable of tasks from web server creation to graphics.
Runtime Support
Garbage Collection, Tail Calls, and Space Safety
Racket can use three different garbage collectors:- Originally, the conservative Boehm garbage collector was used. However, conservative collection is impractical for long-running processes such as a web server—such processes tend to slowly leak memory. In addition, there are pathological cases where a conservative collector leaks memory fast enough to make certain programs impossible to run. For example, when traversing an infinite list, a single conservative mistake of retaining a pointer leads to keeping the complete list in memory, quickly overflowing available memory. This collector is often referred to as "CGC" in the Racket community.
- SenoraGC is an alternative conservative garbage collector that is intended mainly for debugging and memory tracing.
- The moving memory manager is a precise garbage collector, and it has been Racket's default collector since 2007. This collector is a generational one, and it supports memory accounting via custodians. The collector is implemented as a C source transformer that is itself written in Racket. Therefore, the build process uses the conservative collector for bootstrapping.
System Interface and Scripting
Racket's system interface includes asynchronous non-blocking I/O, green threads, synchronization channels, semaphores, sub-processes, and TCP sockets.The following program starts an "echo server" on port 12345.
- lang racket
)
;; handle an incoming connection in a thread
))
;; and immediately loop back to accept additional clients
)
The combination of dynamic compilation and a rich system interface makes Racket a capable scripting language, similar to Perl or Python.
The following example demonstrates walking a directory tree, starting at the current directory. It uses the
in-directory
function to construct a sequence that walks the tree. The for
form binds path
to each path in the sequence, and regexp-match?
tests these paths against the given regexp pattern.- lang racket
))
The next example uses a hash table to record previously seen lines and print only unique ones.
- lang racket
)
))
Both of these programs can be run in DrRacket, or on the command line, via the
racket
executable. Racket ignores an initial shebang line, making it possible to turn such programs to executable scripts. The following script demonstrates this, in addition to using Racket's library for command-line argument parsing:- !/usr/bin/env racket
- lang racket
#:when
)))
The script is a grep-like utility, expecting three command-line arguments: a base directory, a [filename extension, and a regular expression. It scans the base directory for files with the given suffix, and print lines matching the regexp pattern.
Resource Management and Sandboxing
Racket features the concept of a "custodian": a kind of value that acts as a resource manager. This is often used in network servers, where each connection is dealt with in a new custodian, making it easy to "clean-up" all resources that might have been left open by the handler. The following extends the "echo server" example with such a custodian use:- lang racket
)
)
))))
)
Custodians, combined with the memory accounting feature of the 3m garbage collector, and a number of additional runtime parameters that control additional aspects of the runtime, make it possible to create completely safe sandboxed execution contexts. The
racket/sandbox
library provides this kind of functionality in a simple way. The following example creates a "REPL server" on the specified port; connecting to this port will look like a plain Racket REPL, except that the evaluation is subject to the various protection aspects of the sandbox. For example, it is not possible to access the filesystem from this REPL, create network connection, run subprocesses, or use too much time or memory. - lang racket
))
Web and Network Programming
The next example implements a web server using theweb-server/insta
language. Each time a connection is made to the server, the start
function is called to get the HTML to send back to the client.- lang web-server/insta
)
Racket also includes the functions you would need to write scrapers and robots. As an example, the following function would list the Google results for a search string.
- lang racket
.*?
))
The library also includes support for protocols other than http:
- lang racket
) ; wait 3h 45m
"Parking meter alert!"
null null
')
Graphics
Graphic capabilities come in several different flavors that are intended for different audiences. The2htdp/image
library provides convenient functions for constructing images. This library is mainly used by students in HtDP-based courses. In the following example, a sierpinski
function is defined and called to generate a Sierpinski triangle of depth 8.- lang racket
)))
DrRacket editors can contain images, and DrRacket displays image values just like any other type of value. Running the above program, for example, actually displays a Sierpinski triangle, which can be cut and pasted into another program.
The
plot
library constructs image values for more mature audiences and needs. For example, the following program plots the sum of two Gaussians, as concentric, partially transparent surfaces:- lang racket
;; Note: this example requires Racket 5.2 or later
;; Returns an R x R x R -> R Gaussian function centered at
)))
;; Lifts + to operate on three-argument functions
))
;; Constructs an image value representing the sum of two Gaussians
)
-1 2.5 -2.5 1 -1 1
#:label "g")) ; labeling adds a legend
Here, the
isosurfaces3d
function requires a three-argument function for its first argument, which the curried f3+
supplies. Besides constructing image values, plot
can also write files in PNG, PDF, PostScript and SVG formats.GUI Programming
Racket implements a portable GUI layer which the libraries mentioned above build on. It is implemented via the native Windows API, via Cocoa on Mac OS X, and via GTK+ on Linux and others. The Racket API is a class-based toolkit, somewhat related to wxWidgets which was used originally.The following simple guessing game demonstrates coding with the GUI toolkit. The
frame%
class implements a top-level window, and button%
implements a button. The check
function defined here produces a function that is used for the button's callback action.- lang racket/gui
; toplevel window
; horizontal container
"Too small"]
))
)) ; success => close window
p ))
; show the window to start the application
The GUI can be hand-coded in this way or with the help of a GUI designer program available on PLaneT.
Slideshow
-based presentations can also be developed in Racket using theslideshow
language, much like Beamer, but with Racket's programmatic facilities. Elements of the slides are pictures that can be combined.For example, the following program displays in full-screen a title slide, followed by a slide with some pictures. The
vc-append
and hc-append
functions combine pictures vertically and horizontally, respectively, and centered on the other axis.- lang slideshow
)
))))
Extension packages also exist on PLaneT, for example to include LaTeX elements.
Foreign Function Interface
Racket features a foreign function interface that is based on libffi. The interface allows writing unsafe low-level C-like code, that can allocate memory, dereference pointers, call out to functions in shared libraries, and send out callbacks to Racket functions. The core implementation is a thin layer atop libffi, and the full interface is then implemented via Racket code. The interface uses macros extensively, resulting in an expressive Racket-based interface description language. This language has a number of useful features, such as uniform representation for higher-order functions, struct definitions that are similar to plain Racket structs, and custom function types that can represent input and output pointers, implicit arguments. By using this interface to access underlying GUI toolkits, Racket implements its own GUI layer completely in Racket.The FFI can be used in a number of different ways: from writing a complete glue layer for a library, to quickly pulling out a single foreign function. An example of the latter approach:
- lang racket/base
Language Extensions
Racket's most notable feature is its ability to build new domain-specific and general-purpose languages. This is the result of combining a number of important features:- a flexible module system that is used for linking code and for namespace management,
- an extensive macro system—functioning as a compiler-API—that can create new syntactic forms,
- a rich runtime system, providing features that language implementors can use, like continuations, resource management, etc.,
- a way to specify parsers for new language syntaxes.
Such languages are used extensively in the Racket distribution and in user libraries. In fact, creating a new language is so straightforward, that there are some languages that have less than a handful of uses.
Racket comes with a number of useful languages, some are very different from Racket's default language.
Scribble
Scribble, Racket's documentation system, comes in the form of a number of languages that are used to write prose. It is used for Racket's documentation, as well as writing books and articles. Actually, rather than a single "scribble" language, it is a family of dialects, each for a different purpose.To run the following example, copy it into DrRacket and click one of the two scribble rendering buttons that will appear. Alternatively, use the
scribble
executable on the file.- lang scribble/base
@
@title
In case you need some @emph in your life.
@
@item))
The most striking feature of the Scribble languages is their use of a new syntax, which is designed specifically for textually rich code. The syntax allows free-form text, string interpolation, customizable quotations, and is useful in other applications such as preprocessing text, generating text, and HTML template systems. Note that the syntax extends plain S-expressions, and is implemented as an alternative input for such expressions.
- lang scribble/text
I'm a text file -- run me.
@
@thrice!
@thrice!
Typed Racket
Typed Racket is a statically typed variant of Racket. The type system that it implements is unique in that the motivation in developing it was accommodating as much idiomatic Racket code as possible—as a result, it includes subtypes, unions, and much more. Another goal of Typed Racket is to allow migration of parts of a program into the typed language, so it accommodates calling typed code from untyped code and vice versa, generating dynamic contracts to enforce type invariants. This is considered a desirable feature of an application's lifetime stages, as it matures from "a script" to "an application", where static typing helps in maintenance of a large body of code.- lang typed/racket
)
)
Lazy Racket
Thelazy
language is a language with lazy evaluation semantics, similar to Haskell. In the following example, fibs
is an infinite list whose 1000th element will not be computed until its value is needed for the printout.- lang lazy
)
;; Print the 1000th Fibonacci number:
Logic Programming
Racket comes with three logic programming languages: Racklog, a Prolog-like language; a Datalog implementation; and a miniKanren port. Unlike the Scribble syntax, the first two of these languages use a completely new syntax rather than an extension of S-expressions. If you use it in DrRacket, you'll see that it provides proper highlighting, the usual host of tools check syntax, and a Prolog/Datalog REPL.- lang datalog
ancestor :-
parent, D = C, ancestor.
parent.
parent.
ancestor?
Educational Tools
The PLT group which develops Racket has traditionally been involved in education at all levels. One of the earliest research ideas that the group promoted is the use of language levels, which restrict new students while providing them with helpful error messages that fit the student's level of knowledge. This approach is heavily used in How to Design Programs, the textbook that several PLT developers have authored, as well as in the ProgramByDesign project. The following program uses thehtdp/bsl
—the "beginning student language". It uses the 2htdp/image
library for creating pictures in the teaching languages, and the 2htdp/universe
library for interactive animations.- lang htdp/bsl
)
)
)
Algol
Racket comes with a complete implementation of the ALGOL 60 language.- lang algol60
integer procedure SIGMA;
value n;
integer x, i, n;
begin
integer sum;
sum := 0;
for i := 1 step 1 until n do
sum := sum + x;
SIGMA := sum;
end;
integer q;
printnln;
end
Plai and plai-typed
Another supported language is plai which like racket can be typed or untyped. "Modules written in plai export every definition." "The Typed PLAI language differs from traditional Racket most importantly by being statically typed. It also gives you some useful new constructs: define-type, type-case, and test."
Additional Languages
Finally, the following example is an implementation of a new language:- lang racket
))
This language:
- provides everything from the
racket
language, so it is a somewhat similar variant, - except for two special "hook macros" that implement unbound variable lookup and function calls, instead of these, new forms are provided to
- * implicitly quote all unknown variables
- * allow hash tables to be used as functions, where the arguments are used for hash-table lookup.
mylang.rkt
file, you can use it as follows:- lang s-exp "mylang.rkt" ; sexpr syntax, using mylang semantics
; the hash table is used as a function