I’m excited to share that my paper, Fast and forward stable randomized algorithms for linear least-squares problems has been released as a preprint on arXiv.
With the release of this paper, now seemed like a great time to discuss a topic I’ve been wanting to write about for a while: sketching. For the past two decades, sketching has become a widely used algorithmic tool in matrix computations. Despite this long history, questions still seem to be lingering about whether sketching really works:
In this post, I want to take a critical look at the question “does sketching work”? Answering this question requires answering two basic questions:
- What is sketching?
- What would it mean for sketching to work?
I think a large part of the disagreement over the efficacy of sketching boils down to different answers to these questions. By considering different possible answers to these questions, I hope to provide a balanced perspective on the utility of sketching as an algorithmic primitive for solving linear algebra problems.
Sketching
In matrix computations, sketching is really a synonym for (linear) dimensionality reduction. Suppose we are solving a problem involving one or more high-dimensional vectors
or perhaps a tall matrix
. A sketching matrix is a
matrix
where
. When multiplied into a high-dimensional vector
or tall matrix
, the sketching matrix
produces compressed or “sketched” versions
and
that are much smaller than the original vector
and matrix
.
Let
be a collection of vectors. For
to be a “good” sketching matrix for
, we require that
preserves the lengths of every vector in
up to a distortion parameter
:
(1) ![Rendered by QuickLaTeX.com \[(1-\varepsilon) \norm{x}\le\norm{Sx}\le(1+\varepsilon)\norm{x} \]](https://www.ethanepperly.com/wp-content/ql-cache/quicklatex.com-cc4b68a1603c32270b2c2e68ad7bc573_l3.png)
for every
data:image/s3,"s3://crabby-images/4121f/4121fc07e0bb36fa4ebbd4c73c017ba11306272a" alt="Rendered by QuickLaTeX.com x"
in
data:image/s3,"s3://crabby-images/4798e/4798ecd94ec871c82a6e42ef4afd39acdf3d5713" alt="Rendered by QuickLaTeX.com \mathsf{E}"
.
For linear algebra problems, we often want to sketch a matrix
. In this case, the appropriate set
that we want our sketch to be “good” for is the column space of the matrix
, defined to be
![Rendered by QuickLaTeX.com \[\operatorname{col}(A) \coloneqq \{ Ax : x \in \real^k \}.\]](https://www.ethanepperly.com/wp-content/ql-cache/quicklatex.com-b5b1a6d9d03c0b015432e7cdd42a617d_l3.png)
Remarkably, there exist many sketching matrices that achieve distortion
data:image/s3,"s3://crabby-images/2900b/2900b889e9bd76bf047d807e88ab46d6952c1d71" alt="Rendered by QuickLaTeX.com \varepsilon"
for
data:image/s3,"s3://crabby-images/aa089/aa089b1d3fbe086d1fe56d0b14395b4bdc5c3051" alt="Rendered by QuickLaTeX.com \mathsf{E}=\operatorname{col}(A)"
with an output dimension of roughly
data:image/s3,"s3://crabby-images/ac8fa/ac8fa18a0af1aa3a0f31e68d545cf34a8c5cf282" alt="Rendered by QuickLaTeX.com d \approx k / \varepsilon^2"
. In particular, the sketching dimension
data:image/s3,"s3://crabby-images/37053/3705361758c7a5e2926354dd1dcedfd6a496670e" alt="Rendered by QuickLaTeX.com d"
is proportional to the number of columns
data:image/s3,"s3://crabby-images/23b8e/23b8ee99e33f6c55fe41ed5c75af1fa7b7781f26" alt="Rendered by QuickLaTeX.com k"
of
data:image/s3,"s3://crabby-images/c1217/c12179b93ac1291850be714789a02a34bdcd4ea5" alt="Rendered by QuickLaTeX.com A"
. This is pretty neat! We can design a single sketching matrix
data:image/s3,"s3://crabby-images/4ee9e/4ee9ed83ac08d9b083442e4ef72c75da995ebe73" alt="Rendered by QuickLaTeX.com S"
which preserves the lengths of all infinitely-many vectors
data:image/s3,"s3://crabby-images/a582d/a582d62413bc92615446a0dfad28956d0c2f7893" alt="Rendered by QuickLaTeX.com Ax"
in the column space of
data:image/s3,"s3://crabby-images/c1217/c12179b93ac1291850be714789a02a34bdcd4ea5" alt="Rendered by QuickLaTeX.com A"
.
Sketching Matrices
There are many types of sketching matrices, each with different benefits and drawbacks. Many sketching matrices are based on randomized constructions in the sense that entries of
are chosen to be random numbers. Broadly, sketching matrices can be classified into two types:
- Data-dependent sketches. The sketching matrix
is constructed to work for a specific set of input vectors
. - Oblivious sketches. The sketching matrix
is designed to work for an arbitrary set of input vectors
of a given size (i.e.,
has
elements) or dimension (
is a
-dimensional linear subspace).
We will only discuss oblivious sketching for this post. We will look at three types of sketching matrices: Gaussian embeddings, subsampled randomized trignometric transforms, and sparse sign embeddings.
The details of how these sketching matrices are built and their strengths and weaknesses can be a little bit technical. All three constructions are independent from the rest of this article and can be skipped on a first reading. The main point is that good sketching matrices exist and are fast to apply: Reducing
to
requires roughly
operations, rather than the
operations we would expect to multiply a
matrix and a vector of length
.
The simplest type of sketching matrix
is obtained by (independently) setting every entry of
to be a Gaussian random number with mean zero and variance
. Such a sketching matrix is called a Gaussian embedding.
Benefits. Gaussian embeddings are simple to code up, requiring only a standard matrix product to apply to a vector
or matrix
. Gaussian embeddings admit a clean theoretical analysis, and their mathematical properties are well-understood.
Drawbacks. Computing
for a Gaussian embedding costs
operations, significantly slower than the other sketching matrices we will consider below. Additionally, generating and storing a Gaussian embedding can be computationally expensive.
The subsampled randomized trigonometric transform (SRTT) sketching matrix takes a more complicated form. The sketching matrix is defined to be a scaled product of three matrices
![Rendered by QuickLaTeX.com \[S = \sqrt{\frac{n}{d}} \cdot R \cdot F \cdot D.\]](https://www.ethanepperly.com/wp-content/ql-cache/quicklatex.com-c5a94a1ac9f7597c1a0d751105604bbe_l3.png)
These matrices have the following definitions:
is a diagonal matrix whose entries are each a random
(chosen independently with equal probability).
is a fast trigonometric transform such as a fast discrete cosine transform.
is a selection matrix. To generate
, let
be a random subset of
, selected without replacement.
is defined to be a matrix for which
for every vector
.
To store
on a computer, it is sufficient to store the
diagonal entries of
and the
selected coordinates
defining
. Multiplication of
against a vector
should be carried out by applying each of the matrices
,
, and
in sequence, such as in the following MATLAB code:
% Generate randomness for S
signs = 2*randi(2,m,1)-3; % diagonal entries of D
idx = randsample(m,d); % indices i_1,...,i_d defining R
% Multiply S against b
c = signs .* b % multiply by D
c = dct(c) % multiply by F
c = c(idx) % multiply by R
c = sqrt(n/d) * c % scale
Benefits.
can be applied to a vector
in
operations, a significant improvement over the
cost of a Gaussian embedding. The SRTT has the lowest memory and random number generation requirements of any of the three sketches we discuss in this post.
Drawbacks. Applying
to a vector requires a good implementation of a fast trigonometric transform. Even with a high-quality trig transform, SRTTs can be significantly slower than sparse sign embeddings (defined below). SRTTs are hard to parallelize. In theory, the sketching dimension should be chosen to be
, larger than for a Gaussian sketch.
A sparse sign embedding takes the form
![Rendered by QuickLaTeX.com \[S = \frac{1}{\sqrt{\zeta}} \begin{bmatrix} s_1 & s_2 & \cdots & s_n \end{bmatrix}.\]](https://www.ethanepperly.com/wp-content/ql-cache/quicklatex.com-cc514af4074faf14379b710b8e2b0a21_l3.png)
Here, each column
data:image/s3,"s3://crabby-images/45735/45735830dbca79e83ad85b655282daa42a93ed90" alt="Rendered by QuickLaTeX.com s_i"
is an independently generated random vector with exactly
data:image/s3,"s3://crabby-images/c0c6a/c0c6ac1590cb1bf36f518f8d65ff2a1cf93288e9" alt="Rendered by QuickLaTeX.com \zeta"
nonzero entries with random
data:image/s3,"s3://crabby-images/7cca8/7cca8b33fc8748f6a2c247062186150f3a367b3d" alt="Rendered by QuickLaTeX.com \pm 1"
values in uniformly random positions. The result is a
data:image/s3,"s3://crabby-images/8b962/8b962944cd5c8e9be6da864309760eabb1ae3cdf" alt="Rendered by QuickLaTeX.com d\times n"
matrix with only
data:image/s3,"s3://crabby-images/175d8/175d8afe5b79a090acfd9bef166b666f48220649" alt="Rendered by QuickLaTeX.com \zeta \cdot n"
nonzero entries. The parameter
data:image/s3,"s3://crabby-images/c0c6a/c0c6ac1590cb1bf36f518f8d65ff2a1cf93288e9" alt="Rendered by QuickLaTeX.com \zeta"
is often set to a small constant like
data:image/s3,"s3://crabby-images/67a1c/67a1c9edc3168acbd1d94b41192273b1e897351b" alt="Rendered by QuickLaTeX.com 8"
in practice.
Benefits. By using a dedicated sparse matrix library,
can be very fast to apply to a vector
(either
or
operations) to apply to a vector, depending on parameter choices (see below). With a good sparse matrix library, sparse sign embeddings are often the fastest sketching matrix by a wide margin.
Drawbacks. To be fast, sparse sign embeddings requires a good sparse matrix library. They require generating and storing roughly
random numbers, higher than SRTTs (roughly
numbers) but much less than Gaussian embeddings (
numbers). In theory, the sketching dimension should be chosen to be
and the sparsity should be set to
; the theoretically sanctioned sketching dimension (at least according to existing theory) is larger than for a Gaussian sketch. In practice, we can often get away with using
and
.
Summary
Using either SRTTs or sparse maps, a sketching a vector of length
down to
dimensions requires only
to
operations. To apply a sketch to an entire
matrix
thus requires roughly
operations. Therefore, sketching offers the promise of speeding up linear algebraic computations involving
, which typically take
operations.
How Can You Use Sketching?
The simplest way to use sketching is to first apply the sketch to dimensionality-reduce all of your data and then apply a standard algorithm to solve the problem using the reduced data. This approach to using sketching is called sketch-and-solve.
As an example, let’s apply sketch-and-solve to the least-squares problem:
(2) ![Rendered by QuickLaTeX.com \[\operatorname*{minimize}_{x\in\real^k} \norm{Ax - b}. \]](https://www.ethanepperly.com/wp-content/ql-cache/quicklatex.com-bcfff26e64b8d516a8c0d33dfcb45ee1_l3.png)
We assume this problem is highly overdetermined with
data:image/s3,"s3://crabby-images/c1217/c12179b93ac1291850be714789a02a34bdcd4ea5" alt="Rendered by QuickLaTeX.com A"
having many more rows
data:image/s3,"s3://crabby-images/9341e/9341eed35b78f6f0b327faf8ad562bbcda8b662c" alt="Rendered by QuickLaTeX.com n"
than columns
data:image/s3,"s3://crabby-images/d693b/d693be20a6089d551bca5d251195b4a91182a897" alt="Rendered by QuickLaTeX.com m"
.
To solve this problem with sketch-and-solve, generate a good sketching matrix
for the set
. Applying
to our data
and
, we get a dimensionality-reduced least-squares problem
(3) ![Rendered by QuickLaTeX.com \[\operatorname*{minimize}_{\hat{x}\in\real^k} \norm{(SA)\hat{x} - Sb}. \]](https://www.ethanepperly.com/wp-content/ql-cache/quicklatex.com-92ba828310f4ef0dacea49bcd75727ed_l3.png)
The solution
data:image/s3,"s3://crabby-images/7d4c4/7d4c46024d8077b38760cae5de63cfb9872dacf4" alt="Rendered by QuickLaTeX.com \hat{x}"
is the sketch-and-solve solution to the least-squares problem, which we can use as an approximate solution to the original least-squares problem.
Least-squares is just one example of the sketch-and-solve paradigm. We can also use sketching to accelerate other algorithms. For instance, we could apply sketch-and-solve to clustering. To cluster data points
, first apply sketching to obtain
and then apply an out-of-the-box clustering algorithms like k-means to the sketched data points.
Does Sketching Work?
Most often, when sketching critics say “sketching doesn’t work”, what they mean is “sketch-and-solve doesn’t work”.
To address this question in a more concrete setting, let’s go back to the least-squares problem (2). Let
denote the optimal least-squares solution and let
be the sketch-and-solve solution (3). Then, using the distortion condition (1), one can show that
![Rendered by QuickLaTeX.com \[\norm{A\hat{x} - b} \le \frac{1+\varepsilon}{1-\varepsilon} \norm{Ax - b}.\]](https://www.ethanepperly.com/wp-content/ql-cache/quicklatex.com-c03f037b5206dd786bd8b4c179d3c789_l3.png)
If we use a sketching matrix with a distortion of
data:image/s3,"s3://crabby-images/0de00/0de00cee2843381000e050eec260118878e40854" alt="Rendered by QuickLaTeX.com \varepsilon = 1/3"
, then this bound tells us that
(4) ![Rendered by QuickLaTeX.com \[\norm{A\hat{x} - b} \le 2\norm{Ax_\star - b}. \]](https://www.ethanepperly.com/wp-content/ql-cache/quicklatex.com-6ad93cba2f8e5dfec4b865e8b94dd39e_l3.png)
Is this a good result or a bad result? Ultimately, it depends. In some applications, the quality of a putative least-squares solution
is can be assessed from the residual norm
. For such applications, the bound (4) ensures that
is at most twice
. Often, this means
is a pretty decent approximate solution to the least-squares problem.
For other problems, the appropriate measure of accuracy is the so-called forward error
, measuring how close
is to
. For these cases, it is possible that
might be large even though the residuals are comparable (4).
Let’s see an example, using the MATLAB code from my paper:
[A, b, x, r] = random_ls_problem(1e4, 1e2, 1e8, 1e-4); % Random LS problem
S = sparsesign(4e2, 1e4, 8); % Sparse sign embedding
sketch_and_solve = (S*A) \ (S*b); % Sketch-and-solve
direct = A \ b; % MATLAB mldivide
Here, we generate a random least-squares problem of size 10,000 by 100 (with condition number
and residual norm
). Then, we generate a sparse sign embedding of dimension
(corresponding to a distortion of roughly
). Then, we compute the sketch-and-solve solution and, as reference, a “direct” solution by MATLAB’s \.
We compare the quality of the sketch-and-solve solution to the direct solution, using both the residual and forward error:
fprintf('Residuals: sketch-and-solve %.2e, direct %.2e, optimal %.2e\n',...
norm(b-A*sketch_and_solve), norm(b-A*direct), norm(r))
fprintf('Forward errors: sketch-and-solve %.2e, direct %.2e\n',...
norm(x-sketch_and_solve), norm(x-direct))
Here’s the output:
Residuals: sketch-and-solve 1.13e-04, direct 1.00e-04, optimal 1.00e-04
Forward errors: sketch-and-solve 1.06e+03, direct 8.08e-07
The sketch-and-solve solution has a residual norm of
, close to direct method’s residual norm of
. However, the forward error of sketch-and-solve is
nine orders of magnitude larger than the direct method’s forward error of
.
Does sketch-and-solve work? Ultimately, it’s a question of what kind of accuracy you need for your application. If a small-enough residual is all that’s needed, then sketch-and-solve is perfectly adequate. If small forward error is needed, sketch-and-solve can be quite bad.
One way sketch-and-solve can be improved is by increasing the sketching dimension
and lowering the distortion
. Unfortunately, improving the distortion of the sketch is expensive. Because of the relation
, to decrease the distortion by a factor of ten requires increasing the sketching dimension
by a factor of one hundred! Thus, sketch-and-solve is really only appropriate when a low degree of distortion
is necessary.
Iterative Sketching: Combining Sketching with Iteration
Sketch-and-solve is a fast way to get a low-accuracy solution to a least-squares problem. But it’s not the only way to use sketching for least-squares. One can also use sketching to obtain high-accuracy solutions by combining sketching with an iterative method.
There are many iterative methods for least-square problems. Iterative methods generate a sequence of approximate solutions
that we hope will converge at a rapid rate to the true least-squares solution,
.
To using sketching to solve least-squares problems iteratively, we can use the following observation:
If
is a sketching matrix for
, then
.
Therefore, if we compute a QR factorization
![Rendered by QuickLaTeX.com \[SA = QR,\]](https://www.ethanepperly.com/wp-content/ql-cache/quicklatex.com-e8775c0801a1253b7a8822784e7495be_l3.png)
then
![Rendered by QuickLaTeX.com \[A^\top A \approx (SA)^\top (SA) = R^\top Q^\top Q R = R^\top R.\]](https://www.ethanepperly.com/wp-content/ql-cache/quicklatex.com-aabb17d9e49992a08c48837c21e8caf2_l3.png)
Notice that we used the fact that
data:image/s3,"s3://crabby-images/fe883/fe883a98d3bfc663250ab8918d7543373d31601a" alt="Rendered by QuickLaTeX.com Q^\top Q = I"
since
data:image/s3,"s3://crabby-images/5b529/5b529bb6a598a1c0e99729d141c1e82fef953f53" alt="Rendered by QuickLaTeX.com Q"
has orthonormal columns. The conclusion is that
data:image/s3,"s3://crabby-images/c6b87/c6b872c51a31ff1f02fef17a213dad9784d80039" alt="Rendered by QuickLaTeX.com R^\top R \approx A^\top A"
.
Let’s use the approximation
to solve the least-squares problem iteratively. Start off with the normal equations
(5) ![Rendered by QuickLaTeX.com \[(A^\top A)x = A^\top b. \]](https://www.ethanepperly.com/wp-content/ql-cache/quicklatex.com-3e08632dc9739f5394c70af2f2890fe4_l3.png)
We can obtain an approximate solution to the least-squares problem by replacing
data:image/s3,"s3://crabby-images/3bc00/3bc0086c1de96c44330859886e14f348075463d2" alt="Rendered by QuickLaTeX.com A^\top A"
by
data:image/s3,"s3://crabby-images/11223/11223aafd5714459b6cf598264dd52b95f9c6d88" alt="Rendered by QuickLaTeX.com R^\top R"
in (5) and solving. The resulting solution is
![Rendered by QuickLaTeX.com \[x_0 = R^{-1} (R^{-\top}(A^\top b)).\]](https://www.ethanepperly.com/wp-content/ql-cache/quicklatex.com-005b51f6dd4c033bea707996974d294d_l3.png)
This solution
will typically not be a good solution to the least-squares problem (2), so we need to iterate. To do so, we’ll try and solve for the error
. To derive an equation for the error, subtract
from both sides of the normal equations (5), yielding
![Rendered by QuickLaTeX.com \[(A^\top A)(x-x_0) = A^\top (b-Ax_0).\]](https://www.ethanepperly.com/wp-content/ql-cache/quicklatex.com-97a5ef135e29969666e364c48f392167_l3.png)
Now, to solve for the error, substitute
data:image/s3,"s3://crabby-images/11223/11223aafd5714459b6cf598264dd52b95f9c6d88" alt="Rendered by QuickLaTeX.com R^\top R"
for
data:image/s3,"s3://crabby-images/3bc00/3bc0086c1de96c44330859886e14f348075463d2" alt="Rendered by QuickLaTeX.com A^\top A"
again and solve for
data:image/s3,"s3://crabby-images/4121f/4121fc07e0bb36fa4ebbd4c73c017ba11306272a" alt="Rendered by QuickLaTeX.com x"
, obtaining a new approximate solution
data:image/s3,"s3://crabby-images/1f521/1f521349e1ac1a15b34245cb76dcb32f4a9f93d7" alt="Rendered by QuickLaTeX.com x_1"
:
![Rendered by QuickLaTeX.com \[x\approx x_1 \coloneqq x_0 + R^{-\top}(R^{-1}(A^\top(b-Ax_0))).\]](https://www.ethanepperly.com/wp-content/ql-cache/quicklatex.com-f4f205cf644dc7c31027ff4f19c01563_l3.png)
We can now go another step: Derive an equation for the error
, approximate
, and obtain a new approximate solution
. Continuing this process, we obtain an iteration
(6) ![Rendered by QuickLaTeX.com \[x_{i+1} = x_i + R^{-\top}(R^{-1}(A^\top(b-Ax_i))).\]](https://www.ethanepperly.com/wp-content/ql-cache/quicklatex.com-403a2c522b754eabc3f908af0d8b0a9c_l3.png)
This iteration is known as the
iterative sketching method. If the distortion is small enough, this method converges at an exponential rate, yielding a high-accuracy least squares solution after a few iterations.
Let’s apply iterative sketching to the example we considered above. We show the forward error of the sketch-and-solve and direct methods as horizontal dashed purple and red lines. Iterative sketching begins at roughly the forward error of sketch-and-solve, with the error decreasing at an exponential rate until it reaches that of the direct method over the course of fourteen iterations. For this problem, at least, iterative sketching gives high-accuracy solutions to the least-squares problem!
To summarize, we’ve now seen two very different ways of using sketching:
- Sketch-and-solve. Sketch the data
and
and solve the sketched least-squares problem (3). The resulting solution
is cheap to obtain, but may have low accuracy. - Iterative sketching. Sketch the matrix
and obtain an approximation
to
. Use the approximation
to produce a sequence of better-and-better least-squares solutions
by the iteration (6). If we run for enough iterations
, the accuracy of the iterative sketching solution
can be quite high.
By combining sketching with more sophisticated iterative methods such as conjugate gradient and LSQR, we can get an even faster-converging least-squares algorithm, known as sketch-and-precondition. Here’s the same plot from above with sketch-and-precondition added; we see that sketch-and-precondition converges even faster than iterative sketching does!
“Does sketching work?” Even for a simple problem like least-squares, the answer is complicated:
A direct use of sketching (i.e., sketch-and-solve) leads to a fast, low-accuracy solution to least-squares problems. But sketching can achieve much higher accuracy for least-squares problems by combining sketching with an iterative method (iterative sketching and sketch-and-precondition).
We’ve focused on least-squares problems in this section, but these conclusions could hold more generally. If “sketching doesn’t work” in your application, maybe it would if it was combined with an iterative method.
Just How Accurate Can Sketching Be?
We left our discussion of sketching-plus-iterative-methods in the previous section on a positive note, but there is one last lingering question that remains to be answered. We stated that iterative sketching (and sketch-and-precondition) converge at an exponential rate. But our computers store numbers to only so much precision; in practice, the accuracy of an iterative method has to saturate at some point.
An (iterative) least-squares solver is said to be forward stable if, when run for a sufficient number
of iterations, the final accuracy
is comparable to accuracy of a standard direct method for the least-squares problem like MATLAB’s \ command or Python’s scipy.linalg.lstsq. Forward stability is not about speed or rate of convergence but about the maximum achievable accuracy.
The stability of sketch-and-precondition was studied in a recent paper by Meier, Nakatsukasa, Townsend, and Webb. They demonstrated that, with the initial iterate
, sketch-and-precondition is not forward stable. The maximum achievable accuracy was worse than standard solvers by orders of magnitude! Maybe sketching doesn’t work after all?
Fortunately, there is good news:
- The iterative sketching method is provably forward stable. This result is shown in my newly released paper; check it out if you’re interested!
- If we use the sketch-and-solve method as the initial iterate
for sketch-and-precondition, then sketch-and-precondition appears to be forward stable in practice. No theoretical analysis supporting this finding is known at present.
These conclusions are pretty nuanced. To see what’s going, it can be helpful to look at a graph:
The performance of different methods can be summarized as follows: Sketch-and-solve can have very poor forward error. Sketch-and-precondition with the zero initialization
is better, but still much worse than the direct method. Iterative sketching and sketch-and-precondition with
fair much better, eventually achieving an accuracy comparable to the direct method.
Put more simply, appropriately implemented, sketching works after all!
Conclusion
Sketching is a computational tool, just like the fast Fourier transform or the randomized SVD. Sketching can be used effectively to solve some problems. But, like any computational tool, sketching is not a silver bullet. Sketching allows you to dimensionality-reduce matrices and vectors, but it distorts them by an appreciable amount. Whether or not this distortion is something you can live with depends on your problem (how much accuracy do you need?) and how you use the sketch (sketch-and-solve or with an iterative method).