Immediate and Delayed Definitions
You may have noticed that there are two different ways to make assignments in
Mathematica:
lhs=rhs and
lhs:=rhs. The basic difference between these forms is
when the expression
rhs is evaluated.
lhs=rhs is an
immediate assignment, in which
rhs is evaluated at the time when the assignment is made.
lhs:=rhs, 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 is requested.
| 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 ex. |
Because := 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.
| Out[3]= |  |
|
The definition now stored is the result of the Expand command. |
When you execute ex, the Expand is performed.
| Out[5]= |  |
|
iex simply substitutes its argument into the already expanded form, giving a different answer.
| Out[6]= |  |
|
As you can see from the example above, both
= and
:= 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
= one.
| lhs=rhs | rhs is intended to be the "final value" of lhs (e.g., f[x_]=1-x^2) |
| 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.
Although
:= 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
x, you often want to go on and find results for various specific values of
x. One way to do this is to use the
/. operator to apply appropriate rules for
x in each case. It is usually more convenient, however, to use
= to define a function whose argument is
x.
Here is an expression involving x.
| Out[7]= |  |
|
This defines a function whose argument is the value to be taken for x.
| Out[8]= |  |
|
Here is the result when x is taken to be 1+a.
| Out[9]= |  |
|
An important point to notice in the example above is that there is nothing special about the name
x that appears in the
x_ pattern. It is just a symbol, indistinguishable from an
x 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
= and
:= not only to define functions, but also to assign values to variables. If you type
x=value, then
value is immediately evaluated, and the result is assigned to
x. On the other hand, if you type
x:=value, then
value is not immediately evaluated. Instead, it is maintained in an unevaluated form, and is evaluated afresh each time
x is used.
This evaluates RandomReal[] to find a pseudorandom number, then assigns this number to r1.
| Out[10]= |  |
|
Here RandomReal[] is maintained in an unevaluated form, to be evaluated afresh each time r2 is used. |
Here are values for r1 and r2.
| Out[12]= |  |
|
The value of r1 never changes. Every time r2 is used, however, a new pseudorandom number is generated.
| Out[13]= |  |
|
The distinction between immediate and delayed assignments is particularly important when you set up chains of assignments.
| Out[14]= |  |
|
Here a+2 is evaluated to give 3, and the result is assigned to be the value of ri.
| Out[15]= |  |
|
Here a+2 is maintained in an unevaluated form, to be evaluated every time the value of rd is requested. |
In this case, ri and rd give the same values.
| Out[17]= |  |
|
Now the value of a is changed.
| Out[18]= |  |
|
Now rd uses the new value for a, while ri keeps its original value.
| Out[19]= |  |
|
You can use delayed assignments such as
t:=rhs to set up variables whose values you can find in a variety of different "environments". Every time you ask for
t, 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. |
This sets a to 4, then finds the value of t.
| Out[21]= |  |
|
| Out[22]= |  |
|
In the example above, the symbol
a acts as a "global variable", whose value affects the value of
t. 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.
| Out[23]= |  |
|
A rule like this is probably not particularly useful.
| Out[24]= |  |
|
Here the right-hand side of the rule is maintained in an unevaluated form, to be evaluated every time the rule is used.
| Out[25]= |  |
|
Applying the rule causes the expansion to be done.
| Out[26]= |  |
|
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.