Common Lisp the Language
2nd Edition
Next: Numbers
Up: Packages
Previous: Modules

Most
users will want to load and use packages but will never need to
build one. Often a user will load a number of packages into the
user package whenever using Common Lisp. Typically an implementation
might provide some sort of initialization file mechanism to make such setup
automatic when the Lisp starts up. Table 11-1
shows such an initialization file
one that simply
causes other facilities to be loaded.

X3J13 voted in March 1989 (LISP-PACKAGE-NAME) to specify that
the forthcoming ANSI Common Lisp will use the package name common-lisp-user
instead of user.
----------------------------------------------------------------
Table 11-1: An Initialization File
;;;; Lisp init file for I. Newton.
;;; Set up the USER package the way I like it.
(require 'calculus) ;I use CALCULUS a lot; load it.
(use-package 'calculus) ;Get easy access to its
; exported symbols.
(require 'newtonian-mechanics) ;Same thing for NEWTONIAN-MECHANICS
(use-package 'newtonian-mechanics)
;;; I just want a few things from RELATIVITY
;;; and other things conflict.
;;; Import only what I need into the USER package.
(require 'relativity)
(import '(relativity:speed-of-light
relativity:ignore-small-errors))
;;; These are worth loading
but I will use qualified names
;;; such as PHLOGISTON:MAKE-FIRE-BOTTLE
to get at any symbols
;;; I might need from these packages.
(require 'phlogiston)
(require 'alchemy)
;;; End of Lisp init file for I. Newton.
----------------------------------------------------------------
When each of two files uses some symbols from the other
the author
of those files must be
careful to arrange the contents of the file in the proper order.
Typically each file contains a single package that is a complete module.
The contents of such a file should include the following items
in
order:
-
A call to provide that announces the module name.
-
A call to in-package that establishes the package.
-
A call to shadow that establishes any local symbols that will shadow
symbols that would otherwise be inherited from packages that this
package will use.
-
A call to export that establishes all of this package's external
symbols.
-
Any number of calls to require to load other modules that the
contents of this file might want to use or refer to. (Because the
calls to require follow the calls to in-package
shadow
and export
it is possible for the packages that may
be loaded to refer to external symbols in this package.)
-
Any number of calls to use-package
to make external
symbols from other packages accessible in this package.
-
Any number of calls to import
to make
symbols from other packages present in this package.
-
Finally
the definitions making up the contents of this package/module.
The following mnemonic sentence may be helpful in remembering
the proper order of these calls:
Put in seven extremely random user interface commands.
Each word of the sentence corresponds to one item in the above ordering:
Put Provide
IN IN-package
Seven Shadow
EXtremely EXport
Random Require
USEr USE-package
Interface Import
COmmands COntents of package/module
The sentence says what it helps you to do.

The most distressing aspect of the X3J13 vote to eliminate
provide and require
(REQUIRE-PATHNAME-DEFAULTS)
is of course that it completely ruins the mnemonic sentence.
Now
suppose for the sake of example
that the phlogiston and alchemy packages are
single-file
single-package modules as described above. The phlogiston
package needs to use the alchemy package
and the alchemy package
needs to use several
external symbols from the phlogiston package.
The definitions in the alchemy and phlogiston files
(see tables 11-2 and 11-3)
allow a
user to specify require statements for either of these modules
or for
both of them in either order
and all relevant information will be
loaded automatically and in the correct order.
----------------------------------------------------------------
Table 11-2: File alchemy
;;;; Alchemy functions
written and maintained by Merlin
Inc.
(provide 'alchemy) ;The module is named ALCHEMY.
(in-package 'alchemy) ;So is the package.
;;; There is nothing to shadow.
;;; Here is the external interface.
(export '(lead-to-gold gold-to-lead
antimony-to-zinc elixir-of-life))
;;; This package/module needs a function from
;;; the PHLOGISTON package/module.
(require 'phlogiston)
;;; We don't frequently need most of the external symbols from
;;; PHLOGISTON
so it's not worth doing a USE-PACKAGE on it.
;;; We'll just use qualified names as needed. But we use
;;; one function
MAKE-FIRE-BOTTLE
a lot
so import it.
;;; It's external in PHLOGISTON and so can be referred to
;;; here using ":" qualified-name syntax.
(import '(phlogiston:make-fire-bottle))
;;; Now for the real contents of this file.
(defun lead-to-gold (x)
Takes a quantity of lead and returns gold.
(when (> (phlogiston:heat-flow 5 x x) ;Using a qualified symbol
3)
(make-fire-bottle x)) ;Using an imported symbol
(gild x))
;;; And so on ...
----------------------------------------------------------------
----------------------------------------------------------------
Table 11-3: File phlogiston
;;;; Phlogiston functions
by Thermofluidics
Ltd.
(provide 'phlogiston) ;The module is named PHLOGISTON.
(in-package 'phlogiston) ;So is the package.
;;; There is nothing to shadow.
;;; Here is the external interface.
(export '(heat-flow cold-flow mix-fluids separate-fluids
burn make-fire-bottle))
;;; This file uses functions from the ALCHEMY package/module.
(require 'alchemy)
;;; We use alchemy functions a lot
so use the package.
;;; This will allow symbols exported from the ALCHEMY package
;;; to be referred to here without the need for qualified names.
(use-package 'alchemy)
;;; No calls to IMPORT are needed here.
;;; The real contents of this package/module.
(defvar *feeling-weak* nil)
(defun heat-flow (amount x y)
Make some amount of heat flow from x to y.
(when *feeling-weak*
(quaff (elixir-of-life))) ;No qualifier is needed.
(push-heat amount x y))
;;; And so on ...
----------------------------------------------------------------

For very large modules whose contents are spread over several files
(the lisp package is an example)
it is recommended that the user
create the package and declare all of the shadows and external symbols
in a separate file
so that this can be loaded before anything that
might use symbols from this package.

Indeed
the defpackage macro
approved by X3J13 in January 1989
(DEFPACKAGE)
encourages the use of such a separate file.
(By the way
X3J13 voted in March 1989 (LISP-PACKAGE-NAME) to specify that
the forthcoming ANSI Common Lisp will use the package name common-lisp
instead of lisp.)
Let's take a look at a revision
of I. Newton's files using defpackage.
The new version of the initialization file avoids using require;
instead
we assume that load will do the job
(see table 11-4).
----------------------------------------------------------------
Table 11-4: An Initialization File When defpackage Is Used
;;;; Lisp init file for I. Newton.
;;; Set up the USER package the way I like it.
(load "calculus") ;I use CALCULUS a lot; load it.
(use-package 'calculus) ;Get easy access to its
; exported symbols.
(load "newtonian-mechanics") ;Ditto for NEWTONIAN-MECHANICS
(use-package 'newtonian-mechanics)
;;; I just want a few things from RELATIVITY
;;; and other things conflict.
;;; Import only what I need into the USER package.
(load "relativity")
(import '(relativity:speed-of-light
relativity:ignore-small-errors))
;;; These are worth loading
but I will use qualified names
;;; such as PHLOGISTON:MAKE-FIRE-BOTTLE
to get at any symbols
;;; I might need from these packages.
(load "phlogiston")
(load "alchemy")
;;; End of Lisp init file for I. Newton.
----------------------------------------------------------------
The other files have each been split into two parts
one that
establishes the package and one that defines the contents.
This example uses a simple convention that for any file
named
say
``foo'' the file named ``foo-package''
contains the necessary defpackage and/or other package-establishing
code. The idiom
(unless (find-package "FOO")
(load "foo-package"))
is conventionally used to load a package definition but only if the package
has not already been defined. (This is a bit clumsy
and there are other
ways to arrange things so that a package is defined no more than once.)
The file alchemy-package is shown in
table 11-5.
The tricky point here is that the alchemy and phlogiston
packages contain mutual references (each imports from the other)
and so defpackage alone cannot do the job. Therefore
the phlogiston package is not mentioned in a :use option
in the defpackage for the alchemy package. Instead
the function use-package is called explicitly
after the package definition
for phlogiston has been loaded. Note that this file has
been coded with excruciating care so as to operate correctly even if
the package current when the file is loaded does not inherit from
the common-lisp package. In particular
the standard load-package-definition
idiom has been peppered with package qualifiers:
(cl:unless (cl:find-package "PHLOGISTON")
(cl:load "phlogiston-package"))
Note the use of the nickname cl for the common-lisp package.
The alchemy file
shown in table 11-6
simply loads the alchemy package definition
makes that package current
and then defines the ``real contents''
of the package.
----------------------------------------------------------------
Table 11-5: File alchemy-package Using defpackage
;;;; Alchemy package
written and maintained by Merlin
Inc.
(cl:defpackage "ALCHEMY"
(:export "LEAD-TO-GOLD" "GOLD-TO-LEAD"
ANTIMONY-TO-ZINC
ELIXIR-OF-LIFE
)
)
;;; This package needs a function from the PHLOGISTON package.
;;; Load the definition of the PHLOGISTON package if necessary.
(cl:unless (cl:find-package "PHLOGISTON")
(cl:load "phlogiston-package"))
;;; We don't frequently need most of the external symbols from
;;; PHLOGISTON
so it's not worth doing a USE-PACKAGE on it.
;;; We'll just use qualified names as needed. But we use
;;; one function
MAKE-FIRE-BOTTLE
a lot
so import it.
;;; It's external in PHLOGISTON and so can be referred to
;;; here using ":" qualified-name syntax.
(cl:import '(phlogiston:make-fire-bottle))
----------------------------------------------------------------
----------------------------------------------------------------
Table 11-6: File alchemy Using defpackage
;;;; Alchemy functions
written and maintained by Merlin
Inc.
(unless (find-package "ALCHEMY")
(load "alchemy-package"))
(in-package 'alchemy)
(defun lead-to-gold (x)
Takes a quantity of lead and returns gold.
(when (> (phlogiston:heat-flow 5 x x) ;Using a qualified symbol
3)
(make-fire-bottle x)) ;Using an imported symbol
(gild x))
;;; And so on ...
----------------------------------------------------------------
The file phlogiston-package is shown in
table 11-7.
This one is a little more straightforward than the file alchemy-package
because the latter bears the responsibility for breaking the
circular package references.
This file simply makes sure that the alchemy package is defined
and then performs a defpackage for the phlogiston package.
----------------------------------------------------------------
Table 11-7: File phlogiston-package Using defpackage
;;;; Phlogiston package definition
by Thermofluidics
Ltd.
;;; This package uses functions from the ALCHEMY package.
(cl:unless (cl:find-package "ALCHEMY")
(cl:load "alchemy-package"))
(cl:defpackage "PHLOGISTON"
(:use "COMMON-LISP" "ALCHEMY")
(:export "HEAT-FLOW"
COLD-FLOW
MIX-FLUIDS
SEPARATE-FLUIDS
BURN
MAKE-FIRE-BOTTLE
)
)
----------------------------------------------------------------
The phlogiston file
shown in table 11-8
simply loads the phlogiston package definition
makes that package current
and then defines the ``real contents''
of the package.
----------------------------------------------------------------
Table 11-8: File phlogiston Using defpackage
;;;; Phlogiston functions
by Thermofluidics
Ltd.
(unless (find-package "PHLOGISTON")
(load "phlogiston-package"))
(in-package 'phlogiston)
(defvar *feeling-weak* nil)
(defun heat-flow (amount x y)
Make some amount of heat flow from x to y.
(when *feeling-weak*
(quaff (elixir-of-life))) ;No qualifier is needed.
(push-heat amount x y))
;;; And so on ...
----------------------------------------------------------------
Let's look at the question of package circularity in
this example a little more closely.
Suppose that the file alchemy-package is loaded first.
It defines the alchemy package and then loads
file phlogiston-package. That file in turn finds that the
package alchemy has already been defined and therefore does
not attempt to load file alchemy-package again; it merely
defines package phlogiston. The file alchemy-package
then has a chance to import phlogiston:make-fire-bottle and everything
is fine.
On the other hand
suppose that the file phlogiston-package is
loaded first. It finds that the
package alchemy has not already been defined
and therefore
it immediately loads file alchemy-package.
That file in turn defines the alchemy package; then it
finds that package phlogiston is not yet defined and so
loads file phlogiston-package again (indeed
in nested fashion).
This time file phlogiston-package does find that the
package alchemy has already been defined
so it simply defines
package phlogiston and terminates.
The file alchemy-package then imports phlogiston:make-fire-bottle
and terminates.
Finally
the outer loading of file phlogiston-package
re-defines package phlogiston. Oh
dear. Fortunately the
two definitions of package phlogiston agree in every detail
so
everything ought to be all right. Still
it looks a bit dicey; I
certainly don't have the same warm
fuzzy feeling that I would if
no package were defined more than once.
Conclusion: defpackage goes a long way
but it certainly doesn't
solve all the possible problems of package and file management.
Neither did require and provide. Perhaps further experimentation
will yield facilities appropriate for future standardization.
Next: Numbers
Up: Packages
Previous: Modules
AI.Repository@cs.cmu.edu