---
title: "Functions `wedge()` and `wedge2()` in the `stokes` package"
author: "Robin K. S. Hankin"
output: html_vignette
bibliography: stokes.bib
link-citations: true
vignette: >
%\VignetteEngine{knitr::rmarkdown}
%\VignetteIndexEntry{wedge}
%\usepackage[utf8]{inputenc}
---
```{r setup, include=FALSE}
set.seed(0)
library("stokes")
library("spray") # needed for spraycross()
options(rmarkdown.html_vignette.check_title = FALSE)
knitr::opts_chunk$set(echo = TRUE)
knit_print.function <- function(x, ...){dput(x)}
registerS3method(
"knit_print", "function", knit_print.function,
envir = asNamespace("knitr")
)
```
```{r out.width='20%', out.extra='style="float:right; padding:10px"',echo=FALSE}
knitr::include_graphics(system.file("help/figures/stokes.png", package = "stokes"))
```
```{r, label=showwedge,comment=""}
wedge
wedge2
```
To cite the `stokes` package in publications, please use
@hankin2022_stokes. In a memorable passage, @spivak1965 states:
$\ldots$ we would like a theorem analogous to 4.1 [the dimensionality
of $k$-fold tensor products is $n^k$]. Of course, if
$\omega\in\Lambda^k(V)$ and $\eta\in\Lambda^l(V)$, then
$\omega\otimes\eta$ is usually not in $\Lambda^{k+l}(V)$. We will
therefore define a new product, the wedge product
$\omega\wedge\eta\in\Lambda^{k+l}(V)$ by
$$
\omega\wedge\eta=\frac{\left(k+l\right)!}{k!l!}\operatorname{Alt}(\omega\otimes\eta),\qquad\omega\in\Lambda^k(V),\eta\in\Lambda^l(V)
$$
(The reason for the strange coefficient will appear later).
- Michael Spivak, 1969 (Calculus on Manifolds, Perseus books). Page 79
Function `wedge()` returns the wedge product of any number of
$k$-forms; function `wedge2()` returns the wedge product of two
$k$-forms. The idiom of `wedge2()` is somewhat opaque, especially the
"strange" combinatorial coefficient $(k+l)!/(k!l!)$, which is
discussed in detail below.
## Digression: function `spraycross()`
Function `wedge()` is essentially a convenience wrapper for
`spraycross()`; the meat of `wedge2()` is the last line:
`kform(spraycross(K1, K2))`. Function `spraycross()` is part of the
`spray` package and gives a tensor product of sparse arrays,
interpreted as multivariate polynomials:
```{r}
(a <- spray(matrix(1:4,2,2),c(2,5)))
(b <- spray(matrix(c(10,11,12,13),2,2),c(7,11)))
spraycross(a,b)
spraycross(b,a)
```
Observe that `spraycross()` (and by association `wedge()`) is
associative and distributive but not commutative.
### Cut to the chase: `wedge2()`
Function `wedge2()` takes two kforms and we will start with a very
simple example:
```{r}
(x <- as.kform(cbind(1,2),5))
(y <- as.kform(cbind(3,4,7),7))
wedge2(x,y)
```
It looks like the combinatorial term has not been included but it has.
We will express `x` and `y` as tensors (objects of class `ktensor`)
and show how the combinatorial term arises.
```{r}
tx <- as.ktensor(x) # "tx" = tensor 'x'
(ty <- as.ktensor(y)) # "ty" = tensor 'y'
```
As functions, `y` and `ty` are identical:
```{r}
M <- matrix(round(rnorm(21),2),7,3) # member of (R^7)^3
c(as.function(y)(M),as.function(ty)(M))
```
Both are equivalent to
```{r}
7*(
+M[3,1]*M[4,2]*M[7,3]
-M[3,1]*M[4,3]*M[7,2]
-M[3,2]*M[4,1]*M[7,3]
+M[3,2]*M[4,3]*M[7,1]
+M[3,3]*M[4,1]*M[7,2]
-M[3,3]*M[4,2]*M[7,1]
)
```
We can see that `y` is a more compact and efficient representation of
`ty`: both are alternating tensors but `y` has alternatingness built in
to its evaluation, while `ty` is alternating by virtue of including
all permutations of its arguments, with the sign of the permutation.
We can evaluate Spivak's formula (but without the combinatorial term)
for $x\wedge y$ by coercing to ktensors and using `tensorprod()`:
```{r}
(z <- tensorprod(as.ktensor(x),as.ktensor(y)))
```
Above, each coefficient is equal to $\pm 35$ (the sign coming from the
sign of the permutation), and we have $2!3!=12$ rows. We can now
calculate $\operatorname{Alt}(z)$, which would have $5!=120$ rows, one
per permutation of $[5]$, each with coefficient $\pm\frac{12\times
35}{5!}=\pm 3.5$.
We define $x\wedge y$ to be $\frac{5!}{3!2!}\operatorname{Alt}(z)$, so
each coefficient would be $\pm\frac{5!}{3!2!}\cdot\frac{12\times
35}{5!}=35$. We know that $x\wedge y$ is an alternating form. So to
represent it as an object of class `kform`, we need a `kform` object
with _single_ index entry `1 2 3 4 7`. This would need coefficient
35, on the grounds that it is linear, alternating, and maps
$\begin{pmatrix}
1&0&0&0&0\\
0&1&0&0&0\\
0&0&1&0&0\\
0&0&0&1&0\\
0&0&0&0&0\\
0&0&0&0&0\\
0&0&0&0&1
\end{pmatrix}$ to $35$; and indeed this is what we see:
```{r}
wedge(x,y)
```
So to conclude, the combinatorial term is present in the R idiom, it
is just difficult to see at first glance.
# Algebraic properties
First of all we should note that $\Lambda^k(V)$ is a vector space
(this is considered in the `kform` vignette). If
$\omega,\omega_i\in\Lambda^k(V)$ and $\eta,\eta_i\in\Lambda^l(V)$ then
\begin{eqnarray}
(\omega_1+\omega_2)\wedge\eta &=& \omega_1\wedge\eta+\omega_2\wedge\eta\\
\omega\wedge(\eta_1+\eta_2) &=&\omega\wedge\eta_1 + \omega\wedge\eta_2\\
\end{eqnarray}
(that is, the wedge product is left- and right- distributive); if
$a\in\mathbb{R}$ then
\begin{equation}
a\omega\wedge\eta = \omega\wedge a\eta=a(\omega\wedge\eta)
\end{equation}
and
\begin{equation}
\omega\wedge\eta = (-1)^{kl}\eta\wedge\omega.
\end{equation}
These rules make expansion of wedge products possible by expressing a
general kform in terms of a basis for $\Lambda^k(V)$. @spivak1965
tells us that, if $v_1,\ldots,v_k$ is a basis for $V$, then the set of
all
\begin{equation}
\phi_{i_1}\wedge\phi_{i_2}\wedge\cdots\wedge\phi_{i_k}\qquad 1\leq i_1 < \cdots < i_k\leq n
\end{equation}
is a basis for $\Lambda^k(V)$ where $\phi_i(v_j)=\delta_{ij}$. The
package expresses a $k$-form in terms of this basis as in the
following example:
```{r}
(omega <- as.kform(rbind(c(1,2,8),c(1,3,7)),5:6))
```
In algebraic notation, `omega` (or $\omega$) would be
$5\phi_1\wedge\phi_2\wedge\phi_8+6\phi_1\wedge\phi_3\wedge\phi_7$ and
we may write this as $\omega=5\phi_{128}+6\phi_{137}$. To take a
wedge product of this with $\eta=2\phi_{235}+3\phi_{356}$ we would
write
\begin{eqnarray}
\omega\wedge\eta &=& (5\phi_{128}+6\phi_{137})\wedge (2\phi_{235}+3\phi_{356})\\
&=& 10\phi_{128}\wedge\phi_{235} + 15\phi_{128}\wedge\phi_{356} +
12\phi_{137}\wedge\phi_{235} + 18\phi_{137}\wedge\phi_{356}\\
&=&
10\phi_1\wedge\phi_2\wedge\phi_8\wedge\phi_2\wedge\phi_3\wedge\phi_5 +
15\phi_1\wedge\phi_2\wedge\phi_8\wedge\phi_3\wedge\phi_5\wedge\phi_6\\&{}&\qquad +
12\phi_1\wedge\phi_3\wedge\phi_7\wedge\phi_2\wedge\phi_3\wedge\phi_5 +
18\phi_1\wedge\phi_3\wedge\phi_7\wedge\phi_3\wedge\phi_5\wedge\phi_6\\
&=& 0+ 15\phi_1\wedge\phi_2\wedge\phi_8\wedge\phi_3\wedge\phi_5\wedge\phi_6+0+0\\
&=& -15\phi_1\wedge\phi_2\wedge\phi_3\wedge\phi_5\wedge\phi_6\wedge\phi_8
\end{eqnarray}
where we have used the rules repeatedly (especially the fact that
$\omega\wedge\omega=0$ for _any_ alternating form). Package idiom
would be:
```{r}
eta <- as.kform(rbind(c(2,3,5),c(3,5,6)),2:3)
wedge(omega,eta)
```
See how function `wedge()` does the legwork.
# References