In [ ]:
#@title Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# https://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
Automatic differentiation is useful for implementing machine learning algorithms such as backpropagation for training neural networks.
In this guide, we will discuss ways you can compute gradients with TensorFlow, especially in eager execution.
In [ ]:
import numpy as np
import matplotlib.pyplot as plt
import tensorflow as tf
TensorFlow provides the tf.GradientTape API for automatic differentiation; that is, computing the gradient of a computation with respect to some inputs, usually tf.Variable
s.
TensorFlow "records" relevant operations executed inside the context of a tf.GradientTape
onto a "tape". TensorFlow then uses that tape to compute the gradients of a "recorded" computation using reverse mode differentiation.
Here is a simple example:
In [ ]:
x = tf.Variable(3.0)
with tf.GradientTape() as tape:
y = x**2
Once you've recorded some operations, use GradientTape.gradient(target, sources)
to calculate the gradient of some target (often a loss) relative to some source (often the model's variables).
In [ ]:
# dy = 2x * dx
dy_dx = tape.gradient(y, x)
dy_dx.numpy()
The above example uses scalars, but tf.GradientTape
works as easily on any tensor:
In [ ]:
w = tf.Variable(tf.random.normal((3, 2)), name='w')
b = tf.Variable(tf.zeros(2, dtype=tf.float32), name='b')
x = [[1., 2., 3.]]
with tf.GradientTape(persistent=True) as tape:
y = x @ w + b
loss = tf.reduce_mean(y**2)
To get the gradient of y
with respect to both variables, you can pass both as sources to the gradient
method. The tape is flexible about how sources are passed and will accept any nested combination of lists or dictionaries and return the gradient structured the same way (see tf.nest
).
In [ ]:
[dl_dw, dl_db] = tape.gradient(loss, [w, b])
The gradient with respect to each source has the shape of the source:
In [ ]:
print(w.shape)
print(dl_dw.shape)
Here is the gradient calculation again, this time passing a dictionary of variables:
In [ ]:
my_vars = {
'w': tf.Variable(tf.random.normal((3, 2)), name='w'),
'b': tf.Variable(tf.zeros(2, dtype=tf.float32), name='b')
}
grad = tape.gradient(loss, my_vars)
grad['b']
It's common to collect tf.Variables
into a tf.Module
or one of its subclasses (layers.Layer
, keras.Model
) for checkpointing and exporting.
In most cases, you will want to calculate gradients with respect to a model's trainable variables. Since all subclasses of tf.Module
aggregate their variables in the Module.trainable_variables
property, you can calculate these gradients in a few lines of code:
In [ ]:
layer = tf.keras.layers.Dense(2, activation='relu')
x = tf.constant([[1., 2., 3.]])
with tf.GradientTape() as tape:
# Forward pass
y = layer(x)
loss = tf.reduce_mean(y**2)
# Calculate gradients with respect to every trainable variable
grad = tape.gradient(loss, layer.trainable_variables)
In [ ]:
for var, g in zip(layer.trainable_variables, grad):
print(f'{var.name}, shape: {g.shape}')
The default behavior is to record all operations after accessing a trainable tf.Variable
. The reasons for this are:
For example the following fails to calculate a gradient because the tf.Tensor
is not "watched" by default, and the tf.Variable
is not trainable:
In [ ]:
# A trainable variable
x0 = tf.Variable(3.0, name='x0')
# Not trainable
x1 = tf.Variable(3.0, name='x1', trainable=False)
# Not a Variable: A variable + tensor returns a tensor.
x2 = tf.Variable(2.0, name='x2') + 1.0
# Not a variable
x3 = tf.constant(3.0, name='x3')
with tf.GradientTape() as tape:
y = (x0**2) + (x1**2) + (x2**2)
grad = tape.gradient(y, [x0, x1, x2, x3])
for g in grad:
print(g)
You can list the variables being watched by the tape using the GradientTape.watched_variables
method:
In [ ]:
[var.name for var in tape.watched_variables()]
tf.GradientTape
provides hooks that give the user control over what is or is not watched.
To record gradients with respect to a tf.Tensor
, you need to call GradientTape.watch(x)
:
In [ ]:
x = tf.constant(3.0)
with tf.GradientTape() as tape:
tape.watch(x)
y = x**2
# dy = 2x * dx
dy_dx = tape.gradient(y, x)
print(dy_dx.numpy())
Conversely, to disable the default behavior of watching all tf.Variables
, set watch_accessed_variables=False
when creating the gradient tape. This calculation uses two variables, but only connects the gradient for one of the variables:
In [ ]:
x0 = tf.Variable(0.0)
x1 = tf.Variable(10.0)
with tf.GradientTape(watch_accessed_variables=False) as tape:
tape.watch(x1)
y0 = tf.math.sin(x0)
y1 = tf.nn.softplus(x1)
y = y0 + y1
ys = tf.reduce_sum(y)
Since GradientTape.watch
was not called on x0
, no gradient is computed with respect to it:
In [ ]:
# dy = 2x * dx
grad = tape.gradient(ys, {'x0': x0, 'x1': x1})
print('dy/dx0:', grad['x0'])
print('dy/dx1:', grad['x1'].numpy())
In [ ]:
x = tf.constant(3.0)
with tf.GradientTape() as tape:
tape.watch(x)
y = x * x
z = y * y
# Use the tape to compute the gradient of z with respect to the
# intermediate value y.
# dz_dx = 2 * y, where y = x ** 2
print(tape.gradient(z, y).numpy())
By default, the resources held by a GradientTape
are released as soon as GradientTape.gradient()
method is called. To compute multiple gradients over the same computation, create a persistent
gradient tape. This allows multiple calls to the gradient()
method as resources are released when the tape object is garbage collected. For example:
In [ ]:
x = tf.constant([1, 3.0])
with tf.GradientTape(persistent=True) as tape:
tape.watch(x)
y = x * x
z = y * y
print(tape.gradient(z, x).numpy()) # 108.0 (4 * x**3 at x = 3)
print(tape.gradient(y, x).numpy()) # 6.0 (2 * x)
In [ ]:
del tape # Drop the reference to the tape
There is a tiny overhead associated with doing operations inside a gradient tape context. For most eager execution this will not be a noticeable cost, but you should still use tape context around the areas only where it is required.
Gradient tapes use memory to store intermediate results, including inputs and outputs, for use during the backwards pass.
For efficiency, some ops (like ReLU
) don't need to keep their intermediate results and they are pruned during the forward pass. However, if you use persistent=True
on your tape, nothing is discarded and your peak memory usage will be higher.
A gradient is fundamentally an operation on a scalar.
In [ ]:
x = tf.Variable(2.0)
with tf.GradientTape(persistent=True) as tape:
y0 = x**2
y1 = 1 / x
print(tape.gradient(y0, x).numpy())
print(tape.gradient(y1, x).numpy())
Thus, if you ask for the gradient of multiple targets, the result for each source is:
In [ ]:
x = tf.Variable(2.0)
with tf.GradientTape() as tape:
y0 = x**2
y1 = 1 / x
print(tape.gradient({'y0': y0, 'y1': y1}, x).numpy())
Similarly, if the target(s) are not scalar the gradient of the sum is calculated:
In [ ]:
x = tf.Variable(2.)
with tf.GradientTape() as tape:
y = x * [3., 4.]
print(tape.gradient(y, x).numpy())
This makes it simple to take the gradient of the sum of a collection of losses, or the gradient of the sum of an element-wise loss calculation.
If you need a separate gradient for each item see Jacobians.
In some cases you can skip the Jacobian. For an element-wise calculation, the gradient of the sum gives the derivative of each element with respect to its input-element, since each element is independent:
In [ ]:
x = tf.linspace(-10.0, 10.0, 200+1)
with tf.GradientTape() as tape:
tape.watch(x)
y = tf.nn.sigmoid(x)
dy_dx = tape.gradient(y, x)
In [ ]:
plt.plot(x, y, label='y')
plt.plot(x, dy_dx, label='dy/dx')
plt.legend()
_ = plt.xlabel('x')
In [ ]:
x = tf.constant(1.0)
v0 = tf.Variable(2.0)
v1 = tf.Variable(2.0)
with tf.GradientTape(persistent=True) as tape:
tape.watch(x)
if x > 0.0:
result = v0
else:
result = v1**2
dv0, dv1 = tape.gradient(result, [v0, v1])
print(dv0)
print(dv1)
Just remember that the control statements themselves are not differentiable, so they are invisible to gradient-based optimizers.
Depending on the value of x
in the above example, the tape either records result = v0
or result = v1**2
. The gradient with respect to x
is always None
.
In [ ]:
dx = tape.gradient(result, x)
print(dx)
In [ ]:
x = tf.Variable(2.)
y = tf.Variable(3.)
with tf.GradientTape() as tape:
z = y * y
print(tape.gradient(z, x))
Here z
is obviously not connected to x
, but there are several less-obvious ways that a gradient can be disconnected.
In the section on "controlling what the tape watches" you saw that the tape will automatically watch a tf.Variable
but not a tf.Tensor
.
One common error is to inadvertently replace a tf.Variable
with a tf.Tensor
, instead of using Variable.assign
to update the tf.Variable
. Here is an example:
In [ ]:
x = tf.Variable(2.0)
for epoch in range(2):
with tf.GradientTape() as tape:
y = x+1
print(type(x).__name__, ":", tape.gradient(y, x))
x = x + 1 # This should be `x.assign_add(1)`
In [ ]:
x = tf.Variable([[1.0, 2.0],
[3.0, 4.0]], dtype=tf.float32)
with tf.GradientTape() as tape:
x2 = x**2
# This step is calculated with NumPy
y = np.mean(x2, axis=0)
# Like most ops, reduce_mean will cast the NumPy array to a constant tensor
# using `tf.convert_to_tensor`.
y = tf.reduce_mean(y, axis=0)
print(tape.gradient(y, x))
Integers and strings are not differentiable. If a calculation path uses these data types there will be no gradient.
Nobody expects strings to be differentiable, but it's easy to accidentally create an int
constant or variable if you don't specify the dtype
.
In [ ]:
# The x0 variable has an `int` dtype.
x = tf.Variable([[2, 2],
[2, 2]])
with tf.GradientTape() as tape:
# The path to x1 is blocked by the `int` dtype here.
y = tf.cast(x, tf.float32)
y = tf.reduce_sum(x)
print(tape.gradient(y, x))
TensorFlow doesn't automatically cast between types, so in practice you'll often get a type error instead of a missing gradient.
State stops gradients. When you read from a stateful object the tape can only see the current state, not the history that lead to it.
A tf.Tensor
is immutable. You can't change a tensor once it's created. It has a value, but no state. All the operations discussed so far are also stateless: The output of a tf.matmul
only depends on its inputs.
A tf.Variable
has internal state, its value. When you use the variable, the state is read. It's normal to calculate a gradient with respect to a variable, but the variable's state blocks gradient calculations from going farther back. For example:
In [ ]:
x0 = tf.Variable(3.0)
x1 = tf.Variable(0.0)
with tf.GradientTape() as tape:
# Update x1 = x1 + x0.
x1.assign_add(x0)
# The tape starts recording from x1.
y = x1**2 # y = (x1 + x0)**2
# This doesn't work.
print(tape.gradient(y, x0)) #dy/dx0 = 2*(x1 + x2)
Similarly tf.data.Dataset
iterators and tf.queue
s are stateful, and will stop all gradients on tensors that pass through them.
Some tf.Operation
s are registered as being non-differentiable and will return None
. Others have no gradient registered.
The tf.raw_ops page shows which low-level ops have gradients registered.
If you attempt to take a gradient through a float op that has no gradient registered the tape will throw an error instead of silently returning None
. This way you know something has gone wrong.
For example the tf.image.adjust_contrast
function wraps raw_ops.AdjustContrastv2 which could have a gradient but the gradient is not implemented:
In [ ]:
image = tf.Variable([[[0.5, 0.0, 0.0]]])
delta = tf.Variable(0.1)
with tf.GradientTape() as tape:
new_image = tf.image.adjust_contrast(image, delta)
try:
print(tape.gradient(new_image, [image, delta]))
assert False # This should not happen.
except LookupError as e:
print(f'{type(e).__name__}: {e}')
If you need to differentiate through this op you'll either need to implement the gradient and register it (using tf.RegisterGradient
), or re-implement the function using other ops.
In some cases it would be convenient to get 0 instead of None
for unconnected gradients. You can decided what to return when you have unconnected gradients using the unconnected_gradients
argument:
In [ ]:
x = tf.Variable([2., 2.])
y = tf.Variable(3.)
with tf.GradientTape() as tape:
z = y**2
print(tape.gradient(z, x, unconnected_gradients=tf.UnconnectedGradients.ZERO))