from geomalgebra.basisblades import ga2d, BasisBlade
from geomalgebra.multivectors import Multivector, add
Products
# For convenience, we unpack the basis blades into the global scope
= ga2d scalar, e1, e2, e12
The Geometric Product
We have seen the dot product and the wedge product, but the most important product in GA is the geometric product.
For arbitrary vectors (1-blades),
Although we saw what the outer product does geometrically, we glossed over it’s properties. The outer product is associative and distributitve, and also
The dot products works like it does in linear algebra. Perpendicular vectors have a dot product of
To better understand the geometric product, try the following questions on your own. If you get stuck, you can reveal a walkthrough of the answer.
- What is
?
The geometric product of two (non-parallel) vectors is a bivector.
- Find
(i.e. ):
Interesting. Here we have an entity,
- Last example, calculate
:
Also very interesting. We have an entity that squares to -1, like an imaginary number.
See Imaginary Numbers Are Not Real for more on this result, space-time physics, and some history of Geometric Algebra.
Implementation for BasisBlades
Based on the calculations above,
This is pretty similar to the XOR bitwise function. If you have 1 XOR 1, it cancels to 0. Also two different bits, like 1 XOR 0, will become 1.
^
is the XOR operator in Python.
For example,
# Correct
print(BasisBlade.bit_bases2d.get(e1 ^ e2))
# Correct
print(BasisBlade.bit_bases2d.get(e1 ^ e1))
# Incorrect! Wrong sign! Should be -s (i.e. -1.0 scalar)
print(BasisBlade.bit_bases2d.get(e12 ^ e12))
e12
s
s
The XOR is close, but isn’t quite right. When manipulating the vectors algebraically we made sure to change the sign when swapping the order of operands (
But how do we encode the sign change with the bit-representation of basis blades? First let’s assume canonical order is postive (canonical order is when the subscript of the bases are in increasing order).
Then if we are presented with bases out of canonical order, say
For example,
The first number,
Another example,
Again, there is a
A computer can’t visually inspect binary numbers like we can, but we can simulate the same logic by sliding bits in the the first number towards the right, and using the bitwise AND
operator and a bit_count
function to count how many
def canonical_sign(basis_1: int, basis_2: int) -> int:
"""Count the number of basis blade swaps required to get 'a' and 'b' in canonical order
Canonical order means increasing order is positive, i.e: e1^e2 is positive, e2^e1 is negative
"""
= basis_1 >> 1
basis_1 = 0
num_swaps while basis_1:
# Count how many bases in basis_1 are canonically greater than those in basis_2
+= int.bit_count(basis_1 & basis_2)
num_swaps = basis_1 >> 1
basis_1 = (num_swaps & 1) == 0
num_swaps_is_odd return 1. if (num_swaps_is_odd) else -1.
canonical_sign
canonical_sign (basis_1:int, basis_2:int)
Count the number of basis blade swaps required to get ‘a’ and ‘b’ in canonical order Canonical order means increasing order is positive, i.e: e1^e2 is positive, e2^e1 is negative
assert canonical_sign(ga2d.e1, ga2d.e2) == 1.
assert canonical_sign(ga2d.e2, ga2d.e1) == -1.
Scalar multiplication works like usual, so the geometric product for basis blades is:
def basis_blade_gp(b1: BasisBlade, b2: BasisBlade):
"""Geometric Product for basis blades"""
= canonical_sign(b1.basis, b2.basis)
sign return BasisBlade(sign * b1.weight * b2.weight, b1.basis ^ b2.basis)
basis_blade_gp
basis_blade_gp (b1:geomalgebra.basisblades.BasisBlade, b2:geomalgebra.basisblades.BasisBlade)
Geometric Product for basis blades
# Basis Bivector squares to -1 under the geometric product - just like imaginary numbers!
print(basis_blade_gp(BasisBlade(1., e12), BasisBlade(1., e12)))
-1.0s
Implementation for Multivectors
The geometric product for multivectors is:
- associative:
- commutative under scalar multiplication:
(where is a scalar) - distributive:
where
In terms of implementation, this just means we need to multiply all of the terms of one multivector by all of the terms of the other multivectors. It’s really just the same as Polynomial multiplication.
For simplicity, we won’t try to implement any fancy multiplication algorithm that runs better than
.
def gp(m1: Multivector|BasisBlade, m2: Multivector|BasisBlade) -> Multivector:
"""Geometric product for Multivectors and Basis Blades"""
list[BasisBlade] = list()
basis_blades: if isinstance(m1, BasisBlade): m1 = Multivector([m1])
if isinstance(m2, BasisBlade): m2 = Multivector([m2])
for basis_1 in m1.blades.values():
for basis_2 in m2.blades.values():
= [*basis_blades, basis_blade_gp(basis_1, basis_2)]
basis_blades return Multivector(basis_blades)
gp
gp (m1:geomalgebra.multivectors.Multivector|geomalgebra.basisblades.Basis Blade, m2:geomalgebra.multivectors.Multivector|geomalgebra.basisblade s.BasisBlade)
Geometric product for Multivectors and Basis Blades
= Multivector([BasisBlade(2., scalar), BasisBlade(3., e1)])
m1 = Multivector([BasisBlade(4., scalar), BasisBlade(2., e1)])
m2 print(f'({m1})({m2}) = {gp(m1, m2)}')
(2.0s + 3.0e1)(4.0s + 2.0e1) = 14.0s + 16.0e1
We can verify this result by hand:
= Multivector([BasisBlade(-9., e12), BasisBlade(6., scalar)])
m1 = Multivector([BasisBlade(5, e2), BasisBlade(-2., e1)])
m2 print(f'({m1})({m2}) = {gp(m1, m2)}')
(-9.0e12 + 6.0s)(5e2 + -2.0e1) = -57.0e1 + 12.0e2
Again, by hand:
Readings
Geometric Algebra Primer (Suter, 2003)
- Chapter 3.1 The Geometric Product
- Chapter 3.3 The Geometric Product Continued
What’s Next?
This is the end for now! Thank you for taking the time to learn from these tutorials :)
In the future, I plan to add more on applications of Geometric Algebra to geometry, physics, and mathematics.