Free resolutions¶
Let \(R\) be a commutative ring. A finite free resolution of an \(R\)-module \(M\) is a chain complex of free \(R\)-modules
terminating with a zero module at the end that is exact (all homology groups are zero) such that the image of \(d_1\) is \(M\).
EXAMPLES:
sage: from sage.homology.free_resolution import FreeResolution
sage: S.<x,y,z,w> = PolynomialRing(QQ)
sage: m = matrix(S, 1, [z^2 - y*w, y*z - x*w, y^2 - x*z]).transpose()
sage: r = FreeResolution(m, name='S'); r
S^1 <-- S^3 <-- S^2 <-- 0
sage: I = S.ideal([y*w - z^2, -x*w + y*z, x*z - y^2])
sage: r = I.free_resolution(); r
S^1 <-- S^3 <-- S^2 <-- 0
>>> from sage.all import *
>>> from sage.homology.free_resolution import FreeResolution
>>> S = PolynomialRing(QQ, names=('x', 'y', 'z', 'w',)); (x, y, z, w,) = S._first_ngens(4)
>>> m = matrix(S, Integer(1), [z**Integer(2) - y*w, y*z - x*w, y**Integer(2) - x*z]).transpose()
>>> r = FreeResolution(m, name='S'); r
S^1 <-- S^3 <-- S^2 <-- 0
>>> I = S.ideal([y*w - z**Integer(2), -x*w + y*z, x*z - y**Integer(2)])
>>> r = I.free_resolution(); r
S^1 <-- S^3 <-- S^2 <-- 0
from sage.homology.free_resolution import FreeResolution S.<x,y,z,w> = PolynomialRing(QQ) m = matrix(S, 1, [z^2 - y*w, y*z - x*w, y^2 - x*z]).transpose() r = FreeResolution(m, name='S'); r I = S.ideal([y*w - z^2, -x*w + y*z, x*z - y^2]) r = I.free_resolution(); r
sage: S.<x,y,z,w> = PolynomialRing(QQ)
sage: I = S.ideal([y*w - z^2, -x*w + y*z, x*z - y^2])
sage: r = I.graded_free_resolution(); r
S(0) <-- S(-2)⊕S(-2)⊕S(-2) <-- S(-3)⊕S(-3) <-- 0
>>> from sage.all import *
>>> S = PolynomialRing(QQ, names=('x', 'y', 'z', 'w',)); (x, y, z, w,) = S._first_ngens(4)
>>> I = S.ideal([y*w - z**Integer(2), -x*w + y*z, x*z - y**Integer(2)])
>>> r = I.graded_free_resolution(); r
S(0) <-- S(-2)⊕S(-2)⊕S(-2) <-- S(-3)⊕S(-3) <-- 0
S.<x,y,z,w> = PolynomialRing(QQ) I = S.ideal([y*w - z^2, -x*w + y*z, x*z - y^2]) r = I.graded_free_resolution(); r
An example of a minimal free resolution from [CLO2005]:
sage: R.<x,y,z,w> = QQ[]
sage: I = R.ideal([y*z - x*w, y^3 - x^2*z, x*z^2 - y^2*w, z^3 - y*w^2])
sage: r = I.free_resolution(); r
S^1 <-- S^4 <-- S^4 <-- S^1 <-- 0
sage: len(r)
3
sage: r.matrix(2)
[-z^2 -x*z y*w -y^2]
[ y 0 -x 0]
[ -w y z x]
[ 0 w 0 z]
>>> from sage.all import *
>>> R = QQ['x, y, z, w']; (x, y, z, w,) = R._first_ngens(4)
>>> I = R.ideal([y*z - x*w, y**Integer(3) - x**Integer(2)*z, x*z**Integer(2) - y**Integer(2)*w, z**Integer(3) - y*w**Integer(2)])
>>> r = I.free_resolution(); r
S^1 <-- S^4 <-- S^4 <-- S^1 <-- 0
>>> len(r)
3
>>> r.matrix(Integer(2))
[-z^2 -x*z y*w -y^2]
[ y 0 -x 0]
[ -w y z x]
[ 0 w 0 z]
R.<x,y,z,w> = QQ[] I = R.ideal([y*z - x*w, y^3 - x^2*z, x*z^2 - y^2*w, z^3 - y*w^2]) r = I.free_resolution(); r len(r) r.matrix(2)
AUTHORS:
Kwankyu Lee (2022-05-13): initial version
Travis Scrimshaw (2022-08-23): refactored for free module inputs
- class sage.homology.free_resolution.FiniteFreeResolution(module, name='S', **kwds)[source]¶
Bases:
FreeResolution
Finite free resolutions.
The matrix at index \(i\) in the list defines the differential map from \((i + 1)\)-th free module to the \(i\)-th free module over the base ring by multiplication on the left. The number of matrices in the list is the length of the resolution. The number of rows and columns of the matrices define the ranks of the free modules in the resolution.
Note that the first matrix in the list defines the differential map at homological index \(1\).
A subclass must provide a
_maps
attribute that contains a list of the maps defining the resolution.A subclass can define
_initial_differential
attribute that contains the \(0\)-th differential map whose codomain is the target of the free resolution.EXAMPLES:
sage: from sage.homology.free_resolution import FreeResolution sage: S.<x,y,z,w> = PolynomialRing(QQ) sage: I = S.ideal([y*w - z^2, -x*w + y*z, x*z - y^2]) sage: r = FreeResolution(I) sage: r.differential(0) Coercion map: From: Ambient free module of rank 1 over the integral domain Multivariate Polynomial Ring in x, y, z, w over Rational Field To: Quotient module by Submodule of Ambient free module of rank 1 over the integral domain Multivariate Polynomial Ring in x, y, z, w over Rational Field Generated by the rows of the matrix: [-z^2 + y*w] [ y*z - x*w] [-y^2 + x*z]
>>> from sage.all import * >>> from sage.homology.free_resolution import FreeResolution >>> S = PolynomialRing(QQ, names=('x', 'y', 'z', 'w',)); (x, y, z, w,) = S._first_ngens(4) >>> I = S.ideal([y*w - z**Integer(2), -x*w + y*z, x*z - y**Integer(2)]) >>> r = FreeResolution(I) >>> r.differential(Integer(0)) Coercion map: From: Ambient free module of rank 1 over the integral domain Multivariate Polynomial Ring in x, y, z, w over Rational Field To: Quotient module by Submodule of Ambient free module of rank 1 over the integral domain Multivariate Polynomial Ring in x, y, z, w over Rational Field Generated by the rows of the matrix: [-z^2 + y*w] [ y*z - x*w] [-y^2 + x*z]
from sage.homology.free_resolution import FreeResolution S.<x,y,z,w> = PolynomialRing(QQ) I = S.ideal([y*w - z^2, -x*w + y*z, x*z - y^2]) r = FreeResolution(I) r.differential(0)
- chain_complex()[source]¶
Return this resolution as a chain complex.
A chain complex in Sage has its own useful methods.
EXAMPLES:
sage: S.<x,y,z,w> = PolynomialRing(QQ) sage: I = S.ideal([y*w - z^2, -x*w + y*z, x*z - y^2]) sage: r = I.graded_free_resolution() sage: unicode_art(r.chain_complex()) ⎛-y x⎞ ⎜ z -y⎟ (z^2 - y*w y*z - x*w y^2 - x*z) ⎝-w z⎠ 0 <── C_0 <────────────────────────────── C_1 <────── C_2 <── 0
>>> from sage.all import * >>> S = PolynomialRing(QQ, names=('x', 'y', 'z', 'w',)); (x, y, z, w,) = S._first_ngens(4) >>> I = S.ideal([y*w - z**Integer(2), -x*w + y*z, x*z - y**Integer(2)]) >>> r = I.graded_free_resolution() >>> unicode_art(r.chain_complex()) ⎛-y x⎞ ⎜ z -y⎟ (z^2 - y*w y*z - x*w y^2 - x*z) ⎝-w z⎠ 0 <── C_0 <────────────────────────────── C_1 <────── C_2 <── 0
S.<x,y,z,w> = PolynomialRing(QQ) I = S.ideal([y*w - z^2, -x*w + y*z, x*z - y^2]) r = I.graded_free_resolution() unicode_art(r.chain_complex())
- differential(i)[source]¶
Return the \(i\)-th differential map.
INPUT:
i
– positive integer
EXAMPLES:
sage: S.<x,y,z,w> = PolynomialRing(QQ) sage: I = S.ideal([y*w - z^2, -x*w + y*z, x*z - y^2]) sage: r = I.graded_free_resolution() sage: r S(0) <-- S(-2)⊕S(-2)⊕S(-2) <-- S(-3)⊕S(-3) <-- 0 sage: r.differential(3) Free module morphism defined as left-multiplication by the matrix [] Domain: Ambient free module of rank 0 over the integral domain Multivariate Polynomial Ring in x, y, z, w over Rational Field Codomain: Ambient free module of rank 2 over the integral domain Multivariate Polynomial Ring in x, y, z, w over Rational Field sage: r.differential(2) Free module morphism defined as left-multiplication by the matrix [-y x] [ z -y] [-w z] Domain: Ambient free module of rank 2 over the integral domain Multivariate Polynomial Ring in x, y, z, w over Rational Field Codomain: Ambient free module of rank 3 over the integral domain Multivariate Polynomial Ring in x, y, z, w over Rational Field sage: r.differential(1) Free module morphism defined as left-multiplication by the matrix [z^2 - y*w y*z - x*w y^2 - x*z] Domain: Ambient free module of rank 3 over the integral domain Multivariate Polynomial Ring in x, y, z, w over Rational Field Codomain: Ambient free module of rank 1 over the integral domain Multivariate Polynomial Ring in x, y, z, w over Rational Field sage: r.differential(0) Coercion map: From: Ambient free module of rank 1 over the integral domain Multivariate Polynomial Ring in x, y, z, w over Rational Field To: Quotient module by Submodule of Ambient free module of rank 1 over the integral domain Multivariate Polynomial Ring in x, y, z, w over Rational Field Generated by the rows of the matrix: [-z^2 + y*w] [ y*z - x*w] [-y^2 + x*z]
>>> from sage.all import * >>> S = PolynomialRing(QQ, names=('x', 'y', 'z', 'w',)); (x, y, z, w,) = S._first_ngens(4) >>> I = S.ideal([y*w - z**Integer(2), -x*w + y*z, x*z - y**Integer(2)]) >>> r = I.graded_free_resolution() >>> r S(0) <-- S(-2)⊕S(-2)⊕S(-2) <-- S(-3)⊕S(-3) <-- 0 >>> r.differential(Integer(3)) Free module morphism defined as left-multiplication by the matrix [] Domain: Ambient free module of rank 0 over the integral domain Multivariate Polynomial Ring in x, y, z, w over Rational Field Codomain: Ambient free module of rank 2 over the integral domain Multivariate Polynomial Ring in x, y, z, w over Rational Field >>> r.differential(Integer(2)) Free module morphism defined as left-multiplication by the matrix [-y x] [ z -y] [-w z] Domain: Ambient free module of rank 2 over the integral domain Multivariate Polynomial Ring in x, y, z, w over Rational Field Codomain: Ambient free module of rank 3 over the integral domain Multivariate Polynomial Ring in x, y, z, w over Rational Field >>> r.differential(Integer(1)) Free module morphism defined as left-multiplication by the matrix [z^2 - y*w y*z - x*w y^2 - x*z] Domain: Ambient free module of rank 3 over the integral domain Multivariate Polynomial Ring in x, y, z, w over Rational Field Codomain: Ambient free module of rank 1 over the integral domain Multivariate Polynomial Ring in x, y, z, w over Rational Field >>> r.differential(Integer(0)) Coercion map: From: Ambient free module of rank 1 over the integral domain Multivariate Polynomial Ring in x, y, z, w over Rational Field To: Quotient module by Submodule of Ambient free module of rank 1 over the integral domain Multivariate Polynomial Ring in x, y, z, w over Rational Field Generated by the rows of the matrix: [-z^2 + y*w] [ y*z - x*w] [-y^2 + x*z]
S.<x,y,z,w> = PolynomialRing(QQ) I = S.ideal([y*w - z^2, -x*w + y*z, x*z - y^2]) r = I.graded_free_resolution() r r.differential(3) r.differential(2) r.differential(1) r.differential(0)
- matrix(i)[source]¶
Return the matrix representing the \(i\)-th differential map.
INPUT:
i
– positive integer
EXAMPLES:
sage: S.<x,y,z,w> = PolynomialRing(QQ) sage: I = S.ideal([y*w - z^2, -x*w + y*z, x*z - y^2]) sage: r = I.graded_free_resolution(); r S(0) <-- S(-2)⊕S(-2)⊕S(-2) <-- S(-3)⊕S(-3) <-- 0 sage: r.matrix(3) [] sage: r.matrix(2) [-y x] [ z -y] [-w z] sage: r.matrix(1) [z^2 - y*w y*z - x*w y^2 - x*z]
>>> from sage.all import * >>> S = PolynomialRing(QQ, names=('x', 'y', 'z', 'w',)); (x, y, z, w,) = S._first_ngens(4) >>> I = S.ideal([y*w - z**Integer(2), -x*w + y*z, x*z - y**Integer(2)]) >>> r = I.graded_free_resolution(); r S(0) <-- S(-2)⊕S(-2)⊕S(-2) <-- S(-3)⊕S(-3) <-- 0 >>> r.matrix(Integer(3)) [] >>> r.matrix(Integer(2)) [-y x] [ z -y] [-w z] >>> r.matrix(Integer(1)) [z^2 - y*w y*z - x*w y^2 - x*z]
S.<x,y,z,w> = PolynomialRing(QQ) I = S.ideal([y*w - z^2, -x*w + y*z, x*z - y^2]) r = I.graded_free_resolution(); r r.matrix(3) r.matrix(2) r.matrix(1)
- class sage.homology.free_resolution.FiniteFreeResolution_free_module(module, name='S', **kwds)[source]¶
Bases:
FiniteFreeResolution
Free resolutions of a free module.
INPUT:
module
– a free module or ideal over a PIDname
– the name of the base ring
EXAMPLES:
sage: R.<x> = QQ[] sage: M = R^3 sage: v = M([x^2, 2*x^2, 3*x^2]) sage: w = M([0, x, 2*x]) sage: S = M.submodule([v, w]); S Free module of degree 3 and rank 2 over Univariate Polynomial Ring in x over Rational Field Echelon basis matrix: [ x^2 2*x^2 3*x^2] [ 0 x 2*x] sage: res = S.free_resolution(); res S^3 <-- S^2 <-- 0 sage: ascii_art(res.chain_complex()) [ x^2 0] [2*x^2 x] [3*x^2 2*x] 0 <-- C_0 <-------------- C_1 <-- 0 sage: R.<x> = PolynomialRing(QQ) sage: I = R.ideal([x^4 + 3*x^2 + 2]) sage: res = I.free_resolution(); res S^1 <-- S^1 <-- 0
>>> from sage.all import * >>> R = QQ['x']; (x,) = R._first_ngens(1) >>> M = R**Integer(3) >>> v = M([x**Integer(2), Integer(2)*x**Integer(2), Integer(3)*x**Integer(2)]) >>> w = M([Integer(0), x, Integer(2)*x]) >>> S = M.submodule([v, w]); S Free module of degree 3 and rank 2 over Univariate Polynomial Ring in x over Rational Field Echelon basis matrix: [ x^2 2*x^2 3*x^2] [ 0 x 2*x] >>> res = S.free_resolution(); res S^3 <-- S^2 <-- 0 >>> ascii_art(res.chain_complex()) [ x^2 0] [2*x^2 x] [3*x^2 2*x] 0 <-- C_0 <-------------- C_1 <-- 0 >>> R = PolynomialRing(QQ, names=('x',)); (x,) = R._first_ngens(1) >>> I = R.ideal([x**Integer(4) + Integer(3)*x**Integer(2) + Integer(2)]) >>> res = I.free_resolution(); res S^1 <-- S^1 <-- 0
R.<x> = QQ[] M = R^3 v = M([x^2, 2*x^2, 3*x^2]) w = M([0, x, 2*x]) S = M.submodule([v, w]); S res = S.free_resolution(); res ascii_art(res.chain_complex()) R.<x> = PolynomialRing(QQ) I = R.ideal([x^4 + 3*x^2 + 2]) res = I.free_resolution(); res
- class sage.homology.free_resolution.FiniteFreeResolution_singular(module, name='S', algorithm='heuristic', **kwds)[source]¶
Bases:
FiniteFreeResolution
Minimal free resolutions of ideals or submodules of free modules of multivariate polynomial rings implemented in Singular.
INPUT:
module
– a submodule of a free module \(M\) of rank \(n\) over \(S\) or an ideal of a multi-variate polynomial ringname
– string (optional); name of the base ringalgorithm
– (default:'heuristic'
) Singular algorithm to compute a resolution ofideal
OUTPUT: a minimal free resolution of the ideal
If
module
is an ideal of \(S\), it is considered as a submodule of a free module of rank \(1\) over \(S\).The available algorithms and the corresponding Singular commands are shown below:
algorithm
Singular commands
minimal
mres(ideal)
shreyer
minres(sres(std(ideal)))
standard
minres(nres(std(ideal)))
heuristic
minres(res(std(ideal)))
EXAMPLES:
sage: from sage.homology.free_resolution import FreeResolution sage: S.<x,y,z,w> = PolynomialRing(QQ) sage: I = S.ideal([y*w - z^2, -x*w + y*z, x*z - y^2]) sage: r = FreeResolution(I); r S^1 <-- S^3 <-- S^2 <-- 0 sage: len(r) 2
>>> from sage.all import * >>> from sage.homology.free_resolution import FreeResolution >>> S = PolynomialRing(QQ, names=('x', 'y', 'z', 'w',)); (x, y, z, w,) = S._first_ngens(4) >>> I = S.ideal([y*w - z**Integer(2), -x*w + y*z, x*z - y**Integer(2)]) >>> r = FreeResolution(I); r S^1 <-- S^3 <-- S^2 <-- 0 >>> len(r) 2
from sage.homology.free_resolution import FreeResolution S.<x,y,z,w> = PolynomialRing(QQ) I = S.ideal([y*w - z^2, -x*w + y*z, x*z - y^2]) r = FreeResolution(I); r len(r)
sage: FreeResolution(I, algorithm='minimal') S^1 <-- S^3 <-- S^2 <-- 0 sage: FreeResolution(I, algorithm='shreyer') S^1 <-- S^3 <-- S^2 <-- 0 sage: FreeResolution(I, algorithm='standard') S^1 <-- S^3 <-- S^2 <-- 0 sage: FreeResolution(I, algorithm='heuristic') S^1 <-- S^3 <-- S^2 <-- 0
>>> from sage.all import * >>> FreeResolution(I, algorithm='minimal') S^1 <-- S^3 <-- S^2 <-- 0 >>> FreeResolution(I, algorithm='shreyer') S^1 <-- S^3 <-- S^2 <-- 0 >>> FreeResolution(I, algorithm='standard') S^1 <-- S^3 <-- S^2 <-- 0 >>> FreeResolution(I, algorithm='heuristic') S^1 <-- S^3 <-- S^2 <-- 0
FreeResolution(I, algorithm='minimal') FreeResolution(I, algorithm='shreyer') FreeResolution(I, algorithm='standard') FreeResolution(I, algorithm='heuristic')
We can also construct a resolution by passing in a matrix defining the initial differential:
sage: m = matrix(S, 1, [z^2 - y*w, y*z - x*w, y^2 - x*z]).transpose() sage: r = FreeResolution(m, name='S'); r S^1 <-- S^3 <-- S^2 <-- 0 sage: r.matrix(1) [z^2 - y*w y*z - x*w y^2 - x*z]
>>> from sage.all import * >>> m = matrix(S, Integer(1), [z**Integer(2) - y*w, y*z - x*w, y**Integer(2) - x*z]).transpose() >>> r = FreeResolution(m, name='S'); r S^1 <-- S^3 <-- S^2 <-- 0 >>> r.matrix(Integer(1)) [z^2 - y*w y*z - x*w y^2 - x*z]
m = matrix(S, 1, [z^2 - y*w, y*z - x*w, y^2 - x*z]).transpose() r = FreeResolution(m, name='S'); r r.matrix(1)
An additional construction is using a submodule of a free module:
sage: M = m.image() sage: r = FreeResolution(M, name='S'); r S^1 <-- S^3 <-- S^2 <-- 0
>>> from sage.all import * >>> M = m.image() >>> r = FreeResolution(M, name='S'); r S^1 <-- S^3 <-- S^2 <-- 0
M = m.image() r = FreeResolution(M, name='S'); r
A nonhomogeneous ideal:
sage: I = S.ideal([z^2 - y*w, y*z - x*w, y^2 - x]) sage: R = FreeResolution(I); R S^1 <-- S^3 <-- S^3 <-- S^1 <-- 0 sage: R.matrix(2) [ y*z - x*w y^2 - x 0] [-z^2 + y*w 0 y^2 - x] [ 0 -z^2 + y*w -y*z + x*w] sage: R.matrix(3) [ y^2 - x] [-y*z + x*w] [ z^2 - y*w]
>>> from sage.all import * >>> I = S.ideal([z**Integer(2) - y*w, y*z - x*w, y**Integer(2) - x]) >>> R = FreeResolution(I); R S^1 <-- S^3 <-- S^3 <-- S^1 <-- 0 >>> R.matrix(Integer(2)) [ y*z - x*w y^2 - x 0] [-z^2 + y*w 0 y^2 - x] [ 0 -z^2 + y*w -y*z + x*w] >>> R.matrix(Integer(3)) [ y^2 - x] [-y*z + x*w] [ z^2 - y*w]
I = S.ideal([z^2 - y*w, y*z - x*w, y^2 - x]) R = FreeResolution(I); R R.matrix(2) R.matrix(3)
- class sage.homology.free_resolution.FreeResolution(module, name='S', **kwds)[source]¶
Bases:
SageObject
A free resolution.
Let \(R\) be a commutative ring. A free resolution of an \(R\)-module \(M\) is a (possibly infinite) chain complex of free \(R\)-modules
\[\cdots \rightarrow R^{n_k} \xrightarrow{d_k} \cdots \xrightarrow{d_2} R^{n_1} \xrightarrow{d_1} R^{n_0}\]that is exact (all homology groups are zero) such that the image of \(d_1\) is \(M\).
- target()[source]¶
Return the codomain of the \(0\)-th differential map.
The codomain of the \(0\)-th differential map is the cokernel of the first differential map.
EXAMPLES:
sage: S.<x,y,z,w> = PolynomialRing(QQ) sage: I = S.ideal([y*w - z^2, -x*w + y*z, x*z - y^2]) sage: r = I.graded_free_resolution() sage: r S(0) <-- S(-2)⊕S(-2)⊕S(-2) <-- S(-3)⊕S(-3) <-- 0 sage: r.target() Quotient module by Submodule of Ambient free module of rank 1 over the integral domain Multivariate Polynomial Ring in x, y, z, w over Rational Field Generated by the rows of the matrix: [-z^2 + y*w] [ y*z - x*w] [-y^2 + x*z]
>>> from sage.all import * >>> S = PolynomialRing(QQ, names=('x', 'y', 'z', 'w',)); (x, y, z, w,) = S._first_ngens(4) >>> I = S.ideal([y*w - z**Integer(2), -x*w + y*z, x*z - y**Integer(2)]) >>> r = I.graded_free_resolution() >>> r S(0) <-- S(-2)⊕S(-2)⊕S(-2) <-- S(-3)⊕S(-3) <-- 0 >>> r.target() Quotient module by Submodule of Ambient free module of rank 1 over the integral domain Multivariate Polynomial Ring in x, y, z, w over Rational Field Generated by the rows of the matrix: [-z^2 + y*w] [ y*z - x*w] [-y^2 + x*z]
S.<x,y,z,w> = PolynomialRing(QQ) I = S.ideal([y*w - z^2, -x*w + y*z, x*z - y^2]) r = I.graded_free_resolution() r r.target()