Immediate and Delayed Definitions
You may have noticed that there are two different ways to make assignments in Mathematica
. The basic difference between these forms is when
the expression rhs
is an immediate assignment
, in which rhs
is evaluated at the time when the assignment is made.
, on the other hand, is a delayed assignment
, in which rhs
is not evaluated when the assignment is made, but is instead evaluated each time the value of lhs
|lhs=rhs (immediate assignment)||rhs is evaluated when the assignment is made|
|lhs:=rhs (delayed assignment)||rhs is evaluated each time the value of lhs is requested|
The two types of assignments in Mathematica.
This uses the
operator to define the function
was used, the definition is maintained in an unevaluated form.
When you make an assignment with the
operator, the right-hand side is evaluated immediately.
The definition now stored is the result of the Expand
When you execute
, the Expand
simply substitutes its argument into the already expanded form, giving a different answer.
As you can see from the example above, both
can be useful in defining functions, but they have different meanings, and you must be careful about which one to use in a particular case.
One rule of thumb is the following. If you think of an assignment as giving the final "value" of an expression, use the
operator. If instead you think of the assignment as specifying a "command" for finding the value, use the
operator. If in doubt, it is usually better to use the
operator than the
|lhs=rhs||rhs is intended to be the "final value" of lhs (e.g. )|
|lhs:=rhs||rhs gives a "command" or "program" to be executed whenever you ask for the value of lhs (e.g. f[x_]:=Expand[1-x^2])|
Interpretations of assignments with the and operators.
is probably used more often than
in defining functions, there is one important case in which you must use
to define a function. If you do a calculation, and get an answer in terms of a symbolic parameter
, you often want to go on and find results for various specific values of
. One way to do this is to use the
operator to apply appropriate rules for
in each case. It is usually more convenient, however, to use
to define a function whose argument is
Here is an expression involving
This defines a function whose argument is the value to be taken for
Here is the result when
is taken to be
An important point to notice in the example above is that there is nothing special about the name
that appears in the
pattern. It is just a symbol, indistinguishable from an
that appears in any other expression.
|f[x_]=expr||define a function which gives the value expr for any particular value of x|
Defining functions for evaluating expressions.
You can use
not only to define functions, but also to assign values to variables. If you type
, then value
is immediately evaluated, and the result is assigned to x
. On the other hand, if you type
, then value
is not immediately evaluated. Instead, it is maintained in an unevaluated form, and is evaluated afresh each time
This evaluates RandomReal
to find a pseudorandom number, then assigns this number to
is maintained in an unevaluated form, to be evaluated afresh each time
Here are values for
The value of
never changes. Every time
is used, however, a new pseudorandom number is generated.
The distinction between immediate and delayed assignments is particularly important when you set up chains of assignments.
is evaluated to give
, and the result is assigned to be the value of
is maintained in an unevaluated form, to be evaluated every time the value of
In this case,
give the same values.
Now the value of
uses the new value for
keeps its original value.
You can use delayed assignments such as
to set up variables whose values you can find in a variety of different "environments". Every time you ask for
, the expression rhs
is evaluated using the current values of the objects on which it depends.
The right-hand side of the delayed assignment is maintained in an unevaluated form.
, then finds the value of
In the example above, the symbol
acts as a "global variable", whose value affects the value of
. When you have a large number of parameters, many of which change only occasionally, you may find this kind of setup convenient. However, you should realize that implicit or hidden dependence of one variable on others can often become quite confusing. When possible, you should make all dependencies explicit, by defining functions which take all necessary parameters as arguments.
|lhs->rhs||rhs is evaluated when the rule is given|
|lhs:>rhs||rhs is evaluated when the rule is used|
Two types of transformation rules in Mathematica.
Just as you can make immediate and delayed assignments in Mathematica
, so you can also set up immediate and delayed transformation rules.
The right-hand side of this rule is evaluated when you give the rule.
A rule like this is probably not particularly useful.
Here the right-hand side of the rule is maintained in an unevaluated form, to be evaluated every time the rule is used.
Applying the rule causes the expansion to be done.
In analogy with assignments, you should typically use
when you want to replace an expression with a definite value, and you should use
when you want to give a command for finding the value.