123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593 |
- @c -*-texinfo-*-
- @c This is part of the GNU Guile Reference Manual.
- @c Copyright (C) 1996-1997, 2000-2005, 2012, 2019
- @c Free Software Foundation, Inc.
- @c See the file guile.texi for copying conditions.
- @node Hello Scheme!
- @chapter Hello Scheme!
- In this chapter, we introduce the basic concepts that underpin the
- elegance and power of the Scheme language.
- Readers who already possess a background knowledge of Scheme may happily
- skip this chapter. For the reader who is new to the language, however,
- the following discussions on data, procedures, expressions and closure
- are designed to provide a minimum level of Scheme understanding that is
- more or less assumed by the chapters that follow.
- The style of this introductory material aims about halfway between the terse
- precision of R5RS and the discursiveness of existing Scheme tutorials. For
- pointers to useful Scheme resources on the web, please see @ref{Further
- Reading}.
- @menu
- * About Data:: Latent typing, types, values and variables.
- * About Procedures:: The representation and use of procedures.
- * About Expressions:: All kinds of expressions and their meaning.
- * About Closure:: Closure, scoping and environments.
- * Further Reading:: Where to find out more about Scheme.
- @end menu
- @node About Data
- @section Data Types, Values and Variables
- This section discusses the representation of data types and values, what
- it means for Scheme to be a @dfn{latently typed} language, and the role
- of variables. We conclude by introducing the Scheme syntaxes for
- defining a new variable, and for changing the value of an existing
- variable.
-
- @menu
- * Latent Typing:: Scheme as a "latently typed" language.
- * Values and Variables:: About data types, values and variables.
- * Definition:: Defining variables and setting their values.
- @end menu
- @node Latent Typing
- @subsection Latent Typing
- The term @dfn{latent typing} is used to describe a computer language,
- such as Scheme, for which you cannot, @emph{in general}, simply look at
- a program's source code and determine what type of data will be
- associated with a particular variable, or with the result of a
- particular expression.
- Sometimes, of course, you @emph{can} tell from the code what the type of
- an expression will be. If you have a line in your program that sets the
- variable @code{x} to the numeric value 1, you can be certain that,
- immediately after that line has executed (and in the absence of multiple
- threads), @code{x} has the numeric value 1. Or if you write a procedure
- that is designed to concatenate two strings, it is likely that the rest
- of your application will always invoke this procedure with two string
- parameters, and quite probable that the procedure would go wrong in some
- way if it was ever invoked with parameters that were not both strings.
- Nevertheless, the point is that there is nothing in Scheme which
- requires the procedure parameters always to be strings, or @code{x}
- always to hold a numeric value, and there is no way of declaring in your
- program that such constraints should always be obeyed. In the same
- vein, there is no way to declare the expected type of a procedure's
- return value.
- Instead, the types of variables and expressions are only known -- in
- general -- at run time. If you @emph{need} to check at some point that
- a value has the expected type, Scheme provides run time procedures that
- you can invoke to do so. But equally, it can be perfectly valid for two
- separate invocations of the same procedure to specify arguments with
- different types, and to return values with different types.
- The next subsection explains what this means in practice, for the ways
- that Scheme programs use data types, values and variables.
- @node Values and Variables
- @subsection Values and Variables
- Scheme provides many data types that you can use to represent your data.
- Primitive types include characters, strings, numbers and procedures.
- Compound types, which allow a group of primitive and compound values to
- be stored together, include lists, pairs, vectors and multi-dimensional
- arrays. In addition, Guile allows applications to define their own data
- types, with the same status as the built-in standard Scheme types.
- As a Scheme program runs, values of all types pop in and out of
- existence. Sometimes values are stored in variables, but more commonly
- they pass seamlessly from being the result of one computation to being
- one of the parameters for the next.
- Consider an example. A string value is created because the interpreter
- reads in a literal string from your program's source code. Then a
- numeric value is created as the result of calculating the length of the
- string. A second numeric value is created by doubling the calculated
- length. Finally the program creates a list with two elements -- the
- doubled length and the original string itself -- and stores this list in
- a program variable.
- All of the values involved here -- in fact, all values in Scheme --
- carry their type with them. In other words, every value ``knows,'' at
- runtime, what kind of value it is. A number, a string, a list,
- whatever.
- A variable, on the other hand, has no fixed type. A variable --
- @code{x}, say -- is simply the name of a location -- a box -- in which
- you can store any kind of Scheme value. So the same variable in a
- program may hold a number at one moment, a list of procedures the next,
- and later a pair of strings. The ``type'' of a variable -- insofar as
- the idea is meaningful at all -- is simply the type of whatever value
- the variable happens to be storing at a particular moment.
- @node Definition
- @subsection Defining and Setting Variables
- To define a new variable, you use Scheme's @code{define} syntax like
- this:
- @lisp
- (define @var{variable-name} @var{value})
- @end lisp
- This makes a new variable called @var{variable-name} and stores
- @var{value} in it as the variable's initial value. For example:
- @lisp
- ;; Make a variable `x' with initial numeric value 1.
- (define x 1)
- ;; Make a variable `organization' with an initial string value.
- (define organization "Free Software Foundation")
- @end lisp
- (In Scheme, a semicolon marks the beginning of a comment that continues
- until the end of the line. So the lines beginning @code{;;} are
- comments.)
- Changing the value of an already existing variable is very similar,
- except that @code{define} is replaced by the Scheme syntax @code{set!},
- like this:
- @lisp
- (set! @var{variable-name} @var{new-value})
- @end lisp
- Remember that variables do not have fixed types, so @var{new-value} may
- have a completely different type from whatever was previously stored in
- the location named by @var{variable-name}. Both of the following
- examples are therefore correct.
- @lisp
- ;; Change the value of `x' to 5.
- (set! x 5)
- ;; Change the value of `organization' to the FSF's street number.
- (set! organization 545)
- @end lisp
- In these examples, @var{value} and @var{new-value} are literal numeric
- or string values. In general, however, @var{value} and @var{new-value}
- can be any Scheme expression. Even though we have not yet covered the
- forms that Scheme expressions can take (@pxref{About Expressions}), you
- can probably guess what the following @code{set!} example does@dots{}
- @lisp
- (set! x (+ x 1))
- @end lisp
- (Note: this is not a complete description of @code{define} and
- @code{set!}, because we need to introduce some other aspects of Scheme
- before the missing pieces can be filled in. If, however, you are
- already familiar with the structure of Scheme, you may like to read
- about those missing pieces immediately by jumping ahead to the following
- references.
- @itemize @bullet
- @item
- @ref{Lambda Alternatives}, to read about an alternative form of the
- @code{define} syntax that can be used when defining new procedures.
- @item
- @ref{Procedures with Setters}, to read about an alternative form of the
- @code{set!} syntax that helps with changing a single value in the depths
- of a compound data structure.)
- @item
- @xref{Internal Definitions}, to read about using @code{define} other
- than at top level in a Scheme program, including a discussion of when it
- works to use @code{define} rather than @code{set!} to change the value
- of an existing variable.
- @end itemize
- @node About Procedures
- @section The Representation and Use of Procedures
- This section introduces the basics of using and creating Scheme
- procedures. It discusses the representation of procedures as just
- another kind of Scheme value, and shows how procedure invocation
- expressions are constructed. We then explain how @code{lambda} is used
- to create new procedures, and conclude by presenting the various
- shorthand forms of @code{define} that can be used instead of writing an
- explicit @code{lambda} expression.
- @menu
- * Procedures as Values:: Procedures are values like everything else.
- * Simple Invocation:: How to write a simple procedure invocation.
- * Creating a Procedure:: How to create your own procedures.
- * Lambda Alternatives:: Other ways of writing procedure definitions.
- @end menu
- @node Procedures as Values
- @subsection Procedures as Values
- One of the great simplifications of Scheme is that a procedure is just
- another type of value, and that procedure values can be passed around
- and stored in variables in exactly the same way as, for example, strings
- and lists. When we talk about a built-in standard Scheme procedure such
- as @code{open-input-file}, what we actually mean is that there is a
- pre-defined top level variable called @code{open-input-file}, whose
- value is a procedure that implements what R5RS says that
- @code{open-input-file} should do.
- Note that this is quite different from many dialects of Lisp ---
- including Emacs Lisp --- in which a program can use the same name with
- two quite separate meanings: one meaning identifies a Lisp function,
- while the other meaning identifies a Lisp variable, whose value need
- have nothing to do with the function that is associated with the first
- meaning. In these dialects, functions and variables are said to live in
- different @dfn{namespaces}.
- In Scheme, on the other hand, all names belong to a single unified
- namespace, and the variables that these names identify can hold any kind
- of Scheme value, including procedure values.
- One consequence of the ``procedures as values'' idea is that, if you
- don't happen to like the standard name for a Scheme procedure, you can
- change it.
- For example, @code{call-with-current-continuation} is a very important
- standard Scheme procedure, but it also has a very long name! So, many
- programmers use the following definition to assign the same procedure
- value to the more convenient name @code{call/cc}.
- @lisp
- (define call/cc call-with-current-continuation)
- @end lisp
- Let's understand exactly how this works. The definition creates a new
- variable @code{call/cc}, and then sets its value to the value of the
- variable @code{call-with-current-continuation}; the latter value is a
- procedure that implements the behavior that R5RS specifies under the
- name ``call-with-current-continuation''. So @code{call/cc} ends up
- holding this value as well.
- Now that @code{call/cc} holds the required procedure value, you could
- choose to use @code{call-with-current-continuation} for a completely
- different purpose, or just change its value so that you will get an
- error if you accidentally use @code{call-with-current-continuation} as a
- procedure in your program rather than @code{call/cc}. For example:
- @lisp
- (set! call-with-current-continuation "Not a procedure any more!")
- @end lisp
- Or you could just leave @code{call-with-current-continuation} as it was.
- It's perfectly fine for more than one variable to hold the same
- procedure value.
- @node Simple Invocation
- @subsection Simple Procedure Invocation
- A procedure invocation in Scheme is written like this:
- @lisp
- (@var{procedure} [@var{arg1} [@var{arg2} @dots{}]])
- @end lisp
- In this expression, @var{procedure} can be any Scheme expression whose
- value is a procedure. Most commonly, however, @var{procedure} is simply
- the name of a variable whose value is a procedure.
- For example, @code{string-append} is a standard Scheme procedure whose
- behavior is to concatenate together all the arguments, which are
- expected to be strings, that it is given. So the expression
- @lisp
- (string-append "/home" "/" "andrew")
- @end lisp
- @noindent
- is a procedure invocation whose result is the string value
- @code{"/home/andrew"}.
- Similarly, @code{string-length} is a standard Scheme procedure that
- returns the length of a single string argument, so
- @lisp
- (string-length "abc")
- @end lisp
- @noindent
- is a procedure invocation whose result is the numeric value 3.
- Each of the parameters in a procedure invocation can itself be any
- Scheme expression. Since a procedure invocation is itself a type of
- expression, we can put these two examples together to get
- @lisp
- (string-length (string-append "/home" "/" "andrew"))
- @end lisp
- @noindent
- --- a procedure invocation whose result is the numeric value 12.
- (You may be wondering what happens if the two examples are combined the
- other way round. If we do this, we can make a procedure invocation
- expression that is @emph{syntactically} correct:
- @lisp
- (string-append "/home" (string-length "abc"))
- @end lisp
- @noindent
- but when this expression is executed, it will cause an error, because
- the result of @code{(string-length "abc")} is a numeric value, and
- @code{string-append} is not designed to accept a numeric value as one of
- its arguments.)
- @node Creating a Procedure
- @subsection Creating and Using a New Procedure
- Scheme has lots of standard procedures, and Guile provides all of these
- via predefined top level variables. All of these standard procedures
- are documented in the later chapters of this reference manual.
- Before very long, though, you will want to create new procedures that
- encapsulate aspects of your own applications' functionality. To do
- this, you can use the famous @code{lambda} syntax.
- For example, the value of the following Scheme expression
- @lisp
- (lambda (name address) @var{body} @dots{})
- @end lisp
- @noindent
- is a newly created procedure that takes two arguments: @code{name} and
- @code{address}. The behavior of the new procedure is determined by the
- sequence of expressions and definitions in the @var{body} of the
- procedure definition. (Typically, @var{body} would use the arguments in
- some way, or else there wouldn't be any point in giving them to the
- procedure.) When invoked, the new procedure returns a value that is the
- value of the last expression in the @var{body}.
- To make things more concrete, let's suppose that the two arguments are
- both strings, and that the purpose of this procedure is to form a
- combined string that includes these arguments. Then the full lambda
- expression might look like this:
- @lisp
- (lambda (name address)
- (string-append "Name=" name ":Address=" address))
- @end lisp
- We noted in the previous subsection that the @var{procedure} part of a
- procedure invocation expression can be any Scheme expression whose value
- is a procedure. But that's exactly what a lambda expression is! So we
- can use a lambda expression directly in a procedure invocation, like
- this:
- @lisp
- ((lambda (name address)
- (string-append "Name=" name ":Address=" address))
- "FSF"
- "Cambridge")
- @end lisp
- @noindent
- This is a valid procedure invocation expression, and its result is the
- string:
- @lisp
- "Name=FSF:Address=Cambridge"
- @end lisp
- It is more common, though, to store the procedure value in a variable ---
- @lisp
- (define make-combined-string
- (lambda (name address)
- (string-append "Name=" name ":Address=" address)))
- @end lisp
- @noindent
- --- and then to use the variable name in the procedure invocation:
- @lisp
- (make-combined-string "FSF" "Cambridge")
- @end lisp
- @noindent
- Which has exactly the same result.
- It's important to note that procedures created using @code{lambda} have
- exactly the same status as the standard built in Scheme procedures, and
- can be invoked, passed around, and stored in variables in exactly the
- same ways.
- @node Lambda Alternatives
- @subsection Lambda Alternatives
- Since it is so common in Scheme programs to want to create a procedure
- and then store it in a variable, there is an alternative form of the
- @code{define} syntax that allows you to do just that.
- A @code{define} expression of the form
- @lisp
- (define (@var{name} [@var{arg1} [@var{arg2} @dots{}]])
- @var{body} @dots{})
- @end lisp
- @noindent
- is exactly equivalent to the longer form
- @lisp
- (define @var{name}
- (lambda ([@var{arg1} [@var{arg2} @dots{}]])
- @var{body} @dots{}))
- @end lisp
- So, for example, the definition of @code{make-combined-string} in the
- previous subsection could equally be written:
- @lisp
- (define (make-combined-string name address)
- (string-append "Name=" name ":Address=" address))
- @end lisp
- This kind of procedure definition creates a procedure that requires
- exactly the expected number of arguments. There are two further forms
- of the @code{lambda} expression, which create a procedure that can
- accept a variable number of arguments:
- @lisp
- (lambda (@var{arg1} @dots{} . @var{args}) @var{body} @dots{})
- (lambda @var{args} @var{body} @dots{})
- @end lisp
- @noindent
- The corresponding forms of the alternative @code{define} syntax are:
- @lisp
- (define (@var{name} @var{arg1} @dots{} . @var{args}) @var{body} @dots{})
- (define (@var{name} . @var{args}) @var{body} @dots{})
- @end lisp
- @noindent
- For details on how these forms work, see @xref{Lambda}.
- Prior to Guile 2.0, Guile provided an extension to @code{define} syntax
- that allowed you to nest the previous extension up to an arbitrary
- depth. These are no longer provided by default, and instead have been
- moved to @ref{Curried Definitions}.
- (It could be argued that the alternative @code{define} forms are rather
- confusing, especially for newcomers to the Scheme language, as they hide
- both the role of @code{lambda} and the fact that procedures are values
- that are stored in variables in the same way as any other kind of value.
- On the other hand, they are very convenient, and they are also a good
- example of another of Scheme's powerful features: the ability to specify
- arbitrary syntactic transformations at run time, which can be applied to
- subsequently read input.)
- @node About Expressions
- @section Expressions and Evaluation
- So far, we have met expressions that @emph{do} things, such as the
- @code{define} expressions that create and initialize new variables, and
- we have also talked about expressions that have @emph{values}, for
- example the value of the procedure invocation expression:
- @lisp
- (string-append "/home" "/" "andrew")
- @end lisp
- @noindent
- but we haven't yet been precise about what causes an expression like
- this procedure invocation to be reduced to its ``value'', or how the
- processing of such expressions relates to the execution of a Scheme
- program as a whole.
- This section clarifies what we mean by an expression's value, by
- introducing the idea of @dfn{evaluation}. It discusses the side effects
- that evaluation can have, explains how each of the various types of
- Scheme expression is evaluated, and describes the behavior and use of
- the Guile REPL as a mechanism for exploring evaluation. The section
- concludes with a very brief summary of Scheme's common syntactic
- expressions.
- @menu
- * Evaluating:: How a Scheme program is executed.
- * Tail Calls:: Space-safe recursion.
- * The REPL:: Interacting with the Guile interpreter.
- * Syntax Summary:: Common syntactic expressions -- in brief.
- @end menu
- @node Evaluating
- @subsection Evaluating Expressions and Executing Programs
- In Scheme, the process of executing an expression is known as
- @dfn{evaluation}. Evaluation has two kinds of result:
- @itemize @bullet
- @item
- the @dfn{value} of the evaluated expression
- @item
- the @dfn{side effects} of the evaluation, which consist of any effects of
- evaluating the expression that are not represented by the value.
- @end itemize
- Of the expressions that we have met so far, @code{define} and
- @code{set!} expressions have side effects --- the creation or
- modification of a variable --- but no value; @code{lambda} expressions
- have values --- the newly constructed procedures --- but no side
- effects; and procedure invocation expressions, in general, have either
- values, or side effects, or both.
- It is tempting to try to define more intuitively what we mean by
- ``value'' and ``side effects'', and what the difference between them is.
- In general, though, this is extremely difficult. It is also
- unnecessary; instead, we can quite happily define the behavior of a
- Scheme program by specifying how Scheme executes a program as a whole,
- and then by describing the value and side effects of evaluation for each
- type of expression individually.
- @noindent
- So, some@footnote{These definitions are approximate. For the whole
- and detailed truth, see @ref{Formal syntax and semantics,R5RS
- syntax,,r5rs,The Revised(5) Report on the Algorithmic Language
- Scheme}.} definitions@dots{}
- @itemize @bullet
- @item
- A Scheme program consists of a sequence of expressions.
- @item
- A Scheme interpreter executes the program by evaluating these
- expressions in order, one by one.
- @item
- An expression can be
- @itemize @bullet
- @item
- a piece of literal data, such as a number @code{2.3} or a string
- @code{"Hello world!"}
- @item
- a variable name
- @item
- a procedure invocation expression
- @item
- one of Scheme's special syntactic expressions.
- @end itemize
- @end itemize
- @noindent
- The following subsections describe how each of these types of expression
- is evaluated.
- @menu
- * Eval Literal:: Evaluating literal data.
- * Eval Variable:: Evaluating variable references.
- * Eval Procedure:: Evaluating procedure invocation expressions.
- * Eval Special:: Evaluating special syntactic expressions.
- @end menu
- @node Eval Literal
- @subsubsection Evaluating Literal Data
- When a literal data expression is evaluated, the value of the expression
- is simply the value that the expression describes. The evaluation of a
- literal data expression has no side effects.
- @noindent
- So, for example,
- @itemize @bullet
- @item
- the value of the expression @code{"abc"} is the string value
- @code{"abc"}
- @item
- the value of the expression @code{3+4i} is the complex number 3 + 4i
- @item
- the value of the expression @code{#(1 2 3)} is a three-element vector
- containing the numeric values 1, 2 and 3.
- @end itemize
- For any data type which can be expressed literally like this, the syntax
- of the literal data expression for that data type --- in other words,
- what you need to write in your code to indicate a literal value of that
- type --- is known as the data type's @dfn{read syntax}. This manual
- specifies the read syntax for each such data type in the section that
- describes that data type.
- Some data types do not have a read syntax. Procedures, for example,
- cannot be expressed as literal data; they must be created using a
- @code{lambda} expression (@pxref{Creating a Procedure}) or implicitly
- using the shorthand form of @code{define} (@pxref{Lambda Alternatives}).
- @node Eval Variable
- @subsubsection Evaluating a Variable Reference
- When an expression that consists simply of a variable name is evaluated,
- the value of the expression is the value of the named variable. The
- evaluation of a variable reference expression has no side effects.
- So, after
- @lisp
- (define key "Paul Evans")
- @end lisp
- @noindent
- the value of the expression @code{key} is the string value @code{"Paul
- Evans"}. If @var{key} is then modified by
- @lisp
- (set! key 3.74)
- @end lisp
- @noindent
- the value of the expression @code{key} is the numeric value 3.74.
- If there is no variable with the specified name, evaluation of the
- variable reference expression signals an error.
- @node Eval Procedure
- @subsubsection Evaluating a Procedure Invocation Expression
- This is where evaluation starts getting interesting! As already noted,
- a procedure invocation expression has the form
- @lisp
- (@var{procedure} [@var{arg1} [@var{arg2} @dots{}]])
- @end lisp
- @noindent
- where @var{procedure} must be an expression whose value, when evaluated,
- is a procedure.
- The evaluation of a procedure invocation expression like this proceeds
- by
- @itemize @bullet
- @item
- evaluating individually the expressions @var{procedure}, @var{arg1},
- @var{arg2}, and so on
- @item
- calling the procedure that is the value of the @var{procedure}
- expression with the list of values obtained from the evaluations of
- @var{arg1}, @var{arg2} etc. as its parameters.
- @end itemize
- For a procedure defined in Scheme, ``calling the procedure with the list
- of values as its parameters'' means binding the values to the
- procedure's formal parameters and then evaluating the sequence of
- expressions that make up the body of the procedure definition. The
- value of the procedure invocation expression is the value of the last
- evaluated expression in the procedure body. The side effects of calling
- the procedure are the combination of the side effects of the sequence of
- evaluations of expressions in the procedure body.
- For a built-in procedure, the value and side-effects of calling the
- procedure are best described by that procedure's documentation.
- Note that the complete side effects of evaluating a procedure invocation
- expression consist not only of the side effects of the procedure call,
- but also of any side effects of the preceding evaluation of the
- expressions @var{procedure}, @var{arg1}, @var{arg2}, and so on.
- To illustrate this, let's look again at the procedure invocation
- expression:
- @lisp
- (string-length (string-append "/home" "/" "andrew"))
- @end lisp
- In the outermost expression, @var{procedure} is @code{string-length} and
- @var{arg1} is @code{(string-append "/home" "/" "andrew")}.
- @itemize @bullet
- @item
- Evaluation of @code{string-length}, which is a variable, gives a
- procedure value that implements the expected behavior for
- ``string-length''.
- @item
- Evaluation of @code{(string-append "/home" "/" "andrew")}, which is
- another procedure invocation expression, means evaluating each of
- @itemize @bullet
- @item
- @code{string-append}, which gives a procedure value that implements the
- expected behavior for ``string-append''
- @item
- @code{"/home"}, which gives the string value @code{"/home"}
- @item
- @code{"/"}, which gives the string value @code{"/"}
- @item
- @code{"andrew"}, which gives the string value @code{"andrew"}
- @end itemize
- and then invoking the procedure value with this list of string values as
- its arguments. The resulting value is a single string value that is the
- concatenation of all the arguments, namely @code{"/home/andrew"}.
- @end itemize
- In the evaluation of the outermost expression, the interpreter can now
- invoke the procedure value obtained from @var{procedure} with the value
- obtained from @var{arg1} as its arguments. The resulting value is a
- numeric value that is the length of the argument string, which is 12.
- @node Eval Special
- @subsubsection Evaluating Special Syntactic Expressions
- When a procedure invocation expression is evaluated, the procedure and
- @emph{all} the argument expressions must be evaluated before the
- procedure can be invoked. Special syntactic expressions are special
- because they are able to manipulate their arguments in an unevaluated
- form, and can choose whether to evaluate any or all of the argument
- expressions.
- Why is this needed? Consider a program fragment that asks the user
- whether or not to delete a file, and then deletes the file if the user
- answers yes.
- @lisp
- (if (string=? (read-answer "Should I delete this file?")
- "yes")
- (delete-file file))
- @end lisp
- If the outermost @code{(if @dots{})} expression here was a procedure
- invocation expression, the expression @code{(delete-file file)}, whose
- side effect is to actually delete a file, would already have been
- evaluated before the @code{if} procedure even got invoked! Clearly this
- is no use --- the whole point of an @code{if} expression is that the
- @dfn{consequent} expression is only evaluated if the condition of the
- @code{if} expression is ``true''.
- Therefore @code{if} must be special syntax, not a procedure. Other
- special syntaxes that we have already met are @code{define}, @code{set!}
- and @code{lambda}. @code{define} and @code{set!} are syntax because
- they need to know the variable @emph{name} that is given as the first
- argument in a @code{define} or @code{set!} expression, not that
- variable's value. @code{lambda} is syntax because it does not
- immediately evaluate the expressions that define the procedure body;
- instead it creates a procedure object that incorporates these
- expressions so that they can be evaluated in the future, when that
- procedure is invoked.
- The rules for evaluating each special syntactic expression are specified
- individually for each special syntax. For a summary of standard special
- syntax, see @xref{Syntax Summary}.
- @node Tail Calls
- @subsection Tail calls
- @cindex tail calls
- @cindex recursion
- Scheme is ``properly tail recursive'', meaning that tail calls or
- recursions from certain contexts do not consume stack space or other
- resources and can therefore be used on arbitrarily large data or for
- an arbitrarily long calculation. Consider for example,
- @example
- (define (foo n)
- (display n)
- (newline)
- (foo (1+ n)))
- (foo 1)
- @print{}
- 1
- 2
- 3
- @dots{}
- @end example
- @code{foo} prints numbers infinitely, starting from the given @var{n}.
- It's implemented by printing @var{n} then recursing to itself to print
- @math{@var{n}+1} and so on. This recursion is a tail call, it's the
- last thing done, and in Scheme such tail calls can be made without
- limit.
- Or consider a case where a value is returned, a version of the SRFI-1
- @code{last} function (@pxref{SRFI-1 Selectors}) returning the last
- element of a list,
- @example
- (define (my-last lst)
- (if (null? (cdr lst))
- (car lst)
- (my-last (cdr lst))))
- (my-last '(1 2 3)) @result{} 3
- @end example
- If the list has more than one element, @code{my-last} applies itself
- to the @code{cdr}. This recursion is a tail call, there's no code
- after it, and the return value is the return value from that call. In
- Scheme this can be used on an arbitrarily long list argument.
- @sp 1
- A proper tail call is only available from certain contexts, namely the
- following special form positions,
- @itemize @bullet
- @item
- @code{and} --- last expression
- @item
- @code{begin} --- last expression
-
- @item
- @code{case} --- last expression in each clause
- @item
- @code{cond} --- last expression in each clause, and the call to a
- @code{=>} procedure is a tail call
- @item
- @code{do} --- last result expression
- @item
- @code{if} --- ``true'' and ``false'' leg expressions
- @item
- @code{lambda} --- last expression in body
- @item
- @code{let}, @code{let*}, @code{letrec}, @code{let-syntax},
- @code{letrec-syntax} --- last expression in body
- @item
- @code{or} --- last expression
- @end itemize
- @noindent
- The following core functions make tail calls,
- @itemize @bullet
- @item
- @code{apply} --- tail call to given procedure
- @item
- @code{call-with-current-continuation} --- tail call to the procedure
- receiving the new continuation
- @item
- @code{call-with-values} --- tail call to the values-receiving
- procedure
- @item
- @code{eval} --- tail call to evaluate the form
- @item
- @code{string-any}, @code{string-every} --- tail call to predicate on
- the last character (if that point is reached)
- @end itemize
- @sp 1
- The above are just core functions and special forms. Tail calls in
- other modules are described with the relevant documentation, for
- example SRFI-1 @code{any} and @code{every} (@pxref{SRFI-1 Searching}).
- It will be noted there are a lot of places which could potentially be
- tail calls, for instance the last call in a @code{for-each}, but only
- those explicitly described are guaranteed.
- @node The REPL
- @subsection Using the Guile REPL
- If you start Guile without specifying a particular program for it to
- execute, Guile enters its standard Read Evaluate Print Loop --- or
- @dfn{REPL} for short. In this mode, Guile repeatedly reads in the next
- Scheme expression that the user types, evaluates it, and prints the
- resulting value.
- The REPL is a useful mechanism for exploring the evaluation behavior
- described in the previous subsection. If you type @code{string-append},
- for example, the REPL replies @code{#<primitive-procedure
- string-append>}, illustrating the relationship between the variable
- @code{string-append} and the procedure value stored in that variable.
- In this manual, the notation @result{} is used to mean ``evaluates
- to''. Wherever you see an example of the form
- @lisp
- @var{expression}
- @result{}
- @var{result}
- @end lisp
- @noindent
- feel free to try it out yourself by typing @var{expression} into the
- REPL and checking that it gives the expected @var{result}.
- @node Syntax Summary
- @subsection Summary of Common Syntax
- This subsection lists the most commonly used Scheme syntactic
- expressions, simply so that you will recognize common special syntax
- when you see it. For a full description of each of these syntaxes,
- follow the appropriate reference.
- @code{lambda} (@pxref{Lambda}) is used to construct procedure objects.
- @code{define} (@pxref{Top Level}) is used to create a new variable and
- set its initial value.
- @code{set!} (@pxref{Top Level}) is used to modify an existing variable's
- value.
- @code{let}, @code{let*} and @code{letrec} (@pxref{Local Bindings})
- create an inner lexical environment for the evaluation of a sequence of
- expressions, in which a specified set of local variables is bound to the
- values of a corresponding set of expressions. For an introduction to
- environments, see @xref{About Closure}.
- @code{begin} (@pxref{begin}) executes a sequence of expressions in order
- and returns the value of the last expression. Note that this is not the
- same as a procedure which returns its last argument, because the
- evaluation of a procedure invocation expression does not guarantee to
- evaluate the arguments in order.
- @code{if} and @code{cond} (@pxref{Conditionals}) provide conditional
- evaluation of argument expressions depending on whether one or more
- conditions evaluate to ``true'' or ``false''.
- @code{case} (@pxref{Conditionals}) provides conditional evaluation of
- argument expressions depending on whether a variable has one of a
- specified group of values.
- @code{and} (@pxref{and or}) executes a sequence of expressions in order
- until either there are no expressions left, or one of them evaluates to
- ``false''.
- @code{or} (@pxref{and or}) executes a sequence of expressions in order
- until either there are no expressions left, or one of them evaluates to
- ``true''.
- @node About Closure
- @section The Concept of Closure
- @cindex closure
- The concept of @dfn{closure} is the idea that a lambda expression
- ``captures'' the variable bindings that are in lexical scope at the
- point where the lambda expression occurs. The procedure created by the
- lambda expression can refer to and mutate the captured bindings, and the
- values of those bindings persist between procedure calls.
- This section explains and explores the various parts of this idea in
- more detail.
- @menu
- * About Environments:: Names, locations, values and environments.
- * Local Variables:: Local variables and local environments.
- * Chaining:: Environment chaining.
- * Lexical Scope:: The meaning of lexical scoping.
- * Closure:: Explaining the concept of closure.
- * Serial Number:: Example 1: a serial number generator.
- * Shared Variable:: Example 2: a shared persistent variable.
- * Callback Closure:: Example 3: the callback closure problem.
- * OO Closure:: Example 4: object orientation.
- @end menu
- @node About Environments
- @subsection Names, Locations, Values and Environments
- @cindex location
- @cindex environment
- @cindex vcell
- @cindex top level environment
- @cindex environment, top level
- We said earlier that a variable name in a Scheme program is associated
- with a location in which any kind of Scheme value may be stored.
- (Incidentally, the term ``vcell'' is often used in Lisp and Scheme
- circles as an alternative to ``location''.) Thus part of what we mean
- when we talk about ``creating a variable'' is in fact establishing an
- association between a name, or identifier, that is used by the Scheme
- program code, and the variable location to which that name refers.
- Although the value that is stored in that location may change, the
- location to which a given name refers is always the same.
- We can illustrate this by breaking down the operation of the
- @code{define} syntax into three parts: @code{define}
- @itemize @bullet
- @item
- creates a new location
- @item
- establishes an association between that location and the name specified
- as the first argument of the @code{define} expression
- @item
- stores in that location the value obtained by evaluating the second
- argument of the @code{define} expression.
- @end itemize
- A collection of associations between names and locations is called an
- @dfn{environment}. When you create a top level variable in a program
- using @code{define}, the name-location association for that variable is
- added to the ``top level'' environment. The ``top level'' environment
- also includes name-location associations for all the procedures that are
- supplied by standard Scheme.
- It is also possible to create environments other than the top level one,
- and to create variable bindings, or name-location associations, in those
- environments. This ability is a key ingredient in the concept of
- closure; the next subsection shows how it is done.
- @node Local Variables
- @subsection Local Variables and Environments
- @cindex local variable
- @cindex variable, local
- @cindex local environment
- @cindex environment, local
- We have seen how to create top level variables using the @code{define}
- syntax (@pxref{Definition}). It is often useful to create variables
- that are more limited in their scope, typically as part of a procedure
- body. In Scheme, this is done using the @code{let} syntax, or one of
- its modified forms @code{let*} and @code{letrec}. These syntaxes are
- described in full later in the manual (@pxref{Local Bindings}). Here
- our purpose is to illustrate their use just enough that we can see how
- local variables work.
- For example, the following code uses a local variable @code{s} to
- simplify the computation of the area of a triangle given the lengths of
- its three sides.
- @lisp
- (define a 5.3)
- (define b 4.7)
- (define c 2.8)
- (define area
- (let ((s (/ (+ a b c) 2)))
- (sqrt (* s (- s a) (- s b) (- s c)))))
- @end lisp
- The effect of the @code{let} expression is to create a new environment
- and, within this environment, an association between the name @code{s}
- and a new location whose initial value is obtained by evaluating
- @code{(/ (+ a b c) 2)}. The expressions in the body of the @code{let},
- namely @code{(sqrt (* s (- s a) (- s b) (- s c)))}, are then evaluated
- in the context of the new environment, and the value of the last
- expression evaluated becomes the value of the whole @code{let}
- expression, and therefore the value of the variable @code{area}.
- @node Chaining
- @subsection Environment Chaining
- @cindex shadowing an imported variable binding
- @cindex chaining environments
- In the example of the previous subsection, we glossed over an important
- point. The body of the @code{let} expression in that example refers not
- only to the local variable @code{s}, but also to the top level variables
- @code{a}, @code{b}, @code{c} and @code{sqrt}. (@code{sqrt} is the
- standard Scheme procedure for calculating a square root.) If the body
- of the @code{let} expression is evaluated in the context of the
- @emph{local} @code{let} environment, how does the evaluation get at the
- values of these top level variables?
- The answer is that the local environment created by a @code{let}
- expression automatically has a reference to its containing environment
- --- in this case the top level environment --- and that the Scheme
- interpreter automatically looks for a variable binding in the containing
- environment if it doesn't find one in the local environment. More
- generally, every environment except for the top level one has a
- reference to its containing environment, and the interpreter keeps
- searching back up the chain of environments --- from most local to top
- level --- until it either finds a variable binding for the required
- identifier or exhausts the chain.
- This description also determines what happens when there is more than
- one variable binding with the same name. Suppose, continuing the
- example of the previous subsection, that there was also a pre-existing
- top level variable @code{s} created by the expression:
- @lisp
- (define s "Some beans, my lord!")
- @end lisp
- Then both the top level environment and the local @code{let} environment
- would contain bindings for the name @code{s}. When evaluating code
- within the @code{let} body, the interpreter looks first in the local
- @code{let} environment, and so finds the binding for @code{s} created by
- the @code{let} syntax. Even though this environment has a reference to
- the top level environment, which also has a binding for @code{s}, the
- interpreter doesn't get as far as looking there. When evaluating code
- outside the @code{let} body, the interpreter looks up variable names in
- the top level environment, so the name @code{s} refers to the top level
- variable.
- Within the @code{let} body, the binding for @code{s} in the local
- environment is said to @dfn{shadow} the binding for @code{s} in the top
- level environment.
- @node Lexical Scope
- @subsection Lexical Scope
- The rules that we have just been describing are the details of how
- Scheme implements ``lexical scoping''. This subsection takes a brief
- diversion to explain what lexical scope means in general and to present
- an example of non-lexical scoping.
- ``Lexical scope'' in general is the idea that
- @itemize @bullet
- @item
- an identifier at a particular place in a program always refers to the
- same variable location --- where ``always'' means ``every time that the
- containing expression is executed'', and that
- @item
- the variable location to which it refers can be determined by static
- examination of the source code context in which that identifier appears,
- without having to consider the flow of execution through the program as
- a whole.
- @end itemize
- In practice, lexical scoping is the norm for most programming languages,
- and probably corresponds to what you would intuitively consider to be
- ``normal''. You may even be wondering how the situation could possibly
- --- and usefully --- be otherwise. To demonstrate that another kind of
- scoping is possible, therefore, and to compare it against lexical
- scoping, the following subsection presents an example of non-lexical
- scoping and examines in detail how its behavior differs from the
- corresponding lexically scoped code.
- @menu
- * Scoping Example:: An example of non-lexical scoping.
- @end menu
-
- @node Scoping Example
- @subsubsection An Example of Non-Lexical Scoping
- To demonstrate that non-lexical scoping does exist and can be useful, we
- present the following example from Emacs Lisp, which is a ``dynamically
- scoped'' language.
- @lisp
- (defvar currency-abbreviation "USD")
- (defun currency-string (units hundredths)
- (concat currency-abbreviation
- (number-to-string units)
- "."
- (number-to-string hundredths)))
- (defun french-currency-string (units hundredths)
- (let ((currency-abbreviation "FRF"))
- (currency-string units hundredths)))
- @end lisp
- The question to focus on here is: what does the identifier
- @code{currency-abbreviation} refer to in the @code{currency-string}
- function? The answer, in Emacs Lisp, is that all variable bindings go
- onto a single stack, and that @code{currency-abbreviation} refers to the
- topmost binding from that stack which has the name
- ``currency-abbreviation''. The binding that is created by the
- @code{defvar} form, to the value @code{"USD"}, is only relevant if none
- of the code that calls @code{currency-string} rebinds the name
- ``currency-abbreviation'' in the meanwhile.
- The second function @code{french-currency-string} works precisely by
- taking advantage of this behavior. It creates a new binding for the
- name ``currency-abbreviation'' which overrides the one established by
- the @code{defvar} form.
- @lisp
- ;; Note! This is Emacs Lisp evaluation, not Scheme!
- (french-currency-string 33 44)
- @result{}
- "FRF33.44"
- @end lisp
- Now let's look at the corresponding, @emph{lexically scoped} Scheme
- code:
- @lisp
- (define currency-abbreviation "USD")
- (define (currency-string units hundredths)
- (string-append currency-abbreviation
- (number->string units)
- "."
- (number->string hundredths)))
- (define (french-currency-string units hundredths)
- (let ((currency-abbreviation "FRF"))
- (currency-string units hundredths)))
- @end lisp
- According to the rules of lexical scoping, the
- @code{currency-abbreviation} in @code{currency-string} refers to the
- variable location in the innermost environment at that point in the code
- which has a binding for @code{currency-abbreviation}, which is the
- variable location in the top level environment created by the preceding
- @code{(define currency-abbreviation @dots{})} expression.
- In Scheme, therefore, the @code{french-currency-string} procedure does
- not work as intended. The variable binding that it creates for
- ``currency-abbreviation'' is purely local to the code that forms the
- body of the @code{let} expression. Since this code doesn't directly use
- the name ``currency-abbreviation'' at all, the binding is pointless.
- @lisp
- (french-currency-string 33 44)
- @result{}
- "USD33.44"
- @end lisp
- This begs the question of how the Emacs Lisp behavior can be
- implemented in Scheme. In general, this is a design question whose
- answer depends upon the problem that is being addressed. In this case,
- the best answer may be that @code{currency-string} should be
- redesigned so that it can take an optional third argument. This third
- argument, if supplied, is interpreted as a currency abbreviation that
- overrides the default.
- It is possible to change @code{french-currency-string} so that it mostly
- works without changing @code{currency-string}, but the fix is inelegant,
- and susceptible to interrupts that could leave the
- @code{currency-abbreviation} variable in the wrong state:
- @lisp
- (define (french-currency-string units hundredths)
- (set! currency-abbreviation "FRF")
- (let ((result (currency-string units hundredths)))
- (set! currency-abbreviation "USD")
- result))
- @end lisp
- The key point here is that the code does not create any local binding
- for the identifier @code{currency-abbreviation}, so all occurrences of
- this identifier refer to the top level variable.
- @node Closure
- @subsection Closure
- Consider a @code{let} expression that doesn't contain any
- @code{lambda}s:
- @lisp
- (let ((s (/ (+ a b c) 2)))
- (sqrt (* s (- s a) (- s b) (- s c))))
- @end lisp
- @noindent
- When the Scheme interpreter evaluates this, it
- @itemize @bullet
- @item
- creates a new environment with a reference to the environment that was
- current when it encountered the @code{let}
- @item
- creates a variable binding for @code{s} in the new environment, with
- value given by @code{(/ (+ a b c) 2)}
- @item
- evaluates the expression in the body of the @code{let} in the context of
- the new local environment, and remembers the value @code{V}
- @item
- forgets the local environment
- @item
- continues evaluating the expression that contained the @code{let}, using
- the value @code{V} as the value of the @code{let} expression, in the
- context of the containing environment.
- @end itemize
- After the @code{let} expression has been evaluated, the local
- environment that was created is simply forgotten, and there is no longer
- any way to access the binding that was created in this environment. If
- the same code is evaluated again, it will follow the same steps again,
- creating a second new local environment that has no connection with the
- first, and then forgetting this one as well.
- If the @code{let} body contains a @code{lambda} expression, however, the
- local environment is @emph{not} forgotten. Instead, it becomes
- associated with the procedure that is created by the @code{lambda}
- expression, and is reinstated every time that that procedure is called.
- In detail, this works as follows.
- @itemize @bullet
- @item
- When the Scheme interpreter evaluates a @code{lambda} expression, to
- create a procedure object, it stores the current environment as part of
- the procedure definition.
- @item
- Then, whenever that procedure is called, the interpreter reinstates the
- environment that is stored in the procedure definition and evaluates the
- procedure body within the context of that environment.
- @end itemize
- The result is that the procedure body is always evaluated in the context
- of the environment that was current when the procedure was created.
- This is what is meant by @dfn{closure}. The next few subsections
- present examples that explore the usefulness of this concept.
- @node Serial Number
- @subsection Example 1: A Serial Number Generator
- This example uses closure to create a procedure with a variable binding
- that is private to the procedure, like a local variable, but whose value
- persists between procedure calls.
- @lisp
- (define (make-serial-number-generator)
- (let ((current-serial-number 0))
- (lambda ()
- (set! current-serial-number (+ current-serial-number 1))
- current-serial-number)))
- (define entry-sn-generator (make-serial-number-generator))
- (entry-sn-generator)
- @result{}
- 1
- (entry-sn-generator)
- @result{}
- 2
- @end lisp
- When @code{make-serial-number-generator} is called, it creates a local
- environment with a binding for @code{current-serial-number} whose
- initial value is 0, then, within this environment, creates a procedure.
- The local environment is stored within the created procedure object and
- so persists for the lifetime of the created procedure.
- Every time the created procedure is invoked, it increments the value of
- the @code{current-serial-number} binding in the captured environment and
- then returns the current value.
- Note that @code{make-serial-number-generator} can be called again to
- create a second serial number generator that is independent of the
- first. Every new invocation of @code{make-serial-number-generator}
- creates a new local @code{let} environment and returns a new procedure
- object with an association to this environment.
- @node Shared Variable
- @subsection Example 2: A Shared Persistent Variable
- This example uses closure to create two procedures, @code{get-balance}
- and @code{deposit}, that both refer to the same captured local
- environment so that they can both access the @code{balance} variable
- binding inside that environment. The value of this variable binding
- persists between calls to either procedure.
- Note that the captured @code{balance} variable binding is private to
- these two procedures: it is not directly accessible to any other code.
- It can only be accessed indirectly via @code{get-balance} or
- @code{deposit}, as illustrated by the @code{withdraw} procedure.
- @lisp
- (define get-balance #f)
- (define deposit #f)
- (let ((balance 0))
- (set! get-balance
- (lambda ()
- balance))
- (set! deposit
- (lambda (amount)
- (set! balance (+ balance amount))
- balance)))
- (define (withdraw amount)
- (deposit (- amount)))
- (get-balance)
- @result{}
- 0
- (deposit 50)
- @result{}
- 50
- (withdraw 75)
- @result{}
- -25
- @end lisp
- An important detail here is that the @code{get-balance} and
- @code{deposit} variables must be set up by @code{define}ing them at top
- level and then @code{set!}ing their values inside the @code{let} body.
- Using @code{define} within the @code{let} body would not work: this
- would create variable bindings within the local @code{let} environment
- that would not be accessible at top level.
- @node Callback Closure
- @subsection Example 3: The Callback Closure Problem
- A frequently used programming model for library code is to allow an
- application to register a callback function for the library to call when
- some particular event occurs. It is often useful for the application to
- make several such registrations using the same callback function, for
- example if several similar library events can be handled using the same
- application code, but the need then arises to distinguish the callback
- function calls that are associated with one callback registration from
- those that are associated with different callback registrations.
- In languages without the ability to create functions dynamically, this
- problem is usually solved by passing a @code{user_data} parameter on the
- registration call, and including the value of this parameter as one of
- the parameters on the callback function. Here is an example of
- declarations using this solution in C:
- @example
- typedef void (event_handler_t) (int event_type,
- void *user_data);
- void register_callback (int event_type,
- event_handler_t *handler,
- void *user_data);
- @end example
- In Scheme, closure can be used to achieve the same functionality without
- requiring the library code to store a @code{user-data} for each callback
- registration.
- @lisp
- ;; In the library:
- (define (register-callback event-type handler-proc)
- @dots{})
- ;; In the application:
- (define (make-handler event-type user-data)
- (lambda ()
- @dots{}
- <code referencing event-type and user-data>
- @dots{}))
- (register-callback event-type
- (make-handler event-type @dots{}))
- @end lisp
- As far as the library is concerned, @code{handler-proc} is a procedure
- with no arguments, and all the library has to do is call it when the
- appropriate event occurs. From the application's point of view, though,
- the handler procedure has used closure to capture an environment that
- includes all the context that the handler code needs ---
- @code{event-type} and @code{user-data} --- to handle the event
- correctly.
- @node OO Closure
- @subsection Example 4: Object Orientation
- Closure is the capture of an environment, containing persistent variable
- bindings, within the definition of a procedure or a set of related
- procedures. This is rather similar to the idea in some object oriented
- languages of encapsulating a set of related data variables inside an
- ``object'', together with a set of ``methods'' that operate on the
- encapsulated data. The following example shows how closure can be used
- to emulate the ideas of objects, methods and encapsulation in Scheme.
- @lisp
- (define (make-account)
- (let ((balance 0))
- (define (get-balance)
- balance)
- (define (deposit amount)
- (set! balance (+ balance amount))
- balance)
- (define (withdraw amount)
- (deposit (- amount)))
- (lambda args
- (apply
- (case (car args)
- ((get-balance) get-balance)
- ((deposit) deposit)
- ((withdraw) withdraw)
- (else (error "Invalid method!")))
- (cdr args)))))
- @end lisp
- Each call to @code{make-account} creates and returns a new procedure,
- created by the expression in the example code that begins ``(lambda
- args''.
- @lisp
- (define my-account (make-account))
- my-account
- @result{}
- #<procedure args>
- @end lisp
- This procedure acts as an account object with methods
- @code{get-balance}, @code{deposit} and @code{withdraw}. To apply one of
- the methods to the account, you call the procedure with a symbol
- indicating the required method as the first parameter, followed by any
- other parameters that are required by that method.
- @lisp
- (my-account 'get-balance)
- @result{}
- 0
- (my-account 'withdraw 5)
- @result{}
- -5
- (my-account 'deposit 396)
- @result{}
- 391
- (my-account 'get-balance)
- @result{}
- 391
- @end lisp
- Note how, in this example, both the current balance and the helper
- procedures @code{get-balance}, @code{deposit} and @code{withdraw}, used
- to implement the guts of the account object's methods, are all stored in
- variable bindings within the private local environment captured by the
- @code{lambda} expression that creates the account object procedure.
- @c Local Variables:
- @c TeX-master: "guile.texi"
- @c End:
|