diff --git a/dev/index.html b/dev/index.html index 29d397ab..d42a9fa8 100644 --- a/dev/index.html +++ b/dev/index.html @@ -1,2 +1,2 @@ -Home · TensorKit.jl

TensorKit.jl

A Julia package for large-scale tensor computations, with a hint of category theory.

Package summary

TensorKit.jl aims to be a generic package for working with tensors as they appear throughout the physical sciences. TensorKit implements a parametric type Tensor (which is actually a specific case of the type TensorMap) and defines for these types a number of vector space operations (scalar multiplication, addition, norms and inner products), index operations (permutations) and linear algebra operations (multiplication, factorizations). Finally, tensor contractions can be performed using the @tensor macro from TensorOperations.jl.

Currently, most effort is oriented towards tensors as they appear in the context of quantum many body physics and in particular the field of tensor networks. Such tensors often have large dimensions and take on a specific structure when symmetries are present. To deal with generic symmetries, we employ notations and concepts from category theory all the way down to the definition of a tensor.

At the same time, TensorKit.jl focusses on computational efficiency and performance. The underlying storage of a tensor's data can be any DenseArray. Currently, certain operations are already multithreaded, either by distributing the different blocks in case of a structured tensor (i.e. with symmetries) or by using multithreading provided by the package Strided.jl. In the future, we also plan to investigate using CuArrays as underlying storage for the tensors data, so as to leverage GPUs for the different operations defined on tensors.

Contents of the manual

Library outline

+Home · TensorKit.jl

TensorKit.jl

A Julia package for large-scale tensor computations, with a hint of category theory.

Package summary

TensorKit.jl aims to be a generic package for working with tensors as they appear throughout the physical sciences. TensorKit implements a parametric type Tensor (which is actually a specific case of the type TensorMap) and defines for these types a number of vector space operations (scalar multiplication, addition, norms and inner products), index operations (permutations) and linear algebra operations (multiplication, factorizations). Finally, tensor contractions can be performed using the @tensor macro from TensorOperations.jl.

Currently, most effort is oriented towards tensors as they appear in the context of quantum many body physics and in particular the field of tensor networks. Such tensors often have large dimensions and take on a specific structure when symmetries are present. To deal with generic symmetries, we employ notations and concepts from category theory all the way down to the definition of a tensor.

At the same time, TensorKit.jl focusses on computational efficiency and performance. The underlying storage of a tensor's data can be any DenseArray. Currently, certain operations are already multithreaded, either by distributing the different blocks in case of a structured tensor (i.e. with symmetries) or by using multithreading provided by the package Strided.jl. In the future, we also plan to investigate using CuArrays as underlying storage for the tensors data, so as to leverage GPUs for the different operations defined on tensors.

Contents of the manual

Library outline

diff --git a/dev/index/index.html b/dev/index/index.html index 52130f47..fabb8501 100644 --- a/dev/index/index.html +++ b/dev/index/index.html @@ -1,2 +1,2 @@ -Index · TensorKit.jl

Index

+Index · TensorKit.jl

Index

diff --git a/dev/lib/sectors/index.html b/dev/lib/sectors/index.html index 3dcf2abc..6a9ed3a7 100644 --- a/dev/lib/sectors/index.html +++ b/dev/lib/sectors/index.html @@ -1,32 +1,32 @@ -Symmetry sectors an fusion trees · TensorKit.jl

Symmetry sectors an fusion trees

Type hierarchy

TensorKit.SectorType
abstract type Sector end

Abstract type for representing the (isomorphism classes of) simple objects in (unitary and pivotal) (pre-)fusion categories, e.g. the irreducible representations of a finite or compact group. Subtypes I<:Sector as the set of labels of a GradedSpace.

Every new I<:Sector should implement the following methods:

  • one(::Type{I}): unit element of I
  • conj(a::I): $a̅$, conjugate or dual label of $a$
  • ⊗(a::I, b::I): iterable with unique fusion outputs of $a ⊗ b$ (i.e. don't repeat in case of multiplicities)
  • Nsymbol(a::I, b::I, c::I): number of times c appears in a ⊗ b, i.e. the multiplicity
  • FusionStyle(::Type{I}): UniqueFusion(), SimpleFusion() or GenericFusion()
  • BraidingStyle(::Type{I}): Bosonic(), Fermionic(), Anyonic(), ...
  • Fsymbol(a::I, b::I, c::I, d::I, e::I, f::I): F-symbol: scalar (in case of UniqueFusion/SimpleFusion) or matrix (in case of GenericFusion)
  • Rsymbol(a::I, b::I, c::I): R-symbol: scalar (in case of UniqueFusion/SimpleFusion) or matrix (in case of GenericFusion)

and optionally

  • dim(a::I): quantum dimension of sector a
  • frobeniusschur(a::I): Frobenius-Schur indicator of a
  • Bsymbol(a::I, b::I, c::I): B-symbol: scalar (in case of UniqueFusion/SimpleFusion) or matrix (in case of GenericFusion)
  • twist(a::I) -> twist of sector a

and optionally, if FusionStyle(I) isa GenericFusion

  • vertex_ind2label(i::Int, a::I, b::I, c::I) -> a custom label for the ith copy of c appearing in a ⊗ b

Furthermore, iterate and Base.IteratorSize should be made to work for the singleton type SectorValues{I}.

source
TensorKit.SectorValuesType
struct SectorValues{I<:Sector}

Singleton type to represent an iterator over the possible values of type I, whose instance is obtained as values(I). For a new I::Sector, the following should be defined

  • Base.iterate(::SectorValues{I}[, state]): iterate over the values
  • Base.IteratorSize(::Type{SectorValues{I}}): HasLenght(), SizeUnkown() or IsInfinite() depending on whether the number of values of type I is finite (and sufficiently small) or infinite; for a large number of values, SizeUnknown() is recommend because this will trigger the use of GenericGradedSpace.

If IteratorSize(I) == HasLength(), also the following must be implemented:

  • Base.length(::SectorValues{I}): the number of different values
  • Base.getindex(::SectorValues{I}, i::Int): a mapping between an index i and an instance of I
  • findindex(::SectorValues{I}, c::I): reverse mapping between a value c::I and an index i::Integer ∈ 1:length(values(I))
source
TensorKit.FusionStyleType
FusionStyle(a::Sector) -> ::FusionStyle
-FusionStyle(I::Type{<:Sector}) -> ::FusionStyle

Return the type of fusion behavior of sectors of type I, which can be either

  • UniqueFusion(): single fusion output when fusing two sectors;
  • SimpleFusion(): multiple outputs, but every output occurs at most one, also known as multiplicity free (e.g. irreps of $SU(2)$);
  • GenericFusion(): multiple outputs that can occur more than once (e.g. irreps of $SU(3)$).

There is an abstract supertype MultipleFusion of which both SimpleFusion and GenericFusion are subtypes. Furthermore, there is a type alias MultiplicityFreeFusion for those fusion types which do not require muliplicity labels, i.e. MultiplicityFreeFusion = Union{UniqueFusion,SimpleFusion}.

source
TensorKit.BraidingStyleType
BraidingStyle(::Sector) -> ::BraidingStyle
-BraidingStyle(I::Type{<:Sector}) -> ::BraidingStyle

Return the type of braiding and twist behavior of sectors of type I, which can be either

  • Bosonic(): symmetric braiding with trivial twist (i.e. identity)
  • Fermionic(): symmetric braiding with non-trivial twist (squares to identity)
  • Anyonic(): general $R_(a,b)^c$ phase or matrix (depending on SimpleFusion or GenericFusion fusion) and arbitrary twists

Note that Bosonic and Fermionic are subtypes of SymmetricBraiding, which means that braids are in fact equivalent to crossings (i.e. braiding twice is an identity: isone(Rsymbol(b,a,c)*Rsymbol(a,b,c)) == true) and permutations are uniquely defined.

source
TensorKit.AbstractIrrepType
abstract type AbstractIrrep{G<:Group} <: Sector end

Abstract supertype for sectors which corresponds to irreps (irreducible representations) of a group G. As we assume unitary representations, these would be finite groups or compact Lie groups. Note that this could also include projective rather than linear representations.

Actual concrete implementations of those irreps can be obtained as Irrep[G], or via their actual name, which generically takes the form (asciiG)Irrep, i.e. the ASCII spelling of the group name followed by Irrep.

All irreps have BraidingStyle equal to Bosonic() and thus trivial twists.

source
TensorKit.ZNIrrepType
ZNIrrep{N}(n::Integer)
-Irrep[ℤ{N}](n::Integer)

Represents irreps of the group $ℤ_N$ for some value of N<64. (We need 2*(N-1) <= 127 in order for a ⊗ b to work correctly.) For N equals 2, 3 or 4, ℤ{N} can be replaced by ℤ₂, ℤ₃, ℤ₄, whereas Parity is a synonym for Irrep{ℤ₂}. An arbitrary Integer n can be provided to the constructor, but only the value mod(n, N) is relevant.

source
TensorKit.U1IrrepType
U1Irrep(j::Real)
-Irrep[U₁](j::Real)

Represents irreps of the group $U₁$. The irrep is labelled by a charge, which should be an integer for a linear representation. However, it is often useful to allow half integers to represent irreps ofU₁subgroups ofSU₂, such as the Sz of spin-1/2 system. Hence, the charge is stored as aHalfIntfrom the package HalfIntegers.jl, but can be entered as arbitraryReal`. The sequence of the charges is: 0, 1/2, -1/2, 1, -1, ...

source
TensorKit.SU2IrrepType
SU2Irrep(j::Real)
-Irrep[SU₂](j::Real)

Represents irreps of the group $SU₂$. The irrep is labelled by a half integer j which can be entered as an abitrary Real, but is stored as a HalfInt from the HalfIntegers.jl package.

source
TensorKit.CU1IrrepType
CU1Irrep(j, s = ifelse(j>zero(j), 2, 0))
-Irrep[CU₁](j, s = ifelse(j>zero(j), 2, 0))

Represents irreps of the group $U₁ ⋊ C$ ($U₁$ and charge conjugation or reflection), which is also known as just O₂. The irrep is labelled by a positive half integer j (the $U₁$ charge) and an integer s indicating the behaviour under charge conjugation. They take values:

  • if j == 0, s = 0 (trivial charge conjugation) or s = 1 (non-trivial charge conjugation)
  • if j > 0, s = 2 (two-dimensional representation)
source
TensorKit.FibonacciAnyonType
struct FibonacciAnyon <: Sector
-FibonacciAnyon(s::Symbol)

Represents the anyons (isomorphism classes of simple objects) of the Fibonacci fusion category. It can take two values, corresponding to the trivial sector FibonacciAnyon(:I) and the non-trivial sector FibonacciAnyon(:τ) with fusion rules $τ ⊗ τ = 1 ⊕ τ$.

source
TensorKit.FusionTreeType
struct FusionTree{I, N, M, L, T}

Represents a fusion tree of sectors of type I<:Sector, fusing (or splitting) N uncoupled sectors to a coupled sector. (It actually represents a splitting tree, but fusion tree is a more common term). The isdual field indicates whether an isomorphism is present (if the corresponding value is true) or not. The field uncoupled contains the sectors coming out of the splitting trees, before the possible 𝑍 isomorphism. This fusion tree has M=max(0, N-2) inner lines. Furthermore, for FusionStyle(I) isa GenericFusion, the L=max(0, N-1) corresponding vertices carry a label of type T. If FusionStyle(I) isa MultiplicityFreeFusion,T = Nothing`.

source

Useful constants

TensorKit.IrrepConstant
const Irrep

A constant of a singleton type used as Irrep[G] with G<:Group a type of group, to construct or obtain a concrete subtype of AbstractIrrep{G} that implements the data structure used to represent irreducible representations of the group G.

source

Methods for defining and characterizing Sector subtypes

Base.oneMethod
one(::Sector) -> Sector
-one(::Type{<:Sector}) -> Sector

Return the unit element within this type of sector.

source
TensorKit.NsymbolFunction
Nsymbol(a::I, b::I, c::I) where {I<:Sector} -> Integer

Return an Integer representing the number of times c appears in the fusion product a ⊗ b. Could be a Bool if FusionStyle(I) == UniqueFusion() or SimpleFusion().

source
TensorKit.FsymbolFunction
Fsymbol(a::I, b::I, c::I, d::I, e::I, f::I) where {I<:Sector}

Return the F-symbol $F^{abc}_d$ that associates the two different fusion orders of sectors a, b and c into an ouput sector d, using either an intermediate sector $a ⊗ b → e$ or $b ⊗ c → f$:

a-<-μ-<-e-<-ν-<-d                                     a-<-λ-<-d
+Symmetry sectors an fusion trees · TensorKit.jl

Symmetry sectors an fusion trees

Type hierarchy

TensorKit.SectorType
abstract type Sector end

Abstract type for representing the (isomorphism classes of) simple objects in (unitary and pivotal) (pre-)fusion categories, e.g. the irreducible representations of a finite or compact group. Subtypes I<:Sector as the set of labels of a GradedSpace.

Every new I<:Sector should implement the following methods:

  • one(::Type{I}): unit element of I
  • conj(a::I): $a̅$, conjugate or dual label of $a$
  • ⊗(a::I, b::I): iterable with unique fusion outputs of $a ⊗ b$ (i.e. don't repeat in case of multiplicities)
  • Nsymbol(a::I, b::I, c::I): number of times c appears in a ⊗ b, i.e. the multiplicity
  • FusionStyle(::Type{I}): UniqueFusion(), SimpleFusion() or GenericFusion()
  • BraidingStyle(::Type{I}): Bosonic(), Fermionic(), Anyonic(), ...
  • Fsymbol(a::I, b::I, c::I, d::I, e::I, f::I): F-symbol: scalar (in case of UniqueFusion/SimpleFusion) or matrix (in case of GenericFusion)
  • Rsymbol(a::I, b::I, c::I): R-symbol: scalar (in case of UniqueFusion/SimpleFusion) or matrix (in case of GenericFusion)

and optionally

  • dim(a::I): quantum dimension of sector a
  • frobeniusschur(a::I): Frobenius-Schur indicator of a
  • Bsymbol(a::I, b::I, c::I): B-symbol: scalar (in case of UniqueFusion/SimpleFusion) or matrix (in case of GenericFusion)
  • twist(a::I) -> twist of sector a

and optionally, if FusionStyle(I) isa GenericFusion

  • vertex_ind2label(i::Int, a::I, b::I, c::I) -> a custom label for the ith copy of c appearing in a ⊗ b

Furthermore, iterate and Base.IteratorSize should be made to work for the singleton type SectorValues{I}.

source
TensorKit.SectorValuesType
struct SectorValues{I<:Sector}

Singleton type to represent an iterator over the possible values of type I, whose instance is obtained as values(I). For a new I::Sector, the following should be defined

  • Base.iterate(::SectorValues{I}[, state]): iterate over the values
  • Base.IteratorSize(::Type{SectorValues{I}}): HasLenght(), SizeUnkown() or IsInfinite() depending on whether the number of values of type I is finite (and sufficiently small) or infinite; for a large number of values, SizeUnknown() is recommend because this will trigger the use of GenericGradedSpace.

If IteratorSize(I) == HasLength(), also the following must be implemented:

  • Base.length(::SectorValues{I}): the number of different values
  • Base.getindex(::SectorValues{I}, i::Int): a mapping between an index i and an instance of I
  • findindex(::SectorValues{I}, c::I): reverse mapping between a value c::I and an index i::Integer ∈ 1:length(values(I))
source
TensorKit.FusionStyleType
FusionStyle(a::Sector) -> ::FusionStyle
+FusionStyle(I::Type{<:Sector}) -> ::FusionStyle

Return the type of fusion behavior of sectors of type I, which can be either

  • UniqueFusion(): single fusion output when fusing two sectors;
  • SimpleFusion(): multiple outputs, but every output occurs at most one, also known as multiplicity free (e.g. irreps of $SU(2)$);
  • GenericFusion(): multiple outputs that can occur more than once (e.g. irreps of $SU(3)$).

There is an abstract supertype MultipleFusion of which both SimpleFusion and GenericFusion are subtypes. Furthermore, there is a type alias MultiplicityFreeFusion for those fusion types which do not require muliplicity labels, i.e. MultiplicityFreeFusion = Union{UniqueFusion,SimpleFusion}.

source
TensorKit.BraidingStyleType
BraidingStyle(::Sector) -> ::BraidingStyle
+BraidingStyle(I::Type{<:Sector}) -> ::BraidingStyle

Return the type of braiding and twist behavior of sectors of type I, which can be either

  • Bosonic(): symmetric braiding with trivial twist (i.e. identity)
  • Fermionic(): symmetric braiding with non-trivial twist (squares to identity)
  • Anyonic(): general $R_(a,b)^c$ phase or matrix (depending on SimpleFusion or GenericFusion fusion) and arbitrary twists

Note that Bosonic and Fermionic are subtypes of SymmetricBraiding, which means that braids are in fact equivalent to crossings (i.e. braiding twice is an identity: isone(Rsymbol(b,a,c)*Rsymbol(a,b,c)) == true) and permutations are uniquely defined.

source
TensorKit.AbstractIrrepType
abstract type AbstractIrrep{G<:Group} <: Sector end

Abstract supertype for sectors which corresponds to irreps (irreducible representations) of a group G. As we assume unitary representations, these would be finite groups or compact Lie groups. Note that this could also include projective rather than linear representations.

Actual concrete implementations of those irreps can be obtained as Irrep[G], or via their actual name, which generically takes the form (asciiG)Irrep, i.e. the ASCII spelling of the group name followed by Irrep.

All irreps have BraidingStyle equal to Bosonic() and thus trivial twists.

source
TensorKit.ZNIrrepType
ZNIrrep{N}(n::Integer)
+Irrep[ℤ{N}](n::Integer)

Represents irreps of the group $ℤ_N$ for some value of N<64. (We need 2*(N-1) <= 127 in order for a ⊗ b to work correctly.) For N equals 2, 3 or 4, ℤ{N} can be replaced by ℤ₂, ℤ₃, ℤ₄, whereas Parity is a synonym for Irrep{ℤ₂}. An arbitrary Integer n can be provided to the constructor, but only the value mod(n, N) is relevant.

source
TensorKit.U1IrrepType
U1Irrep(j::Real)
+Irrep[U₁](j::Real)

Represents irreps of the group $U₁$. The irrep is labelled by a charge, which should be an integer for a linear representation. However, it is often useful to allow half integers to represent irreps ofU₁subgroups ofSU₂, such as the Sz of spin-1/2 system. Hence, the charge is stored as aHalfIntfrom the package HalfIntegers.jl, but can be entered as arbitraryReal`. The sequence of the charges is: 0, 1/2, -1/2, 1, -1, ...

source
TensorKit.SU2IrrepType
SU2Irrep(j::Real)
+Irrep[SU₂](j::Real)

Represents irreps of the group $SU₂$. The irrep is labelled by a half integer j which can be entered as an abitrary Real, but is stored as a HalfInt from the HalfIntegers.jl package.

source
TensorKit.CU1IrrepType
CU1Irrep(j, s = ifelse(j>zero(j), 2, 0))
+Irrep[CU₁](j, s = ifelse(j>zero(j), 2, 0))

Represents irreps of the group $U₁ ⋊ C$ ($U₁$ and charge conjugation or reflection), which is also known as just O₂. The irrep is labelled by a positive half integer j (the $U₁$ charge) and an integer s indicating the behaviour under charge conjugation. They take values:

  • if j == 0, s = 0 (trivial charge conjugation) or s = 1 (non-trivial charge conjugation)
  • if j > 0, s = 2 (two-dimensional representation)
source
TensorKit.FibonacciAnyonType
struct FibonacciAnyon <: Sector
+FibonacciAnyon(s::Symbol)

Represents the anyons (isomorphism classes of simple objects) of the Fibonacci fusion category. It can take two values, corresponding to the trivial sector FibonacciAnyon(:I) and the non-trivial sector FibonacciAnyon(:τ) with fusion rules $τ ⊗ τ = 1 ⊕ τ$.

source
TensorKit.FusionTreeType
struct FusionTree{I, N, M, L, T}

Represents a fusion tree of sectors of type I<:Sector, fusing (or splitting) N uncoupled sectors to a coupled sector. (It actually represents a splitting tree, but fusion tree is a more common term). The isdual field indicates whether an isomorphism is present (if the corresponding value is true) or not. The field uncoupled contains the sectors coming out of the splitting trees, before the possible 𝑍 isomorphism. This fusion tree has M=max(0, N-2) inner lines. Furthermore, for FusionStyle(I) isa GenericFusion, the L=max(0, N-1) corresponding vertices carry a label of type T. If FusionStyle(I) isa MultiplicityFreeFusion,T = Nothing`.

source

Useful constants

TensorKit.IrrepConstant
const Irrep

A constant of a singleton type used as Irrep[G] with G<:Group a type of group, to construct or obtain a concrete subtype of AbstractIrrep{G} that implements the data structure used to represent irreducible representations of the group G.

source

Methods for defining and characterizing Sector subtypes

Base.oneMethod
one(::Sector) -> Sector
+one(::Type{<:Sector}) -> Sector

Return the unit element within this type of sector.

source
TensorKit.NsymbolFunction
Nsymbol(a::I, b::I, c::I) where {I<:Sector} -> Integer

Return an Integer representing the number of times c appears in the fusion product a ⊗ b. Could be a Bool if FusionStyle(I) == UniqueFusion() or SimpleFusion().

source
TensorKit.FsymbolFunction
Fsymbol(a::I, b::I, c::I, d::I, e::I, f::I) where {I<:Sector}

Return the F-symbol $F^{abc}_d$ that associates the two different fusion orders of sectors a, b and c into an ouput sector d, using either an intermediate sector $a ⊗ b → e$ or $b ⊗ c → f$:

a-<-μ-<-e-<-ν-<-d                                     a-<-λ-<-d
     ∨       ∨       -> Fsymbol(a,b,c,d,e,f)[μ,ν,κ,λ]      ∨
     b       c                                             f
                                                           v
                                                       b-<-κ
                                                           ∨
-                                                          c

If FusionStyle(I) is UniqueFusion or SimpleFusion, the F-symbol is a number. Otherwise it is a rank 4 array of size (Nsymbol(a, b, e), Nsymbol(e, c, d), Nsymbol(b, c, f), Nsymbol(a, f, d)).

source
TensorKit.RsymbolFunction
Rsymbol(a::I, b::I, c::I) where {I<:Sector}

Returns the R-symbol $R^{ab}_c$ that maps between $c → a ⊗ b$ and $c → b ⊗ a$ as in

a -<-μ-<- c                                 b -<-ν-<- c
+                                                          c

If FusionStyle(I) is UniqueFusion or SimpleFusion, the F-symbol is a number. Otherwise it is a rank 4 array of size (Nsymbol(a, b, e), Nsymbol(e, c, d), Nsymbol(b, c, f), Nsymbol(a, f, d)).

source
TensorKit.RsymbolFunction
Rsymbol(a::I, b::I, c::I) where {I<:Sector}

Returns the R-symbol $R^{ab}_c$ that maps between $c → a ⊗ b$ and $c → b ⊗ a$ as in

a -<-μ-<- c                                 b -<-ν-<- c
      ∨          -> Rsymbol(a,b,c)[μ,ν]           v
-     b                                           a

If FusionStyle(I) is UniqueFusion() or SimpleFusion(), the R-symbol is a number. Otherwise it is a square matrix with row and column size Nsymbol(a,b,c) == Nsymbol(b,a,c).

source
TensorKit.BsymbolFunction
Bsymbol(a::I, b::I, c::I) where {I<:Sector}

Return the value of $B^{ab}_c$ which appears in transforming a splitting vertex into a fusion vertex using the transformation

a -<-μ-<- c                                                    a -<-ν-<- c
+     b                                           a

If FusionStyle(I) is UniqueFusion() or SimpleFusion(), the R-symbol is a number. Otherwise it is a square matrix with row and column size Nsymbol(a,b,c) == Nsymbol(b,a,c).

source
TensorKit.BsymbolFunction
Bsymbol(a::I, b::I, c::I) where {I<:Sector}

Return the value of $B^{ab}_c$ which appears in transforming a splitting vertex into a fusion vertex using the transformation

a -<-μ-<- c                                                    a -<-ν-<- c
      ∨          -> √(dim(c)/dim(a)) * Bsymbol(a,b,c)[μ,ν]           ∧
-     b                                                            dual(b)

If FusionStyle(I) is UniqueFusion() or SimpleFusion(), the B-symbol is a number. Otherwise it is a square matrix with row and column size Nsymbol(a, b, c) == Nsymbol(c, dual(b), a).

source
TensorKit.twistFunction
twist(a::Sector)

Return the twist of a sector a

source
twist(t::AbstractTensorMap, i::Int; inv::Bool=false)
-    -> t

Apply a twist to the ith index of t and return the result as a new tensor. If inv=true, use the inverse twist.

See twist! for storing the result in place.

source
Base.isrealMethod
isreal(::Type{<:Sector}) -> Bool

Return whether the topological data (Fsymbol, Rsymbol) of the sector is real or not (in which case it is complex).

source
TensorKit.vertex_ind2labelFunction
vertex_ind2label(k::Int, a::I, b::I, c::I) where {I<:Sector}

Convert the index k of the fusion vertex (a,b)->c into a label. For FusionStyle(I) == UniqueFusion() or FusionStyle(I) == MultipleFusion(), where every fusion output occurs only once and k == 1, the default is to suppress vertex labels by setting them equal to nothing. For FusionStyle(I) == GenericFusion(), the default is to just use k, unless a specialized method is provided.

source
TensorKit.:⊠Method
⊠(s₁::Sector, s₂::Sector)
-deligneproduct(s₁::Sector, s₂::Sector)

Given two sectors s₁ and s₂, which label an isomorphism class of simple objects in a fusion category $C₁$ and $C₂$, s1 ⊠ s2 (obtained as oxtimes+TAB) labels the isomorphism class of simple objects in the Deligne tensor product category $C₁ ⊠ C₂$.

The Deligne tensor product also works in the type domain and for spaces and tensors. For group representations, we have Irrep[G₁] ⊠ Irrep[G₂] == Irrep[G₁ × G₂].

source

Methods for manipulating fusion trees or pairs of fusion-splitting trees

The main method for manipulating a fusion-splitting tree pair is

TensorKit.braidMethod
braid(f₁::FusionTree{I}, f₂::FusionTree{I},
+     b                                                            dual(b)

If FusionStyle(I) is UniqueFusion() or SimpleFusion(), the B-symbol is a number. Otherwise it is a square matrix with row and column size Nsymbol(a, b, c) == Nsymbol(c, dual(b), a).

source
TensorKit.twistFunction
twist(a::Sector)

Return the twist of a sector a

source
twist(t::AbstractTensorMap, i::Int; inv::Bool=false)
+    -> t

Apply a twist to the ith index of t and return the result as a new tensor. If inv=true, use the inverse twist.

See twist! for storing the result in place.

source
Base.isrealMethod
isreal(::Type{<:Sector}) -> Bool

Return whether the topological data (Fsymbol, Rsymbol) of the sector is real or not (in which case it is complex).

source
TensorKit.vertex_ind2labelFunction
vertex_ind2label(k::Int, a::I, b::I, c::I) where {I<:Sector}

Convert the index k of the fusion vertex (a,b)->c into a label. For FusionStyle(I) == UniqueFusion() or FusionStyle(I) == MultipleFusion(), where every fusion output occurs only once and k == 1, the default is to suppress vertex labels by setting them equal to nothing. For FusionStyle(I) == GenericFusion(), the default is to just use k, unless a specialized method is provided.

source
TensorKit.:⊠Method
⊠(s₁::Sector, s₂::Sector)
+deligneproduct(s₁::Sector, s₂::Sector)

Given two sectors s₁ and s₂, which label an isomorphism class of simple objects in a fusion category $C₁$ and $C₂$, s1 ⊠ s2 (obtained as oxtimes+TAB) labels the isomorphism class of simple objects in the Deligne tensor product category $C₁ ⊠ C₂$.

The Deligne tensor product also works in the type domain and for spaces and tensors. For group representations, we have Irrep[G₁] ⊠ Irrep[G₂] == Irrep[G₁ × G₂].

source

Methods for manipulating fusion trees or pairs of fusion-splitting trees

The main method for manipulating a fusion-splitting tree pair is

TensorKit.braidMethod
braid(f₁::FusionTree{I}, f₂::FusionTree{I},
         levels1::IndexTuple, levels2::IndexTuple,
         p1::IndexTuple{N₁}, p2::IndexTuple{N₂}) where {I<:Sector, N₁, N₂}
--> <:AbstractDict{Tuple{FusionTree{I, N₁}, FusionTree{I, N₂}}, <:Number}

Input is a fusion-splitting tree pair that describes the fusion of a set of incoming uncoupled sectors to a set of outgoing uncoupled sectors, represented using the splitting tree f₁ and fusion tree f₂, such that the incoming sectors f₂.uncoupled are fused to f₁.coupled == f₂.coupled and then to the outgoing sectors f₁.uncoupled. Compute new trees and corresponding coefficients obtained from repartitioning and braiding the tree such that sectors p1 become outgoing and sectors p2 become incoming. The uncoupled indices in splitting tree f₁ and fusion tree f₂ have levels (or depths) levels1 and levels2 respectively, which determines how indices braid. In particular, if i and j cross, $τ_{i,j}$ is applied if levels[i] < levels[j] and $τ_{j,i}^{-1}$ if levels[i] > levels[j]. This does not allow to encode the most general braid, but a general braid can be obtained by combining such operations.

source

which, for FusionStyle(G) isa SymmetricBraiding, simplifies to

TensorKit.permuteMethod
permute(f₁::FusionTree{I}, f₂::FusionTree{I},
+-> <:AbstractDict{Tuple{FusionTree{I, N₁}, FusionTree{I, N₂}}, <:Number}

Input is a fusion-splitting tree pair that describes the fusion of a set of incoming uncoupled sectors to a set of outgoing uncoupled sectors, represented using the splitting tree f₁ and fusion tree f₂, such that the incoming sectors f₂.uncoupled are fused to f₁.coupled == f₂.coupled and then to the outgoing sectors f₁.uncoupled. Compute new trees and corresponding coefficients obtained from repartitioning and braiding the tree such that sectors p1 become outgoing and sectors p2 become incoming. The uncoupled indices in splitting tree f₁ and fusion tree f₂ have levels (or depths) levels1 and levels2 respectively, which determines how indices braid. In particular, if i and j cross, $τ_{i,j}$ is applied if levels[i] < levels[j] and $τ_{j,i}^{-1}$ if levels[i] > levels[j]. This does not allow to encode the most general braid, but a general braid can be obtained by combining such operations.

source

which, for FusionStyle(G) isa SymmetricBraiding, simplifies to

TensorKit.permuteMethod
permute(f₁::FusionTree{I}, f₂::FusionTree{I},
         p1::NTuple{N₁, Int}, p2::NTuple{N₂, Int}) where {I, N₁, N₂}
--> <:AbstractDict{Tuple{FusionTree{I, N₁}, FusionTree{I, N₂}}, <:Number}

Input is a double fusion tree that describes the fusion of a set of incoming uncoupled sectors to a set of outgoing uncoupled sectors, represented using the individual trees of outgoing (t1) and incoming sectors (t2) respectively (with identical coupled sector t1.coupled == t2.coupled). Computes new trees and corresponding coefficients obtained from repartitioning and permuting the tree such that sectors p1 become outgoing and sectors p2 become incoming.

source

These operations are implemented by composing the following more elementary manipulations

TensorKit.braidMethod
braid(f::FusionTree{<:Sector, N}, levels::NTuple{N, Int}, p::NTuple{N, Int})
--> <:AbstractDict{typeof(t), <:Number}

Perform a braiding of the uncoupled indices of the fusion tree f and return the result as a <:AbstractDict of output trees and corresponding coefficients. The braiding is determined by specifying that the new sector at position k corresponds to the sector that was originally at the position i = p[k], and assigning to every index i of the original fusion tree a distinct level or depth levels[i]. This permutation is then decomposed into elementary swaps between neighbouring indices, where the swaps are applied as braids such that if i and j cross, $τ_{i,j}$ is applied if levels[i] < levels[j] and $τ_{j,i}^{-1}$ if levels[i] > levels[j]. This does not allow to encode the most general braid, but a general braid can be obtained by combining such operations.

source
TensorKit.permuteMethod
permute(f::FusionTree, p::NTuple{N, Int}) -> <:AbstractDict{typeof(t), <:Number}

Perform a permutation of the uncoupled indices of the fusion tree f and returns the result as a <:AbstractDict of output trees and corresponding coefficients; this requires that BraidingStyle(sectortype(f)) isa SymmetricBraiding.

source
TensorKit.repartitionFunction
repartition(f₁::FusionTree{I, N₁}, f₂::FusionTree{I, N₂}, N::Int) where {I, N₁, N₂}
--> <:AbstractDict{Tuple{FusionTree{I, N}, FusionTree{I, N₁+N₂-N}}, <:Number}

Input is a double fusion tree that describes the fusion of a set of incoming uncoupled sectors to a set of outgoing uncoupled sectors, represented using the individual trees of outgoing (f₁) and incoming sectors (f₂) respectively (with identical coupled sector f₁.coupled == f₂.coupled). Computes new trees and corresponding coefficients obtained from repartitioning the tree by bending incoming to outgoing sectors (or vice versa) in order to have N outgoing sectors.

source
TensorKit.artin_braidFunction
artin_braid(f::FusionTree, i; inv::Bool = false) -> <:AbstractDict{typeof(t), <:Number}

Perform an elementary braid (Artin generator) of neighbouring uncoupled indices i and i+1 on a fusion tree f, and returns the result as a dictionary of output trees and corresponding coefficients.

The keyword inv determines whether index i will braid above or below index i+1, i.e. applying artin_braid(f′, i; inv = true) to all the outputs f′ of artin_braid(f, i; inv = false) and collecting the results should yield a single fusion tree with non-zero coefficient, namely f with coefficient 1. This keyword has no effect if BraidingStyle(sectortype(f)) isa SymmetricBraiding.

source

Finally, there are some additional manipulations for internal use

TensorKit.insertatFunction
insertat(f::FusionTree{I, N₁}, i::Int, f₂::FusionTree{I, N₂})
--> <:AbstractDict{<:FusionTree{I, N₁+N₂-1}, <:Number}

Attach a fusion tree f₂ to the uncoupled leg i of the fusion tree f₁ and bring it into a linear combination of fusion trees in standard form. This requires that f₂.coupled == f₁.uncoupled[i] and f₁.isdual[i] == false.

source
TensorKit.splitFunction
split(f::FusionTree{I, N}, M::Int)
--> (::FusionTree{I, M}, ::FusionTree{I, N-M+1})

Split a fusion tree into two. The first tree has as uncoupled sectors the first M uncoupled sectors of the input tree f, whereas its coupled sector corresponds to the internal sector between uncoupled sectors M and M+1 of the original tree f. The second tree has as first uncoupled sector that same internal sector of f, followed by remaining N-M uncoupled sectors of f. It couples to the same sector as f. This operation is the inverse of insertat in the sense that if f₁, f₂ = split(t, M) ⇒ f == insertat(f₂, 1, f₁).

source
TensorKit.mergeFunction
merge(f₁::FusionTree{I, N₁}, f₂::FusionTree{I, N₂}, c::I, μ = nothing)
--> <:AbstractDict{<:FusionTree{I, N₁+N₂}, <:Number}

Merge two fusion trees together to a linear combination of fusion trees whose uncoupled sectors are those of f₁ followed by those of f₂, and where the two coupled sectors of f₁ and f₂ are further fused to c. In case of FusionStyle(I) == GenericFusion(), also a degeneracy label μ for the fusion of the coupled sectors of f₁ and f₂ to c needs to be specified.

source
+-> <:AbstractDict{Tuple{FusionTree{I, N₁}, FusionTree{I, N₂}}, <:Number}

Input is a double fusion tree that describes the fusion of a set of incoming uncoupled sectors to a set of outgoing uncoupled sectors, represented using the individual trees of outgoing (t1) and incoming sectors (t2) respectively (with identical coupled sector t1.coupled == t2.coupled). Computes new trees and corresponding coefficients obtained from repartitioning and permuting the tree such that sectors p1 become outgoing and sectors p2 become incoming.

source

These operations are implemented by composing the following more elementary manipulations

TensorKit.braidMethod
braid(f::FusionTree{<:Sector, N}, levels::NTuple{N, Int}, p::NTuple{N, Int})
+-> <:AbstractDict{typeof(t), <:Number}

Perform a braiding of the uncoupled indices of the fusion tree f and return the result as a <:AbstractDict of output trees and corresponding coefficients. The braiding is determined by specifying that the new sector at position k corresponds to the sector that was originally at the position i = p[k], and assigning to every index i of the original fusion tree a distinct level or depth levels[i]. This permutation is then decomposed into elementary swaps between neighbouring indices, where the swaps are applied as braids such that if i and j cross, $τ_{i,j}$ is applied if levels[i] < levels[j] and $τ_{j,i}^{-1}$ if levels[i] > levels[j]. This does not allow to encode the most general braid, but a general braid can be obtained by combining such operations.

source
TensorKit.permuteMethod
permute(f::FusionTree, p::NTuple{N, Int}) -> <:AbstractDict{typeof(t), <:Number}

Perform a permutation of the uncoupled indices of the fusion tree f and returns the result as a <:AbstractDict of output trees and corresponding coefficients; this requires that BraidingStyle(sectortype(f)) isa SymmetricBraiding.

source
TensorKit.repartitionFunction
repartition(f₁::FusionTree{I, N₁}, f₂::FusionTree{I, N₂}, N::Int) where {I, N₁, N₂}
+-> <:AbstractDict{Tuple{FusionTree{I, N}, FusionTree{I, N₁+N₂-N}}, <:Number}

Input is a double fusion tree that describes the fusion of a set of incoming uncoupled sectors to a set of outgoing uncoupled sectors, represented using the individual trees of outgoing (f₁) and incoming sectors (f₂) respectively (with identical coupled sector f₁.coupled == f₂.coupled). Computes new trees and corresponding coefficients obtained from repartitioning the tree by bending incoming to outgoing sectors (or vice versa) in order to have N outgoing sectors.

source
TensorKit.artin_braidFunction
artin_braid(f::FusionTree, i; inv::Bool = false) -> <:AbstractDict{typeof(t), <:Number}

Perform an elementary braid (Artin generator) of neighbouring uncoupled indices i and i+1 on a fusion tree f, and returns the result as a dictionary of output trees and corresponding coefficients.

The keyword inv determines whether index i will braid above or below index i+1, i.e. applying artin_braid(f′, i; inv = true) to all the outputs f′ of artin_braid(f, i; inv = false) and collecting the results should yield a single fusion tree with non-zero coefficient, namely f with coefficient 1. This keyword has no effect if BraidingStyle(sectortype(f)) isa SymmetricBraiding.

source

Finally, there are some additional manipulations for internal use

TensorKit.insertatFunction
insertat(f::FusionTree{I, N₁}, i::Int, f₂::FusionTree{I, N₂})
+-> <:AbstractDict{<:FusionTree{I, N₁+N₂-1}, <:Number}

Attach a fusion tree f₂ to the uncoupled leg i of the fusion tree f₁ and bring it into a linear combination of fusion trees in standard form. This requires that f₂.coupled == f₁.uncoupled[i] and f₁.isdual[i] == false.

source
TensorKit.splitFunction
split(f::FusionTree{I, N}, M::Int)
+-> (::FusionTree{I, M}, ::FusionTree{I, N-M+1})

Split a fusion tree into two. The first tree has as uncoupled sectors the first M uncoupled sectors of the input tree f, whereas its coupled sector corresponds to the internal sector between uncoupled sectors M and M+1 of the original tree f. The second tree has as first uncoupled sector that same internal sector of f, followed by remaining N-M uncoupled sectors of f. It couples to the same sector as f. This operation is the inverse of insertat in the sense that if f₁, f₂ = split(t, M) ⇒ f == insertat(f₂, 1, f₁).

source
TensorKit.mergeFunction
merge(f₁::FusionTree{I, N₁}, f₂::FusionTree{I, N₂}, c::I, μ = nothing)
+-> <:AbstractDict{<:FusionTree{I, N₁+N₂}, <:Number}

Merge two fusion trees together to a linear combination of fusion trees whose uncoupled sectors are those of f₁ followed by those of f₂, and where the two coupled sectors of f₁ and f₂ are further fused to c. In case of FusionStyle(I) == GenericFusion(), also a degeneracy label μ for the fusion of the coupled sectors of f₁ and f₂ to c needs to be specified.

source
diff --git a/dev/lib/spaces/index.html b/dev/lib/spaces/index.html index 1816e9a7..3be96982 100644 --- a/dev/lib/spaces/index.html +++ b/dev/lib/spaces/index.html @@ -1,32 +1,32 @@ -Vector spaces · TensorKit.jl

Vector spaces

Type hierarchy

TensorKit.FieldType
abstract type Field end

Abstract type at the top of the type hierarchy for denoting fields over which vector spaces (or more generally, linear categories) can be defined. Two common fields are and , representing the field of real or complex numbers respectively.

source
TensorKit.VectorSpaceType
abstract type VectorSpace end

Abstract type at the top of the type hierarchy for denoting vector spaces, or, more accurately, 𝕜-linear categories. All instances of subtypes of VectorSpace will represent objects in 𝕜-linear monoidal categories.

source
TensorKit.ElementarySpaceType
abstract type ElementarySpace{𝕜} <: VectorSpace end

Elementary finite-dimensional vector space over a field 𝕜 that can be used as the index space corresponding to the indices of a tensor. ElementarySpace is a super type for all vector spaces (objects) that can be associated with the individual indices of a tensor, as hinted to by its alias IndexSpace.

Every elementary vector space should respond to the methods conj and dual, returning the complex conjugate space and the dual space respectively. The complex conjugate of the dual space is obtained as dual(conj(V)) === conj(dual(V)). These different spaces should be of the same type, so that a tensor can be defined as an element of a homogeneous tensor product of these spaces.

source
TensorKit.GeneralSpaceType
struct GeneralSpace{𝕜} <: ElementarySpace{𝕜}

A finite-dimensional space over an arbitrary field 𝕜 without additional structure. It is thus characterized by its dimension, and whether or not it is the dual and/or conjugate space. For a real field 𝕜, the space and its conjugate are the same.

source
TensorKit.CartesianSpaceType
struct CartesianSpace <: ElementarySpace{ℝ}

A real Euclidean space ℝ^d, which is therefore self-dual. CartesianSpace has no additonal structure and is completely characterised by its dimension d. This is the vector space that is implicitly assumed in most of matrix algebra.

source
TensorKit.ComplexSpaceType
struct ComplexSpace <: ElementarySpace{ℂ}

A standard complex vector space ℂ^d with Euclidean inner product and no additional structure. It is completely characterised by its dimension and whether its the normal space or its dual (which is canonically isomorphic to the conjugate space).

source
TensorKit.GradedSpaceType
struct GradedSpace{I<:Sector, D} <: ElementarySpace{ℂ}
+Vector spaces · TensorKit.jl

Vector spaces

Type hierarchy

TensorKit.FieldType
abstract type Field end

Abstract type at the top of the type hierarchy for denoting fields over which vector spaces (or more generally, linear categories) can be defined. Two common fields are and , representing the field of real or complex numbers respectively.

source
TensorKit.VectorSpaceType
abstract type VectorSpace end

Abstract type at the top of the type hierarchy for denoting vector spaces, or, more accurately, 𝕜-linear categories. All instances of subtypes of VectorSpace will represent objects in 𝕜-linear monoidal categories.

source
TensorKit.ElementarySpaceType
abstract type ElementarySpace{𝕜} <: VectorSpace end

Elementary finite-dimensional vector space over a field 𝕜 that can be used as the index space corresponding to the indices of a tensor. ElementarySpace is a super type for all vector spaces (objects) that can be associated with the individual indices of a tensor, as hinted to by its alias IndexSpace.

Every elementary vector space should respond to the methods conj and dual, returning the complex conjugate space and the dual space respectively. The complex conjugate of the dual space is obtained as dual(conj(V)) === conj(dual(V)). These different spaces should be of the same type, so that a tensor can be defined as an element of a homogeneous tensor product of these spaces.

source
TensorKit.GeneralSpaceType
struct GeneralSpace{𝕜} <: ElementarySpace{𝕜}

A finite-dimensional space over an arbitrary field 𝕜 without additional structure. It is thus characterized by its dimension, and whether or not it is the dual and/or conjugate space. For a real field 𝕜, the space and its conjugate are the same.

source
TensorKit.CartesianSpaceType
struct CartesianSpace <: ElementarySpace{ℝ}

A real Euclidean space ℝ^d, which is therefore self-dual. CartesianSpace has no additonal structure and is completely characterised by its dimension d. This is the vector space that is implicitly assumed in most of matrix algebra.

source
TensorKit.ComplexSpaceType
struct ComplexSpace <: ElementarySpace{ℂ}

A standard complex vector space ℂ^d with Euclidean inner product and no additional structure. It is completely characterised by its dimension and whether its the normal space or its dual (which is canonically isomorphic to the conjugate space).

source
TensorKit.GradedSpaceType
struct GradedSpace{I<:Sector, D} <: ElementarySpace{ℂ}
     dims::D
     dual::Bool
-end

A complex Euclidean space with a direct sum structure corresponding to labels in a set I, the objects of which have the structure of a monoid with respect to a monoidal product . In practice, we restrict the label set to be a set of superselection sectors of type I<:Sector, e.g. the set of distinct irreps of a finite or compact group, or the isomorphism classes of simple objects of a unitary and pivotal (pre-)fusion category.

Here dims represents the degeneracy or multiplicity of every sector.

The data structure D of dims will depend on the result Base.IteratorElsize(values(I)); if the result is of type HasLength or HasShape, dims will be stored in a NTuple{N,Int} with N = length(values(I)). This requires that a sector s::I can be transformed into an index via s == getindex(values(I), i) and i == findindex(values(I), s). If Base.IteratorElsize(values(I)) results IsInfinite() or SizeUnknown(), a SectorDict{I,Int} is used to store the non-zero degeneracy dimensions with the corresponding sector as key. The parameter D is hidden from the user and should typically be of no concern.

The concrete type GradedSpace{I,D} with correct D can be obtained as Vect[I], or if I == Irrep[G] for some G<:Group, as Rep[G].

source
TensorKit.CompositeSpaceType
abstract type CompositeSpace{S<:ElementarySpace} <: VectorSpace end

Abstract type for composite spaces that are defined in terms of a number of elementary vector spaces of a homogeneous type S<:ElementarySpace{𝕜}.

source
TensorKit.ProductSpaceType
struct ProductSpace{S<:ElementarySpace, N} <: CompositeSpace{S}

A ProductSpace is a tensor product space of N vector spaces of type S<:ElementarySpace. Only tensor products between ElementarySpace objects of the same type are allowed.

source

Useful constants

TensorKit.VectConstant
const Vect

A constant of a singleton type used as Vect[I] with I<:Sector a type of sector, to construct or obtain the concrete type GradedSpace{I,D} instances without having to specify D.

source
TensorKit.RepConstant
const Rep

A constant of a singleton type used as Rep[G] with G<:Group a type of group, to construct or obtain the concrete type GradedSpace{Irrep[G],D} instances without having to specify D. Note that Rep[G] == Vect[Irrep[G]].

See also Irrep and Vect.

source
Missing docstring.

Missing docstring for ZNSpace{N}. Check Documenter's build log for details.

TensorKit.Z2SpaceType
struct GradedSpace{I<:Sector, D} <: ElementarySpace{ℂ}
+end

A complex Euclidean space with a direct sum structure corresponding to labels in a set I, the objects of which have the structure of a monoid with respect to a monoidal product . In practice, we restrict the label set to be a set of superselection sectors of type I<:Sector, e.g. the set of distinct irreps of a finite or compact group, or the isomorphism classes of simple objects of a unitary and pivotal (pre-)fusion category.

Here dims represents the degeneracy or multiplicity of every sector.

The data structure D of dims will depend on the result Base.IteratorElsize(values(I)); if the result is of type HasLength or HasShape, dims will be stored in a NTuple{N,Int} with N = length(values(I)). This requires that a sector s::I can be transformed into an index via s == getindex(values(I), i) and i == findindex(values(I), s). If Base.IteratorElsize(values(I)) results IsInfinite() or SizeUnknown(), a SectorDict{I,Int} is used to store the non-zero degeneracy dimensions with the corresponding sector as key. The parameter D is hidden from the user and should typically be of no concern.

The concrete type GradedSpace{I,D} with correct D can be obtained as Vect[I], or if I == Irrep[G] for some G<:Group, as Rep[G].

source
TensorKit.CompositeSpaceType
abstract type CompositeSpace{S<:ElementarySpace} <: VectorSpace end

Abstract type for composite spaces that are defined in terms of a number of elementary vector spaces of a homogeneous type S<:ElementarySpace{𝕜}.

source
TensorKit.ProductSpaceType
struct ProductSpace{S<:ElementarySpace, N} <: CompositeSpace{S}

A ProductSpace is a tensor product space of N vector spaces of type S<:ElementarySpace. Only tensor products between ElementarySpace objects of the same type are allowed.

source

Useful constants

TensorKit.VectConstant
const Vect

A constant of a singleton type used as Vect[I] with I<:Sector a type of sector, to construct or obtain the concrete type GradedSpace{I,D} instances without having to specify D.

source
TensorKit.RepConstant
const Rep

A constant of a singleton type used as Rep[G] with G<:Group a type of group, to construct or obtain the concrete type GradedSpace{Irrep[G],D} instances without having to specify D. Note that Rep[G] == Vect[Irrep[G]].

See also Irrep and Vect.

source
Missing docstring.

Missing docstring for ZNSpace{N}. Check Documenter's build log for details.

TensorKit.Z2SpaceType
struct GradedSpace{I<:Sector, D} <: ElementarySpace{ℂ}
     dims::D
     dual::Bool
-end

A complex Euclidean space with a direct sum structure corresponding to labels in a set I, the objects of which have the structure of a monoid with respect to a monoidal product . In practice, we restrict the label set to be a set of superselection sectors of type I<:Sector, e.g. the set of distinct irreps of a finite or compact group, or the isomorphism classes of simple objects of a unitary and pivotal (pre-)fusion category.

Here dims represents the degeneracy or multiplicity of every sector.

The data structure D of dims will depend on the result Base.IteratorElsize(values(I)); if the result is of type HasLength or HasShape, dims will be stored in a NTuple{N,Int} with N = length(values(I)). This requires that a sector s::I can be transformed into an index via s == getindex(values(I), i) and i == findindex(values(I), s). If Base.IteratorElsize(values(I)) results IsInfinite() or SizeUnknown(), a SectorDict{I,Int} is used to store the non-zero degeneracy dimensions with the corresponding sector as key. The parameter D is hidden from the user and should typically be of no concern.

The concrete type GradedSpace{I,D} with correct D can be obtained as Vect[I], or if I == Irrep[G] for some G<:Group, as Rep[G].

source
TensorKit.Z3SpaceType
struct GradedSpace{I<:Sector, D} <: ElementarySpace{ℂ}
+end

A complex Euclidean space with a direct sum structure corresponding to labels in a set I, the objects of which have the structure of a monoid with respect to a monoidal product . In practice, we restrict the label set to be a set of superselection sectors of type I<:Sector, e.g. the set of distinct irreps of a finite or compact group, or the isomorphism classes of simple objects of a unitary and pivotal (pre-)fusion category.

Here dims represents the degeneracy or multiplicity of every sector.

The data structure D of dims will depend on the result Base.IteratorElsize(values(I)); if the result is of type HasLength or HasShape, dims will be stored in a NTuple{N,Int} with N = length(values(I)). This requires that a sector s::I can be transformed into an index via s == getindex(values(I), i) and i == findindex(values(I), s). If Base.IteratorElsize(values(I)) results IsInfinite() or SizeUnknown(), a SectorDict{I,Int} is used to store the non-zero degeneracy dimensions with the corresponding sector as key. The parameter D is hidden from the user and should typically be of no concern.

The concrete type GradedSpace{I,D} with correct D can be obtained as Vect[I], or if I == Irrep[G] for some G<:Group, as Rep[G].

source
TensorKit.Z3SpaceType
struct GradedSpace{I<:Sector, D} <: ElementarySpace{ℂ}
     dims::D
     dual::Bool
-end

A complex Euclidean space with a direct sum structure corresponding to labels in a set I, the objects of which have the structure of a monoid with respect to a monoidal product . In practice, we restrict the label set to be a set of superselection sectors of type I<:Sector, e.g. the set of distinct irreps of a finite or compact group, or the isomorphism classes of simple objects of a unitary and pivotal (pre-)fusion category.

Here dims represents the degeneracy or multiplicity of every sector.

The data structure D of dims will depend on the result Base.IteratorElsize(values(I)); if the result is of type HasLength or HasShape, dims will be stored in a NTuple{N,Int} with N = length(values(I)). This requires that a sector s::I can be transformed into an index via s == getindex(values(I), i) and i == findindex(values(I), s). If Base.IteratorElsize(values(I)) results IsInfinite() or SizeUnknown(), a SectorDict{I,Int} is used to store the non-zero degeneracy dimensions with the corresponding sector as key. The parameter D is hidden from the user and should typically be of no concern.

The concrete type GradedSpace{I,D} with correct D can be obtained as Vect[I], or if I == Irrep[G] for some G<:Group, as Rep[G].

source
TensorKit.Z4SpaceType
struct GradedSpace{I<:Sector, D} <: ElementarySpace{ℂ}
+end

A complex Euclidean space with a direct sum structure corresponding to labels in a set I, the objects of which have the structure of a monoid with respect to a monoidal product . In practice, we restrict the label set to be a set of superselection sectors of type I<:Sector, e.g. the set of distinct irreps of a finite or compact group, or the isomorphism classes of simple objects of a unitary and pivotal (pre-)fusion category.

Here dims represents the degeneracy or multiplicity of every sector.

The data structure D of dims will depend on the result Base.IteratorElsize(values(I)); if the result is of type HasLength or HasShape, dims will be stored in a NTuple{N,Int} with N = length(values(I)). This requires that a sector s::I can be transformed into an index via s == getindex(values(I), i) and i == findindex(values(I), s). If Base.IteratorElsize(values(I)) results IsInfinite() or SizeUnknown(), a SectorDict{I,Int} is used to store the non-zero degeneracy dimensions with the corresponding sector as key. The parameter D is hidden from the user and should typically be of no concern.

The concrete type GradedSpace{I,D} with correct D can be obtained as Vect[I], or if I == Irrep[G] for some G<:Group, as Rep[G].

source
TensorKit.Z4SpaceType
struct GradedSpace{I<:Sector, D} <: ElementarySpace{ℂ}
     dims::D
     dual::Bool
-end

A complex Euclidean space with a direct sum structure corresponding to labels in a set I, the objects of which have the structure of a monoid with respect to a monoidal product . In practice, we restrict the label set to be a set of superselection sectors of type I<:Sector, e.g. the set of distinct irreps of a finite or compact group, or the isomorphism classes of simple objects of a unitary and pivotal (pre-)fusion category.

Here dims represents the degeneracy or multiplicity of every sector.

The data structure D of dims will depend on the result Base.IteratorElsize(values(I)); if the result is of type HasLength or HasShape, dims will be stored in a NTuple{N,Int} with N = length(values(I)). This requires that a sector s::I can be transformed into an index via s == getindex(values(I), i) and i == findindex(values(I), s). If Base.IteratorElsize(values(I)) results IsInfinite() or SizeUnknown(), a SectorDict{I,Int} is used to store the non-zero degeneracy dimensions with the corresponding sector as key. The parameter D is hidden from the user and should typically be of no concern.

The concrete type GradedSpace{I,D} with correct D can be obtained as Vect[I], or if I == Irrep[G] for some G<:Group, as Rep[G].

source
TensorKit.U1SpaceType
struct GradedSpace{I<:Sector, D} <: ElementarySpace{ℂ}
+end

A complex Euclidean space with a direct sum structure corresponding to labels in a set I, the objects of which have the structure of a monoid with respect to a monoidal product . In practice, we restrict the label set to be a set of superselection sectors of type I<:Sector, e.g. the set of distinct irreps of a finite or compact group, or the isomorphism classes of simple objects of a unitary and pivotal (pre-)fusion category.

Here dims represents the degeneracy or multiplicity of every sector.

The data structure D of dims will depend on the result Base.IteratorElsize(values(I)); if the result is of type HasLength or HasShape, dims will be stored in a NTuple{N,Int} with N = length(values(I)). This requires that a sector s::I can be transformed into an index via s == getindex(values(I), i) and i == findindex(values(I), s). If Base.IteratorElsize(values(I)) results IsInfinite() or SizeUnknown(), a SectorDict{I,Int} is used to store the non-zero degeneracy dimensions with the corresponding sector as key. The parameter D is hidden from the user and should typically be of no concern.

The concrete type GradedSpace{I,D} with correct D can be obtained as Vect[I], or if I == Irrep[G] for some G<:Group, as Rep[G].

source
TensorKit.U1SpaceType
struct GradedSpace{I<:Sector, D} <: ElementarySpace{ℂ}
     dims::D
     dual::Bool
-end

A complex Euclidean space with a direct sum structure corresponding to labels in a set I, the objects of which have the structure of a monoid with respect to a monoidal product . In practice, we restrict the label set to be a set of superselection sectors of type I<:Sector, e.g. the set of distinct irreps of a finite or compact group, or the isomorphism classes of simple objects of a unitary and pivotal (pre-)fusion category.

Here dims represents the degeneracy or multiplicity of every sector.

The data structure D of dims will depend on the result Base.IteratorElsize(values(I)); if the result is of type HasLength or HasShape, dims will be stored in a NTuple{N,Int} with N = length(values(I)). This requires that a sector s::I can be transformed into an index via s == getindex(values(I), i) and i == findindex(values(I), s). If Base.IteratorElsize(values(I)) results IsInfinite() or SizeUnknown(), a SectorDict{I,Int} is used to store the non-zero degeneracy dimensions with the corresponding sector as key. The parameter D is hidden from the user and should typically be of no concern.

The concrete type GradedSpace{I,D} with correct D can be obtained as Vect[I], or if I == Irrep[G] for some G<:Group, as Rep[G].

source
TensorKit.SU2SpaceType
struct GradedSpace{I<:Sector, D} <: ElementarySpace{ℂ}
+end

A complex Euclidean space with a direct sum structure corresponding to labels in a set I, the objects of which have the structure of a monoid with respect to a monoidal product . In practice, we restrict the label set to be a set of superselection sectors of type I<:Sector, e.g. the set of distinct irreps of a finite or compact group, or the isomorphism classes of simple objects of a unitary and pivotal (pre-)fusion category.

Here dims represents the degeneracy or multiplicity of every sector.

The data structure D of dims will depend on the result Base.IteratorElsize(values(I)); if the result is of type HasLength or HasShape, dims will be stored in a NTuple{N,Int} with N = length(values(I)). This requires that a sector s::I can be transformed into an index via s == getindex(values(I), i) and i == findindex(values(I), s). If Base.IteratorElsize(values(I)) results IsInfinite() or SizeUnknown(), a SectorDict{I,Int} is used to store the non-zero degeneracy dimensions with the corresponding sector as key. The parameter D is hidden from the user and should typically be of no concern.

The concrete type GradedSpace{I,D} with correct D can be obtained as Vect[I], or if I == Irrep[G] for some G<:Group, as Rep[G].

source
TensorKit.SU2SpaceType
struct GradedSpace{I<:Sector, D} <: ElementarySpace{ℂ}
     dims::D
     dual::Bool
-end

A complex Euclidean space with a direct sum structure corresponding to labels in a set I, the objects of which have the structure of a monoid with respect to a monoidal product . In practice, we restrict the label set to be a set of superselection sectors of type I<:Sector, e.g. the set of distinct irreps of a finite or compact group, or the isomorphism classes of simple objects of a unitary and pivotal (pre-)fusion category.

Here dims represents the degeneracy or multiplicity of every sector.

The data structure D of dims will depend on the result Base.IteratorElsize(values(I)); if the result is of type HasLength or HasShape, dims will be stored in a NTuple{N,Int} with N = length(values(I)). This requires that a sector s::I can be transformed into an index via s == getindex(values(I), i) and i == findindex(values(I), s). If Base.IteratorElsize(values(I)) results IsInfinite() or SizeUnknown(), a SectorDict{I,Int} is used to store the non-zero degeneracy dimensions with the corresponding sector as key. The parameter D is hidden from the user and should typically be of no concern.

The concrete type GradedSpace{I,D} with correct D can be obtained as Vect[I], or if I == Irrep[G] for some G<:Group, as Rep[G].

source
TensorKit.CU1SpaceType
struct GradedSpace{I<:Sector, D} <: ElementarySpace{ℂ}
+end

A complex Euclidean space with a direct sum structure corresponding to labels in a set I, the objects of which have the structure of a monoid with respect to a monoidal product . In practice, we restrict the label set to be a set of superselection sectors of type I<:Sector, e.g. the set of distinct irreps of a finite or compact group, or the isomorphism classes of simple objects of a unitary and pivotal (pre-)fusion category.

Here dims represents the degeneracy or multiplicity of every sector.

The data structure D of dims will depend on the result Base.IteratorElsize(values(I)); if the result is of type HasLength or HasShape, dims will be stored in a NTuple{N,Int} with N = length(values(I)). This requires that a sector s::I can be transformed into an index via s == getindex(values(I), i) and i == findindex(values(I), s). If Base.IteratorElsize(values(I)) results IsInfinite() or SizeUnknown(), a SectorDict{I,Int} is used to store the non-zero degeneracy dimensions with the corresponding sector as key. The parameter D is hidden from the user and should typically be of no concern.

The concrete type GradedSpace{I,D} with correct D can be obtained as Vect[I], or if I == Irrep[G] for some G<:Group, as Rep[G].

source
TensorKit.CU1SpaceType
struct GradedSpace{I<:Sector, D} <: ElementarySpace{ℂ}
     dims::D
     dual::Bool
-end

A complex Euclidean space with a direct sum structure corresponding to labels in a set I, the objects of which have the structure of a monoid with respect to a monoidal product . In practice, we restrict the label set to be a set of superselection sectors of type I<:Sector, e.g. the set of distinct irreps of a finite or compact group, or the isomorphism classes of simple objects of a unitary and pivotal (pre-)fusion category.

Here dims represents the degeneracy or multiplicity of every sector.

The data structure D of dims will depend on the result Base.IteratorElsize(values(I)); if the result is of type HasLength or HasShape, dims will be stored in a NTuple{N,Int} with N = length(values(I)). This requires that a sector s::I can be transformed into an index via s == getindex(values(I), i) and i == findindex(values(I), s). If Base.IteratorElsize(values(I)) results IsInfinite() or SizeUnknown(), a SectorDict{I,Int} is used to store the non-zero degeneracy dimensions with the corresponding sector as key. The parameter D is hidden from the user and should typically be of no concern.

The concrete type GradedSpace{I,D} with correct D can be obtained as Vect[I], or if I == Irrep[G] for some G<:Group, as Rep[G].

source

Methods

Methods often apply similar to e.g. spaces and corresponding tensors or tensor maps, e.g.:

TensorKit.fieldFunction
field(V::VectorSpace) -> Field

Return the field type over which a vector space is defined.

source
TensorKit.sectortypeFunction
sectortype(a) -> Type{<:Sector}

Return the type of sector over which object a (e.g. a representation space or a tensor) is defined. Also works in type domain.

source
TensorKit.sectorsFunction
sectors(V::ElementarySpace)

Return an iterator over the different sectors of V.

source
sectors(P::ProductSpace{S, N}) where {S<:ElementarySpace}

Return an iterator over all possible combinations of sectors (represented as an NTuple{N, sectortype(S)}) that can appear within the tensor product space P.

source
TensorKit.hassectorFunction
hassector(V::VectorSpace, a::Sector) -> Bool

Return whether a vector space V has a subspace corresponding to sector a with non-zero dimension, i.e. dim(V, a) > 0.

source
hassector(P::ProductSpace{S, N}, s::NTuple{N, sectortype(S)}) where {S<:ElementarySpace}
--> Bool

Query whether P has a non-zero degeneracy of sector s, representing a combination of sectors on the individual tensor indices.

source
TensorKit.dimFunction
dim(V::VectorSpace) -> Int

Return the total dimension of the vector space V as an Int.

source
TensorKit.dimsFunction
dims(::ProductSpace{S, N}) -> Dims{N} = NTuple{N, Int}

Return the dimensions of the spaces in the tensor product space as a tuple of integers.

source
dims(P::ProductSpace{S, N}, s::NTuple{N, sectortype(S)}) where {S<:ElementarySpace}
--> Dims{N} = NTuple{N, Int}

Return the degeneracy dimensions corresponding to a tuple of sectors s for each of the spaces in the tensor product P.

source
TensorKit.blocksectorsFunction
blocksectors(P::ProductSpace)

Return an iterator over the different unique coupled sector labels, i.e. the different fusion outputs that can be obtained by fusing the sectors present in the different spaces that make up the ProductSpace instance.

source
blocksectors(W::HomSpace)

Return an iterator over the different unique coupled sector labels, i.e. the intersection of the different fusion outputs that can be obtained by fusing the sectors present in the domain, as well as from the codomain.

See also hasblock.

source
TensorKit.blockdimFunction
blockdim(P::ProductSpace, c::Sector)

Return the total dimension of a coupled sector c in the product space, by summing over all dim(P, s) for all tuples of sectors s::NTuple{N, <:Sector} that can fuse to c, counted with the correct multiplicity (i.e. number of ways in which s can fuse to c).

See also hasblock and blocksectors.

source

The following methods act specifically on ElementarySpace spaces:

TensorKit.isdualFunction
isdual(V::ElementarySpace) -> Bool

Return wether an ElementarySpace V is normal or rather a dual space. Always returns false for spaces where V == dual(V).

source
TensorKit.dualFunction
dual(V::VectorSpace) -> VectorSpace

Return the dual space of V; also obtained via V'. This should satisfy dual(dual(V)) == V. It is assumed that typeof(V) == typeof(V').

source
Base.conjFunction
conj(V::S) where {S<:ElementarySpace} -> S

Return the conjugate space of V. This should satisfy conj(conj(V)) == V.

For field(V)==ℝ, conj(V) == V. It is assumed that typeof(V) == typeof(conj(V)).

source
TensorKit.flipFunction
flip(V::S) where {S<:ElementarySpace} -> S

Return a single vector space of type S that has the same value of isdual as dual(V), but yet is isomorphic to V rather than to dual(V). The spaces flip(V) and dual(V) only differ in the case of GradedSpace{I}.

source
TensorKit.:⊕Function
⊕(V₁::S, V₂::S, V₃::S...) where {S<:ElementarySpace} -> S

Return the corresponding vector space of type S that represents the direct sum sum of the spaces V₁, V₂, ... Note that all the individual spaces should have the same value for isdual, as otherwise the direct sum is not defined.

source
Base.oneunitFunction
oneunit(V::S) where {S<:ElementarySpace} -> S

Return the corresponding vector space of type S that represents the trivial one-dimensional space, i.e. the space that is isomorphic to the corresponding field. Note that this is different from one(V::S), which returns the empty product space ProductSpace{S,0}(()).

source
TensorKit.supremumFunction
supremum(V₁::ElementarySpace, V₂::ElementarySpace, V₃::ElementarySpace...)

Return the supremum of a number of elementary spaces, i.e. an instance V::ElementarySpace such that V ≿ V₁, V ≿ V₂, ... and no other W ≺ V has this property. This requires that all arguments have the same value of isdual( ), and also the return value V will have the same value.

source
TensorKit.infimumFunction
infimum(V₁::ElementarySpace, V₂::ElementarySpace, V₃::ElementarySpace...)

Return the infimum of a number of elementary spaces, i.e. an instance V::ElementarySpace such that V ≾ V₁, V ≾ V₂, ... and no other W ≻ V has this property. This requires that all arguments have the same value of isdual( ), and also the return value V will have the same value.

source

while the following also work on both ElementarySpace and ProductSpace

TensorKit.fuseFunction
fuse(V₁::S, V₂::S, V₃::S...) where {S<:ElementarySpace} -> S
-fuse(P::ProductSpace{S}) where {S<:ElementarySpace} -> S

Return a single vector space of type S that is isomorphic to the fusion product of the individual spaces V₁, V₂, ..., or the spaces contained in P.

source
TensorKit.:⊗Function
⊗(V₁::S, V₂::S, V₃::S...) where {S<:ElementarySpace} -> S

Create a ProductSpace{S}(V₁, V₂, V₃...) representing the tensor product of several elementary vector spaces. For convience, Julia's regular multiplication operator * applied to vector spaces has the same effect.

The tensor product structure is preserved, see fuse for returning a single elementary space of type S that is isomorphic to this tensor product.

source
TensorKit.:⊠Function
⊠(s₁::Sector, s₂::Sector)
-deligneproduct(s₁::Sector, s₂::Sector)

Given two sectors s₁ and s₂, which label an isomorphism class of simple objects in a fusion category $C₁$ and $C₂$, s1 ⊠ s2 (obtained as oxtimes+TAB) labels the isomorphism class of simple objects in the Deligne tensor product category $C₁ ⊠ C₂$.

The Deligne tensor product also works in the type domain and for spaces and tensors. For group representations, we have Irrep[G₁] ⊠ Irrep[G₂] == Irrep[G₁ × G₂].

source
⊠(V₁::VectorSpace, V₂::VectorSpace)

Given two vector spaces V₁ and V₂ (ElementarySpace or ProductSpace), or thus, objects of corresponding fusion categories $C₁$ and $C₂$, $V₁ ⊠ V₂$ constructs the Deligne tensor product, an object in $C₁ ⊠ C₂$ which is the natural tensor product of those categories. In particular, the corresponding type of sectors (simple objects) is given by sectortype(V₁ ⊠ V₂) == sectortype(V₁) ⊠ sectortype(V₂) and can be thought of as a tuple of the individual sectors.

The Deligne tensor product also works in the type domain and for sectors and tensors. For group representations, we have Rep[G₁] ⊠ Rep[G₂] == Rep[G₁ × G₂], i.e. these are the natural representation spaces of the direct product of two groups.

source
Base.oneFunction
one(::Sector) -> Sector
-one(::Type{<:Sector}) -> Sector

Return the unit element within this type of sector.

source
one(::S) where {S<:ElementarySpace} -> ProductSpace{S, 0}
-one(::ProductSpace{S}) where {S<:ElementarySpace} -> ProductSpace{S, 0}

Return a tensor product of zero spaces of type S, i.e. this is the unit object under the tensor product operation, such that V ⊗ one(V) == V.

source
TensorKit.ismonomorphicFunction
ismonomorphic(V₁::VectorSpace, V₂::VectorSpace)
-V₁ ≾ V₂

Return whether there exist monomorphisms from V₁ to V₂, i.e. 'injective' morphisms with left inverses.

source
TensorKit.isepimorphicFunction
isepimorphic(V₁::VectorSpace, V₂::VectorSpace)
-V₁ ≿ V₂

Return whether there exist epimorphisms from V₁ to V₂, i.e. 'surjective' morphisms with right inverses.

source
TensorKit.isisomorphicFunction
isisomorphic(V₁::VectorSpace, V₂::VectorSpace)
-V₁ ≅ V₂

Return if V₁ and V₂ are isomorphic, meaning that there exists isomorphisms from V₁ to V₂, i.e. morphisms with left and right inverses.

source
TensorKit.insertunitFunction
insertunit(P::ProductSpace, i::Int = length(P)+1; dual = false, conj = false)

For P::ProductSpace{S,N}, this adds an extra tensor product factor at position 1 <= i <= N+1 (last position by default) which is just the S-equivalent of the underlying field of scalars, i.e. oneunit(S). With the keyword arguments, one can choose to insert the conjugated or dual space instead, which are all isomorphic to the field of scalars.

source
+end

A complex Euclidean space with a direct sum structure corresponding to labels in a set I, the objects of which have the structure of a monoid with respect to a monoidal product . In practice, we restrict the label set to be a set of superselection sectors of type I<:Sector, e.g. the set of distinct irreps of a finite or compact group, or the isomorphism classes of simple objects of a unitary and pivotal (pre-)fusion category.

Here dims represents the degeneracy or multiplicity of every sector.

The data structure D of dims will depend on the result Base.IteratorElsize(values(I)); if the result is of type HasLength or HasShape, dims will be stored in a NTuple{N,Int} with N = length(values(I)). This requires that a sector s::I can be transformed into an index via s == getindex(values(I), i) and i == findindex(values(I), s). If Base.IteratorElsize(values(I)) results IsInfinite() or SizeUnknown(), a SectorDict{I,Int} is used to store the non-zero degeneracy dimensions with the corresponding sector as key. The parameter D is hidden from the user and should typically be of no concern.

The concrete type GradedSpace{I,D} with correct D can be obtained as Vect[I], or if I == Irrep[G] for some G<:Group, as Rep[G].

source

Methods

Methods often apply similar to e.g. spaces and corresponding tensors or tensor maps, e.g.:

TensorKit.fieldFunction
field(V::VectorSpace) -> Field

Return the field type over which a vector space is defined.

source
TensorKit.sectortypeFunction
sectortype(a) -> Type{<:Sector}

Return the type of sector over which object a (e.g. a representation space or a tensor) is defined. Also works in type domain.

source
TensorKit.sectorsFunction
sectors(V::ElementarySpace)

Return an iterator over the different sectors of V.

source
sectors(P::ProductSpace{S, N}) where {S<:ElementarySpace}

Return an iterator over all possible combinations of sectors (represented as an NTuple{N, sectortype(S)}) that can appear within the tensor product space P.

source
TensorKit.hassectorFunction
hassector(V::VectorSpace, a::Sector) -> Bool

Return whether a vector space V has a subspace corresponding to sector a with non-zero dimension, i.e. dim(V, a) > 0.

source
hassector(P::ProductSpace{S, N}, s::NTuple{N, sectortype(S)}) where {S<:ElementarySpace}
+-> Bool

Query whether P has a non-zero degeneracy of sector s, representing a combination of sectors on the individual tensor indices.

source
TensorKit.dimFunction
dim(V::VectorSpace) -> Int

Return the total dimension of the vector space V as an Int.

source
TensorKit.dimsFunction
dims(::ProductSpace{S, N}) -> Dims{N} = NTuple{N, Int}

Return the dimensions of the spaces in the tensor product space as a tuple of integers.

source
dims(P::ProductSpace{S, N}, s::NTuple{N, sectortype(S)}) where {S<:ElementarySpace}
+-> Dims{N} = NTuple{N, Int}

Return the degeneracy dimensions corresponding to a tuple of sectors s for each of the spaces in the tensor product P.

source
TensorKit.blocksectorsFunction
blocksectors(P::ProductSpace)

Return an iterator over the different unique coupled sector labels, i.e. the different fusion outputs that can be obtained by fusing the sectors present in the different spaces that make up the ProductSpace instance.

source
blocksectors(W::HomSpace)

Return an iterator over the different unique coupled sector labels, i.e. the intersection of the different fusion outputs that can be obtained by fusing the sectors present in the domain, as well as from the codomain.

See also hasblock.

source
TensorKit.blockdimFunction
blockdim(P::ProductSpace, c::Sector)

Return the total dimension of a coupled sector c in the product space, by summing over all dim(P, s) for all tuples of sectors s::NTuple{N, <:Sector} that can fuse to c, counted with the correct multiplicity (i.e. number of ways in which s can fuse to c).

See also hasblock and blocksectors.

source

The following methods act specifically on ElementarySpace spaces:

TensorKit.isdualFunction
isdual(V::ElementarySpace) -> Bool

Return wether an ElementarySpace V is normal or rather a dual space. Always returns false for spaces where V == dual(V).

source
TensorKit.dualFunction
dual(V::VectorSpace) -> VectorSpace

Return the dual space of V; also obtained via V'. This should satisfy dual(dual(V)) == V. It is assumed that typeof(V) == typeof(V').

source
Base.conjFunction
conj(V::S) where {S<:ElementarySpace} -> S

Return the conjugate space of V. This should satisfy conj(conj(V)) == V.

For field(V)==ℝ, conj(V) == V. It is assumed that typeof(V) == typeof(conj(V)).

source
TensorKit.flipFunction
flip(V::S) where {S<:ElementarySpace} -> S

Return a single vector space of type S that has the same value of isdual as dual(V), but yet is isomorphic to V rather than to dual(V). The spaces flip(V) and dual(V) only differ in the case of GradedSpace{I}.

source
TensorKit.:⊕Function
⊕(V₁::S, V₂::S, V₃::S...) where {S<:ElementarySpace} -> S

Return the corresponding vector space of type S that represents the direct sum sum of the spaces V₁, V₂, ... Note that all the individual spaces should have the same value for isdual, as otherwise the direct sum is not defined.

source
Base.oneunitFunction
oneunit(V::S) where {S<:ElementarySpace} -> S

Return the corresponding vector space of type S that represents the trivial one-dimensional space, i.e. the space that is isomorphic to the corresponding field. Note that this is different from one(V::S), which returns the empty product space ProductSpace{S,0}(()).

source
TensorKit.supremumFunction
supremum(V₁::ElementarySpace, V₂::ElementarySpace, V₃::ElementarySpace...)

Return the supremum of a number of elementary spaces, i.e. an instance V::ElementarySpace such that V ≿ V₁, V ≿ V₂, ... and no other W ≺ V has this property. This requires that all arguments have the same value of isdual( ), and also the return value V will have the same value.

source
TensorKit.infimumFunction
infimum(V₁::ElementarySpace, V₂::ElementarySpace, V₃::ElementarySpace...)

Return the infimum of a number of elementary spaces, i.e. an instance V::ElementarySpace such that V ≾ V₁, V ≾ V₂, ... and no other W ≻ V has this property. This requires that all arguments have the same value of isdual( ), and also the return value V will have the same value.

source

while the following also work on both ElementarySpace and ProductSpace

TensorKit.fuseFunction
fuse(V₁::S, V₂::S, V₃::S...) where {S<:ElementarySpace} -> S
+fuse(P::ProductSpace{S}) where {S<:ElementarySpace} -> S

Return a single vector space of type S that is isomorphic to the fusion product of the individual spaces V₁, V₂, ..., or the spaces contained in P.

source
TensorKit.:⊗Function
⊗(V₁::S, V₂::S, V₃::S...) where {S<:ElementarySpace} -> S

Create a ProductSpace{S}(V₁, V₂, V₃...) representing the tensor product of several elementary vector spaces. For convience, Julia's regular multiplication operator * applied to vector spaces has the same effect.

The tensor product structure is preserved, see fuse for returning a single elementary space of type S that is isomorphic to this tensor product.

source
TensorKit.:⊠Function
⊠(s₁::Sector, s₂::Sector)
+deligneproduct(s₁::Sector, s₂::Sector)

Given two sectors s₁ and s₂, which label an isomorphism class of simple objects in a fusion category $C₁$ and $C₂$, s1 ⊠ s2 (obtained as oxtimes+TAB) labels the isomorphism class of simple objects in the Deligne tensor product category $C₁ ⊠ C₂$.

The Deligne tensor product also works in the type domain and for spaces and tensors. For group representations, we have Irrep[G₁] ⊠ Irrep[G₂] == Irrep[G₁ × G₂].

source
⊠(V₁::VectorSpace, V₂::VectorSpace)

Given two vector spaces V₁ and V₂ (ElementarySpace or ProductSpace), or thus, objects of corresponding fusion categories $C₁$ and $C₂$, $V₁ ⊠ V₂$ constructs the Deligne tensor product, an object in $C₁ ⊠ C₂$ which is the natural tensor product of those categories. In particular, the corresponding type of sectors (simple objects) is given by sectortype(V₁ ⊠ V₂) == sectortype(V₁) ⊠ sectortype(V₂) and can be thought of as a tuple of the individual sectors.

The Deligne tensor product also works in the type domain and for sectors and tensors. For group representations, we have Rep[G₁] ⊠ Rep[G₂] == Rep[G₁ × G₂], i.e. these are the natural representation spaces of the direct product of two groups.

source
Base.oneFunction
one(::Sector) -> Sector
+one(::Type{<:Sector}) -> Sector

Return the unit element within this type of sector.

source
one(::S) where {S<:ElementarySpace} -> ProductSpace{S, 0}
+one(::ProductSpace{S}) where {S<:ElementarySpace} -> ProductSpace{S, 0}

Return a tensor product of zero spaces of type S, i.e. this is the unit object under the tensor product operation, such that V ⊗ one(V) == V.

source
TensorKit.ismonomorphicFunction
ismonomorphic(V₁::VectorSpace, V₂::VectorSpace)
+V₁ ≾ V₂

Return whether there exist monomorphisms from V₁ to V₂, i.e. 'injective' morphisms with left inverses.

source
TensorKit.isepimorphicFunction
isepimorphic(V₁::VectorSpace, V₂::VectorSpace)
+V₁ ≿ V₂

Return whether there exist epimorphisms from V₁ to V₂, i.e. 'surjective' morphisms with right inverses.

source
TensorKit.isisomorphicFunction
isisomorphic(V₁::VectorSpace, V₂::VectorSpace)
+V₁ ≅ V₂

Return if V₁ and V₂ are isomorphic, meaning that there exists isomorphisms from V₁ to V₂, i.e. morphisms with left and right inverses.

source
TensorKit.insertunitFunction
insertunit(P::ProductSpace, i::Int = length(P)+1; dual = false, conj = false)

For P::ProductSpace{S,N}, this adds an extra tensor product factor at position 1 <= i <= N+1 (last position by default) which is just the S-equivalent of the underlying field of scalars, i.e. oneunit(S). With the keyword arguments, one can choose to insert the conjugated or dual space instead, which are all isomorphic to the field of scalars.

source
diff --git a/dev/lib/tensors/index.html b/dev/lib/tensors/index.html index 972e6de9..1031e04a 100644 --- a/dev/lib/tensors/index.html +++ b/dev/lib/tensors/index.html @@ -1,25 +1,25 @@ -Tensors · TensorKit.jl

Tensors

Type hierarchy

TensorKit.AbstractTensorMapType
abstract type AbstractTensorMap{S<:IndexSpace, N₁, N₂} end

Abstract supertype of all tensor maps, i.e. linear maps between tensor products of vector spaces of type S<:IndexSpace. An AbstractTensorMap maps from an input space of type ProductSpace{S, N₂} to an output space of type ProductSpace{S, N₁}.

source
TensorKit.AbstractTensorType
AbstractTensor{S<:IndexSpace, N} = AbstractTensorMap{S, N, 0}

Abstract supertype of all tensors, i.e. elements in the tensor product space of type ProductSpace{S, N}, built from elementary spaces of type S<:IndexSpace.

An AbstractTensor{S, N} is actually a special case AbstractTensorMap{S, N, 0}, i.e. a tensor map with only a non-trivial output space.

source
TensorKit.TensorMapType
struct TensorMap{S<:IndexSpace, N₁, N₂, ...} <: AbstractTensorMap{S, N₁, N₂}

Specific subtype of AbstractTensorMap for representing tensor maps (morphisms in a tensor category) whose data is stored in blocks of some subtype of DenseMatrix.

source

Specific TensorMap constructors

TensorKit.idFunction
id([A::Type{<:DenseMatrix} = Matrix{Float64},] space::VectorSpace) -> TensorMap

Construct the identity endomorphism on space space, i.e. return a t::TensorMap with domain(t) == codomain(t) == V, where storagetype(t) = A can be specified.

source
TensorKit.isomorphismFunction
isomorphism([A::Type{<:DenseMatrix} = Matrix{Float64},]
+Tensors · TensorKit.jl

Tensors

Type hierarchy

TensorKit.AbstractTensorMapType
abstract type AbstractTensorMap{S<:IndexSpace, N₁, N₂} end

Abstract supertype of all tensor maps, i.e. linear maps between tensor products of vector spaces of type S<:IndexSpace. An AbstractTensorMap maps from an input space of type ProductSpace{S, N₂} to an output space of type ProductSpace{S, N₁}.

source
TensorKit.AbstractTensorType
AbstractTensor{S<:IndexSpace, N} = AbstractTensorMap{S, N, 0}

Abstract supertype of all tensors, i.e. elements in the tensor product space of type ProductSpace{S, N}, built from elementary spaces of type S<:IndexSpace.

An AbstractTensor{S, N} is actually a special case AbstractTensorMap{S, N, 0}, i.e. a tensor map with only a non-trivial output space.

source
TensorKit.TensorMapType
struct TensorMap{S<:IndexSpace, N₁, N₂, ...} <: AbstractTensorMap{S, N₁, N₂}

Specific subtype of AbstractTensorMap for representing tensor maps (morphisms in a tensor category) whose data is stored in blocks of some subtype of DenseMatrix.

source

Specific TensorMap constructors

TensorKit.idFunction
id([A::Type{<:DenseMatrix} = Matrix{Float64},] space::VectorSpace) -> TensorMap

Construct the identity endomorphism on space space, i.e. return a t::TensorMap with domain(t) == codomain(t) == V, where storagetype(t) = A can be specified.

source
TensorKit.isomorphismFunction
isomorphism([A::Type{<:DenseMatrix} = Matrix{Float64},]
                 cod::VectorSpace, dom::VectorSpace)
--> TensorMap

Return a t::TensorMap that implements a specific isomorphism between the codomain cod and the domain dom, and for which storagetype(t) can optionally be chosen to be of type A. If the two spaces do not allow for such an isomorphism, and are thus not isomorphic, and error will be thrown. When they are isomorphic, there is no canonical choice for a specific isomorphism, but the current choice is such that isomorphism(cod, dom) == inv(isomorphism(dom, cod)).

See also unitary when InnerProductStyle(cod) === EuclideanProduct().

source
TensorKit.unitaryFunction
unitary([A::Type{<:DenseMatrix} = Matrix{Float64},] cod::VectorSpace, dom::VectorSpace)
--> TensorMap

Return a t::TensorMap that implements a specific unitary isomorphism between the codomain cod and the domain dom, for which spacetype(dom) (== spacetype(cod)) must have an inner product. Furthermore, storagetype(t) can optionally be chosen to be of type A. If the two spaces do not allow for such an isomorphism, and are thus not isomorphic, and error will be thrown. When they are isomorphic, there is no canonical choice for a specific isomorphism, but the current choice is such that unitary(cod, dom) == inv(unitary(dom, cod)) = adjoint(unitary(dom, cod)).

source
TensorKit.isometryFunction
isometry([A::Type{<:DenseMatrix} = Matrix{Float64},] cod::VectorSpace, dom::VectorSpace)
--> TensorMap

Return a t::TensorMap that implements a specific isometry that embeds the domain dom into the codomain cod, and which requires that spacetype(dom) (== spacetype(cod)) has an Euclidean inner product. An isometry t is such that its adjoint t' is the left inverse of t, i.e. t'*t = id(dom), while t*t' is some idempotent endomorphism of cod, i.e. it squares to itself. When dom and cod do not allow for such an isometric inclusion, an error will be thrown.

source

TensorMap operations

Missing docstring.

Missing docstring for permute(t::TensorMap{S}, p1::IndexTuple, p2::IndexTuple) where {S}. Check Documenter's build log for details.

Base.permute!Function
permute!(tdst::AbstractTensorMap{S,N₁,N₂}, tsrc::AbstractTensorMap{S},
+-> TensorMap

Return a t::TensorMap that implements a specific isomorphism between the codomain cod and the domain dom, and for which storagetype(t) can optionally be chosen to be of type A. If the two spaces do not allow for such an isomorphism, and are thus not isomorphic, and error will be thrown. When they are isomorphic, there is no canonical choice for a specific isomorphism, but the current choice is such that isomorphism(cod, dom) == inv(isomorphism(dom, cod)).

See also unitary when InnerProductStyle(cod) === EuclideanProduct().

source
TensorKit.unitaryFunction
unitary([A::Type{<:DenseMatrix} = Matrix{Float64},] cod::VectorSpace, dom::VectorSpace)
+-> TensorMap

Return a t::TensorMap that implements a specific unitary isomorphism between the codomain cod and the domain dom, for which spacetype(dom) (== spacetype(cod)) must have an inner product. Furthermore, storagetype(t) can optionally be chosen to be of type A. If the two spaces do not allow for such an isomorphism, and are thus not isomorphic, and error will be thrown. When they are isomorphic, there is no canonical choice for a specific isomorphism, but the current choice is such that unitary(cod, dom) == inv(unitary(dom, cod)) = adjoint(unitary(dom, cod)).

source
TensorKit.isometryFunction
isometry([A::Type{<:DenseMatrix} = Matrix{Float64},] cod::VectorSpace, dom::VectorSpace)
+-> TensorMap

Return a t::TensorMap that implements a specific isometry that embeds the domain dom into the codomain cod, and which requires that spacetype(dom) (== spacetype(cod)) has an Euclidean inner product. An isometry t is such that its adjoint t' is the left inverse of t, i.e. t'*t = id(dom), while t*t' is some idempotent endomorphism of cod, i.e. it squares to itself. When dom and cod do not allow for such an isometric inclusion, an error will be thrown.

source

TensorMap operations

Missing docstring.

Missing docstring for permute(t::TensorMap{S}, p1::IndexTuple, p2::IndexTuple) where {S}. Check Documenter's build log for details.

Base.permute!Function
permute!(tdst::AbstractTensorMap{S,N₁,N₂}, tsrc::AbstractTensorMap{S},
          (p₁, p₂)::Tuple{IndexTuple{N₁},IndexTuple{N₂}}) where {S,N₁,N₂}
-    -> tdst

Write into tdst the result of permuting the indices of tsrc. The codomain and domain of tdst correspond to the indices in p₁ and p₂ of tsrc respectively.

See permute for creating a new tensor and add_permute! for a more general version.

source
TensorKit.braidFunction
braid(f::FusionTree{<:Sector, N}, levels::NTuple{N, Int}, p::NTuple{N, Int})
--> <:AbstractDict{typeof(t), <:Number}

Perform a braiding of the uncoupled indices of the fusion tree f and return the result as a <:AbstractDict of output trees and corresponding coefficients. The braiding is determined by specifying that the new sector at position k corresponds to the sector that was originally at the position i = p[k], and assigning to every index i of the original fusion tree a distinct level or depth levels[i]. This permutation is then decomposed into elementary swaps between neighbouring indices, where the swaps are applied as braids such that if i and j cross, $τ_{i,j}$ is applied if levels[i] < levels[j] and $τ_{j,i}^{-1}$ if levels[i] > levels[j]. This does not allow to encode the most general braid, but a general braid can be obtained by combining such operations.

source
braid(f₁::FusionTree{I}, f₂::FusionTree{I},
+    -> tdst

Write into tdst the result of permuting the indices of tsrc. The codomain and domain of tdst correspond to the indices in p₁ and p₂ of tsrc respectively.

See permute for creating a new tensor and add_permute! for a more general version.

source
TensorKit.braidFunction
braid(f::FusionTree{<:Sector, N}, levels::NTuple{N, Int}, p::NTuple{N, Int})
+-> <:AbstractDict{typeof(t), <:Number}

Perform a braiding of the uncoupled indices of the fusion tree f and return the result as a <:AbstractDict of output trees and corresponding coefficients. The braiding is determined by specifying that the new sector at position k corresponds to the sector that was originally at the position i = p[k], and assigning to every index i of the original fusion tree a distinct level or depth levels[i]. This permutation is then decomposed into elementary swaps between neighbouring indices, where the swaps are applied as braids such that if i and j cross, $τ_{i,j}$ is applied if levels[i] < levels[j] and $τ_{j,i}^{-1}$ if levels[i] > levels[j]. This does not allow to encode the most general braid, but a general braid can be obtained by combining such operations.

source
braid(f₁::FusionTree{I}, f₂::FusionTree{I},
         levels1::IndexTuple, levels2::IndexTuple,
         p1::IndexTuple{N₁}, p2::IndexTuple{N₂}) where {I<:Sector, N₁, N₂}
--> <:AbstractDict{Tuple{FusionTree{I, N₁}, FusionTree{I, N₂}}, <:Number}

Input is a fusion-splitting tree pair that describes the fusion of a set of incoming uncoupled sectors to a set of outgoing uncoupled sectors, represented using the splitting tree f₁ and fusion tree f₂, such that the incoming sectors f₂.uncoupled are fused to f₁.coupled == f₂.coupled and then to the outgoing sectors f₁.uncoupled. Compute new trees and corresponding coefficients obtained from repartitioning and braiding the tree such that sectors p1 become outgoing and sectors p2 become incoming. The uncoupled indices in splitting tree f₁ and fusion tree f₂ have levels (or depths) levels1 and levels2 respectively, which determines how indices braid. In particular, if i and j cross, $τ_{i,j}$ is applied if levels[i] < levels[j] and $τ_{j,i}^{-1}$ if levels[i] > levels[j]. This does not allow to encode the most general braid, but a general braid can be obtained by combining such operations.

source
braid(tsrc::AbstractTensorMap{S}, (p₁, p₂)::Tuple{IndexTuple{N₁},IndexTuple{N₂}}, levels::Tuple;
+-> <:AbstractDict{Tuple{FusionTree{I, N₁}, FusionTree{I, N₂}}, <:Number}

Input is a fusion-splitting tree pair that describes the fusion of a set of incoming uncoupled sectors to a set of outgoing uncoupled sectors, represented using the splitting tree f₁ and fusion tree f₂, such that the incoming sectors f₂.uncoupled are fused to f₁.coupled == f₂.coupled and then to the outgoing sectors f₁.uncoupled. Compute new trees and corresponding coefficients obtained from repartitioning and braiding the tree such that sectors p1 become outgoing and sectors p2 become incoming. The uncoupled indices in splitting tree f₁ and fusion tree f₂ have levels (or depths) levels1 and levels2 respectively, which determines how indices braid. In particular, if i and j cross, $τ_{i,j}$ is applied if levels[i] < levels[j] and $τ_{j,i}^{-1}$ if levels[i] > levels[j]. This does not allow to encode the most general braid, but a general braid can be obtained by combining such operations.

source
braid(tsrc::AbstractTensorMap{S}, (p₁, p₂)::Tuple{IndexTuple{N₁},IndexTuple{N₂}}, levels::Tuple;
       copy::Bool = false) where {S,N₁,N₂}
-    -> tdst::TensorMap{S,N₁,N₂}

Return tensor tdst obtained by braiding the indices of tsrc. The codomain and domain of tdst correspond to the indices in p₁ and p₂ of tsrc respectively. Here, levels is a tuple of length numind(tsrc) that assigns a level or height to the indices of tsrc, which determines whether they will braid over or under any other index with which they have to change places.

If copy=false, tdst might share data with tsrc whenever possible. Otherwise, a copy is always made.

To braid into an existing destination, see braid! and add_braid!

source
TensorKit.braid!Function
braid!(tdst::AbstractTensorMap{S,N₁,N₂}, tsrc::AbstractTensorMap{S},
+    -> tdst::TensorMap{S,N₁,N₂}

Return tensor tdst obtained by braiding the indices of tsrc. The codomain and domain of tdst correspond to the indices in p₁ and p₂ of tsrc respectively. Here, levels is a tuple of length numind(tsrc) that assigns a level or height to the indices of tsrc, which determines whether they will braid over or under any other index with which they have to change places.

If copy=false, tdst might share data with tsrc whenever possible. Otherwise, a copy is always made.

To braid into an existing destination, see braid! and add_braid!

source
TensorKit.braid!Function
braid!(tdst::AbstractTensorMap{S,N₁,N₂}, tsrc::AbstractTensorMap{S},
        (p₁, p₂)::Tuple{IndexTuple{N₁},IndexTuple{N₂}}, levels::Tuple) where {S,N₁,N₂}
-    -> tdst

Write into tdst the result of braiding the indices of tsrc. The codomain and domain of tdst correspond to the indices in p₁ and p₂ of tsrc respectively. Here, levels is a tuple of length numind(tsrc) that assigns a level or height to the indices of tsrc, which determines whether they will braid over or under any other index with which they have to change places.

See braid for creating a new tensor and add_braid! for a more general version.

source
Missing docstring.

Missing docstring for twist. Check Documenter's build log for details.

TensorKit.twist!Function
twist!(t::AbstractTensorMap, i::Int; inv::Bool=false)
-    -> t

Apply a twist to the ith index of t, storing the result in t. If inv=true, use the inverse twist.

See twist for creating a new tensor.

source
Missing docstring.

Missing docstring for add!. Check Documenter's build log for details.

Missing docstring.

Missing docstring for trace!. Check Documenter's build log for details.

Missing docstring.

Missing docstring for contract!. Check Documenter's build log for details.

TensorMap factorizations

TensorKit.leftorthFunction
leftorth(t::AbstractTensorMap, leftind::Tuple, rightind::Tuple;
-            alg::OrthogonalFactorizationAlgorithm = QRpos()) -> Q, R

Create orthonormal basis Q for indices in leftind, and remainder R such that permute(t, (leftind, rightind)) = Q*R.

If leftind and rightind are not specified, the current partition of left and right indices of t is used. In that case, less memory is allocated if one allows the data in t to be destroyed/overwritten, by using leftorth!(t, alg = QRpos()).

Different algorithms are available, namely QR(), QRpos(), SVD() and Polar(). QR() and QRpos() use a standard QR decomposition, producing an upper triangular matrix R. Polar() produces a Hermitian and positive semidefinite R. QRpos() corrects the standard QR decomposition such that the diagonal elements of R are positive. Only QRpos() and Polar() are unique (no residual freedom) so that they always return the same result for the same input tensor t.

Orthogonality requires InnerProductStyle(t) <: HasInnerProduct, and leftorth(!) is currently only implemented for InnerProductStyle(t) === EuclideanProduct().

source
TensorKit.rightorthFunction
rightorth(t::AbstractTensorMap, leftind::Tuple, rightind::Tuple;
-            alg::OrthogonalFactorizationAlgorithm = LQpos()) -> L, Q

Create orthonormal basis Q for indices in rightind, and remainder L such that permute(t, (leftind, rightind)) = L*Q.

If leftind and rightind are not specified, the current partition of left and right indices of t is used. In that case, less memory is allocated if one allows the data in t to be destroyed/overwritten, by using rightorth!(t, alg = LQpos()).

Different algorithms are available, namely LQ(), LQpos(), RQ(), RQpos(), SVD() and Polar(). LQ() and LQpos() produce a lower triangular matrix L and are computed using a QR decomposition of the transpose. RQ() and RQpos() produce an upper triangular remainder L and only works if the total left dimension is smaller than or equal to the total right dimension. LQpos() and RQpos() add an additional correction such that the diagonal elements of L are positive. Polar() produces a Hermitian and positive semidefinite L. Only LQpos(), RQpos() and Polar() are unique (no residual freedom) so that they always return the same result for the same input tensor t.

Orthogonality requires InnerProductStyle(t) <: HasInnerProduct, and rightorth(!) is currently only implemented for InnerProductStyle(t) === EuclideanProduct().

source
TensorKit.leftnullFunction
leftnull(t::AbstractTensor, leftind::Tuple, rightind::Tuple;
-            alg::OrthogonalFactorizationAlgorithm = QRpos()) -> N

Create orthonormal basis for the orthogonal complement of the support of the indices in leftind, such that N' * permute(t, (leftind, rightind)) = 0.

If leftind and rightind are not specified, the current partition of left and right indices of t is used. In that case, less memory is allocated if one allows the data in t to be destroyed/overwritten, by using leftnull!(t, alg = QRpos()).

Different algorithms are available, namely QR() (or equivalently, QRpos()), SVD() and SDD(). The first assumes that the matrix is full rank and requires iszero(atol) and iszero(rtol). With SVD() and SDD(), rightnull will use the corresponding singular value decomposition, and one can specify an absolute or relative tolerance for which singular values are to be considered zero, where max(atol, norm(t)*rtol) is used as upper bound.

Orthogonality requires InnerProductStyle(t) <: HasInnerProduct, and leftnull(!) is currently only implemented for InnerProductStyle(t) === EuclideanProduct().

source
TensorKit.rightnullFunction
rightnull(t::AbstractTensor, leftind::Tuple, rightind::Tuple;
+    -> tdst

Write into tdst the result of braiding the indices of tsrc. The codomain and domain of tdst correspond to the indices in p₁ and p₂ of tsrc respectively. Here, levels is a tuple of length numind(tsrc) that assigns a level or height to the indices of tsrc, which determines whether they will braid over or under any other index with which they have to change places.

See braid for creating a new tensor and add_braid! for a more general version.

source
Missing docstring.

Missing docstring for twist. Check Documenter's build log for details.

TensorKit.twist!Function
twist!(t::AbstractTensorMap, i::Int; inv::Bool=false)
+    -> t

Apply a twist to the ith index of t, storing the result in t. If inv=true, use the inverse twist.

See twist for creating a new tensor.

source
Missing docstring.

Missing docstring for add!. Check Documenter's build log for details.

Missing docstring.

Missing docstring for trace!. Check Documenter's build log for details.

Missing docstring.

Missing docstring for contract!. Check Documenter's build log for details.

TensorMap factorizations

TensorKit.leftorthFunction
leftorth(t::AbstractTensorMap, leftind::Tuple, rightind::Tuple;
+            alg::OrthogonalFactorizationAlgorithm = QRpos()) -> Q, R

Create orthonormal basis Q for indices in leftind, and remainder R such that permute(t, (leftind, rightind)) = Q*R.

If leftind and rightind are not specified, the current partition of left and right indices of t is used. In that case, less memory is allocated if one allows the data in t to be destroyed/overwritten, by using leftorth!(t, alg = QRpos()).

Different algorithms are available, namely QR(), QRpos(), SVD() and Polar(). QR() and QRpos() use a standard QR decomposition, producing an upper triangular matrix R. Polar() produces a Hermitian and positive semidefinite R. QRpos() corrects the standard QR decomposition such that the diagonal elements of R are positive. Only QRpos() and Polar() are unique (no residual freedom) so that they always return the same result for the same input tensor t.

Orthogonality requires InnerProductStyle(t) <: HasInnerProduct, and leftorth(!) is currently only implemented for InnerProductStyle(t) === EuclideanProduct().

source
TensorKit.rightorthFunction
rightorth(t::AbstractTensorMap, leftind::Tuple, rightind::Tuple;
+            alg::OrthogonalFactorizationAlgorithm = LQpos()) -> L, Q

Create orthonormal basis Q for indices in rightind, and remainder L such that permute(t, (leftind, rightind)) = L*Q.

If leftind and rightind are not specified, the current partition of left and right indices of t is used. In that case, less memory is allocated if one allows the data in t to be destroyed/overwritten, by using rightorth!(t, alg = LQpos()).

Different algorithms are available, namely LQ(), LQpos(), RQ(), RQpos(), SVD() and Polar(). LQ() and LQpos() produce a lower triangular matrix L and are computed using a QR decomposition of the transpose. RQ() and RQpos() produce an upper triangular remainder L and only works if the total left dimension is smaller than or equal to the total right dimension. LQpos() and RQpos() add an additional correction such that the diagonal elements of L are positive. Polar() produces a Hermitian and positive semidefinite L. Only LQpos(), RQpos() and Polar() are unique (no residual freedom) so that they always return the same result for the same input tensor t.

Orthogonality requires InnerProductStyle(t) <: HasInnerProduct, and rightorth(!) is currently only implemented for InnerProductStyle(t) === EuclideanProduct().

source
TensorKit.leftnullFunction
leftnull(t::AbstractTensor, leftind::Tuple, rightind::Tuple;
+            alg::OrthogonalFactorizationAlgorithm = QRpos()) -> N

Create orthonormal basis for the orthogonal complement of the support of the indices in leftind, such that N' * permute(t, (leftind, rightind)) = 0.

If leftind and rightind are not specified, the current partition of left and right indices of t is used. In that case, less memory is allocated if one allows the data in t to be destroyed/overwritten, by using leftnull!(t, alg = QRpos()).

Different algorithms are available, namely QR() (or equivalently, QRpos()), SVD() and SDD(). The first assumes that the matrix is full rank and requires iszero(atol) and iszero(rtol). With SVD() and SDD(), rightnull will use the corresponding singular value decomposition, and one can specify an absolute or relative tolerance for which singular values are to be considered zero, where max(atol, norm(t)*rtol) is used as upper bound.

Orthogonality requires InnerProductStyle(t) <: HasInnerProduct, and leftnull(!) is currently only implemented for InnerProductStyle(t) === EuclideanProduct().

source
TensorKit.rightnullFunction
rightnull(t::AbstractTensor, leftind::Tuple, rightind::Tuple;
             alg::OrthogonalFactorizationAlgorithm = LQ(),
             atol::Real = 0.0,
-            rtol::Real = eps(real(float(one(scalartype(t)))))*iszero(atol)) -> N

Create orthonormal basis for the orthogonal complement of the support of the indices in rightind, such that permute(t, (leftind, rightind))*N' = 0.

If leftind and rightind are not specified, the current partition of left and right indices of t is used. In that case, less memory is allocated if one allows the data in t to be destroyed/overwritten, by using rightnull!(t, alg = LQpos()).

Different algorithms are available, namely LQ() (or equivalently, LQpos), SVD() and SDD(). The first assumes that the matrix is full rank and requires iszero(atol) and iszero(rtol). With SVD() and SDD(), rightnull will use the corresponding singular value decomposition, and one can specify an absolute or relative tolerance for which singular values are to be considered zero, where max(atol, norm(t)*rtol) is used as upper bound.

Orthogonality requires InnerProductStyle(t) <: HasInnerProduct, and rightnull(!) is currently only implemented for InnerProductStyle(t) === EuclideanProduct().

source
TensorKit.tsvdFunction
tsvd(t::AbstractTensorMap, leftind::Tuple, rightind::Tuple;
+            rtol::Real = eps(real(float(one(scalartype(t)))))*iszero(atol)) -> N

Create orthonormal basis for the orthogonal complement of the support of the indices in rightind, such that permute(t, (leftind, rightind))*N' = 0.

If leftind and rightind are not specified, the current partition of left and right indices of t is used. In that case, less memory is allocated if one allows the data in t to be destroyed/overwritten, by using rightnull!(t, alg = LQpos()).

Different algorithms are available, namely LQ() (or equivalently, LQpos), SVD() and SDD(). The first assumes that the matrix is full rank and requires iszero(atol) and iszero(rtol). With SVD() and SDD(), rightnull will use the corresponding singular value decomposition, and one can specify an absolute or relative tolerance for which singular values are to be considered zero, where max(atol, norm(t)*rtol) is used as upper bound.

Orthogonality requires InnerProductStyle(t) <: HasInnerProduct, and rightnull(!) is currently only implemented for InnerProductStyle(t) === EuclideanProduct().

source
TensorKit.tsvdFunction
tsvd(t::AbstractTensorMap, leftind::Tuple, rightind::Tuple;
     trunc::TruncationScheme = notrunc(), p::Real = 2, alg::Union{SVD, SDD} = SDD())
-    -> U, S, V, ϵ

Compute the (possibly truncated) singular value decomposition such that norm(permute(t, (leftind, rightind)) - U * S * V) ≈ ϵ, where ϵ thus represents the truncation error.

If leftind and rightind are not specified, the current partition of left and right indices of t is used. In that case, less memory is allocated if one allows the data in t to be destroyed/overwritten, by using tsvd!(t, trunc = notrunc(), p = 2).

A truncation parameter trunc can be specified for the new internal dimension, in which case a truncated singular value decomposition will be computed. Choices are:

  • notrunc(): no truncation (default);
  • truncerr(η::Real): truncates such that the p-norm of the truncated singular values is smaller than η times the p-norm of all singular values;
  • truncdim(χ::Int): truncates such that the equivalent total dimension of the internal vector space is no larger than χ;
  • truncspace(V): truncates such that the dimension of the internal vector space is smaller than that of V in any sector.
  • truncbelow(χ::Real): truncates such that every singular value is larger then χ ;

The method tsvd also returns the truncation error ϵ, computed as the p norm of the singular values that were truncated.

THe keyword alg can be equal to SVD() or SDD(), corresponding to the underlying LAPACK algorithm that computes the decomposition (_gesvd or _gesdd).

Orthogonality requires InnerProductStyle(t) <: HasInnerProduct, and tsvd(!) is currently only implemented for InnerProductStyle(t) === EuclideanProduct().

source
TensorKit.eighFunction
eigh(t::AbstractTensorMap, leftind::Tuple, rightind::Tuple) -> D, V

Compute eigenvalue factorization of tensor t as linear map from rightind to leftind. The function eigh assumes that the linear map is hermitian and D and V tensors with the same scalartype as t. See eig and eigen for non-hermitian tensors. Hermiticity requires that the tensor acts on inner product spaces, and the current implementation requires InnerProductStyle(t) === EuclideanProduct().

If leftind and rightind are not specified, the current partition of left and right indices of t is used. In that case, less memory is allocated if one allows the data in t to be destroyed/overwritten, by using eigh!(t). Note that the permuted tensor on which eigh! is called should have equal domain and codomain, as otherwise the eigenvalue decomposition is meaningless and cannot satisfy

permute(t, (leftind, rightind)) * V = V * D

See also eigen and eig.

source
TensorKit.eigFunction
eig(t::AbstractTensor, leftind::Tuple, rightind::Tuple; kwargs...) -> D, V

Compute eigenvalue factorization of tensor t as linear map from rightind to leftind. The function eig assumes that the linear map is not hermitian and returns type stable complex valued D and V tensors for both real and complex valued t. See eigh for hermitian linear maps

If leftind and rightind are not specified, the current partition of left and right indices of t is used. In that case, less memory is allocated if one allows the data in t to be destroyed/overwritten, by using eig!(t). Note that the permuted tensor on which eig! is called should have equal domain and codomain, as otherwise the eigenvalue decomposition is meaningless and cannot satisfy

permute(t, (leftind, rightind)) * V = V * D

Accepts the same keyword arguments scale, permute and sortby as eigen of dense matrices. See the corresponding documentation for more information.

See also eigen and eigh.

source
LinearAlgebra.eigenFunction
eigen(t::AbstractTensor, leftind::Tuple, rightind::Tuple; kwargs...) -> D, V

Compute eigenvalue factorization of tensor t as linear map from rightind to leftind.

If leftind and rightind are not specified, the current partition of left and right indices of t is used. In that case, less memory is allocated if one allows the data in t to be destroyed/overwritten, by using eigen!(t). Note that the permuted tensor on which eigen! is called should have equal domain and codomain, as otherwise the eigenvalue decomposition is meaningless and cannot satisfy

permute(t, (leftind, rightind)) * V = V * D

Accepts the same keyword arguments scale, permute and sortby as eigen of dense matrices. See the corresponding documentation for more information.

See also eig and eigh

source
+ -> U, S, V, ϵ

Compute the (possibly truncated) singular value decomposition such that norm(permute(t, (leftind, rightind)) - U * S * V) ≈ ϵ, where ϵ thus represents the truncation error.

If leftind and rightind are not specified, the current partition of left and right indices of t is used. In that case, less memory is allocated if one allows the data in t to be destroyed/overwritten, by using tsvd!(t, trunc = notrunc(), p = 2).

A truncation parameter trunc can be specified for the new internal dimension, in which case a truncated singular value decomposition will be computed. Choices are:

  • notrunc(): no truncation (default);
  • truncerr(η::Real): truncates such that the p-norm of the truncated singular values is smaller than η times the p-norm of all singular values;
  • truncdim(χ::Int): truncates such that the equivalent total dimension of the internal vector space is no larger than χ;
  • truncspace(V): truncates such that the dimension of the internal vector space is smaller than that of V in any sector.
  • truncbelow(χ::Real): truncates such that every singular value is larger then χ ;

The method tsvd also returns the truncation error ϵ, computed as the p norm of the singular values that were truncated.

THe keyword alg can be equal to SVD() or SDD(), corresponding to the underlying LAPACK algorithm that computes the decomposition (_gesvd or _gesdd).

Orthogonality requires InnerProductStyle(t) <: HasInnerProduct, and tsvd(!) is currently only implemented for InnerProductStyle(t) === EuclideanProduct().

source
TensorKit.eighFunction
eigh(t::AbstractTensorMap, leftind::Tuple, rightind::Tuple) -> D, V

Compute eigenvalue factorization of tensor t as linear map from rightind to leftind. The function eigh assumes that the linear map is hermitian and D and V tensors with the same scalartype as t. See eig and eigen for non-hermitian tensors. Hermiticity requires that the tensor acts on inner product spaces, and the current implementation requires InnerProductStyle(t) === EuclideanProduct().

If leftind and rightind are not specified, the current partition of left and right indices of t is used. In that case, less memory is allocated if one allows the data in t to be destroyed/overwritten, by using eigh!(t). Note that the permuted tensor on which eigh! is called should have equal domain and codomain, as otherwise the eigenvalue decomposition is meaningless and cannot satisfy

permute(t, (leftind, rightind)) * V = V * D

See also eigen and eig.

source
TensorKit.eigFunction
eig(t::AbstractTensor, leftind::Tuple, rightind::Tuple; kwargs...) -> D, V

Compute eigenvalue factorization of tensor t as linear map from rightind to leftind. The function eig assumes that the linear map is not hermitian and returns type stable complex valued D and V tensors for both real and complex valued t. See eigh for hermitian linear maps

If leftind and rightind are not specified, the current partition of left and right indices of t is used. In that case, less memory is allocated if one allows the data in t to be destroyed/overwritten, by using eig!(t). Note that the permuted tensor on which eig! is called should have equal domain and codomain, as otherwise the eigenvalue decomposition is meaningless and cannot satisfy

permute(t, (leftind, rightind)) * V = V * D

Accepts the same keyword arguments scale, permute and sortby as eigen of dense matrices. See the corresponding documentation for more information.

See also eigen and eigh.

source
LinearAlgebra.eigenFunction
eigen(t::AbstractTensor, leftind::Tuple, rightind::Tuple; kwargs...) -> D, V

Compute eigenvalue factorization of tensor t as linear map from rightind to leftind.

If leftind and rightind are not specified, the current partition of left and right indices of t is used. In that case, less memory is allocated if one allows the data in t to be destroyed/overwritten, by using eigen!(t). Note that the permuted tensor on which eigen! is called should have equal domain and codomain, as otherwise the eigenvalue decomposition is meaningless and cannot satisfy

permute(t, (leftind, rightind)) * V = V * D

Accepts the same keyword arguments scale, permute and sortby as eigen of dense matrices. See the corresponding documentation for more information.

See also eig and eigh

source
diff --git a/dev/man/categories/index.html b/dev/man/categories/index.html index 8adfde56..4c81071c 100644 --- a/dev/man/categories/index.html +++ b/dev/man/categories/index.html @@ -6,4 +6,4 @@ Springer Science & Business Media.
  • kitaev
      Kitaev, A. (2006). Anyons in an exactly solved model and beyond.
             Annals of Physics, 321(1), 2-111.
  • beer
        From categories to anyons: a travelogue
             Kerstin Beer, Dmytro Bondarenko, Alexander Hahn, Maria Kalabakov, Nicole Knust, Laura Niermann, Tobias J. Osborne, Christin Schridde, Stefan Seckmeyer, Deniz E. Stiegemann, and Ramona Wolf
    -        [https://arxiv.org/abs/1811.06670](https://arxiv.org/abs/1811.06670)
  • + [https://arxiv.org/abs/1811.06670](https://arxiv.org/abs/1811.06670) diff --git a/dev/man/intro/index.html b/dev/man/intro/index.html index 54b6d078..fc1192f7 100644 --- a/dev/man/intro/index.html +++ b/dev/man/intro/index.html @@ -1,2 +1,2 @@ -Introduction · TensorKit.jl

    Introduction

    Before providing a typical "user guide" and discussing the implementation of TensorKit.jl on the next pages, let us discuss some of the rationale behind this package.

    What is a tensor?

    At the very start we should ponder about the most suitable and sufficiently general definition of a tensor. A good starting point is the following:

    • A tensor $t$ is an element from the tensor product of $N$ vector spaces $V_1 , V_2, …, V_N$, where $N$ is referred to as the rank or order of the tensor, i.e.

      $t ∈ V_1 ⊗ V_2 ⊗ … ⊗ V_N.$

    If you think of a tensor as an object with indices, a rank $N$ tensor has $N$ indices where every index is associated with the corresponding vector space in that it labels a particular basis in that space. We will return to index notation at the very end of this manual.

    As the tensor product of vector spaces is itself a vector space, this implies that a tensor behaves as a vector, i.e. tensors from the same tensor product space can be added and multiplied by scalars. The tensor product is only defined for vector spaces over the same field of scalars, e.g. there is no meaning in $ℝ^5 ⊗ ℂ^3$. When all the vector spaces in the tensor product have an inner product, this also implies an inner product for the tensor product space. It is hence clear that the different vector spaces in the tensor product should have some form of homogeneity in their structure, yet they do not need to be all equal and can e.g. have different dimensions. It goes without saying that defining the vector spaces and their properties will be an important part of the definition of a tensor. As a consequence, this also constitutes a significant part of the implementation, and is discussed in the section on Vector spaces.

    Aside from the interpretation of a tensor as a vector, we also want to interpret it as a matrix (or more correctly, a linear map) in order to decompose tensors using linear algebra factorisations (e.g. eigenvalue or singular value decomposition). Henceforth, we use the term "tensor map" as follows:

    • A tensor map $t$ is a linear map from a source or domain $W_1 ⊗ W_2 ⊗ … ⊗ W_{N_2}$ to a target or codomain $V_1 ⊗ V_2 ⊗ … ⊗ V_{N_1}$, i.e.

      $t:W_1 ⊗ W_2 ⊗ … ⊗ W_{N_2} → V_1 ⊗ V_2 ⊗ … ⊗ V_{N_1}.$

    A tensor of rank $N$ is then just a special case of a tensor map with $N_1 = N$ and $N_2 = 0$. A contraction between two tensors is just a composition of linear maps (i.e. matrix multiplication), where the contracted indices correspond to the domain of the first tensor and the codomain of the second tensor.

    In order to allow for arbitrary tensor contractions or decompositions, we need to be able to reorganise which vector spaces appear in the domain and the codomain of the tensor map, and in which order. This amounts to defining canonical isomorphisms between the different ways to order and partition the tensor indices (i.e. the vector spaces). For example, a linear map $W → V$ is often denoted as a rank 2 tensor in $V ⊗ W^*$, where $W^*$ corresponds to the dual space of $W$. This simple example introduces two new concepts.

    1. Typical vector spaces can appear in the domain and codomain in different related forms, e.g. as normal space or dual space. In fact, the most generic case is that every vector space $V$ has associated with it a dual space $V^*$, a conjugate space $\overline{V}$ and a conjugate dual space $\overline{V}^*$. The four different vector spaces $V$, $V^*$, $\overline{V}$ and $\overline{V}^*$ correspond to the representation spaces of respectively the fundamental, dual or contragredient, complex conjugate and dual complex conjugate representation of the general linear group $\mathsf{GL}(V)$. In index notation these spaces are denoted with respectively contravariant (upper), covariant (lower), dotted contravariant and dotted covariant indices.

      For real vector spaces, the conjugate (dual) space is identical to the normal (dual) space and we only have upper and lower indices, i.e. this is the setting of e.g. general relativity. For (complex) vector spaces with a sesquilinear inner product $\overline{V} ⊗ V → ℂ$, the inner product allows to define an isomorphism from the conjugate space to the dual space (known as Riesz representation theorem in the more general context of Hilbert spaces).

      In particular, in spaces with a Euclidean inner product (the setting of e.g. quantum mechanics), the conjugate and dual space are naturally isomorphic (because the dual and conjugate representation of the unitary group are the same). Again we only need upper and lower indices (or kets and bras).

      Finally, in $ℝ^d$ with a Euclidean inner product, these four different spaces are all equivalent and we only need one type of index. The space is completely characterized by its dimension $d$. This is the setting of much of classical mechanics and we refer to such tensors as cartesian tensors and the corresponding space as cartesian space. These are the tensors that can equally well be represented as multidimensional arrays (i.e. using some AbstractArray{<:Real,N} in Julia) without loss of structure.

      The implementation of all of this is discussed in Vector spaces.

    2. In the generic case, the identification between maps $W → V$ and tensors in $V ⊗ W^*$ is not an equivalence but an isomorphism, which needs to be defined. Similarly, there is an isomorphism between between $V ⊗ W$ and $W ⊗ V$ that can be non-trivial (e.g. in the case of fermions / super vector spaces). The correct formalism here is provided by theory of monoidal categories, which is introduced on the next page. Nonetheless, we try to hide these canonical isomorphisms from the user wherever possible, and one does not need to know category theory to be able to use this package.

    This brings us to our final (yet formal) definition

    • A tensor (map) is a homomorphism between two objects from the category $\mathbf{Vect}$ (or some subcategory thereof). In practice, this will be $\mathbf{FinVect}$, the category of finite dimensional vector spaces. More generally even, our concept of a tensor makes sense, in principle, for any linear (a.k.a. $\mathbf{Vect}$-enriched) monoidal category. We refer to the next page on "Monoidal categories and their properties".

    Symmetries and block sparsity

    Physical problems often have some symmetry, i.e. the setup is invariant under the action of a group $\mathsf{G}$ which acts on the vector spaces $V$ in the problem according to a certain representation. Having quantum mechanics in mind, TensorKit.jl is so far restricted to unitary representations. A general representation space $V$ can be specified as the number of times every irreducible representation (irrep) $a$ of $\mathsf{G}$ appears, i.e.

    $V = \bigoplus_{a} ℂ^{n_a} ⊗ R_a$

    with $R_a$ the space associated with irrep $a$ of $\mathsf{G}$, which itself has dimension $d_a$ (often called the quantum dimension), and $n_a$ the number of times this irrep appears in $V$. If the unitary irrep $a$ for $g ∈ \mathsf{G}$ is given by $u_a(g)$, then there exists a specific basis for $V$ such that the group action of $\mathsf{G}$ on $V$ is given by the unitary representation

    $u(g) = \bigoplus_{a} 𝟙_{n_a} ⊗ u_a(g)$

    with $𝟙_{n_a}$ the $n_a × n_a$ identity matrix. The total dimension of $V$ is given by $∑_a n_a d_a$.

    The reason for implementing symmetries is to exploit the computation and memory gains resulting from restricting to tensor maps $t:W_1 ⊗ W_2 ⊗ … ⊗ W_{N_2} → V_1 ⊗ V_2 ⊗ … ⊗ V_{N_1}$ that are equivariant under the symmetry, i.e. that act as intertwiners between the symmetry action on the domain and the codomain. Indeed, such tensors should be block diagonal because of Schur's lemma, but only after we couple the individual irreps in the spaces $W_i$ to a joint irrep, which is then again split into the individual irreps of the spaces $V_i$. The basis change from the tensor product of irreps in the (co)domain to the joint irrep is implemented by a sequence of Clebsch–Gordan coefficients, also known as a fusion (or splitting) tree. We implement the necessary machinery to manipulate these fusion trees under index permutations and repartitions for arbitrary groups $\mathsf{G}$. In particular, this fits with the formalism of monoidal categories, and more specifically fusion categories, and only requires the topological data of the group, i.e. the fusion rules of the irreps, their quantum dimensions and the F-symbol (6j-symbol or more precisely Racah's W-symbol in the case of $\mathsf{SU}_2$). In particular, we don't actually need the Clebsch–Gordan coefficients themselves (but they can be useful for checking purposes).

    Hence, a second major part of TensorKit.jl is the interface and implementation for specifying symmetries, and further details are provided in Sectors, representation spaces and fusion trees.

    +Introduction · TensorKit.jl

    Introduction

    Before providing a typical "user guide" and discussing the implementation of TensorKit.jl on the next pages, let us discuss some of the rationale behind this package.

    What is a tensor?

    At the very start we should ponder about the most suitable and sufficiently general definition of a tensor. A good starting point is the following:

    • A tensor $t$ is an element from the tensor product of $N$ vector spaces $V_1 , V_2, …, V_N$, where $N$ is referred to as the rank or order of the tensor, i.e.

      $t ∈ V_1 ⊗ V_2 ⊗ … ⊗ V_N.$

    If you think of a tensor as an object with indices, a rank $N$ tensor has $N$ indices where every index is associated with the corresponding vector space in that it labels a particular basis in that space. We will return to index notation at the very end of this manual.

    As the tensor product of vector spaces is itself a vector space, this implies that a tensor behaves as a vector, i.e. tensors from the same tensor product space can be added and multiplied by scalars. The tensor product is only defined for vector spaces over the same field of scalars, e.g. there is no meaning in $ℝ^5 ⊗ ℂ^3$. When all the vector spaces in the tensor product have an inner product, this also implies an inner product for the tensor product space. It is hence clear that the different vector spaces in the tensor product should have some form of homogeneity in their structure, yet they do not need to be all equal and can e.g. have different dimensions. It goes without saying that defining the vector spaces and their properties will be an important part of the definition of a tensor. As a consequence, this also constitutes a significant part of the implementation, and is discussed in the section on Vector spaces.

    Aside from the interpretation of a tensor as a vector, we also want to interpret it as a matrix (or more correctly, a linear map) in order to decompose tensors using linear algebra factorisations (e.g. eigenvalue or singular value decomposition). Henceforth, we use the term "tensor map" as follows:

    • A tensor map $t$ is a linear map from a source or domain $W_1 ⊗ W_2 ⊗ … ⊗ W_{N_2}$ to a target or codomain $V_1 ⊗ V_2 ⊗ … ⊗ V_{N_1}$, i.e.

      $t:W_1 ⊗ W_2 ⊗ … ⊗ W_{N_2} → V_1 ⊗ V_2 ⊗ … ⊗ V_{N_1}.$

    A tensor of rank $N$ is then just a special case of a tensor map with $N_1 = N$ and $N_2 = 0$. A contraction between two tensors is just a composition of linear maps (i.e. matrix multiplication), where the contracted indices correspond to the domain of the first tensor and the codomain of the second tensor.

    In order to allow for arbitrary tensor contractions or decompositions, we need to be able to reorganise which vector spaces appear in the domain and the codomain of the tensor map, and in which order. This amounts to defining canonical isomorphisms between the different ways to order and partition the tensor indices (i.e. the vector spaces). For example, a linear map $W → V$ is often denoted as a rank 2 tensor in $V ⊗ W^*$, where $W^*$ corresponds to the dual space of $W$. This simple example introduces two new concepts.

    1. Typical vector spaces can appear in the domain and codomain in different related forms, e.g. as normal space or dual space. In fact, the most generic case is that every vector space $V$ has associated with it a dual space $V^*$, a conjugate space $\overline{V}$ and a conjugate dual space $\overline{V}^*$. The four different vector spaces $V$, $V^*$, $\overline{V}$ and $\overline{V}^*$ correspond to the representation spaces of respectively the fundamental, dual or contragredient, complex conjugate and dual complex conjugate representation of the general linear group $\mathsf{GL}(V)$. In index notation these spaces are denoted with respectively contravariant (upper), covariant (lower), dotted contravariant and dotted covariant indices.

      For real vector spaces, the conjugate (dual) space is identical to the normal (dual) space and we only have upper and lower indices, i.e. this is the setting of e.g. general relativity. For (complex) vector spaces with a sesquilinear inner product $\overline{V} ⊗ V → ℂ$, the inner product allows to define an isomorphism from the conjugate space to the dual space (known as Riesz representation theorem in the more general context of Hilbert spaces).

      In particular, in spaces with a Euclidean inner product (the setting of e.g. quantum mechanics), the conjugate and dual space are naturally isomorphic (because the dual and conjugate representation of the unitary group are the same). Again we only need upper and lower indices (or kets and bras).

      Finally, in $ℝ^d$ with a Euclidean inner product, these four different spaces are all equivalent and we only need one type of index. The space is completely characterized by its dimension $d$. This is the setting of much of classical mechanics and we refer to such tensors as cartesian tensors and the corresponding space as cartesian space. These are the tensors that can equally well be represented as multidimensional arrays (i.e. using some AbstractArray{<:Real,N} in Julia) without loss of structure.

      The implementation of all of this is discussed in Vector spaces.

    2. In the generic case, the identification between maps $W → V$ and tensors in $V ⊗ W^*$ is not an equivalence but an isomorphism, which needs to be defined. Similarly, there is an isomorphism between between $V ⊗ W$ and $W ⊗ V$ that can be non-trivial (e.g. in the case of fermions / super vector spaces). The correct formalism here is provided by theory of monoidal categories, which is introduced on the next page. Nonetheless, we try to hide these canonical isomorphisms from the user wherever possible, and one does not need to know category theory to be able to use this package.

    This brings us to our final (yet formal) definition

    • A tensor (map) is a homomorphism between two objects from the category $\mathbf{Vect}$ (or some subcategory thereof). In practice, this will be $\mathbf{FinVect}$, the category of finite dimensional vector spaces. More generally even, our concept of a tensor makes sense, in principle, for any linear (a.k.a. $\mathbf{Vect}$-enriched) monoidal category. We refer to the next page on "Monoidal categories and their properties".

    Symmetries and block sparsity

    Physical problems often have some symmetry, i.e. the setup is invariant under the action of a group $\mathsf{G}$ which acts on the vector spaces $V$ in the problem according to a certain representation. Having quantum mechanics in mind, TensorKit.jl is so far restricted to unitary representations. A general representation space $V$ can be specified as the number of times every irreducible representation (irrep) $a$ of $\mathsf{G}$ appears, i.e.

    $V = \bigoplus_{a} ℂ^{n_a} ⊗ R_a$

    with $R_a$ the space associated with irrep $a$ of $\mathsf{G}$, which itself has dimension $d_a$ (often called the quantum dimension), and $n_a$ the number of times this irrep appears in $V$. If the unitary irrep $a$ for $g ∈ \mathsf{G}$ is given by $u_a(g)$, then there exists a specific basis for $V$ such that the group action of $\mathsf{G}$ on $V$ is given by the unitary representation

    $u(g) = \bigoplus_{a} 𝟙_{n_a} ⊗ u_a(g)$

    with $𝟙_{n_a}$ the $n_a × n_a$ identity matrix. The total dimension of $V$ is given by $∑_a n_a d_a$.

    The reason for implementing symmetries is to exploit the computation and memory gains resulting from restricting to tensor maps $t:W_1 ⊗ W_2 ⊗ … ⊗ W_{N_2} → V_1 ⊗ V_2 ⊗ … ⊗ V_{N_1}$ that are equivariant under the symmetry, i.e. that act as intertwiners between the symmetry action on the domain and the codomain. Indeed, such tensors should be block diagonal because of Schur's lemma, but only after we couple the individual irreps in the spaces $W_i$ to a joint irrep, which is then again split into the individual irreps of the spaces $V_i$. The basis change from the tensor product of irreps in the (co)domain to the joint irrep is implemented by a sequence of Clebsch–Gordan coefficients, also known as a fusion (or splitting) tree. We implement the necessary machinery to manipulate these fusion trees under index permutations and repartitions for arbitrary groups $\mathsf{G}$. In particular, this fits with the formalism of monoidal categories, and more specifically fusion categories, and only requires the topological data of the group, i.e. the fusion rules of the irreps, their quantum dimensions and the F-symbol (6j-symbol or more precisely Racah's W-symbol in the case of $\mathsf{SU}_2$). In particular, we don't actually need the Clebsch–Gordan coefficients themselves (but they can be useful for checking purposes).

    Hence, a second major part of TensorKit.jl is the interface and implementation for specifying symmetries, and further details are provided in Sectors, representation spaces and fusion trees.

    diff --git a/dev/man/sectors/index.html b/dev/man/sectors/index.html index 33824999..d3016c03 100644 --- a/dev/man/sectors/index.html +++ b/dev/man/sectors/index.html @@ -244,7 +244,7 @@ FusionTree{Irrep[SU₂]}((1/2, 1/2, 1/2, 1/2, 1/2), 1/2, (true, false, false, true, false), (0, 1/2, 1)) FusionTree{Irrep[SU₂]}((1/2, 1/2, 1/2, 1/2, 1/2), 1/2, (true, false, false, true, false), (1, 1/2, 0)) FusionTree{Irrep[SU₂]}((1/2, 1/2, 1/2, 1/2, 1/2), 1/2, (true, false, false, true, false), (1, 1/2, 1)) - FusionTree{Irrep[SU₂]}((1/2, 1/2, 1/2, 1/2, 1/2), 1/2, (true, false, false, true, false), (1, 3/2, 1))
    julia> iter = fusiontrees(ntuple(n->s, 16))TensorKit.FusionTreeIterator{SU2Irrep, 16}((Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2)), Irrep[SU₂](0), (false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false))
    julia> sum(n->1, iter)1430
    julia> length(iter)1430
    julia> @elapsed sum(n->1, iter)0.038106202
    julia> @elapsed length(iter)9.3202e-5
    julia> s2 = s ⊠ s(Irrep[SU₂](1/2) ⊠ Irrep[SU₂](1/2))
    julia> collect(fusiontrees((s2,s2,s2,s2)))4-element Vector{FusionTree{TensorKit.ProductSector{Tuple{SU2Irrep, SU2Irrep}}, 4, 2, 3, Nothing}}: + FusionTree{Irrep[SU₂]}((1/2, 1/2, 1/2, 1/2, 1/2), 1/2, (true, false, false, true, false), (1, 3/2, 1))
    julia> iter = fusiontrees(ntuple(n->s, 16))TensorKit.FusionTreeIterator{SU2Irrep, 16}((Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2), Irrep[SU₂](1/2)), Irrep[SU₂](0), (false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false))
    julia> sum(n->1, iter)1430
    julia> length(iter)1430
    julia> @elapsed sum(n->1, iter)0.039145518
    julia> @elapsed length(iter)9.3701e-5
    julia> s2 = s ⊠ s(Irrep[SU₂](1/2) ⊠ Irrep[SU₂](1/2))
    julia> collect(fusiontrees((s2,s2,s2,s2)))4-element Vector{FusionTree{TensorKit.ProductSector{Tuple{SU2Irrep, SU2Irrep}}, 4, 2, 3, Nothing}}: FusionTree{Irrep[SU₂ × SU₂]}(((1/2, 1/2), (1/2, 1/2), (1/2, 1/2), (1/2, 1/2)), (0, 0), (false, false, false, false), ((0, 0), (1/2, 1/2))) FusionTree{Irrep[SU₂ × SU₂]}(((1/2, 1/2), (1/2, 1/2), (1/2, 1/2), (1/2, 1/2)), (0, 0), (false, false, false, false), ((1, 0), (1/2, 1/2))) FusionTree{Irrep[SU₂ × SU₂]}(((1/2, 1/2), (1/2, 1/2), (1/2, 1/2), (1/2, 1/2)), (0, 0), (false, false, false, false), ((0, 1), (1/2, 1/2))) @@ -320,4 +320,4 @@ 0.618034 0.786151 0.786151 -0.618034
    julia> Fτ'*Fτ2×2 Matrix{Float64}: 1.0 0.0 - 0.0 1.0
    julia> polar(x) = rationalize.((abs(x), angle(x)/(2pi)))polar (generic function with 1 method)
    julia> Rsymbol(τ,τ,𝟙) |> polar(1//1, 2//5)
    julia> Rsymbol(τ,τ,τ) |> polar(1//1, -3//10)
    julia> twist(τ) |> polar(1//1, -2//5) + 0.0 1.0
    julia> polar(x) = rationalize.((abs(x), angle(x)/(2pi)))polar (generic function with 1 method)
    julia> Rsymbol(τ,τ,𝟙) |> polar(1//1, 2//5)
    julia> Rsymbol(τ,τ,τ) |> polar(1//1, -3//10)
    julia> twist(τ) |> polar(1//1, -2//5) diff --git a/dev/man/spaces/index.html b/dev/man/spaces/index.html index bc9df0b7..64c8d83e 100644 --- a/dev/man/spaces/index.html +++ b/dev/man/spaces/index.html @@ -31,4 +31,4 @@ codomain::P1 domain::P2 end

    and can create it as either domain → codomain or codomain ← domain (where the arrows are obtained as \to+TAB or \leftarrow+TAB, and as \rightarrow+TAB respectively). The reason for first listing the codomain and than the domain will become clear in the section on tensor maps.

    Note that HomSpace is not a subtype of VectorSpace, i.e. we restrict the latter to denote certain categories and their objects, and keep HomSpace distinct. However, HomSpace has a number of properties defined, which we illustrate via examples

    julia> W = ℂ^2 ⊗ ℂ^3 → ℂ^3 ⊗ dual(ℂ^4)(ℂ^3 ⊗ (ℂ^4)') ← (ℂ^2 ⊗ ℂ^3)
    julia> field(W)
    julia> dual(W)((ℂ^3)' ⊗ (ℂ^2)') ← (ℂ^4 ⊗ (ℂ^3)')
    julia> adjoint(W)(ℂ^2 ⊗ ℂ^3) ← (ℂ^3 ⊗ (ℂ^4)')
    julia> spacetype(W)ComplexSpace
    julia> spacetype(typeof(W))ComplexSpace
    julia> W[1]ℂ^3
    julia> W[2](ℂ^4)'
    julia> W[3](ℂ^2)'
    julia> W[4](ℂ^3)'
    julia> dim(W)72

    Note that indexing W yields first the spaces in the codomain, followed by the dual of the spaces in the domain. This particular convention is useful in combination with the instances of type TensorMap, which represent morphisms living in such a HomSpace. Also note that dim(W) here seems to be the product of the dimensions of the individual spaces, but that this is no longer true once symmetries are involved. At any time will dim(::HomSpace) represent the number of linearly independent morphisms in this space.

    Partial order among vector spaces

    Vector spaces of the same spacetype can be given a partial order, based on whether there exist injective morphisms (a.k.a monomorphisms) or surjective morphisms (a.k.a. epimorphisms) between them. In particular, we define ismonomorphic(V1, V2), with Unicode synonym V1 ≾ V2 (obtained as \precsim+TAB), to express whether there exist monomorphisms in V1→V2. Similarly, we define isepimorphic(V1, V2), with Unicode synonym V1 ≿ V2 (obtained as \succsim+TAB), to express whether there exist epimorphisms in V1→V2. Finally, we define isisomorphic(V1, V2), with Unicode alternative V1 ≅ V2 (obtained as \cong+TAB), to express whether there exist isomorphism in V1→V2. In particular V1 ≅ V2 if and only if V1 ≾ V2 && V1 ≿ V2.

    For completeness, we also export the strict comparison operators and (\prec+TAB and \succ+TAB), with definitions

    ≺(V1::VectorSpace, V2::VectorSpace) = V1 ≾ V2 && !(V1 ≿ V2)
    -≻(V1::VectorSpace, V2::VectorSpace) = V1 ≿ V2 && !(V1 ≾ V2)

    However, as we expect these to be less commonly used, no ASCII alternative is provided.

    In the context of InnerProductStyle(V) <: EuclideanProduct, V1 ≾ V2 implies that there exists isometries $W:V1 → V2$ such that $W^† ∘ W = \mathrm{id}_{V1}$, while V1 ≅ V2 implies that there exist unitaries $U:V1→V2$ such that $U^† ∘ U = \mathrm{id}_{V1}$ and $U ∘ U^† = \mathrm{id}_{V2}$.

    Note that spaces that are isomorphic are not necessarily equal. One can be a dual space, and the other a normal space, or one can be an instance of ProductSpace, while the other is an ElementarySpace. There will exist (infinitely) many isomorphisms between the corresponding spaces, but in general none of those will be canonical.

    There are also a number of convenience functions to create isomorphic spaces. The function fuse(V1, V2, ...) or fuse(V1 ⊗ V2 ⊗ ...) returns an elementary space that is isomorphic to V1 ⊗ V2 ⊗ .... The function flip(V::ElementarySpace) returns a space that is isomorphic to V but has isdual(flip(V)) == isdual(V'), i.e. if V is a normal space than flip(V) is a dual space. flip(V) is different from dual(V) in the case of GradedSpace. It is useful to flip a tensor index from a ket to a bra (or vice versa), by contracting that index with a unitary map from V1 to flip(V1). We refer to Index operations for further information. Some examples:

    julia> ℝ^3 ≾ ℝ^5true
    julia> ℂ^3 ≾ (ℂ^5)'true
    julia> (ℂ^5) ≅ (ℂ^5)'true
    julia> fuse(ℝ^5, ℝ^3)ℝ^15
    julia> fuse(ℂ^3, (ℂ^5)' ⊗ ℂ^2)ℂ^30
    julia> fuse(ℂ^3, (ℂ^5)') ⊗ ℂ^2 ≅ fuse(ℂ^3, (ℂ^5)', ℂ^2) ≅ ℂ^3 ⊗ (ℂ^5)' ⊗ ℂ^2true
    julia> flip(ℂ^4)(ℂ^4)'
    julia> flip(ℂ^4) ≅ ℂ^4true
    julia> flip(ℂ^4) == ℂ^4false

    We also define the direct sum V1 and V2 as V1 ⊕ V2, where is obtained by typing \oplus+TAB. This is possible only if isdual(V1) == isdual(V2). With a little pun on Julia Base, oneunit applied to an elementary space (in the value or type domain) returns the one-dimensional space, which is isomorphic to the scalar field of the space itself. Some examples illustrate this better

    julia> ℝ^5 ⊕ ℝ^3ℝ^8
    julia> ℂ^5 ⊕ ℂ^3ℂ^8
    julia> ℂ^5 ⊕ (ℂ^3)'ERROR: SpaceMismatch("Direct sum of a vector space and its dual does not exist")
    julia> oneunit(ℝ^3)ℝ^1
    julia> ℂ^5 ⊕ oneunit(ComplexSpace)ℂ^6
    julia> oneunit((ℂ^3)')ℂ^1
    julia> (ℂ^5) ⊕ oneunit((ℂ^5))ℂ^6
    julia> (ℂ^5)' ⊕ oneunit((ℂ^5)')ERROR: SpaceMismatch("Direct sum of a vector space and its dual does not exist")

    Finally, while spaces have a partial order, there is no unique infimum or supremum of a two or more spaces. However, if V1 and V2 are two ElementarySpace instances with isdual(V1) == isdual(V2), then we can define a unique infimum V::ElementarySpace with the same value of isdual that satisfies V ≾ V1 and V ≾ V2, as well as a unique supremum W::ElementarySpace with the same value of isdual that satisfies W ≿ V1 and W ≿ V2. For CartesianSpace and ComplexSpace, this simply amounts to the space with minimal or maximal dimension, i.e.

    julia> infimum(ℝ^5, ℝ^3)ℝ^3
    julia> supremum(ℂ^5, ℂ^3)ℂ^5
    julia> supremum(ℂ^5, (ℂ^3)')ERROR: SpaceMismatch("Supremum of space and dual space does not exist")

    The names infimum and supremum are especially suited in the case of GradedSpace, as the infimum of two spaces might be different from either of those two spaces, and similar for the supremum.

    +≻(V1::VectorSpace, V2::VectorSpace) = V1 ≿ V2 && !(V1 ≾ V2)

    However, as we expect these to be less commonly used, no ASCII alternative is provided.

    In the context of InnerProductStyle(V) <: EuclideanProduct, V1 ≾ V2 implies that there exists isometries $W:V1 → V2$ such that $W^† ∘ W = \mathrm{id}_{V1}$, while V1 ≅ V2 implies that there exist unitaries $U:V1→V2$ such that $U^† ∘ U = \mathrm{id}_{V1}$ and $U ∘ U^† = \mathrm{id}_{V2}$.

    Note that spaces that are isomorphic are not necessarily equal. One can be a dual space, and the other a normal space, or one can be an instance of ProductSpace, while the other is an ElementarySpace. There will exist (infinitely) many isomorphisms between the corresponding spaces, but in general none of those will be canonical.

    There are also a number of convenience functions to create isomorphic spaces. The function fuse(V1, V2, ...) or fuse(V1 ⊗ V2 ⊗ ...) returns an elementary space that is isomorphic to V1 ⊗ V2 ⊗ .... The function flip(V::ElementarySpace) returns a space that is isomorphic to V but has isdual(flip(V)) == isdual(V'), i.e. if V is a normal space than flip(V) is a dual space. flip(V) is different from dual(V) in the case of GradedSpace. It is useful to flip a tensor index from a ket to a bra (or vice versa), by contracting that index with a unitary map from V1 to flip(V1). We refer to Index operations for further information. Some examples:

    julia> ℝ^3 ≾ ℝ^5true
    julia> ℂ^3 ≾ (ℂ^5)'true
    julia> (ℂ^5) ≅ (ℂ^5)'true
    julia> fuse(ℝ^5, ℝ^3)ℝ^15
    julia> fuse(ℂ^3, (ℂ^5)' ⊗ ℂ^2)ℂ^30
    julia> fuse(ℂ^3, (ℂ^5)') ⊗ ℂ^2 ≅ fuse(ℂ^3, (ℂ^5)', ℂ^2) ≅ ℂ^3 ⊗ (ℂ^5)' ⊗ ℂ^2true
    julia> flip(ℂ^4)(ℂ^4)'
    julia> flip(ℂ^4) ≅ ℂ^4true
    julia> flip(ℂ^4) == ℂ^4false

    We also define the direct sum V1 and V2 as V1 ⊕ V2, where is obtained by typing \oplus+TAB. This is possible only if isdual(V1) == isdual(V2). With a little pun on Julia Base, oneunit applied to an elementary space (in the value or type domain) returns the one-dimensional space, which is isomorphic to the scalar field of the space itself. Some examples illustrate this better

    julia> ℝ^5 ⊕ ℝ^3ℝ^8
    julia> ℂ^5 ⊕ ℂ^3ℂ^8
    julia> ℂ^5 ⊕ (ℂ^3)'ERROR: SpaceMismatch("Direct sum of a vector space and its dual does not exist")
    julia> oneunit(ℝ^3)ℝ^1
    julia> ℂ^5 ⊕ oneunit(ComplexSpace)ℂ^6
    julia> oneunit((ℂ^3)')ℂ^1
    julia> (ℂ^5) ⊕ oneunit((ℂ^5))ℂ^6
    julia> (ℂ^5)' ⊕ oneunit((ℂ^5)')ERROR: SpaceMismatch("Direct sum of a vector space and its dual does not exist")

    Finally, while spaces have a partial order, there is no unique infimum or supremum of a two or more spaces. However, if V1 and V2 are two ElementarySpace instances with isdual(V1) == isdual(V2), then we can define a unique infimum V::ElementarySpace with the same value of isdual that satisfies V ≾ V1 and V ≾ V2, as well as a unique supremum W::ElementarySpace with the same value of isdual that satisfies W ≿ V1 and W ≿ V2. For CartesianSpace and ComplexSpace, this simply amounts to the space with minimal or maximal dimension, i.e.

    julia> infimum(ℝ^5, ℝ^3)ℝ^3
    julia> supremum(ℂ^5, ℂ^3)ℂ^5
    julia> supremum(ℂ^5, (ℂ^3)')ERROR: SpaceMismatch("Supremum of space and dual space does not exist")

    The names infimum and supremum are especially suited in the case of GradedSpace, as the infimum of two spaces might be different from either of those two spaces, and similar for the supremum.

    diff --git a/dev/man/tensors/index.html b/dev/man/tensors/index.html index 800ff59e..5adf546e 100644 --- a/dev/man/tensors/index.html +++ b/dev/man/tensors/index.html @@ -4,45 +4,45 @@ TensorMap(undef, codomain, domain) TensorMap(undef, eltype::Type{<:Number}, codomain, domain)

    Here, in the first form, f can be any function or object that is called with an argument of type Dims{2} = Tuple{Int,Int} and is such that f((m,n)) creates a DenseMatrix instance with size(f(m,n)) == (m,n). In the second form, f is called as f(eltype,(m,n)). Possibilities for f are randn and rand from Julia Base. TensorKit.jl provides randnormal and randuniform as an synonym for randn and rand, as well as the new function randisometry, alternatively called randhaar, that creates a random isometric m × n matrix w satisfying w'*w ≈ I distributed according to the Haar measure (this requires m>= n). The third and fourth calling syntax use the UndefInitializer from Julia Base and generates a TensorMap with unitialized data, which could thus contain NaNs.

    In all of these constructors, the last two arguments can be replaced by domain→codomain or codomain←domain, where the arrows are obtained as \rightarrow+TAB and \leftarrow+TAB and create a HomSpace as explained in the section on Spaces of morphisms. Some examples are perhaps in order

    julia> t1 = TensorMap(randnormal, ℂ^2 ⊗ ℂ^3, ℂ^2)TensorMap((ℂ^2 ⊗ ℂ^3) ← ℂ^2):
     [:, :, 1] =
    - 1.7191397731406066  -0.4693943589815249   1.4858094600247482
    - 1.2405680118764764  -0.5107663898838336  -1.518264368594122
    + -0.7472809746347745   0.6335647272083453  -1.107434356410526
    +  0.5932188131741438  -0.4983746079709879  -0.6365568621528584
     
     [:, :, 2] =
    -  0.456822121280905   -0.2310298284043067   -0.545959009501283
    - -1.0718911512964926  -0.15240867455516627   0.15304223829819164
    julia> t2 = TensorMap(randisometry, Float32, ℂ^2 ⊗ ℂ^3 ← ℂ^2)TensorMap((ℂ^2 ⊗ ℂ^3) ← ℂ^2): + -1.437066456744795 -0.85819009296136 -0.11512621208204923 + 0.7158213544343481 0.04718407401355761 0.8162132536609322
    julia> t2 = TensorMap(randisometry, Float32, ℂ^2 ⊗ ℂ^3 ← ℂ^2)TensorMap((ℂ^2 ⊗ ℂ^3) ← ℂ^2): [:, :, 1] = - -0.2121085f0 0.34501296f0 0.7413397f0 - 0.39508098f0 -0.29063663f0 0.21408598f0 + 0.41395319f0 0.06401364f0 -0.18103658f0 + 0.70660937f0 -0.5346894f0 -0.081124745f0 [:, :, 2] = - -0.4675541f0 -0.18374991f0 -0.31698054f0 - 0.37376285f0 0.29893264f0 0.6466015f0
    julia> t3 = TensorMap(undef, ℂ^2 → ℂ^2 ⊗ ℂ^3)TensorMap((ℂ^2 ⊗ ℂ^3) ← ℂ^2): + 0.18927628f0 -0.05099631f0 0.61169755f0 + -0.061518073f0 -0.032171447f0 -0.7632694f0
    julia> t3 = TensorMap(undef, ℂ^2 → ℂ^2 ⊗ ℂ^3)TensorMap((ℂ^2 ⊗ ℂ^3) ← ℂ^2): [:, :, 1] = - 6.90822043087865e-310 6.90823278476325e-310 6.908239026093e-310 - 0.0 0.0 0.0 + 6.9047795953426e-310 6.90478448915925e-310 6.9047844891782e-310 + 6.90478448916874e-310 6.90477950706926e-310 6.90477976153995e-310 [:, :, 2] = - 6.90823903371345e-310 6.90824007056356e-310 6.90822043089446e-310 - 0.0 0.0 0.0
    julia> domain(t1) == domain(t2) == domain(t3)true
    julia> codomain(t1) == codomain(t2) == codomain(t3)true
    julia> disp(x) = show(IOContext(Core.stdout, :compact=>false), "text/plain", trunc.(x; digits = 3));
    julia> t1[] |> disp2×3×2 StridedViews.StridedView{Float64, 3, Array{Float64, 3}, typeof(identity)}: + 6.9047797615423e-310 6.90478448919087e-310 6.9047844891972e-310 + 6.9047797615447e-310 6.90478448919403e-310 6.90477976153757e-310
    julia> domain(t1) == domain(t2) == domain(t3)true
    julia> codomain(t1) == codomain(t2) == codomain(t3)true
    julia> disp(x) = show(IOContext(Core.stdout, :compact=>false), "text/plain", trunc.(x; digits = 3));
    julia> t1[] |> disp2×3×2 StridedViews.StridedView{Float64, 3, Array{Float64, 3}, typeof(identity)}: [:, :, 1] = - 1.719 -0.469 1.485 - 1.24 -0.51 -1.518 + -0.747 0.633 -1.107 + 0.593 -0.498 -0.636 [:, :, 2] = - 0.456 -0.231 -0.545 - -1.071 -0.152 0.153
    julia> block(t1, Trivial()) |> disp6×2 Array{Float64, 2}: - 1.719 0.456 - 1.24 -1.071 - -0.469 -0.231 - -0.51 -0.152 - 1.485 -0.545 - -1.518 0.153
    julia> reshape(t1[], dim(codomain(t1)), dim(domain(t1))) |> disp6×2 Array{Float64, 2}: - 1.719 0.456 - 1.24 -1.071 - -0.469 -0.231 - -0.51 -0.152 - 1.485 -0.545 - -1.518 0.153

    Finally, all constructors can also be replaced by Tensor(..., codomain), in which case the domain is assumed to be the empty ProductSpace{S,0}(), which can easily be obtained as one(codomain). Indeed, the empty product space is the unit object of the monoidal category, equivalent to the field of scalars 𝕜, and thus the multiplicative identity (especially since * also acts as tensor product on vector spaces).

    The matrices created by f are the matrices $B_c$ discussed above, i.e. those returned by block(t, c). Only numerical matrices of type DenseMatrix are accepted, which in practice just means Julia's intrinsic Matrix{T} for some T<:Number. In the future, we will add support for CuMatrix from CuArrays.jl to harness GPU computing power, and maybe SharedArray from the Julia's SharedArrays standard library.

    Support for static or sparse data is currently unavailable, and if it would be implemented, it would lead to new subtypes of AbstractTensorMap which are distinct from TensorMap. Future implementations of e.g. SparseTensorMap or StaticTensorMap could be useful. Furthermore, there could be specific implementations for tensors whose blocks are Diagonal.

    Tensor maps from existing data

    To create a TensorMap with existing data, one can use the aforementioned form but with the function f replaced with the actual data, i.e. TensorMap(data, codomain, domain) or any of its equivalents.

    Here, data can be of two types. It can be a dictionary (any Associative subtype) which has blocksectors c of type sectortype(codomain) as keys, and the corresponding matrix blocks as value, i.e. data[c] is some DenseMatrix of size (blockdim(codomain, c), blockdim(domain, c)). This is the form of how the data is stored within the TensorMap objects.

    For those space types for which a TensorMap can be converted to a plain multidimensional array, the data can also be a general DenseArray, either of rank N₁+N₂ and with matching size (dims(codomain)..., dims(domain)...), or just as a DenseMatrix with size (dim(codomain), dim(domain)). This is true in particular if the sector type is Trivial, e.g. for CartesianSpace or ComplexSpace. Then the data array is just reshaped into matrix form and referred to as such in the resulting TensorMap instance. When spacetype is GradedSpace, the TensorMap constructor will try to reconstruct the tensor data such that the resulting tensor t satisfies data == convert(Array, t). This might not be possible, if the data does not respect the symmetry structure. Let's sketch this with a simple example

    julia> data = zeros(2,2,2,2)2×2×2×2 Array{Float64, 4}:
    + -1.437  -0.858  -0.115
    +  0.715   0.047   0.816
    julia> block(t1, Trivial()) |> disp6×2 Array{Float64, 2}: + -0.747 -1.437 + 0.593 0.715 + 0.633 -0.858 + -0.498 0.047 + -1.107 -0.115 + -0.636 0.816
    julia> reshape(t1[], dim(codomain(t1)), dim(domain(t1))) |> disp6×2 Array{Float64, 2}: + -0.747 -1.437 + 0.593 0.715 + 0.633 -0.858 + -0.498 0.047 + -1.107 -0.115 + -0.636 0.816

    Finally, all constructors can also be replaced by Tensor(..., codomain), in which case the domain is assumed to be the empty ProductSpace{S,0}(), which can easily be obtained as one(codomain). Indeed, the empty product space is the unit object of the monoidal category, equivalent to the field of scalars 𝕜, and thus the multiplicative identity (especially since * also acts as tensor product on vector spaces).

    The matrices created by f are the matrices $B_c$ discussed above, i.e. those returned by block(t, c). Only numerical matrices of type DenseMatrix are accepted, which in practice just means Julia's intrinsic Matrix{T} for some T<:Number. In the future, we will add support for CuMatrix from CuArrays.jl to harness GPU computing power, and maybe SharedArray from the Julia's SharedArrays standard library.

    Support for static or sparse data is currently unavailable, and if it would be implemented, it would lead to new subtypes of AbstractTensorMap which are distinct from TensorMap. Future implementations of e.g. SparseTensorMap or StaticTensorMap could be useful. Furthermore, there could be specific implementations for tensors whose blocks are Diagonal.

    Tensor maps from existing data

    To create a TensorMap with existing data, one can use the aforementioned form but with the function f replaced with the actual data, i.e. TensorMap(data, codomain, domain) or any of its equivalents.

    Here, data can be of two types. It can be a dictionary (any Associative subtype) which has blocksectors c of type sectortype(codomain) as keys, and the corresponding matrix blocks as value, i.e. data[c] is some DenseMatrix of size (blockdim(codomain, c), blockdim(domain, c)). This is the form of how the data is stored within the TensorMap objects.

    For those space types for which a TensorMap can be converted to a plain multidimensional array, the data can also be a general DenseArray, either of rank N₁+N₂ and with matching size (dims(codomain)..., dims(domain)...), or just as a DenseMatrix with size (dim(codomain), dim(domain)). This is true in particular if the sector type is Trivial, e.g. for CartesianSpace or ComplexSpace. Then the data array is just reshaped into matrix form and referred to as such in the resulting TensorMap instance. When spacetype is GradedSpace, the TensorMap constructor will try to reconstruct the tensor data such that the resulting tensor t satisfies data == convert(Array, t). This might not be possible, if the data does not respect the symmetry structure. Let's sketch this with a simple example

    julia> data = zeros(2,2,2,2)2×2×2×2 Array{Float64, 4}:
     [:, :, 1, 1] =
      0.0  0.0
      0.0  0.0
    @@ -115,192 +115,192 @@
      1.0

    Hence, we recognize that the Heisenberg interaction has eigenvalue $-1$ in the coupled spin zero sector (SUIrrep(0)), and eigenvalue $+1$ in the coupled spin 1 sector (SU2Irrep(1)). Using Irrep[U₁] instead, we observe that both coupled charge U1Irrep(+1) and U1Irrep(-1) have eigenvalue $+1$. The coupled charge U1Irrep(0) sector is two-dimensional, and has an eigenvalue $+1$ and an eigenvalue $-1$.

    To construct the proper data in more complicated cases, one has to know where to find each sector in the range 1:dim(V) of every index i with associated space V, as well as the internal structure of the representation space when the corresponding sector c has dim(c)>1, i.e. in the case of FusionStyle(c) isa MultipleFusion. Currently, the only non- abelian sectors are Irrep[SU₂] and Irrep[CU₁], for which the internal structure is the natural one.

    There are some tools available to facilate finding the proper range of sector c in space V, namely axes(V, c). This also works on a ProductSpace, with a tuple of sectors. An example

    julia> V = SU2Space(0=>3, 1=>2, 2=>1)Rep[SU₂](0=>3, 1=>2, 2=>1)
    julia> P = V ⊗ V ⊗ V(Rep[SU₂](0=>3, 1=>2, 2=>1) ⊗ Rep[SU₂](0=>3, 1=>2, 2=>1) ⊗ Rep[SU₂](0=>3, 1=>2, 2=>1))
    julia> axes(P, (SU2Irrep(1), SU2Irrep(0), SU2Irrep(2)))(4:9, 1:3, 10:14)

    Note that the length of the range is the degeneracy dimension of that sector, times the dimension of the internal representation space, i.e. the quantum dimension of that sector.

    Constructing similar tensors

    A third way to construct a TensorMap instance is to use Base.similar, i.e.

    similar(t [, T::Type{<:Number}, codomain, domain])

    where T is a possibly different eltype for the tensor data, and codomain and domain optionally define a new codomain and domain for the resulting tensor. By default, these values just take the value from the input tensor t. The result will be a new TensorMap instance, with undef data, but whose data is stored in the same subtype of DenseMatrix (e.g. Matrix or CuMatrix or ...) as t. In particular, this uses the methods storagetype(t) and TensorKit.similarstoragetype(t, T).

    Special purpose constructors

    Finally, there are methods zero, one, id, isomorphism, unitary and isometry to create specific new tensors. Tensor maps behave as vectors and can be added (if they have the same domain and codomain); zero(t) is the additive identity, i.e. a TensorMap instance where all entries are zero. For a t::TensorMap with domain(t) == codomain(t), i.e. an endomorphism, one(t) creates the identity tensor, i.e. the identity under composition. As discussed in the section on linear algebra operations, we denote composition of tensor maps with the mutliplication operator *, such that one(t) is the multiplicative identity. Similarly, it can be created as id(V) with V the relevant vector space, e.g. one(t) == id(domain(t)). The identity tensor is currently represented with dense data, and one can use id(A::Type{<:DenseMatrix}, V) to specify the type of DenseMatrix (and its eltype), e.g. A = Matrix{Float64}. Finally, it often occurs that we want to construct a specific isomorphism between two spaces that are isomorphic but not equal, and for which there is no canonical choice. Hereto, one can use the method u = isomorphism([A::Type{<:DenseMatrix}, ] codomain, domain), which will explicitly check that the domain and codomain are isomorphic, and return an error otherwise. Again, an optional first argument can be given to specify the specific type of DenseMatrix that is currently used to store the rather trivial data of this tensor. If InnerProductStyle(u) <: EuclideanProduct, the same result can be obtained with the method u = unitary([A::Type{<:DenseMatrix}, ] codomain, domain). Note that reversing the domain and codomain yields the inverse morphism, which in the case of EuclideanProduct coincides with the adjoint morphism, i.e. isomorphism(A, domain, codomain) == adjoint(u) == inv(u), where inv and adjoint will be further discussed below. Finally, if two spaces V1 and V2 are such that V2 can be embedded in V1, i.e. there exists an inclusion with a left inverse, and furthermore they represent tensor products of some ElementarySpace with EuclideanProduct, the function w = isometry([A::Type{<:DenseMatrix}, ], V1, V2) creates one specific isometric embedding, such that adjoint(w) * w == id(V2) and w * adjoint(w) is some hermitian idempotent (a.k.a. orthogonal projector) acting on V1. An error will be thrown if such a map cannot be constructed for the given domain and codomain.

    Let's conclude this section with some examples with GradedSpace.

    julia> V1 = ℤ₂Space(0=>3,1=>2)Rep[ℤ₂](0=>3, 1=>2)
    julia> V2 = ℤ₂Space(0=>2,1=>1)Rep[ℤ₂](0=>2, 1=>1)
    julia> # First a `TensorMap{ℤ₂Space, 1, 1}` m = TensorMap(randn, V1, V2)TensorMap(Rep[ℤ₂](0=>3, 1=>2) ← Rep[ℤ₂](0=>2, 1=>1)): * Data for sector (Irrep[ℤ₂](0),) ← (Irrep[ℤ₂](0),): - -1.142562502165892 1.7287955318919281 - -0.25486755056997557 0.6755998429890282 - -0.0020021003026748944 -0.08580790459667396 + 0.8834951722723807 -0.33495630101402757 + 0.03832207409691113 -1.1240373205445602 + -0.6835612273495983 0.829170725035153 * Data for sector (Irrep[ℤ₂](1),) ← (Irrep[ℤ₂](1),): - -0.016572965982553464 - -1.2401538093800264
    julia> convert(Array, m) |> disp5×3 Array{Float64, 2}: - -1.142 1.728 0.0 - -0.254 0.675 0.0 - -0.002 -0.085 0.0 - 0.0 0.0 -0.016 - 0.0 0.0 -1.24
    julia> # compare with: + 0.19994200531540873 + -0.028705900129868317
    julia> convert(Array, m) |> disp5×3 Array{Float64, 2}: + 0.883 -0.334 0.0 + 0.038 -1.124 0.0 + -0.683 0.829 0.0 + 0.0 0.0 0.199 + 0.0 0.0 -0.028
    julia> # compare with: block(m, Irrep[ℤ₂](0)) |> disp3×2 Array{Float64, 2}: - -1.142 1.728 - -0.254 0.675 - -0.002 -0.085
    julia> block(m, Irrep[ℤ₂](1)) |> disp2×1 Array{Float64, 2}: - -0.016 - -1.24
    julia> # Now a `TensorMap{ℤ₂Space, 2, 2}` + 0.883 -0.334 + 0.038 -1.124 + -0.683 0.829
    julia> block(m, Irrep[ℤ₂](1)) |> disp2×1 Array{Float64, 2}: + 0.199 + -0.028
    julia> # Now a `TensorMap{ℤ₂Space, 2, 2}` t = TensorMap(randn, V1 ⊗ V1, V2 ⊗ V2')TensorMap((Rep[ℤ₂](0=>3, 1=>2) ⊗ Rep[ℤ₂](0=>3, 1=>2)) ← (Rep[ℤ₂](0=>2, 1=>1) ⊗ Rep[ℤ₂](0=>2, 1=>1)')): * Data for sector (Irrep[ℤ₂](0), Irrep[ℤ₂](0)) ← (Irrep[ℤ₂](0), Irrep[ℤ₂](0)): [:, :, 1, 1] = - 0.2677959372693652 -0.8687542790601263 0.7229652924029744 - 0.2973856399534342 1.0380052102422985 -0.8723253553092314 - -0.16525915084630247 -0.7375749414228974 -0.16720289191100565 + 0.16966408211124556 -0.8823574649445491 0.7295057908053834 + 0.17390249429456406 -0.06774995815575079 -0.00013985099537990945 + 1.094625234503748 -0.04833402751970194 1.689647360563786 [:, :, 2, 1] = - 0.6930492905586744 -2.0416769698883606 -0.43213740455789384 - 0.38825452996928755 -0.37786379889832206 2.544801428554545 - 1.241908654868515 0.7132588694651693 -1.6488682267975983 + -0.5747027623265746 0.7716942259823745 -0.7781451727696169 + 0.896375814954789 -1.056119482455283 0.2951093409929376 + 0.06594049088658152 -1.7417286606395141 -1.195096043909179 [:, :, 1, 2] = - 0.4973254423441147 0.3104260044437105 0.5721882751340138 - -0.24114269458775842 0.2627326459819454 -1.308358807419044 - -1.550885101560961 -0.9087157098980331 1.5378204354699374 + -0.5201867833688448 -0.22366784334720116 -0.15078732821036006 + -0.5445743432702214 1.1577667040804547 0.22541361866668091 + 0.8762211045281265 0.7660751208045433 0.915992189455053 [:, :, 2, 2] = - 0.8865184877148872 0.28539706304368845 1.3730212908809072 - -1.9308774676653828 0.8430994977042172 -0.07717247628346523 - 0.16599858674845816 0.747284829042743 0.9432494178944454 + -1.274801377299435 0.4000598477627681 -0.39849089382296465 + -1.0365367387311883 -1.4055837498932662 0.1656677205908873 + 0.6879787584741845 0.21529631996890428 1.3559401919906124 * Data for sector (Irrep[ℤ₂](1), Irrep[ℤ₂](1)) ← (Irrep[ℤ₂](0), Irrep[ℤ₂](0)): [:, :, 1, 1] = - -0.21407898600785352 -0.7956380730420182 - 0.6505392118375859 0.832007198479789 + -0.6819851793518098 -0.6083020151099267 + 0.6141539979723087 -1.0287296343885743 [:, :, 2, 1] = - -0.9563346358764868 0.2794749620099953 - -0.15745811808884988 0.8881052418552147 + 0.42296146713248645 0.4374150190653562 + 0.5887762944689704 0.6676436243368421 [:, :, 1, 2] = - 0.37672598941462404 -1.1239936979609941 - -0.3321893394950498 -1.001897907312974 + 0.5745558235669471 -1.0784604320731563 + -0.9865945734832214 -2.020160647698571 [:, :, 2, 2] = - 0.07900836126268095 0.40143777447445644 - 1.1424789373513524 -1.15004281130414 + -1.3245620945976586 -0.14573547217250876 + 0.355260495987531 -0.6506710575793228 * Data for sector (Irrep[ℤ₂](0), Irrep[ℤ₂](0)) ← (Irrep[ℤ₂](1), Irrep[ℤ₂](1)): [:, :, 1, 1] = - 0.5163195279417994 0.3693061884776015 0.004016317902156877 - -0.6281432207406468 0.506081572766194 1.4406250115070645 - -0.7385511055083152 0.8027289365642909 0.35851218690562436 + 0.01467039544231305 -1.7814105616468172 0.4801888631511623 + 0.39360038117787766 -0.06211124022120183 -0.9883435584883784 + 1.8529220941440965 1.824919332083236 -0.15022157852072654 * Data for sector (Irrep[ℤ₂](1), Irrep[ℤ₂](1)) ← (Irrep[ℤ₂](1), Irrep[ℤ₂](1)): [:, :, 1, 1] = - 2.2326603420419704 0.5919719154168709 - 1.0420107907368146 1.057971552676971 + 0.12844112029321297 0.015658415678976547 + -0.4599140150054338 -1.4387524865769385 * Data for sector (Irrep[ℤ₂](1), Irrep[ℤ₂](0)) ← (Irrep[ℤ₂](0), Irrep[ℤ₂](1)): [:, :, 1, 1] = - 1.4466104095622931 1.0554255945736668 -0.7175767958398785 - 1.9922731875582356 1.4684442414536885 -0.770428286019265 + -0.7033192146106706 -1.202584753166928 -1.0546963899731032 + 0.6686523547296848 -0.4283015273650146 1.360673126342998 [:, :, 2, 1] = - 0.3234907922378424 -0.7769881482501443 0.19225157400300893 - 1.0119192580794014 0.16043343053099637 0.7228427540962465 + 0.2772331206441533 0.5253501191099913 0.10158210504913548 + 1.0591542923288704 1.4329647930530116 0.9759242734413163 * Data for sector (Irrep[ℤ₂](0), Irrep[ℤ₂](1)) ← (Irrep[ℤ₂](0), Irrep[ℤ₂](1)): [:, :, 1, 1] = - 0.5364982671302465 -2.1528740869199505 - 0.7804351332271342 0.8565210625395749 - -3.5592153042812824 -0.5914062398528355 + 0.7798952940803601 -0.07302529079491826 + 1.9376612657098726 -0.5500991703789327 + -0.5235424661538854 0.9163693178882915 [:, :, 2, 1] = - 0.8633195387320285 1.1064573799248878 - 0.3076862144331437 0.9985876024186126 - -0.19527980863170866 0.42514689291867086 + -0.9123966204191521 1.7472381894136517 + 0.9449467575544456 1.033560650421486 + -1.5483269389758771 -0.24831339552242349 * Data for sector (Irrep[ℤ₂](1), Irrep[ℤ₂](0)) ← (Irrep[ℤ₂](1), Irrep[ℤ₂](0)): [:, :, 1, 1] = - -0.6692774743500223 0.35982264022759625 -0.47858902110354884 - 0.09190294417398195 -0.7444241929787857 -0.7936723374547249 + 0.7201489417191368 -1.1099615339953404 -0.07350888334325303 + -0.009207520782859174 -1.4026269091165238 -0.20974630748768702 [:, :, 1, 2] = - 0.11433016476732384 -1.3326041029525713 0.14413430052449974 - -0.10470529134175087 -1.232954435111801 -1.0463502151525292 + -0.49926351895573956 0.7572192069201215 0.49828547589834205 + -0.057142739747096515 -0.42622504567358577 -1.1231661575104264 * Data for sector (Irrep[ℤ₂](0), Irrep[ℤ₂](1)) ← (Irrep[ℤ₂](1), Irrep[ℤ₂](0)): [:, :, 1, 1] = - 0.49841558773110356 -1.1062087815917674 - 0.39962396675166195 -0.6841716977478202 - -0.4129674181436516 -0.3681410470825964 + 1.3504449909899512 -0.9241721664002996 + 2.070438648840842 0.3197200798870968 + -1.4739996534136373 2.311210730865384 [:, :, 1, 2] = - -0.4702450308321265 0.4803455387641434 - -0.7800343364818398 1.134429749494955 - -0.44565453625837026 -0.4406671401950615
    julia> (array = convert(Array, t)) |> disp5×5×3×3 Array{Float64, 4}: + -0.3249272561689486 0.7944744120012226 + 1.322118927012488 -0.7110540713030596 + 2.5494293208382643 1.3651875520448318
    julia> (array = convert(Array, t)) |> disp5×5×3×3 Array{Float64, 4}: [:, :, 1, 1] = - 0.267 -0.868 0.722 0.0 0.0 - 0.297 1.038 -0.872 0.0 0.0 - -0.165 -0.737 -0.167 0.0 0.0 - 0.0 0.0 0.0 -0.214 -0.795 - 0.0 0.0 0.0 0.65 0.832 + 0.169 -0.882 0.729 0.0 0.0 + 0.173 -0.067 -0.0 0.0 0.0 + 1.094 -0.048 1.689 0.0 0.0 + 0.0 0.0 0.0 -0.681 -0.608 + 0.0 0.0 0.0 0.614 -1.028 [:, :, 2, 1] = - 0.693 -2.041 -0.432 0.0 0.0 - 0.388 -0.377 2.544 0.0 0.0 - 1.241 0.713 -1.648 0.0 0.0 - 0.0 0.0 0.0 -0.956 0.279 - 0.0 0.0 0.0 -0.157 0.888 + -0.574 0.771 -0.778 0.0 0.0 + 0.896 -1.056 0.295 0.0 0.0 + 0.065 -1.741 -1.195 0.0 0.0 + 0.0 0.0 0.0 0.422 0.437 + 0.0 0.0 0.0 0.588 0.667 [:, :, 3, 1] = - 0.0 0.0 0.0 0.498 -1.106 - 0.0 0.0 0.0 0.399 -0.684 - 0.0 0.0 0.0 -0.412 -0.368 - -0.669 0.359 -0.478 0.0 0.0 - 0.091 -0.744 -0.793 0.0 0.0 + 0.0 0.0 0.0 1.35 -0.924 + 0.0 0.0 0.0 2.07 0.319 + 0.0 0.0 0.0 -1.473 2.311 + 0.72 -1.109 -0.073 0.0 0.0 + -0.009 -1.402 -0.209 0.0 0.0 [:, :, 1, 2] = - 0.497 0.31 0.572 0.0 0.0 - -0.241 0.262 -1.308 0.0 0.0 - -1.55 -0.908 1.537 0.0 0.0 - 0.0 0.0 0.0 0.376 -1.123 - 0.0 0.0 0.0 -0.332 -1.001 + -0.52 -0.223 -0.15 0.0 0.0 + -0.544 1.157 0.225 0.0 0.0 + 0.876 0.766 0.915 0.0 0.0 + 0.0 0.0 0.0 0.574 -1.078 + 0.0 0.0 0.0 -0.986 -2.02 [:, :, 2, 2] = - 0.886 0.285 1.373 0.0 0.0 - -1.93 0.843 -0.077 0.0 0.0 - 0.165 0.747 0.943 0.0 0.0 - 0.0 0.0 0.0 0.079 0.401 - 0.0 0.0 0.0 1.142 -1.15 + -1.274 0.4 -0.398 0.0 0.0 + -1.036 -1.405 0.165 0.0 0.0 + 0.687 0.215 1.355 0.0 0.0 + 0.0 0.0 0.0 -1.324 -0.145 + 0.0 0.0 0.0 0.355 -0.65 [:, :, 3, 2] = - 0.0 0.0 0.0 -0.47 0.48 - 0.0 0.0 0.0 -0.78 1.134 - 0.0 0.0 0.0 -0.445 -0.44 - 0.114 -1.332 0.144 0.0 0.0 - -0.104 -1.232 -1.046 0.0 0.0 + 0.0 0.0 0.0 -0.324 0.794 + 0.0 0.0 0.0 1.322 -0.711 + 0.0 0.0 0.0 2.549 1.365 + -0.499 0.757 0.498 0.0 0.0 + -0.057 -0.426 -1.123 0.0 0.0 [:, :, 1, 3] = - 0.0 0.0 0.0 0.536 -2.152 - 0.0 0.0 0.0 0.78 0.856 - 0.0 0.0 0.0 -3.559 -0.591 - 1.446 1.055 -0.717 0.0 0.0 - 1.992 1.468 -0.77 0.0 0.0 + 0.0 0.0 0.0 0.779 -0.073 + 0.0 0.0 0.0 1.937 -0.55 + 0.0 0.0 0.0 -0.523 0.916 + -0.703 -1.202 -1.054 0.0 0.0 + 0.668 -0.428 1.36 0.0 0.0 [:, :, 2, 3] = - 0.0 0.0 0.0 0.863 1.106 - 0.0 0.0 0.0 0.307 0.998 - 0.0 0.0 0.0 -0.195 0.425 - 0.323 -0.776 0.192 0.0 0.0 - 1.011 0.16 0.722 0.0 0.0 + 0.0 0.0 0.0 -0.912 1.747 + 0.0 0.0 0.0 0.944 1.033 + 0.0 0.0 0.0 -1.548 -0.248 + 0.277 0.525 0.101 0.0 0.0 + 1.059 1.432 0.975 0.0 0.0 [:, :, 3, 3] = - 0.516 0.369 0.004 0.0 0.0 - -0.628 0.506 1.44 0.0 0.0 - -0.738 0.802 0.358 0.0 0.0 - 0.0 0.0 0.0 2.232 0.591 - 0.0 0.0 0.0 1.042 1.057
    julia> d1 = dim(codomain(t))25
    julia> d2 = dim(domain(t))9
    julia> (matrix = reshape(array, d1, d2)) |> disp25×9 Array{Float64, 2}: - 0.267 0.693 0.0 0.497 0.886 0.0 0.0 0.0 0.516 - 0.297 0.388 0.0 -0.241 -1.93 0.0 0.0 0.0 -0.628 - -0.165 1.241 0.0 -1.55 0.165 0.0 0.0 0.0 -0.738 - 0.0 0.0 -0.669 0.0 0.0 0.114 1.446 0.323 0.0 - 0.0 0.0 0.091 0.0 0.0 -0.104 1.992 1.011 0.0 - -0.868 -2.041 0.0 0.31 0.285 0.0 0.0 0.0 0.369 - 1.038 -0.377 0.0 0.262 0.843 0.0 0.0 0.0 0.506 - -0.737 0.713 0.0 -0.908 0.747 0.0 0.0 0.0 0.802 - 0.0 0.0 0.359 0.0 0.0 -1.332 1.055 -0.776 0.0 - 0.0 0.0 -0.744 0.0 0.0 -1.232 1.468 0.16 0.0 - 0.722 -0.432 0.0 0.572 1.373 0.0 0.0 0.0 0.004 - -0.872 2.544 0.0 -1.308 -0.077 0.0 0.0 0.0 1.44 - -0.167 -1.648 0.0 1.537 0.943 0.0 0.0 0.0 0.358 - 0.0 0.0 -0.478 0.0 0.0 0.144 -0.717 0.192 0.0 - 0.0 0.0 -0.793 0.0 0.0 -1.046 -0.77 0.722 0.0 - 0.0 0.0 0.498 0.0 0.0 -0.47 0.536 0.863 0.0 - 0.0 0.0 0.399 0.0 0.0 -0.78 0.78 0.307 0.0 - 0.0 0.0 -0.412 0.0 0.0 -0.445 -3.559 -0.195 0.0 - -0.214 -0.956 0.0 0.376 0.079 0.0 0.0 0.0 2.232 - 0.65 -0.157 0.0 -0.332 1.142 0.0 0.0 0.0 1.042 - 0.0 0.0 -1.106 0.0 0.0 0.48 -2.152 1.106 0.0 - 0.0 0.0 -0.684 0.0 0.0 1.134 0.856 0.998 0.0 - 0.0 0.0 -0.368 0.0 0.0 -0.44 -0.591 0.425 0.0 - -0.795 0.279 0.0 -1.123 0.401 0.0 0.0 0.0 0.591 - 0.832 0.888 0.0 -1.001 -1.15 0.0 0.0 0.0 1.057
    julia> (u = reshape(convert(Array, unitary(codomain(t), fuse(codomain(t)))), d1, d1)) |> disp25×25 Array{Float64, 2}: + 0.014 -1.781 0.48 0.0 0.0 + 0.393 -0.062 -0.988 0.0 0.0 + 1.852 1.824 -0.15 0.0 0.0 + 0.0 0.0 0.0 0.128 0.015 + 0.0 0.0 0.0 -0.459 -1.438
    julia> d1 = dim(codomain(t))25
    julia> d2 = dim(domain(t))9
    julia> (matrix = reshape(array, d1, d2)) |> disp25×9 Array{Float64, 2}: + 0.169 -0.574 0.0 -0.52 -1.274 0.0 0.0 0.0 0.014 + 0.173 0.896 0.0 -0.544 -1.036 0.0 0.0 0.0 0.393 + 1.094 0.065 0.0 0.876 0.687 0.0 0.0 0.0 1.852 + 0.0 0.0 0.72 0.0 0.0 -0.499 -0.703 0.277 0.0 + 0.0 0.0 -0.009 0.0 0.0 -0.057 0.668 1.059 0.0 + -0.882 0.771 0.0 -0.223 0.4 0.0 0.0 0.0 -1.781 + -0.067 -1.056 0.0 1.157 -1.405 0.0 0.0 0.0 -0.062 + -0.048 -1.741 0.0 0.766 0.215 0.0 0.0 0.0 1.824 + 0.0 0.0 -1.109 0.0 0.0 0.757 -1.202 0.525 0.0 + 0.0 0.0 -1.402 0.0 0.0 -0.426 -0.428 1.432 0.0 + 0.729 -0.778 0.0 -0.15 -0.398 0.0 0.0 0.0 0.48 + -0.0 0.295 0.0 0.225 0.165 0.0 0.0 0.0 -0.988 + 1.689 -1.195 0.0 0.915 1.355 0.0 0.0 0.0 -0.15 + 0.0 0.0 -0.073 0.0 0.0 0.498 -1.054 0.101 0.0 + 0.0 0.0 -0.209 0.0 0.0 -1.123 1.36 0.975 0.0 + 0.0 0.0 1.35 0.0 0.0 -0.324 0.779 -0.912 0.0 + 0.0 0.0 2.07 0.0 0.0 1.322 1.937 0.944 0.0 + 0.0 0.0 -1.473 0.0 0.0 2.549 -0.523 -1.548 0.0 + -0.681 0.422 0.0 0.574 -1.324 0.0 0.0 0.0 0.128 + 0.614 0.588 0.0 -0.986 0.355 0.0 0.0 0.0 -0.459 + 0.0 0.0 -0.924 0.0 0.0 0.794 -0.073 1.747 0.0 + 0.0 0.0 0.319 0.0 0.0 -0.711 -0.55 1.033 0.0 + 0.0 0.0 2.311 0.0 0.0 1.365 0.916 -0.248 0.0 + -0.608 0.437 0.0 -1.078 -0.145 0.0 0.0 0.0 0.015 + -1.028 0.667 0.0 -2.02 -0.65 0.0 0.0 0.0 -1.438
    julia> (u = reshape(convert(Array, unitary(codomain(t), fuse(codomain(t)))), d1, d1)) |> disp25×25 Array{Float64, 2}: 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 @@ -335,256 +335,256 @@ 0.0 0.0 0.0 0.0 0.0 0.0 0.0 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 1.0 0.0 0.0 0.0 0.0 1.0 0.0 0.0 0.0 0.0
    julia> u'*u ≈ I ≈ v'*vtrue
    julia> (u'*matrix*v) |> disp25×9 Array{Float64, 2}: - 0.267 0.693 0.497 0.886 0.516 0.0 0.0 0.0 0.0 - 0.297 0.388 -0.241 -1.93 -0.628 0.0 0.0 0.0 0.0 - -0.165 1.241 -1.55 0.165 -0.738 0.0 0.0 0.0 0.0 - -0.868 -2.041 0.31 0.285 0.369 0.0 0.0 0.0 0.0 - 1.038 -0.377 0.262 0.843 0.506 0.0 0.0 0.0 0.0 - -0.737 0.713 -0.908 0.747 0.802 0.0 0.0 0.0 0.0 - 0.722 -0.432 0.572 1.373 0.004 0.0 0.0 0.0 0.0 - -0.872 2.544 -1.308 -0.077 1.44 0.0 0.0 0.0 0.0 - -0.167 -1.648 1.537 0.943 0.358 0.0 0.0 0.0 0.0 - -0.214 -0.956 0.376 0.079 2.232 0.0 0.0 0.0 0.0 - 0.65 -0.157 -0.332 1.142 1.042 0.0 0.0 0.0 0.0 - -0.795 0.279 -1.123 0.401 0.591 0.0 0.0 0.0 0.0 - 0.832 0.888 -1.001 -1.15 1.057 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 0.0 -0.669 0.114 1.446 0.323 - 0.0 0.0 0.0 0.0 0.0 0.091 -0.104 1.992 1.011 - 0.0 0.0 0.0 0.0 0.0 0.359 -1.332 1.055 -0.776 - 0.0 0.0 0.0 0.0 0.0 -0.744 -1.232 1.468 0.16 - 0.0 0.0 0.0 0.0 0.0 -0.478 0.144 -0.717 0.192 - 0.0 0.0 0.0 0.0 0.0 -0.793 -1.046 -0.77 0.722 - 0.0 0.0 0.0 0.0 0.0 0.498 -0.47 0.536 0.863 - 0.0 0.0 0.0 0.0 0.0 0.399 -0.78 0.78 0.307 - 0.0 0.0 0.0 0.0 0.0 -0.412 -0.445 -3.559 -0.195 - 0.0 0.0 0.0 0.0 0.0 -1.106 0.48 -2.152 1.106 - 0.0 0.0 0.0 0.0 0.0 -0.684 1.134 0.856 0.998 - 0.0 0.0 0.0 0.0 0.0 -0.368 -0.44 -0.591 0.425
    julia> # compare with: + 0.169 -0.574 -0.52 -1.274 0.014 0.0 0.0 0.0 0.0 + 0.173 0.896 -0.544 -1.036 0.393 0.0 0.0 0.0 0.0 + 1.094 0.065 0.876 0.687 1.852 0.0 0.0 0.0 0.0 + -0.882 0.771 -0.223 0.4 -1.781 0.0 0.0 0.0 0.0 + -0.067 -1.056 1.157 -1.405 -0.062 0.0 0.0 0.0 0.0 + -0.048 -1.741 0.766 0.215 1.824 0.0 0.0 0.0 0.0 + 0.729 -0.778 -0.15 -0.398 0.48 0.0 0.0 0.0 0.0 + -0.0 0.295 0.225 0.165 -0.988 0.0 0.0 0.0 0.0 + 1.689 -1.195 0.915 1.355 -0.15 0.0 0.0 0.0 0.0 + -0.681 0.422 0.574 -1.324 0.128 0.0 0.0 0.0 0.0 + 0.614 0.588 -0.986 0.355 -0.459 0.0 0.0 0.0 0.0 + -0.608 0.437 -1.078 -0.145 0.015 0.0 0.0 0.0 0.0 + -1.028 0.667 -2.02 -0.65 -1.438 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 0.0 0.72 -0.499 -0.703 0.277 + 0.0 0.0 0.0 0.0 0.0 -0.009 -0.057 0.668 1.059 + 0.0 0.0 0.0 0.0 0.0 -1.109 0.757 -1.202 0.525 + 0.0 0.0 0.0 0.0 0.0 -1.402 -0.426 -0.428 1.432 + 0.0 0.0 0.0 0.0 0.0 -0.073 0.498 -1.054 0.101 + 0.0 0.0 0.0 0.0 0.0 -0.209 -1.123 1.36 0.975 + 0.0 0.0 0.0 0.0 0.0 1.35 -0.324 0.779 -0.912 + 0.0 0.0 0.0 0.0 0.0 2.07 1.322 1.937 0.944 + 0.0 0.0 0.0 0.0 0.0 -1.473 2.549 -0.523 -1.548 + 0.0 0.0 0.0 0.0 0.0 -0.924 0.794 -0.073 1.747 + 0.0 0.0 0.0 0.0 0.0 0.319 -0.711 -0.55 1.033 + 0.0 0.0 0.0 0.0 0.0 2.311 1.365 0.916 -0.248
    julia> # compare with: block(t, Z2Irrep(0)) |> disp13×5 Array{Float64, 2}: - 0.267 0.693 0.497 0.886 0.516 - 0.297 0.388 -0.241 -1.93 -0.628 - -0.165 1.241 -1.55 0.165 -0.738 - -0.868 -2.041 0.31 0.285 0.369 - 1.038 -0.377 0.262 0.843 0.506 - -0.737 0.713 -0.908 0.747 0.802 - 0.722 -0.432 0.572 1.373 0.004 - -0.872 2.544 -1.308 -0.077 1.44 - -0.167 -1.648 1.537 0.943 0.358 - -0.214 -0.956 0.376 0.079 2.232 - 0.65 -0.157 -0.332 1.142 1.042 - -0.795 0.279 -1.123 0.401 0.591 - 0.832 0.888 -1.001 -1.15 1.057
    julia> block(t, Z2Irrep(1)) |> disp12×4 Array{Float64, 2}: - -0.669 0.114 1.446 0.323 - 0.091 -0.104 1.992 1.011 - 0.359 -1.332 1.055 -0.776 - -0.744 -1.232 1.468 0.16 - -0.478 0.144 -0.717 0.192 - -0.793 -1.046 -0.77 0.722 - 0.498 -0.47 0.536 0.863 - 0.399 -0.78 0.78 0.307 - -0.412 -0.445 -3.559 -0.195 - -1.106 0.48 -2.152 1.106 - -0.684 1.134 0.856 0.998 - -0.368 -0.44 -0.591 0.425

    Here, we illustrated some additional concepts. Firstly, note that we convert a TensorMap to an Array. This only works when sectortype(t) supports fusiontensor, and in particular when BraidingStyle(sectortype(t)) == Bosonic(), e.g. the case of trivial tensors (the category $\mathbf{Vect}$) and group representations (the category $\mathbf{Rep}_{\mathsf{G}}$, which can be interpreted as a subcategory of $\mathbf{Vect}$). Here, we are in this case with $\mathsf{G} = ℤ₂$. For a TensorMap{S,1,1}, the blocks directly correspond to the diagonal blocks in the block diagonal structure of its representation as an Array, there is no basis transform in between. This is no longer the case for TensorMap{S,N₁,N₂} with different values of N₁ and N₂. Here, we use the operation fuse(V), which creates an ElementarySpace which is isomorphic to a given space V (of type ProductSpace or ElementarySpace). The specific map between those two spaces constructed using the specific method unitary implements precisely the basis change from the product basis to the coupled basis. In this case, for a group G with FusionStyle(Irrep[G]) isa UniqueFusion, it is a permutation matrix. Specifically choosing V equal to the codomain and domain of t, we can construct the explicit basis transforms that bring t into block diagonal form.

    Let's repeat the same exercise for I = Irrep[SU₂], which has FusionStyle(I) isa MultipleFusion.

    julia> V1 = SU₂Space(0=>2,1=>1)Rep[SU₂](0=>2, 1=>1)
    julia> V2 = SU₂Space(0=>1,1=>1)Rep[SU₂](0=>1, 1=>1)
    julia> # First a `TensorMap{SU₂Space, 1, 1}` + 0.169 -0.574 -0.52 -1.274 0.014 + 0.173 0.896 -0.544 -1.036 0.393 + 1.094 0.065 0.876 0.687 1.852 + -0.882 0.771 -0.223 0.4 -1.781 + -0.067 -1.056 1.157 -1.405 -0.062 + -0.048 -1.741 0.766 0.215 1.824 + 0.729 -0.778 -0.15 -0.398 0.48 + -0.0 0.295 0.225 0.165 -0.988 + 1.689 -1.195 0.915 1.355 -0.15 + -0.681 0.422 0.574 -1.324 0.128 + 0.614 0.588 -0.986 0.355 -0.459 + -0.608 0.437 -1.078 -0.145 0.015 + -1.028 0.667 -2.02 -0.65 -1.438
    julia> block(t, Z2Irrep(1)) |> disp12×4 Array{Float64, 2}: + 0.72 -0.499 -0.703 0.277 + -0.009 -0.057 0.668 1.059 + -1.109 0.757 -1.202 0.525 + -1.402 -0.426 -0.428 1.432 + -0.073 0.498 -1.054 0.101 + -0.209 -1.123 1.36 0.975 + 1.35 -0.324 0.779 -0.912 + 2.07 1.322 1.937 0.944 + -1.473 2.549 -0.523 -1.548 + -0.924 0.794 -0.073 1.747 + 0.319 -0.711 -0.55 1.033 + 2.311 1.365 0.916 -0.248

    Here, we illustrated some additional concepts. Firstly, note that we convert a TensorMap to an Array. This only works when sectortype(t) supports fusiontensor, and in particular when BraidingStyle(sectortype(t)) == Bosonic(), e.g. the case of trivial tensors (the category $\mathbf{Vect}$) and group representations (the category $\mathbf{Rep}_{\mathsf{G}}$, which can be interpreted as a subcategory of $\mathbf{Vect}$). Here, we are in this case with $\mathsf{G} = ℤ₂$. For a TensorMap{S,1,1}, the blocks directly correspond to the diagonal blocks in the block diagonal structure of its representation as an Array, there is no basis transform in between. This is no longer the case for TensorMap{S,N₁,N₂} with different values of N₁ and N₂. Here, we use the operation fuse(V), which creates an ElementarySpace which is isomorphic to a given space V (of type ProductSpace or ElementarySpace). The specific map between those two spaces constructed using the specific method unitary implements precisely the basis change from the product basis to the coupled basis. In this case, for a group G with FusionStyle(Irrep[G]) isa UniqueFusion, it is a permutation matrix. Specifically choosing V equal to the codomain and domain of t, we can construct the explicit basis transforms that bring t into block diagonal form.

    Let's repeat the same exercise for I = Irrep[SU₂], which has FusionStyle(I) isa MultipleFusion.

    julia> V1 = SU₂Space(0=>2,1=>1)Rep[SU₂](0=>2, 1=>1)
    julia> V2 = SU₂Space(0=>1,1=>1)Rep[SU₂](0=>1, 1=>1)
    julia> # First a `TensorMap{SU₂Space, 1, 1}` m = TensorMap(randn, V1, V2)TensorMap(Rep[SU₂](0=>2, 1=>1) ← Rep[SU₂](0=>1, 1=>1)): * Data for fusiontree FusionTree{Irrep[SU₂]}((0,), 0, (false,), ()) ← FusionTree{Irrep[SU₂]}((0,), 0, (false,), ()): - -0.4804035342466387 - -1.7232087396823772 + -0.5677419546332747 + -1.7628536678287654 * Data for fusiontree FusionTree{Irrep[SU₂]}((1,), 1, (false,), ()) ← FusionTree{Irrep[SU₂]}((1,), 1, (false,), ()): - 0.808832372309538
    julia> convert(Array, m) |> disp5×4 Array{Float64, 2}: - -0.48 0.0 0.0 0.0 - -1.723 0.0 0.0 0.0 - 0.0 0.808 0.0 0.0 - 0.0 0.0 0.808 0.0 - 0.0 0.0 0.0 0.808
    julia> # compare with: + -0.23838780405371376
    julia> convert(Array, m) |> disp5×4 Array{Float64, 2}: + -0.567 0.0 0.0 0.0 + -1.762 0.0 0.0 0.0 + 0.0 -0.238 0.0 0.0 + 0.0 0.0 -0.238 0.0 + 0.0 0.0 0.0 -0.238
    julia> # compare with: block(m, Irrep[SU₂](0)) |> disp2×1 Array{Float64, 2}: - -0.48 - -1.723
    julia> block(m, Irrep[SU₂](1)) |> disp1×1 Array{Float64, 2}: - 0.808
    julia> # Now a `TensorMap{SU₂Space, 2, 2}` + -0.567 + -1.762
    julia> block(m, Irrep[SU₂](1)) |> disp1×1 Array{Float64, 2}: + -0.238
    julia> # Now a `TensorMap{SU₂Space, 2, 2}` t = TensorMap(randn, V1 ⊗ V1, V2 ⊗ V2')TensorMap((Rep[SU₂](0=>2, 1=>1) ⊗ Rep[SU₂](0=>2, 1=>1)) ← (Rep[SU₂](0=>1, 1=>1) ⊗ Rep[SU₂](0=>1, 1=>1)')): * Data for fusiontree FusionTree{Irrep[SU₂]}((0, 0), 0, (false, false), ()) ← FusionTree{Irrep[SU₂]}((0, 0), 0, (false, true), ()): [:, :, 1, 1] = - 0.30150948038367575 -0.45488117404798817 - -0.7890800170012612 -0.15805956424025974 + 0.8531160773410976 0.832758448612936 + 0.9516949012853989 1.4967304591411645 * Data for fusiontree FusionTree{Irrep[SU₂]}((1, 1), 0, (false, false), ()) ← FusionTree{Irrep[SU₂]}((0, 0), 0, (false, true), ()): [:, :, 1, 1] = - 1.5591591870708312 + -0.5581140808213874 * Data for fusiontree FusionTree{Irrep[SU₂]}((0, 0), 0, (false, false), ()) ← FusionTree{Irrep[SU₂]}((1, 1), 0, (false, true), ()): [:, :, 1, 1] = - 1.5944308516363672 -2.180658880286796 - -0.07360915059422922 0.9797043167142565 + 0.4637934463942614 -1.1649679755387266 + 1.5289370649305727 -0.004059168895202379 * Data for fusiontree FusionTree{Irrep[SU₂]}((1, 1), 0, (false, false), ()) ← FusionTree{Irrep[SU₂]}((1, 1), 0, (false, true), ()): [:, :, 1, 1] = - 1.3432793344594418 + 0.01755118723901428 * Data for fusiontree FusionTree{Irrep[SU₂]}((1, 1), 1, (false, false), ()) ← FusionTree{Irrep[SU₂]}((0, 1), 1, (false, true), ()): [:, :, 1, 1] = - 0.040057575321393826 + 0.30573110532205466 * Data for fusiontree FusionTree{Irrep[SU₂]}((1, 0), 1, (false, false), ()) ← FusionTree{Irrep[SU₂]}((0, 1), 1, (false, true), ()): [:, :, 1, 1] = - 1.3078034594174315 -1.7123156547327212 + 0.28884763891953236 -0.01533019730706483 * Data for fusiontree FusionTree{Irrep[SU₂]}((0, 1), 1, (false, false), ()) ← FusionTree{Irrep[SU₂]}((0, 1), 1, (false, true), ()): [:, :, 1, 1] = - -0.4637615260208085 - -0.03157242007176484 + -2.517333921639034 + 0.8154523767753589 * Data for fusiontree FusionTree{Irrep[SU₂]}((1, 1), 1, (false, false), ()) ← FusionTree{Irrep[SU₂]}((1, 1), 1, (false, true), ()): [:, :, 1, 1] = - 0.4492889571340022 + 0.003151027434170872 * Data for fusiontree FusionTree{Irrep[SU₂]}((1, 0), 1, (false, false), ()) ← FusionTree{Irrep[SU₂]}((1, 1), 1, (false, true), ()): [:, :, 1, 1] = - -1.3239384414285096 -1.0578696874192446 + -0.2989536251322713 -0.027696423515437996 * Data for fusiontree FusionTree{Irrep[SU₂]}((0, 1), 1, (false, false), ()) ← FusionTree{Irrep[SU₂]}((1, 1), 1, (false, true), ()): [:, :, 1, 1] = - 0.9062636248222917 - -0.6434330106650527 + 2.0829740916349255 + -0.47862350855627395 * Data for fusiontree FusionTree{Irrep[SU₂]}((1, 1), 1, (false, false), ()) ← FusionTree{Irrep[SU₂]}((1, 0), 1, (false, true), ()): [:, :, 1, 1] = - -0.5170886738293158 + -0.36793003281171294 * Data for fusiontree FusionTree{Irrep[SU₂]}((1, 0), 1, (false, false), ()) ← FusionTree{Irrep[SU₂]}((1, 0), 1, (false, true), ()): [:, :, 1, 1] = - 0.47106069759658514 1.3332353909335106 + 0.6103980863654384 0.16018386135558885 * Data for fusiontree FusionTree{Irrep[SU₂]}((0, 1), 1, (false, false), ()) ← FusionTree{Irrep[SU₂]}((1, 0), 1, (false, true), ()): [:, :, 1, 1] = - 0.7610065976534381 - -0.6531145018589162 + 0.9210957846774901 + 0.4778150522086885 * Data for fusiontree FusionTree{Irrep[SU₂]}((1, 1), 2, (false, false), ()) ← FusionTree{Irrep[SU₂]}((1, 1), 2, (false, true), ()): [:, :, 1, 1] = - -0.5907610564295156
    julia> (array = convert(Array, t)) |> disp5×5×4×4 Array{Float64, 4}: + -1.3479769189597905
    julia> (array = convert(Array, t)) |> disp5×5×4×4 Array{Float64, 4}: [:, :, 1, 1] = - 0.301 -0.454 0.0 0.0 0.0 - -0.789 -0.158 0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 0.9 - 0.0 0.0 0.0 -0.9 0.0 - 0.0 0.0 0.9 0.0 0.0 + 0.853 0.832 0.0 0.0 0.0 + 0.951 1.496 0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 -0.322 + 0.0 0.0 0.0 0.322 0.0 + 0.0 0.0 -0.322 0.0 0.0 [:, :, 2, 1] = - 0.0 0.0 0.761 0.0 0.0 - 0.0 0.0 -0.653 0.0 0.0 - 0.471 1.333 0.0 -0.365 0.0 - 0.0 0.0 0.365 0.0 0.0 - 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 0.921 0.0 0.0 + 0.0 0.0 0.477 0.0 0.0 + 0.61 0.16 0.0 -0.26 0.0 + 0.0 0.0 0.26 0.0 0.0 + 0.0 0.0 0.0 0.0 0.0 [:, :, 3, 1] = - 0.0 0.0 0.0 0.761 0.0 - 0.0 0.0 0.0 -0.653 0.0 - 0.0 0.0 0.0 0.0 -0.365 - 0.471 1.333 0.0 0.0 0.0 - 0.0 0.0 0.365 0.0 0.0 + 0.0 0.0 0.0 0.921 0.0 + 0.0 0.0 0.0 0.477 0.0 + 0.0 0.0 0.0 0.0 -0.26 + 0.61 0.16 0.0 0.0 0.0 + 0.0 0.0 0.26 0.0 0.0 [:, :, 4, 1] = - 0.0 0.0 0.0 0.0 0.761 - 0.0 0.0 0.0 0.0 -0.653 - 0.0 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 -0.365 - 0.471 1.333 0.0 0.365 0.0 + 0.0 0.0 0.0 0.0 0.921 + 0.0 0.0 0.0 0.0 0.477 + 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 -0.26 + 0.61 0.16 0.0 0.26 0.0 [:, :, 1, 2] = - 0.0 0.0 0.0 0.0 -0.463 - 0.0 0.0 0.0 0.0 -0.031 + 0.0 0.0 0.0 0.0 -2.517 + 0.0 0.0 0.0 0.0 0.815 0.0 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 0.028 - 1.307 -1.712 0.0 -0.028 0.0 + 0.0 0.0 0.0 0.0 0.216 + 0.288 -0.015 0.0 -0.216 0.0 [:, :, 2, 2] = - 0.92 -1.259 0.0 0.64 0.0 - -0.042 0.565 0.0 -0.454 0.0 - 0.0 0.0 0.0 0.0 0.573 - -0.936 -0.748 0.0 -0.644 0.0 - 0.0 0.0 0.124 0.0 0.0 + 0.267 -0.672 0.0 1.472 0.0 + 0.882 -0.002 0.0 -0.338 0.0 + 0.0 0.0 0.0 0.0 -0.217 + -0.211 -0.019 0.0 -0.455 0.0 + 0.0 0.0 -0.22 0.0 0.0 [:, :, 3, 2] = - 0.0 0.0 0.0 0.0 0.64 - 0.0 0.0 0.0 0.0 -0.454 - 0.0 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 -0.07 - -0.936 -0.748 0.0 -0.52 0.0 + 0.0 0.0 0.0 0.0 1.472 + 0.0 0.0 0.0 0.0 -0.338 + 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 -0.672 + -0.211 -0.019 0.0 -0.675 0.0 [:, :, 4, 2] = 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 -0.59 + 0.0 0.0 0.0 0.0 -1.347 [:, :, 1, 3] = - 0.0 0.0 0.0 0.463 0.0 - 0.0 0.0 0.0 0.031 0.0 - 0.0 0.0 0.0 0.0 -0.028 - -1.307 1.712 0.0 0.0 0.0 - 0.0 0.0 0.028 0.0 0.0 + 0.0 0.0 0.0 2.517 0.0 + 0.0 0.0 0.0 -0.815 0.0 + 0.0 0.0 0.0 0.0 -0.216 + -0.288 0.015 0.0 0.0 0.0 + 0.0 0.0 0.216 0.0 0.0 [:, :, 2, 3] = - 0.0 0.0 -0.64 0.0 0.0 - 0.0 0.0 0.454 0.0 0.0 - 0.936 0.748 0.0 0.07 0.0 - 0.0 0.0 0.52 0.0 0.0 - 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 -1.472 0.0 0.0 + 0.0 0.0 0.338 0.0 0.0 + 0.211 0.019 0.0 0.672 0.0 + 0.0 0.0 0.675 0.0 0.0 + 0.0 0.0 0.0 0.0 0.0 [:, :, 3, 3] = - 0.92 -1.259 0.0 0.0 0.0 - -0.042 0.565 0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 0.644 - 0.0 0.0 0.0 -0.053 0.0 - 0.0 0.0 0.644 0.0 0.0 + 0.267 -0.672 0.0 0.0 0.0 + 0.882 -0.002 0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 0.455 + 0.0 0.0 0.0 0.892 0.0 + 0.0 0.0 0.455 0.0 0.0 [:, :, 4, 3] = - 0.0 0.0 0.0 0.0 0.64 - 0.0 0.0 0.0 0.0 -0.454 - 0.0 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 0.52 - -0.936 -0.748 0.0 0.07 0.0 + 0.0 0.0 0.0 0.0 1.472 + 0.0 0.0 0.0 0.0 -0.338 + 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 0.675 + -0.211 -0.019 0.0 0.672 0.0 [:, :, 1, 4] = - 0.0 0.0 -0.463 0.0 0.0 - 0.0 0.0 -0.031 0.0 0.0 - 1.307 -1.712 0.0 0.028 0.0 - 0.0 0.0 -0.028 0.0 0.0 + 0.0 0.0 -2.517 0.0 0.0 + 0.0 0.0 0.815 0.0 0.0 + 0.288 -0.015 0.0 0.216 0.0 + 0.0 0.0 -0.216 0.0 0.0 0.0 0.0 0.0 0.0 0.0 [:, :, 2, 4] = - 0.0 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 0.0 - 0.0 0.0 -0.59 0.0 0.0 - 0.0 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 -1.347 0.0 0.0 + 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 0.0 [:, :, 3, 4] = - 0.0 0.0 -0.64 0.0 0.0 - 0.0 0.0 0.454 0.0 0.0 - 0.936 0.748 0.0 -0.52 0.0 - 0.0 0.0 -0.07 0.0 0.0 - 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 -1.472 0.0 0.0 + 0.0 0.0 0.338 0.0 0.0 + 0.211 0.019 0.0 -0.675 0.0 + 0.0 0.0 -0.672 0.0 0.0 + 0.0 0.0 0.0 0.0 0.0 [:, :, 4, 4] = - 0.92 -1.259 0.0 -0.64 0.0 - -0.042 0.565 0.0 0.454 0.0 - 0.0 0.0 0.0 0.0 0.124 - 0.936 0.748 0.0 -0.644 0.0 - 0.0 0.0 0.573 0.0 0.0
    julia> d1 = dim(codomain(t))25
    julia> d2 = dim(domain(t))16
    julia> (matrix = reshape(array, d1, d2)) |> disp25×16 Array{Float64, 2}: - 0.301 0.0 0.0 0.0 0.0 0.92 0.0 0.0 0.0 0.0 0.92 0.0 0.0 0.0 0.0 0.92 - -0.789 0.0 0.0 0.0 0.0 -0.042 0.0 0.0 0.0 0.0 -0.042 0.0 0.0 0.0 0.0 -0.042 - 0.0 0.471 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.936 0.0 0.0 1.307 0.0 0.936 0.0 - 0.0 0.0 0.471 0.0 0.0 -0.936 0.0 0.0 -1.307 0.0 0.0 0.0 0.0 0.0 0.0 0.936 - 0.0 0.0 0.0 0.471 1.307 0.0 -0.936 0.0 0.0 0.0 0.0 -0.936 0.0 0.0 0.0 0.0 - -0.454 0.0 0.0 0.0 0.0 -1.259 0.0 0.0 0.0 0.0 -1.259 0.0 0.0 0.0 0.0 -1.259 - -0.158 0.0 0.0 0.0 0.0 0.565 0.0 0.0 0.0 0.0 0.565 0.0 0.0 0.0 0.0 0.565 - 0.0 1.333 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.748 0.0 0.0 -1.712 0.0 0.748 0.0 - 0.0 0.0 1.333 0.0 0.0 -0.748 0.0 0.0 1.712 0.0 0.0 0.0 0.0 0.0 0.0 0.748 - 0.0 0.0 0.0 1.333 -1.712 0.0 -0.748 0.0 0.0 0.0 0.0 -0.748 0.0 0.0 0.0 0.0 - 0.0 0.761 0.0 0.0 0.0 0.0 0.0 0.0 0.0 -0.64 0.0 0.0 -0.463 0.0 -0.64 0.0 - 0.0 -0.653 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.454 0.0 0.0 -0.031 0.0 0.454 0.0 - 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 -0.59 0.0 0.0 - 0.0 0.365 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.52 0.0 0.0 -0.028 0.0 -0.07 0.0 - 0.9 0.0 0.365 0.0 0.0 0.124 0.0 0.0 0.028 0.0 0.644 0.0 0.0 0.0 0.0 0.573 - 0.0 0.0 0.761 0.0 0.0 0.64 0.0 0.0 0.463 0.0 0.0 0.0 0.0 0.0 0.0 -0.64 - 0.0 0.0 -0.653 0.0 0.0 -0.454 0.0 0.0 0.031 0.0 0.0 0.0 0.0 0.0 0.0 0.454 - 0.0 -0.365 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.07 0.0 0.0 0.028 0.0 -0.52 0.0 - -0.9 0.0 0.0 0.0 0.0 -0.644 0.0 0.0 0.0 0.0 -0.053 0.0 0.0 0.0 0.0 -0.644 - 0.0 0.0 0.0 0.365 -0.028 0.0 -0.52 0.0 0.0 0.0 0.0 0.07 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 0.761 -0.463 0.0 0.64 0.0 0.0 0.0 0.0 0.64 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 -0.653 -0.031 0.0 -0.454 0.0 0.0 0.0 0.0 -0.454 0.0 0.0 0.0 0.0 - 0.9 0.0 -0.365 0.0 0.0 0.573 0.0 0.0 -0.028 0.0 0.644 0.0 0.0 0.0 0.0 0.124 - 0.0 0.0 0.0 -0.365 0.028 0.0 -0.07 0.0 0.0 0.0 0.0 0.52 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 0.0 0.0 0.0 -0.59 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0
    julia> (u = reshape(convert(Array, unitary(codomain(t), fuse(codomain(t)))), d1, d1)) |> disp25×25 Array{Float64, 2}: + 0.267 -0.672 0.0 -1.472 0.0 + 0.882 -0.002 0.0 0.338 0.0 + 0.0 0.0 0.0 0.0 -0.22 + 0.211 0.019 0.0 -0.455 0.0 + 0.0 0.0 -0.217 0.0 0.0
    julia> d1 = dim(codomain(t))25
    julia> d2 = dim(domain(t))16
    julia> (matrix = reshape(array, d1, d2)) |> disp25×16 Array{Float64, 2}: + 0.853 0.0 0.0 0.0 0.0 0.267 0.0 0.0 0.0 0.0 0.267 0.0 0.0 0.0 0.0 0.267 + 0.951 0.0 0.0 0.0 0.0 0.882 0.0 0.0 0.0 0.0 0.882 0.0 0.0 0.0 0.0 0.882 + 0.0 0.61 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.211 0.0 0.0 0.288 0.0 0.211 0.0 + 0.0 0.0 0.61 0.0 0.0 -0.211 0.0 0.0 -0.288 0.0 0.0 0.0 0.0 0.0 0.0 0.211 + 0.0 0.0 0.0 0.61 0.288 0.0 -0.211 0.0 0.0 0.0 0.0 -0.211 0.0 0.0 0.0 0.0 + 0.832 0.0 0.0 0.0 0.0 -0.672 0.0 0.0 0.0 0.0 -0.672 0.0 0.0 0.0 0.0 -0.672 + 1.496 0.0 0.0 0.0 0.0 -0.002 0.0 0.0 0.0 0.0 -0.002 0.0 0.0 0.0 0.0 -0.002 + 0.0 0.16 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.019 0.0 0.0 -0.015 0.0 0.019 0.0 + 0.0 0.0 0.16 0.0 0.0 -0.019 0.0 0.0 0.015 0.0 0.0 0.0 0.0 0.0 0.0 0.019 + 0.0 0.0 0.0 0.16 -0.015 0.0 -0.019 0.0 0.0 0.0 0.0 -0.019 0.0 0.0 0.0 0.0 + 0.0 0.921 0.0 0.0 0.0 0.0 0.0 0.0 0.0 -1.472 0.0 0.0 -2.517 0.0 -1.472 0.0 + 0.0 0.477 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.338 0.0 0.0 0.815 0.0 0.338 0.0 + 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 -1.347 0.0 0.0 + 0.0 0.26 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.675 0.0 0.0 -0.216 0.0 -0.672 0.0 + -0.322 0.0 0.26 0.0 0.0 -0.22 0.0 0.0 0.216 0.0 0.455 0.0 0.0 0.0 0.0 -0.217 + 0.0 0.0 0.921 0.0 0.0 1.472 0.0 0.0 2.517 0.0 0.0 0.0 0.0 0.0 0.0 -1.472 + 0.0 0.0 0.477 0.0 0.0 -0.338 0.0 0.0 -0.815 0.0 0.0 0.0 0.0 0.0 0.0 0.338 + 0.0 -0.26 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.672 0.0 0.0 0.216 0.0 -0.675 0.0 + 0.322 0.0 0.0 0.0 0.0 -0.455 0.0 0.0 0.0 0.0 0.892 0.0 0.0 0.0 0.0 -0.455 + 0.0 0.0 0.0 0.26 -0.216 0.0 -0.675 0.0 0.0 0.0 0.0 0.672 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.921 -2.517 0.0 1.472 0.0 0.0 0.0 0.0 1.472 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.477 0.815 0.0 -0.338 0.0 0.0 0.0 0.0 -0.338 0.0 0.0 0.0 0.0 + -0.322 0.0 -0.26 0.0 0.0 -0.217 0.0 0.0 -0.216 0.0 0.455 0.0 0.0 0.0 0.0 -0.22 + 0.0 0.0 0.0 -0.26 0.216 0.0 -0.672 0.0 0.0 0.0 0.0 0.675 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 0.0 0.0 0.0 -1.347 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0
    julia> (u = reshape(convert(Array, unitary(codomain(t), fuse(codomain(t)))), d1, d1)) |> disp25×25 Array{Float64, 2}: 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 @@ -626,57 +626,57 @@ 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.999 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 -0.707 0.0 0.0 0.0 0.707 0.0 0.0 0.0 0.0 0.577 0.0 0.0 0.0 0.0 0.0 0.0 0.0 -0.707 0.0 0.0 0.0 0.408 0.0 0.0
    julia> u'*u ≈ I ≈ v'*vtrue
    julia> (u'*matrix*v) |> disp25×16 Array{Float64, 2}: - 0.301 1.594 0.0 0.0 0.0 0.0 0.0 0.0 0.0 -0.0 0.0 0.0 0.0 -0.0 0.0 0.0 - -0.789 -0.073 0.0 0.0 0.0 0.0 0.0 0.0 0.0 -0.0 0.0 0.0 0.0 -0.0 0.0 0.0 - -0.454 -2.18 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 -0.0 0.0 0.0 - -0.158 0.979 0.0 0.0 0.0 0.0 0.0 0.0 0.0 -0.0 0.0 0.0 0.0 0.0 0.0 0.0 - 1.559 1.343 0.0 0.0 0.0 0.0 -0.0 0.0 0.0 0.0 0.0 0.0 0.0 -0.0 0.0 0.0 - 0.0 0.0 0.471 0.0 0.0 1.307 0.0 0.0 -1.323 0.0 0.0 0.0 0.0 0.0 0.0 0.0 - 0.0 -0.0 0.0 0.471 0.0 0.0 1.307 0.0 0.0 -1.323 0.0 0.0 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 0.471 0.0 0.0 1.307 0.0 0.0 -1.323 0.0 0.0 0.0 0.0 0.0 - 0.0 0.0 1.333 0.0 0.0 -1.712 0.0 0.0 -1.057 0.0 0.0 0.0 -0.0 0.0 0.0 0.0 - 0.0 -0.0 0.0 1.333 0.0 0.0 -1.712 0.0 0.0 -1.057 0.0 0.0 0.0 -0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 1.333 0.0 0.0 -1.712 0.0 0.0 -1.057 0.0 0.0 0.0 -0.0 0.0 - 0.0 0.0 0.761 0.0 0.0 -0.463 0.0 0.0 0.906 0.0 0.0 0.0 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 0.761 0.0 0.0 -0.463 0.0 0.0 0.906 0.0 0.0 0.0 -0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 0.761 0.0 0.0 -0.463 0.0 0.0 0.906 0.0 0.0 0.0 0.0 0.0 - 0.0 0.0 -0.653 0.0 0.0 -0.031 0.0 0.0 -0.643 0.0 0.0 0.0 -0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 -0.653 0.0 0.0 -0.031 0.0 0.0 -0.643 0.0 0.0 0.0 -0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 -0.653 0.0 0.0 -0.031 0.0 0.0 -0.643 0.0 0.0 0.0 -0.0 0.0 - 0.0 0.0 -0.517 0.0 0.0 0.04 0.0 0.0 0.449 0.0 0.0 0.0 0.0 0.0 0.0 0.0 - -0.0 -0.0 0.0 -0.517 0.0 0.0 0.04 0.0 0.0 0.449 0.0 0.0 0.0 -0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 -0.517 0.0 0.0 0.04 0.0 0.0 0.449 0.0 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 -0.59 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 0.0 -0.0 0.0 0.0 0.0 0.0 0.0 0.0 -0.59 0.0 0.0 0.0 - -0.0 0.0 0.0 -0.0 0.0 0.0 -0.0 0.0 0.0 -0.0 0.0 0.0 0.0 -0.59 0.0 0.0 - 0.0 0.0 0.0 0.0 0.0 0.0 0.0 -0.0 0.0 0.0 0.0 0.0 0.0 0.0 -0.59 0.0 - 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 -0.59
    julia> # compare with: + 0.853 0.463 0.0 0.0 0.0 0.0 0.0 0.0 0.0 -0.0 0.0 0.0 0.0 -0.0 0.0 0.0 + 0.951 1.528 0.0 0.0 0.0 0.0 0.0 0.0 0.0 -0.0 0.0 0.0 0.0 0.0 0.0 0.0 + 0.832 -1.164 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 + 1.496 -0.004 0.0 0.0 0.0 0.0 0.0 0.0 0.0 -0.0 0.0 0.0 0.0 -0.0 0.0 0.0 + -0.558 0.017 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 0.61 0.0 0.0 0.288 0.0 0.0 -0.298 0.0 0.0 0.0 0.0 0.0 0.0 0.0 + 0.0 -0.0 0.0 0.61 0.0 0.0 0.288 0.0 0.0 -0.298 0.0 0.0 0.0 -0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 0.61 0.0 0.0 0.288 0.0 0.0 -0.298 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 0.16 0.0 0.0 -0.015 0.0 0.0 -0.027 0.0 0.0 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.16 0.0 0.0 -0.015 0.0 0.0 -0.027 0.0 0.0 0.0 -0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 0.16 0.0 0.0 -0.015 0.0 0.0 -0.027 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 0.921 0.0 0.0 -2.517 0.0 0.0 2.082 0.0 0.0 0.0 -0.0 0.0 0.0 0.0 + 0.0 -0.0 0.0 0.921 0.0 0.0 -2.517 0.0 0.0 2.082 0.0 0.0 0.0 -0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 0.921 0.0 0.0 -2.517 0.0 0.0 2.082 0.0 0.0 0.0 -0.0 0.0 + 0.0 0.0 0.477 0.0 0.0 0.815 0.0 0.0 -0.478 0.0 0.0 0.0 -0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.477 0.0 0.0 0.815 0.0 0.0 -0.478 0.0 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 0.477 0.0 0.0 0.815 0.0 0.0 -0.478 0.0 0.0 0.0 -0.0 0.0 + 0.0 0.0 -0.367 0.0 0.0 0.305 0.0 0.0 0.003 0.0 0.0 0.0 0.0 0.0 0.0 0.0 + -0.0 0.0 0.0 -0.367 0.0 0.0 0.305 0.0 0.0 0.003 0.0 0.0 0.0 -0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 -0.367 0.0 0.0 0.305 0.0 0.0 0.003 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 -1.347 0.0 0.0 0.0 0.0 + 0.0 0.0 -0.0 0.0 0.0 0.0 0.0 0.0 -0.0 0.0 0.0 0.0 -1.347 0.0 0.0 0.0 + -0.0 0.0 0.0 0.0 0.0 0.0 -0.0 0.0 0.0 0.0 0.0 0.0 0.0 -1.347 0.0 0.0 + 0.0 0.0 0.0 0.0 -0.0 0.0 0.0 0.0 0.0 0.0 -0.0 0.0 0.0 0.0 -1.347 0.0 + 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 -1.347
    julia> # compare with: block(t, SU2Irrep(0)) |> disp5×2 Array{Float64, 2}: - 0.301 1.594 - -0.789 -0.073 - -0.454 -2.18 - -0.158 0.979 - 1.559 1.343
    julia> block(t, SU2Irrep(1)) |> disp5×3 Array{Float64, 2}: - 0.471 1.307 -1.323 - 1.333 -1.712 -1.057 - 0.761 -0.463 0.906 - -0.653 -0.031 -0.643 - -0.517 0.04 0.449
    julia> block(t, SU2Irrep(2)) |> disp1×1 Array{Float64, 2}: - -0.59

    Note that the basis transforms u and v are no longer permutation matrices, but are still unitary. Furthermore, note that they render the tensor block diagonal, but that now every element of the diagonal blocks labeled by c comes itself in a tensor product with an identity matrix of size dim(c), i.e. dim(SU2Irrep(1)) = 3 and dim(SU2Irrep(2)) = 5.

    Tensor properties

    Given a t::AbstractTensorMap{S,N₁,N₂}, there are various methods to query its properties. The most important are clearly codomain(t) and domain(t). For t::AbstractTensor{S,N}, i.e. t::AbstractTensorMap{S,N,0}, we can use space(t) as synonym for codomain(t). However, for a general AbstractTensorMap this has no meaning. However, we can query space(t, i), the space associated with the ith index. For i ∈ 1:N₁, this corresponds to codomain(t, i) = codomain(t)[i]. For j = i-N₁ ∈ (1:N₂), this corresponds to dual(domain(t, j)) = dual(domain(t)[j]).

    The total number of indices, i.e. N₁+N₂, is given by numind(t), with N₁ == numout(t) and N₂ == numin(t), the number of outgoing and incoming indices. There are also the unexported methods TensorKit.codomainind(t) and TensorKit.domainind(t) which return the tuples (1, 2, …, N₁) and (N₁+1, …, N₁+N₂), and are useful for internal purposes. The type parameter S<:ElementarySpace can be obtained as spacetype(t); the corresponding sector can directly obtained as sectortype(t) and is Trivial when S != GradedSpace. The underlying field scalars of S can also directly be obtained as field(t). This is different from eltype(t), which returns the type of Number in the tensor data, i.e. the type parameter T in the (subtype of) DenseMatrix{T} in which the matrix blocks are stored. Note that during construction, a (one-time) warning is printed if !(T ⊂ field(S)). The specific DenseMatrix{T} subtype in which the tensor data is stored is obtained as storagetype(t). Each of the methods numind, numout, numin, TensorKit.codomainind, TensorKit.domainind, spacetype, sectortype, field, eltype and storagetype work in the type domain as well, i.e. they are encoded in typeof(t).

    Finally, there are methods to probe the data, which we already encountered. blocksectors(t) returns an iterator over the different coupled sectors that can be obtained from fusing the uncoupled sectors available in the domain, but they must also be obtained from fusing the uncoupled sectors available in the codomain (i.e. it is the intersection of both blocksectors(codomain(t)) and blocksectors(domain(t))). For a specific sector c ∈ blocksectors(t), block(t, c) returns the corresponding data. Both are obtained together with blocks(t), which returns an iterator over the pairs c=>block(t, c). Furthermore, there is fusiontrees(t) which returns an iterator over splitting-fusion tree pairs (f₁,f₂), for which the corresponding data is given by t[f₁,f₂] (i.e. using Base.getindex).

    Let's again illustrate these methods with an example, continuing with the tensor t from the previous example

    julia> typeof(t)TensorMap{GradedSpace{SU2Irrep, TensorKit.SortedVectorDict{SU2Irrep, Int64}}, 2, 2, SU2Irrep, TensorKit.SortedVectorDict{SU2Irrep, Matrix{Float64}}, FusionTree{SU2Irrep, 2, 0, 1, Nothing}, FusionTree{SU2Irrep, 2, 0, 1, Nothing}}
    julia> codomain(t)(Rep[SU₂](0=>2, 1=>1) ⊗ Rep[SU₂](0=>2, 1=>1))
    julia> domain(t)(Rep[SU₂](0=>1, 1=>1) ⊗ Rep[SU₂](0=>1, 1=>1)')
    julia> space(t,1)Rep[SU₂](0=>2, 1=>1)
    julia> space(t,2)Rep[SU₂](0=>2, 1=>1)
    julia> space(t,3)Rep[SU₂](0=>1, 1=>1)'
    julia> space(t,4)Rep[SU₂](0=>1, 1=>1)
    julia> numind(t)4
    julia> numout(t)2
    julia> numin(t)2
    julia> spacetype(t)GradedSpace{SU2Irrep, TensorKit.SortedVectorDict{SU2Irrep, Int64}}
    julia> sectortype(t)SU2Irrep
    julia> field(t)
    julia> eltype(t)Float64
    julia> storagetype(t)Matrix{Float64} (alias for Array{Float64, 2})
    julia> blocksectors(t)3-element Vector{SU2Irrep}: + 0.853 0.463 + 0.951 1.528 + 0.832 -1.164 + 1.496 -0.004 + -0.558 0.017
    julia> block(t, SU2Irrep(1)) |> disp5×3 Array{Float64, 2}: + 0.61 0.288 -0.298 + 0.16 -0.015 -0.027 + 0.921 -2.517 2.082 + 0.477 0.815 -0.478 + -0.367 0.305 0.003
    julia> block(t, SU2Irrep(2)) |> disp1×1 Array{Float64, 2}: + -1.347

    Note that the basis transforms u and v are no longer permutation matrices, but are still unitary. Furthermore, note that they render the tensor block diagonal, but that now every element of the diagonal blocks labeled by c comes itself in a tensor product with an identity matrix of size dim(c), i.e. dim(SU2Irrep(1)) = 3 and dim(SU2Irrep(2)) = 5.

    Tensor properties

    Given a t::AbstractTensorMap{S,N₁,N₂}, there are various methods to query its properties. The most important are clearly codomain(t) and domain(t). For t::AbstractTensor{S,N}, i.e. t::AbstractTensorMap{S,N,0}, we can use space(t) as synonym for codomain(t). However, for a general AbstractTensorMap this has no meaning. However, we can query space(t, i), the space associated with the ith index. For i ∈ 1:N₁, this corresponds to codomain(t, i) = codomain(t)[i]. For j = i-N₁ ∈ (1:N₂), this corresponds to dual(domain(t, j)) = dual(domain(t)[j]).

    The total number of indices, i.e. N₁+N₂, is given by numind(t), with N₁ == numout(t) and N₂ == numin(t), the number of outgoing and incoming indices. There are also the unexported methods TensorKit.codomainind(t) and TensorKit.domainind(t) which return the tuples (1, 2, …, N₁) and (N₁+1, …, N₁+N₂), and are useful for internal purposes. The type parameter S<:ElementarySpace can be obtained as spacetype(t); the corresponding sector can directly obtained as sectortype(t) and is Trivial when S != GradedSpace. The underlying field scalars of S can also directly be obtained as field(t). This is different from eltype(t), which returns the type of Number in the tensor data, i.e. the type parameter T in the (subtype of) DenseMatrix{T} in which the matrix blocks are stored. Note that during construction, a (one-time) warning is printed if !(T ⊂ field(S)). The specific DenseMatrix{T} subtype in which the tensor data is stored is obtained as storagetype(t). Each of the methods numind, numout, numin, TensorKit.codomainind, TensorKit.domainind, spacetype, sectortype, field, eltype and storagetype work in the type domain as well, i.e. they are encoded in typeof(t).

    Finally, there are methods to probe the data, which we already encountered. blocksectors(t) returns an iterator over the different coupled sectors that can be obtained from fusing the uncoupled sectors available in the domain, but they must also be obtained from fusing the uncoupled sectors available in the codomain (i.e. it is the intersection of both blocksectors(codomain(t)) and blocksectors(domain(t))). For a specific sector c ∈ blocksectors(t), block(t, c) returns the corresponding data. Both are obtained together with blocks(t), which returns an iterator over the pairs c=>block(t, c). Furthermore, there is fusiontrees(t) which returns an iterator over splitting-fusion tree pairs (f₁,f₂), for which the corresponding data is given by t[f₁,f₂] (i.e. using Base.getindex).

    Let's again illustrate these methods with an example, continuing with the tensor t from the previous example

    julia> typeof(t)TensorMap{GradedSpace{SU2Irrep, TensorKit.SortedVectorDict{SU2Irrep, Int64}}, 2, 2, SU2Irrep, TensorKit.SortedVectorDict{SU2Irrep, Matrix{Float64}}, FusionTree{SU2Irrep, 2, 0, 1, Nothing}, FusionTree{SU2Irrep, 2, 0, 1, Nothing}}
    julia> codomain(t)(Rep[SU₂](0=>2, 1=>1) ⊗ Rep[SU₂](0=>2, 1=>1))
    julia> domain(t)(Rep[SU₂](0=>1, 1=>1) ⊗ Rep[SU₂](0=>1, 1=>1)')
    julia> space(t,1)Rep[SU₂](0=>2, 1=>1)
    julia> space(t,2)Rep[SU₂](0=>2, 1=>1)
    julia> space(t,3)Rep[SU₂](0=>1, 1=>1)'
    julia> space(t,4)Rep[SU₂](0=>1, 1=>1)
    julia> numind(t)4
    julia> numout(t)2
    julia> numin(t)2
    julia> spacetype(t)GradedSpace{SU2Irrep, TensorKit.SortedVectorDict{SU2Irrep, Int64}}
    julia> sectortype(t)SU2Irrep
    julia> field(t)
    julia> eltype(t)Float64
    julia> storagetype(t)Matrix{Float64} (alias for Array{Float64, 2})
    julia> blocksectors(t)3-element Vector{SU2Irrep}: 0 1 2
    julia> blocks(t)TensorKit.SortedVectorDict{SU2Irrep, Matrix{Float64}} with 3 entries: - 0 => [0.301509 1.59443; -0.78908 -0.0736092; … ; -0.15806 0.979704; 1.55916 1… - 1 => [0.471061 1.3078 -1.32394; 1.33324 -1.71232 -1.05787; … ; -0.653115 -0.0… - 2 => [-0.590761;;]
    julia> block(t, first(blocksectors(t)))5×2 Matrix{Float64}: - 0.301509 1.59443 - -0.78908 -0.0736092 - -0.454881 -2.18066 - -0.15806 0.979704 - 1.55916 1.34328
    julia> fusiontrees(t)TensorKit.TensorKeyIterator{SU2Irrep, FusionTree{SU2Irrep, 2, 0, 1, Nothing}, FusionTree{SU2Irrep, 2, 0, 1, Nothing}}(TensorKit.SortedVectorDict{SU2Irrep, Dict{FusionTree{SU2Irrep, 2, 0, 1, Nothing}, UnitRange{Int64}}}(0 => Dict(FusionTree{Irrep[SU₂]}((0, 0), 0, (false, false), ()) => 1:4, FusionTree{Irrep[SU₂]}((1, 1), 0, (false, false), ()) => 5:5), 1 => Dict(FusionTree{Irrep[SU₂]}((1, 1), 1, (false, false), ()) => 5:5, FusionTree{Irrep[SU₂]}((1, 0), 1, (false, false), ()) => 1:2, FusionTree{Irrep[SU₂]}((0, 1), 1, (false, false), ()) => 3:4), 2 => Dict(FusionTree{Irrep[SU₂]}((1, 1), 2, (false, false), ()) => 1:1)), TensorKit.SortedVectorDict{SU2Irrep, Dict{FusionTree{SU2Irrep, 2, 0, 1, Nothing}, UnitRange{Int64}}}(0 => Dict(FusionTree{Irrep[SU₂]}((0, 0), 0, (false, true), ()) => 1:1, FusionTree{Irrep[SU₂]}((1, 1), 0, (false, true), ()) => 2:2), 1 => Dict(FusionTree{Irrep[SU₂]}((0, 1), 1, (false, true), ()) => 2:2, FusionTree{Irrep[SU₂]}((1, 1), 1, (false, true), ()) => 3:3, FusionTree{Irrep[SU₂]}((1, 0), 1, (false, true), ()) => 1:1), 2 => Dict(FusionTree{Irrep[SU₂]}((1, 1), 2, (false, true), ()) => 1:1)))
    julia> f1, f2 = first(fusiontrees(t))(FusionTree{Irrep[SU₂]}((0, 0), 0, (false, false), ()), FusionTree{Irrep[SU₂]}((0, 0), 0, (false, true), ()))
    julia> t[f1,f2]2×2×1×1 StridedViews.StridedView{Float64, 4, Matrix{Float64}, typeof(identity)}: -[:, :, 1, 1] = - 0.301509 -0.454881 - -0.78908 -0.15806

    Reading and writing tensors: Dict conversion

    There are no custom or dedicated methods for reading, writing or storing TensorMaps, however, there is the possibility to convert a t::AbstractTensorMap into a Dict, simply as convert(Dict, t). The backward conversion convert(TensorMap, dict) will return a tensor that is equal to t, i.e. t == convert(TensorMap, convert(Dict, t)).

    This conversion relies on that the string represenation of objects such as VectorSpace, FusionTree or Sector should be such that it represents valid code to recreate the object. Hence, we store information about the domain and codomain of the tensor, and the sector associated with each data block, as a String obtained with repr. This provides the flexibility to still change the internal structure of such objects, without this breaking the ability to load older data files. The resulting dictionary can then be stored using any of the provided Julia packages such as JLD.jl, JLD2.jl, BSON.jl, JSON.jl, ...

    Vector space and linear algebra operations

    AbstractTensorMap instances t represent linear maps, i.e. homomorphisms in a 𝕜-linear category, just like matrices. To a large extent, they follow the interface of Matrix in Julia's LinearAlgebra standard library. Many methods from LinearAlgebra are (re)exported by TensorKit.jl, and can then us be used without using LinearAlgebra explicitly. In all of the following methods, the implementation acts directly on the underlying matrix blocks (typically using the same method) and never needs to perform any basis transforms.

    In particular, AbstractTensorMap instances can be composed, provided the domain of the first object coincides with the codomain of the second. Composing tensor maps uses the regular multiplication symbol as in t = t1*t2, which is also used for matrix multiplication. TensorKit.jl also supports (and exports) the mutating method mul!(t, t1, t2). We can then also try to invert a tensor map using inv(t), though this can only exist if the domain and codomain are isomorphic, which can e.g. be checked as fuse(codomain(t)) == fuse(domain(t)). If the inverse is composed with another tensor t2, we can use the syntax t1\t2 or t2/t1. However, this syntax also accepts instances t1 whose domain and codomain are not isomorphic, and then amounts to pinv(t1), the Moore-Penrose pseudoinverse. This, however, is only really justified as minimizing the least squares problem if InnerProductStyle(t) <: EuclideanProduct.

    AbstractTensorMap instances behave themselves as vectors (i.e. they are 𝕜-linear) and so they can be multiplied by scalars and, if they live in the same space, i.e. have the same domain and codomain, they can be added to each other. There is also a zero(t), the additive identity, which produces a zero tensor with the same domain and codomain as t. In addition, TensorMap supports basic Julia methods such as fill! and copyto!, as well as copy(t) to create a copy with independent data. Aside from basic + and * operations, TensorKit.jl reexports a number of efficient in-place methods from LinearAlgebra, such as axpy! (for y ← α * x + y), axpby! (for y ← α * x + β * y), lmul! and rmul! (for y ← α*y and y ← y*α, which is typically the same) and mul!, which can also be used for out-of-place scalar multiplication y ← α*x.

    For t::AbstractTensorMap{S} where InnerProductStyle(S) <: EuclideanProduct, we can compute norm(t), and for two such instances, the inner product dot(t1, t2), provided t1 and t2 have the same domain and codomain. Furthermore, there is normalize(t) and normalize!(t) to return a scaled version of t with unit norm. These operations should also exist for InnerProductStyle(S) <: HasInnerProduct, but require an interface for defining a custom inner product in these spaces. Currently, there is no concrete subtype of HasInnerProduct that is not an EuclideanProduct. In particular, CartesianSpace, ComplexSpace and GradedSpace all have InnerProductStyle(V) <: EuclideanProduct.

    With tensors that have InnerProductStyle(t) <: EuclideanProduct there is associated an adjoint operation, given by adjoint(t) or simply t', such that domain(t') == codomain(t) and codomain(t') == domain(t). Note that for an instance t::TensorMap{S,N₁,N₂}, t' is simply stored in a wrapper called AdjointTensorMap{S,N₂,N₁}, which is another subtype of AbstractTensorMap. This should be mostly unvisible to the user, as all methods should work for this type as well. It can be hard to reason about the index order of t', i.e. index i of t appears in t' at index position j = TensorKit.adjointtensorindex(t, i), where the latter method is typically not necessary and hence unexported. There is also a plural TensorKit.adjointtensorindices to convert multiple indices at once. Note that, because the adjoint interchanges domain and codomain, we have space(t', j) == space(t, i)'.

    AbstractTensorMap instances can furthermore be tested for exact (t1 == t2) or approximate (t1 ≈ t2) equality, though the latter requires that norm can be computed.

    When tensor map instances are endomorphisms, i.e. they have the same domain and codomain, there is a multiplicative identity which can be obtained as one(t) or one!(t), where the latter overwrites the contents of t. The multiplicative identity on a space V can also be obtained using id(A, V) as discussed above, such that for a general homomorphism t′, we have t′ == id(codomain(t′))*t′ == t′*id(domain(t′)). Returning to the case of endomorphisms t, we can compute the trace via tr(t) and exponentiate them using exp(t), or if the contents of t can be destroyed in the process, exp!(t). Furthermore, there are a number of tensor factorizations for both endomorphisms and general homomorphism that we discuss below.

    Finally, there are a number of operations that also belong in this paragraph because of their analogy to common matrix operations. The tensor product of two TensorMap instances t1 and t2 is obtained as t1 ⊗ t2 and results in a new TensorMap with codomain(t1⊗t2) = codomain(t1) ⊗ codomain(t2) and domain(t1⊗t2) = domain(t1) ⊗ domain(t2). If we have two TensorMap{S,N,1} instances t1 and t2 with the same codomain, we can combine them in a way that is analoguous to hcat, i.e. we stack them such that the new tensor catdomain(t1, t2) has also the same codomain, but has a domain which is domain(t1) ⊕ domain(t2). Similarly, if t1 and t2 are of type TensorMap{S,1,N} and have the same domain, the operation catcodomain(t1, t2) results in a new tensor with the same domain and a codomain given by codomain(t1) ⊕ codomain(t2), which is the analogy of vcat. Note that direct sum only makes sense between ElementarySpace objects, i.e. there is no way to give a tensor product meaning to a direct sum of tensor product spaces.

    Time for some more examples:

    julia> t == t + zero(t) == t*id(domain(t)) == id(codomain(t))*ttrue
    julia> t2 = TensorMap(randn, ComplexF64, codomain(t), domain(t));
    julia> dot(t2, t)3.2927022497978244 - 10.272417578301411im
    julia> tr(t2'*t)3.2927022497978244 - 10.272417578301411im
    julia> dot(t2, t) ≈ dot(t', t2')true
    julia> dot(t2, t2)49.447233460250516 + 0.0im
    julia> norm(t2)^249.44723346025053
    julia> t3 = copyto!(similar(t, ComplexF64), t);ERROR: MethodError: no method matching copyto!(::TensorMap{GradedSpace{SU2Irrep, TensorKit.SortedVectorDict{SU2Irrep, Int64}}, 2, 2, SU2Irrep, TensorKit.SortedVectorDict{SU2Irrep, Matrix{ComplexF64}}, FusionTree{SU2Irrep, 2, 0, 1, Nothing}, FusionTree{SU2Irrep, 2, 0, 1, Nothing}}, ::TensorMap{GradedSpace{SU2Irrep, TensorKit.SortedVectorDict{SU2Irrep, Int64}}, 2, 2, SU2Irrep, TensorKit.SortedVectorDict{SU2Irrep, Matrix{Float64}}, FusionTree{SU2Irrep, 2, 0, 1, Nothing}, FusionTree{SU2Irrep, 2, 0, 1, Nothing}}) + 0 => [0.853116 0.463793; 0.951695 1.52894; … ; 1.49673 -0.00405917; -0.558114… + 1 => [0.610398 0.288848 -0.298954; 0.160184 -0.0153302 -0.0276964; … ; 0.4778… + 2 => [-1.34798;;]
    julia> block(t, first(blocksectors(t)))5×2 Matrix{Float64}: + 0.853116 0.463793 + 0.951695 1.52894 + 0.832758 -1.16497 + 1.49673 -0.00405917 + -0.558114 0.0175512
    julia> fusiontrees(t)TensorKit.TensorKeyIterator{SU2Irrep, FusionTree{SU2Irrep, 2, 0, 1, Nothing}, FusionTree{SU2Irrep, 2, 0, 1, Nothing}}(TensorKit.SortedVectorDict{SU2Irrep, Dict{FusionTree{SU2Irrep, 2, 0, 1, Nothing}, UnitRange{Int64}}}(0 => Dict(FusionTree{Irrep[SU₂]}((0, 0), 0, (false, false), ()) => 1:4, FusionTree{Irrep[SU₂]}((1, 1), 0, (false, false), ()) => 5:5), 1 => Dict(FusionTree{Irrep[SU₂]}((1, 1), 1, (false, false), ()) => 5:5, FusionTree{Irrep[SU₂]}((1, 0), 1, (false, false), ()) => 1:2, FusionTree{Irrep[SU₂]}((0, 1), 1, (false, false), ()) => 3:4), 2 => Dict(FusionTree{Irrep[SU₂]}((1, 1), 2, (false, false), ()) => 1:1)), TensorKit.SortedVectorDict{SU2Irrep, Dict{FusionTree{SU2Irrep, 2, 0, 1, Nothing}, UnitRange{Int64}}}(0 => Dict(FusionTree{Irrep[SU₂]}((0, 0), 0, (false, true), ()) => 1:1, FusionTree{Irrep[SU₂]}((1, 1), 0, (false, true), ()) => 2:2), 1 => Dict(FusionTree{Irrep[SU₂]}((0, 1), 1, (false, true), ()) => 2:2, FusionTree{Irrep[SU₂]}((1, 1), 1, (false, true), ()) => 3:3, FusionTree{Irrep[SU₂]}((1, 0), 1, (false, true), ()) => 1:1), 2 => Dict(FusionTree{Irrep[SU₂]}((1, 1), 2, (false, true), ()) => 1:1)))
    julia> f1, f2 = first(fusiontrees(t))(FusionTree{Irrep[SU₂]}((0, 0), 0, (false, false), ()), FusionTree{Irrep[SU₂]}((0, 0), 0, (false, true), ()))
    julia> t[f1,f2]2×2×1×1 StridedViews.StridedView{Float64, 4, Matrix{Float64}, typeof(identity)}: +[:, :, 1, 1] = + 0.853116 0.832758 + 0.951695 1.49673

    Reading and writing tensors: Dict conversion

    There are no custom or dedicated methods for reading, writing or storing TensorMaps, however, there is the possibility to convert a t::AbstractTensorMap into a Dict, simply as convert(Dict, t). The backward conversion convert(TensorMap, dict) will return a tensor that is equal to t, i.e. t == convert(TensorMap, convert(Dict, t)).

    This conversion relies on that the string represenation of objects such as VectorSpace, FusionTree or Sector should be such that it represents valid code to recreate the object. Hence, we store information about the domain and codomain of the tensor, and the sector associated with each data block, as a String obtained with repr. This provides the flexibility to still change the internal structure of such objects, without this breaking the ability to load older data files. The resulting dictionary can then be stored using any of the provided Julia packages such as JLD.jl, JLD2.jl, BSON.jl, JSON.jl, ...

    Vector space and linear algebra operations

    AbstractTensorMap instances t represent linear maps, i.e. homomorphisms in a 𝕜-linear category, just like matrices. To a large extent, they follow the interface of Matrix in Julia's LinearAlgebra standard library. Many methods from LinearAlgebra are (re)exported by TensorKit.jl, and can then us be used without using LinearAlgebra explicitly. In all of the following methods, the implementation acts directly on the underlying matrix blocks (typically using the same method) and never needs to perform any basis transforms.

    In particular, AbstractTensorMap instances can be composed, provided the domain of the first object coincides with the codomain of the second. Composing tensor maps uses the regular multiplication symbol as in t = t1*t2, which is also used for matrix multiplication. TensorKit.jl also supports (and exports) the mutating method mul!(t, t1, t2). We can then also try to invert a tensor map using inv(t), though this can only exist if the domain and codomain are isomorphic, which can e.g. be checked as fuse(codomain(t)) == fuse(domain(t)). If the inverse is composed with another tensor t2, we can use the syntax t1\t2 or t2/t1. However, this syntax also accepts instances t1 whose domain and codomain are not isomorphic, and then amounts to pinv(t1), the Moore-Penrose pseudoinverse. This, however, is only really justified as minimizing the least squares problem if InnerProductStyle(t) <: EuclideanProduct.

    AbstractTensorMap instances behave themselves as vectors (i.e. they are 𝕜-linear) and so they can be multiplied by scalars and, if they live in the same space, i.e. have the same domain and codomain, they can be added to each other. There is also a zero(t), the additive identity, which produces a zero tensor with the same domain and codomain as t. In addition, TensorMap supports basic Julia methods such as fill! and copyto!, as well as copy(t) to create a copy with independent data. Aside from basic + and * operations, TensorKit.jl reexports a number of efficient in-place methods from LinearAlgebra, such as axpy! (for y ← α * x + y), axpby! (for y ← α * x + β * y), lmul! and rmul! (for y ← α*y and y ← y*α, which is typically the same) and mul!, which can also be used for out-of-place scalar multiplication y ← α*x.

    For t::AbstractTensorMap{S} where InnerProductStyle(S) <: EuclideanProduct, we can compute norm(t), and for two such instances, the inner product dot(t1, t2), provided t1 and t2 have the same domain and codomain. Furthermore, there is normalize(t) and normalize!(t) to return a scaled version of t with unit norm. These operations should also exist for InnerProductStyle(S) <: HasInnerProduct, but require an interface for defining a custom inner product in these spaces. Currently, there is no concrete subtype of HasInnerProduct that is not an EuclideanProduct. In particular, CartesianSpace, ComplexSpace and GradedSpace all have InnerProductStyle(V) <: EuclideanProduct.

    With tensors that have InnerProductStyle(t) <: EuclideanProduct there is associated an adjoint operation, given by adjoint(t) or simply t', such that domain(t') == codomain(t) and codomain(t') == domain(t). Note that for an instance t::TensorMap{S,N₁,N₂}, t' is simply stored in a wrapper called AdjointTensorMap{S,N₂,N₁}, which is another subtype of AbstractTensorMap. This should be mostly unvisible to the user, as all methods should work for this type as well. It can be hard to reason about the index order of t', i.e. index i of t appears in t' at index position j = TensorKit.adjointtensorindex(t, i), where the latter method is typically not necessary and hence unexported. There is also a plural TensorKit.adjointtensorindices to convert multiple indices at once. Note that, because the adjoint interchanges domain and codomain, we have space(t', j) == space(t, i)'.

    AbstractTensorMap instances can furthermore be tested for exact (t1 == t2) or approximate (t1 ≈ t2) equality, though the latter requires that norm can be computed.

    When tensor map instances are endomorphisms, i.e. they have the same domain and codomain, there is a multiplicative identity which can be obtained as one(t) or one!(t), where the latter overwrites the contents of t. The multiplicative identity on a space V can also be obtained using id(A, V) as discussed above, such that for a general homomorphism t′, we have t′ == id(codomain(t′))*t′ == t′*id(domain(t′)). Returning to the case of endomorphisms t, we can compute the trace via tr(t) and exponentiate them using exp(t), or if the contents of t can be destroyed in the process, exp!(t). Furthermore, there are a number of tensor factorizations for both endomorphisms and general homomorphism that we discuss below.

    Finally, there are a number of operations that also belong in this paragraph because of their analogy to common matrix operations. The tensor product of two TensorMap instances t1 and t2 is obtained as t1 ⊗ t2 and results in a new TensorMap with codomain(t1⊗t2) = codomain(t1) ⊗ codomain(t2) and domain(t1⊗t2) = domain(t1) ⊗ domain(t2). If we have two TensorMap{S,N,1} instances t1 and t2 with the same codomain, we can combine them in a way that is analoguous to hcat, i.e. we stack them such that the new tensor catdomain(t1, t2) has also the same codomain, but has a domain which is domain(t1) ⊕ domain(t2). Similarly, if t1 and t2 are of type TensorMap{S,1,N} and have the same domain, the operation catcodomain(t1, t2) results in a new tensor with the same domain and a codomain given by codomain(t1) ⊕ codomain(t2), which is the analogy of vcat. Note that direct sum only makes sense between ElementarySpace objects, i.e. there is no way to give a tensor product meaning to a direct sum of tensor product spaces.

    Time for some more examples:

    julia> t == t + zero(t) == t*id(domain(t)) == id(codomain(t))*ttrue
    julia> t2 = TensorMap(randn, ComplexF64, codomain(t), domain(t));
    julia> dot(t2, t)0.7217945980133003 + 11.970014061902198im
    julia> tr(t2'*t)0.7217945980133011 + 11.970014061902198im
    julia> dot(t2, t) ≈ dot(t', t2')true
    julia> dot(t2, t2)47.91833800573911 + 0.0im
    julia> norm(t2)^247.91833800573911
    julia> t3 = copyto!(similar(t, ComplexF64), t);ERROR: MethodError: no method matching copyto!(::TensorMap{GradedSpace{SU2Irrep, TensorKit.SortedVectorDict{SU2Irrep, Int64}}, 2, 2, SU2Irrep, TensorKit.SortedVectorDict{SU2Irrep, Matrix{ComplexF64}}, FusionTree{SU2Irrep, 2, 0, 1, Nothing}, FusionTree{SU2Irrep, 2, 0, 1, Nothing}}, ::TensorMap{GradedSpace{SU2Irrep, TensorKit.SortedVectorDict{SU2Irrep, Int64}}, 2, 2, SU2Irrep, TensorKit.SortedVectorDict{SU2Irrep, Matrix{Float64}}, FusionTree{SU2Irrep, 2, 0, 1, Nothing}, FusionTree{SU2Irrep, 2, 0, 1, Nothing}}) Closest candidates are: copyto!(!Matched::AbstractArray, ::Any) @@ -690,44 +690,44 @@ p1::NTuple{N₁′,Int}, p2::NTuple{N₂′,Int})

    and

    permute(t::AbstractTensorMap{S,N₁,N₂},
             p1::NTuple{N₁′,Int}, p2::NTuple{N₂′,Int}; copy = false)

    both of which return an instance of AbstractTensorMap{S,N₁′,N₂′}.

    In these methods, p1 and p2 specify which of the original tensor indices ranging from 1 to N₁+N₂ make up the new codomain (with N₁′ spaces) and new domain (with N₂′ spaces). Hence, (p1..., p2...) should be a valid permutation of 1:(N₁+N₂). Note that, throughout TensorKit.jl, permutations are always specified using tuples of Ints, for reasons of type stability. For braid, we also need to specify levels or depths for each of the indices of the original tensor, which determine whether indices will braid over or underneath each other (use the braiding or its inverse). We refer to the section on manipulating fusion trees for more details.

    When BraidingStyle(sectortype(t)) isa SymmetricBraiding, we can use the simpler interface of permute, which does not require the argument levels. permute accepts a keyword argument copy. When copy == true, the result will be a tensor with newly allocated data that can independently be modified from that of the input tensor t. When copy takes the default value false, permute can try to return the result in a way that it shares its data with the input tensor t, though this is only possible in specific cases (e.g. when sectortype(S) == Trivial and (p1..., p2...) = (1:(N₁+N₂)...)).

    Both braid and permute come in a version where the result is stored in an already existing tensor, i.e. braid!(tdst, tsrc, levels, p1, p2) and permute!(tdst, tsrc, p1, p2).

    Another operation that belongs und index manipulations is taking the transpose of a tensor, i.e. LinearAlgebra.transpose(t) and LinearAlgebra.transpose!(tdst, tsrc), both of which are reexported by TensorKit.jl. Note that transpose(t) is not simply equal to reshuffling domain and codomain with braid(t, (1:(N₁+N₂)...), reverse(domainind(tsrc)), reverse(codomainind(tsrc)))). Indeed, the graphical representation (where we draw the codomain and domain as a single object), makes clear that this introduces an additional (inverse) twist, which is then compensated in the transpose implementation.

    transpose

    In categorical language, the reason for this extra twist is that we use the left coevaluation $η$, but the right evaluation $\tilde{ϵ}$, when repartitioning the indices between domain and codomain.

    There are a number of other index related manipulations. We can apply a twist (or inverse twist) to one of the tensor map indices via twist(t, i; inv = false) or twist!(t, i; inv = false). Note that the latter method does not store the result in a new destination tensor, but just modifies the tensor t in place. Twisting several indices simultaneously can be obtained by using the defining property

    $θ_{V⊗W} = τ_{W,V} ∘ (θ_W ⊗ θ_V) ∘ τ_{V,W} = (θ_V ⊗ θ_W) ∘ τ_{W,V} ∘ τ_{V,W}.$

    but is currently not implemented explicitly.

    For all sector types I with BraidingStyle(I) == Bosonic(), all twists are 1 and thus have no effect. Let us start with some examples, in which we illustrate that, albeit permute might act highly non-trivial on the fusion trees and on the corresponding data, after conversion to a regular Array (when possible), it just acts like permutedims

    julia> domain(t) → codomain(t)(Rep[SU₂](0=>2, 1=>1) ⊗ Rep[SU₂](0=>2, 1=>1)) ← (Rep[SU₂](0=>1, 1=>1) ⊗ Rep[SU₂](0=>1, 1=>1)')
    julia> ta = convert(Array, t);
    julia> t′ = permute(t, (1,2,3,4));
    julia> domain(t′) → codomain(t′)(Rep[SU₂](0=>2, 1=>1) ⊗ Rep[SU₂](0=>2, 1=>1) ⊗ Rep[SU₂](0=>1, 1=>1)' ⊗ Rep[SU₂](0=>1, 1=>1)) ← ProductSpace{GradedSpace{SU2Irrep, TensorKit.SortedVectorDict{SU2Irrep, Int64}}, 0}()
    julia> convert(Array, t′) ≈ tatrue
    julia> t′′ = permute(t, (4,2,3),(1,));
    julia> domain(t′′) → codomain(t′′)(Rep[SU₂](0=>1, 1=>1) ⊗ Rep[SU₂](0=>2, 1=>1) ⊗ Rep[SU₂](0=>1, 1=>1)') ← Rep[SU₂](0=>2, 1=>1)'
    julia> convert(Array, t′′) ≈ permutedims(ta, (4,2,3,1))true
    julia> mTensorMap(Rep[SU₂](0=>2, 1=>1) ← Rep[SU₂](0=>1, 1=>1)): * Data for fusiontree FusionTree{Irrep[SU₂]}((0,), 0, (false,), ()) ← FusionTree{Irrep[SU₂]}((0,), 0, (false,), ()): - -0.4804035342466387 - -1.7232087396823772 + -0.5677419546332747 + -1.7628536678287654 * Data for fusiontree FusionTree{Irrep[SU₂]}((1,), 1, (false,), ()) ← FusionTree{Irrep[SU₂]}((1,), 1, (false,), ()): - 0.808832372309538
    julia> transpose(m)TensorMap(Rep[SU₂](0=>1, 1=>1)' ← Rep[SU₂](0=>2, 1=>1)'): + -0.23838780405371376
    julia> transpose(m)TensorMap(Rep[SU₂](0=>1, 1=>1)' ← Rep[SU₂](0=>2, 1=>1)'): * Data for fusiontree FusionTree{Irrep[SU₂]}((0,), 0, (true,), ()) ← FusionTree{Irrep[SU₂]}((0,), 0, (true,), ()): - -0.4804035342466387 -1.7232087396823772 + -0.5677419546332747 -1.7628536678287654 * Data for fusiontree FusionTree{Irrep[SU₂]}((1,), 1, (true,), ()) ← FusionTree{Irrep[SU₂]}((1,), 1, (true,), ()): - 0.8088323723095379
    julia> convert(Array, transpose(t)) ≈ permutedims(ta,(4,3,2,1))true
    julia> dot(t2, t) ≈ dot(transpose(t2), transpose(t))true
    julia> transpose(transpose(t)) ≈ ttrue
    julia> twist(t, 3) ≈ ttrue
    julia> # as twist acts trivially for + -0.23838780405371374
    julia> convert(Array, transpose(t)) ≈ permutedims(ta,(4,3,2,1))true
    julia> dot(t2, t) ≈ dot(transpose(t2), transpose(t))true
    julia> transpose(transpose(t)) ≈ ttrue
    julia> twist(t, 3) ≈ ttrue
    julia> # as twist acts trivially for BraidingStyle(sectortype(t))Bosonic()

    Note that transpose acts like one would expect on a TensorMap{S,1,1}. On a TensorMap{S,N₁,N₂}, because transpose replaces the codomain with the dual of the domain, which has its tensor product operation reversed, this in the end amounts in a complete reversal of all tensor indices when representing it as a plain mutli-dimensional Array. Also, note that we have not defined the conjugation of TensorMap instances. One definition that one could think of is conj(t) = adjoint(transpose(t)). However note that codomain(adjoint(tranpose(t))) == domain(transpose(t)) == dual(codomain(t)) and similarly domain(adjoint(tranpose(t))) == dual(domain(t)), where dual of a ProductSpace is composed of the dual of the ElementarySpace instances, in reverse order of tensor product. This might be very confusing, and as such we leave tensor conjugation undefined. However, note that we have a conjugation syntax within the context of tensor contractions.

    To show the effect of twist, we now consider a type of sector I for which BraidingStyle{I} != Bosonic(). In particular, we use FibonacciAnyon. We cannot convert the resulting TensorMap to an Array, so we have to rely on indirect tests to verify our results.

    julia> V1 = GradedSpace{FibonacciAnyon}(:I=>3,:τ=>2)Vect[FibonacciAnyon](:I=>3, :τ=>2)
    julia> V2 = GradedSpace{FibonacciAnyon}(:I=>2,:τ=>1)Vect[FibonacciAnyon](:I=>2, :τ=>1)
    julia> m = TensorMap(randn, Float32, V1, V2)TensorMap(Vect[FibonacciAnyon](:I=>3, :τ=>2) ← Vect[FibonacciAnyon](:I=>2, :τ=>1)): * Data for fusiontree FusionTree{FibonacciAnyon}((:I,), :I, (false,), ()) ← FusionTree{FibonacciAnyon}((:I,), :I, (false,), ()): - -1.8741488f0 + 0.0f0im -0.5531331f0 + 0.0f0im - -1.0865633f0 + 0.0f0im -2.8764462f0 + 0.0f0im - 1.2639742f0 + 0.0f0im 1.1575584f0 + 0.0f0im + 0.55932474f0 + 0.0f0im -0.5207317f0 + 0.0f0im + -0.8873631f0 + 0.0f0im -0.039305367f0 + 0.0f0im + -0.09340267f0 + 0.0f0im -0.8626384f0 + 0.0f0im * Data for fusiontree FusionTree{FibonacciAnyon}((:τ,), :τ, (false,), ()) ← FusionTree{FibonacciAnyon}((:τ,), :τ, (false,), ()): - -2.5934753f0 + 0.0f0im - 2.1567361f0 + 0.0f0im
    julia> transpose(m)TensorMap(Vect[FibonacciAnyon](:I=>2, :τ=>1)' ← Vect[FibonacciAnyon](:I=>3, :τ=>2)'): + -0.64308095f0 + 0.0f0im + 0.676147f0 + 0.0f0im
    julia> transpose(m)TensorMap(Vect[FibonacciAnyon](:I=>2, :τ=>1)' ← Vect[FibonacciAnyon](:I=>3, :τ=>2)'): * Data for fusiontree FusionTree{FibonacciAnyon}((:I,), :I, (true,), ()) ← FusionTree{FibonacciAnyon}((:I,), :I, (true,), ()): - -1.8741488f0 + 0.0f0im -1.0865633f0 + 0.0f0im 1.2639742f0 + 0.0f0im - -0.5531331f0 + 0.0f0im -2.8764462f0 + 0.0f0im 1.1575584f0 + 0.0f0im + 0.55932474f0 + 0.0f0im -0.8873631f0 + 0.0f0im -0.09340267f0 + 0.0f0im + -0.5207317f0 + 0.0f0im -0.039305367f0 + 0.0f0im -0.8626384f0 + 0.0f0im * Data for fusiontree FusionTree{FibonacciAnyon}((:τ,), :τ, (true,), ()) ← FusionTree{FibonacciAnyon}((:τ,), :τ, (true,), ()): - -2.5934753f0 + 0.0f0im 2.1567361f0 + 0.0f0im
    julia> twist(braid(m, (1,2), (2,), (1,)), 1)ERROR: AssertionError: length(levels) == numind(t)
    julia> t1 = TensorMap(randn, V1*V2', V2*V1);
    julia> t2 = TensorMap(randn, ComplexF64, V1*V2', V2*V1);
    julia> dot(t1, t2) ≈ dot(transpose(t1), transpose(t2))true
    julia> transpose(transpose(t1)) ≈ t1true

    A final operation that one might expect in this section is to fuse or join indices, and its inverse, to split a given index into two or more indices. For a plain tensor (i.e. with sectortype(t) == Trivial) amount to the equivalent of reshape on the multidimensional data. However, this represents only one possibility, as there is no canonically unique way to embed the tensor product of two spaces V₁ ⊗ V₂ in a new space V = fuse(V₁⊗V₂). Such a mapping can always be accompagnied by a basis transform. However, one particular choice is created by the function isomorphism, or for EuclideanProduct spaces, unitary. Hence, we can join or fuse two indices of a tensor by first constructing u = unitary(fuse(space(t, i) ⊗ space(t, j)), space(t, i) ⊗ space(t, j)) and then contracting this map with indices i and j of t, as explained in the section on contracting tensors. Note, however, that a typical algorithm is not expected to often need to fuse and split indices, as e.g. tensor factorizations can easily be applied without needing to reshape or fuse indices first, as explained in the next section.

    Tensor factorizations

    Eigenvalue decomposition

    As tensors are linear maps, they have various kinds of factorizations. Endomorphism, i.e. tensor maps t with codomain(t) == domain(t), have an eigenvalue decomposition. For this, we overload both LinearAlgebra.eigen(t; kwargs...) and LinearAlgebra.eigen!(t; kwargs...), where the latter destroys t in the process. The keyword arguments are the same that are accepted by LinearAlgebra.eigen(!) for matrices. The result is returned as D, V = eigen(t), such that t*V ≈ V*D. For given t::TensorMap{S,N,N}, V is a TensorMap{S,N,1}, whose codomain corresponds to that of t, but whose domain is a single space S (or more correctly a ProductSpace{S,1}), that corresponds to fuse(codomain(t)). The eigenvalues are encoded in D, a TensorMap{S,1,1}, whose domain and codomain correspond to the domain of V. Indeed, we cannot reasonably associate a tensor product structure with the different eigenvalues. Note that D stores the eigenvalues on the diagonal of a (collection of) DenseMatrix instance(s), as there is currently no dedicated DiagonalTensorMap or diagonal storage support.

    We also define LinearAlgebra.ishermitian(t), which can only return true for instances of AbstractEuclideanTensorMap. In all other cases, as the inner product is not defined, there is no notion of hermiticity (i.e. we are not working in a -category). For instances of EuclideanTensorMap, we also define and export the routines eigh and eigh!, which compute the eigenvalue decomposition under the guarantee (not checked) that the map is hermitian. Hence, eigenvalues will be real and V will be unitary with eltype(V) == eltype(t). We also define and export eig and eig!, which similarly assume that the TensorMap is not hermitian (hence this does not require EuclideanTensorMap), and always returns complex values eigenvalues and eigenvectors. Like for matrices, LinearAlgebra.eigen is type unstable and checks hermiticity at run-time, then falling back to either eig or eigh.

    Orthogonal factorizations

    Other factorizations that are provided by TensorKit.jl are orthogonal or unitary in nature, and thus always require a AbstractEuclideanTensorMap. However, they don't require equal domain and codomain. Let us first discuss the singular value decomposition, for which we define and export the methods tsvd and tsvd! (where as always, the latter destroys the input).

    U, Σ, Vʰ, ϵ = tsvd(t; trunc = notrunc(), p::Real = 2,
    + -0.64308095f0 + 0.0f0im  0.676147f0 + 0.0f0im
    julia> twist(braid(m, (1,2), (2,), (1,)), 1)ERROR: AssertionError: length(levels) == numind(t)
    julia> t1 = TensorMap(randn, V1*V2', V2*V1);
    julia> t2 = TensorMap(randn, ComplexF64, V1*V2', V2*V1);
    julia> dot(t1, t2) ≈ dot(transpose(t1), transpose(t2))true
    julia> transpose(transpose(t1)) ≈ t1true

    A final operation that one might expect in this section is to fuse or join indices, and its inverse, to split a given index into two or more indices. For a plain tensor (i.e. with sectortype(t) == Trivial) amount to the equivalent of reshape on the multidimensional data. However, this represents only one possibility, as there is no canonically unique way to embed the tensor product of two spaces V₁ ⊗ V₂ in a new space V = fuse(V₁⊗V₂). Such a mapping can always be accompagnied by a basis transform. However, one particular choice is created by the function isomorphism, or for EuclideanProduct spaces, unitary. Hence, we can join or fuse two indices of a tensor by first constructing u = unitary(fuse(space(t, i) ⊗ space(t, j)), space(t, i) ⊗ space(t, j)) and then contracting this map with indices i and j of t, as explained in the section on contracting tensors. Note, however, that a typical algorithm is not expected to often need to fuse and split indices, as e.g. tensor factorizations can easily be applied without needing to reshape or fuse indices first, as explained in the next section.

    Tensor factorizations

    Eigenvalue decomposition

    As tensors are linear maps, they have various kinds of factorizations. Endomorphism, i.e. tensor maps t with codomain(t) == domain(t), have an eigenvalue decomposition. For this, we overload both LinearAlgebra.eigen(t; kwargs...) and LinearAlgebra.eigen!(t; kwargs...), where the latter destroys t in the process. The keyword arguments are the same that are accepted by LinearAlgebra.eigen(!) for matrices. The result is returned as D, V = eigen(t), such that t*V ≈ V*D. For given t::TensorMap{S,N,N}, V is a TensorMap{S,N,1}, whose codomain corresponds to that of t, but whose domain is a single space S (or more correctly a ProductSpace{S,1}), that corresponds to fuse(codomain(t)). The eigenvalues are encoded in D, a TensorMap{S,1,1}, whose domain and codomain correspond to the domain of V. Indeed, we cannot reasonably associate a tensor product structure with the different eigenvalues. Note that D stores the eigenvalues on the diagonal of a (collection of) DenseMatrix instance(s), as there is currently no dedicated DiagonalTensorMap or diagonal storage support.

    We also define LinearAlgebra.ishermitian(t), which can only return true for instances of AbstractEuclideanTensorMap. In all other cases, as the inner product is not defined, there is no notion of hermiticity (i.e. we are not working in a -category). For instances of EuclideanTensorMap, we also define and export the routines eigh and eigh!, which compute the eigenvalue decomposition under the guarantee (not checked) that the map is hermitian. Hence, eigenvalues will be real and V will be unitary with eltype(V) == eltype(t). We also define and export eig and eig!, which similarly assume that the TensorMap is not hermitian (hence this does not require EuclideanTensorMap), and always returns complex values eigenvalues and eigenvectors. Like for matrices, LinearAlgebra.eigen is type unstable and checks hermiticity at run-time, then falling back to either eig or eigh.

    Orthogonal factorizations

    Other factorizations that are provided by TensorKit.jl are orthogonal or unitary in nature, and thus always require a AbstractEuclideanTensorMap. However, they don't require equal domain and codomain. Let us first discuss the singular value decomposition, for which we define and export the methods tsvd and tsvd! (where as always, the latter destroys the input).

    U, Σ, Vʰ, ϵ = tsvd(t; trunc = notrunc(), p::Real = 2,
                             alg::OrthogonalFactorizationAlgorithm = SDD())

    This computes a (possibly truncated) singular value decomposition of t::TensorMap{S,N₁,N₂} (with InnerProductStyle(t)<:EuclideanProduct), such that norm(t - U*Σ*Vʰ) ≈ ϵ, where U::TensorMap{S,N₁,1}, S::TensorMap{S,1,1}, Vʰ::TensorMap{S,1,N₂} and ϵ::Real. U is an isometry, i.e. U'*U approximates the identity, whereas U*U' is an idempotent (squares to itself). The same holds for adjoint(Vʰ). The domain of U equals the domain and codomain of Σ and the codomain of . In the case of trunc = notrunc() (default value, see below), this space is given by min(fuse(codomain(t)), fuse(domain(t))). The singular values are contained in Σ and are stored on the diagonal of a (collection of) DenseMatrix instance(s), similar to the eigenvalues before.

    The keyword argument trunc provides a way to control the truncation, and is connected to the keyword argument p. The default value notrunc() implies no truncation, and thus ϵ = 0. Other valid options are

    Furthermore, the alg keyword can be either SVD() or SDD() (default), which corresponds to two different algorithms in LAPACK to compute singular value decompositions. The default value SDD() uses a divide-and-conquer algorithm and is typically the fastest, but can loose some accuracy. The SVD() method uses a QR-iteration scheme and can be more accurate, but is typically slower. Since Julia 1.3, these two algorithms are also available in the LinearAlgebra standard library, where they are specified as LinearAlgebra.DivideAndConquer() and LinearAlgebra.QRIteration().

    Note that we defined the new method tsvd (truncated or tensor singular value decomposition), rather than overloading LinearAlgebra.svd. We (will) also support LinearAlgebra.svd(t) as alternative for tsvd(t; trunc = notrunc()), but note that the return values are then given by U, Σ, V = svd(t) with V = adjoint(Vʰ).

    We also define the following pair of orthogonal factorization algorithms, which are useful when one is not interested in truncating a tensor or knowing the singular values, but only in its image or coimage.

    Furthermore, we can compute an orthonormal basis for the orthogonal complement of the image and of the co-image (i.e. the kernel) with the following methods:

    Note that the methods leftorth, rightorth, leftnull and rightnull also come in a form with exclamation mark, i.e. leftorth!, rightorth!, leftnull! and rightnull!, which destroy the input tensor t.

    Factorizations for custom index bipartions

    Finally, note that each of the factorizations take a single argument, the tensor map t, and a number of keyword arguments. They perform the factorization according to the given codomain and domain of the tensor map. In many cases, we want to perform the factorization according to a different bipartition of the indices. When BraidingStyle(sectortype(t)) isa SymmetricBraiding, we can immediately specify an alternative bipartition of the indices of t in all of these methods, in the form

    factorize(t::AbstracTensorMap, pleft::NTuple{N₁′,Int}, pright::NTuple{N₂′,Int}; kwargs...)

    where pleft will be the indices in the codomain of the new tensor map, and pright the indices of the domain. Here, factorize is any of the methods LinearAlgebra.eigen, eig, eigh, tsvd, LinearAlgebra.svd, leftorth, rightorth, leftnull and rightnull. This signature does not allow for the exclamation mark, because it amounts to

    factorize!(permute(t, pleft, pright; copy = true); kwargs...)

    where permute was introduced and discussed in the previous section. When the braiding is not symmetric, the user should manually apply braid to bring the tensor map in proper form before performing the factorization.

    Some examples to conclude this section

    julia> V1 = SU₂Space(0=>2,1/2=>1)Rep[SU₂](0=>2, 1/2=>1)
    julia> V2 = SU₂Space(0=>1,1/2=>1,1=>1)Rep[SU₂](0=>1, 1/2=>1, 1=>1)
    julia> t = TensorMap(randn, V1 ⊗ V1, V2);
    julia> U, S, W = tsvd(t);
    julia> t ≈ U * S * Wtrue
    julia> D, V = eigh(t'*t);
    julia> D ≈ S*Strue
    julia> U'*U ≈ id(domain(U))true
    julia> STensorMap(Rep[SU₂](0=>1, 1/2=>1, 1=>1) ← Rep[SU₂](0=>1, 1/2=>1, 1=>1)): * Data for fusiontree FusionTree{Irrep[SU₂]}((0,), 0, (false,), ()) ← FusionTree{Irrep[SU₂]}((0,), 0, (false,), ()): - 1.5420788676203365 + 1.6169351297726544 * Data for fusiontree FusionTree{Irrep[SU₂]}((1/2,), 1/2, (false,), ()) ← FusionTree{Irrep[SU₂]}((1/2,), 1/2, (false,), ()): - 1.1567269847294306 + 2.073795142309855 * Data for fusiontree FusionTree{Irrep[SU₂]}((1,), 1, (false,), ()) ← FusionTree{Irrep[SU₂]}((1,), 1, (false,), ()): - 1.2799008389772437
    julia> Q, R = leftorth(t; alg = Polar());
    julia> isposdef(R)true
    julia> Q ≈ U*Wtrue
    julia> R ≈ W'*S*Wtrue
    julia> U2, S2, W2, ε = tsvd(t; trunc = truncspace(V1));
    julia> W2*W2' ≈ id(codomain(W2))true
    julia> S2TensorMap(Rep[SU₂](0=>1, 1/2=>1) ← Rep[SU₂](0=>1, 1/2=>1)): + 0.07542698080487421
    julia> Q, R = leftorth(t; alg = Polar());
    julia> isposdef(R)true
    julia> Q ≈ U*Wtrue
    julia> R ≈ W'*S*Wtrue
    julia> U2, S2, W2, ε = tsvd(t; trunc = truncspace(V1));
    julia> W2*W2' ≈ id(codomain(W2))true
    julia> S2TensorMap(Rep[SU₂](0=>1, 1/2=>1) ← Rep[SU₂](0=>1, 1/2=>1)): * Data for fusiontree FusionTree{Irrep[SU₂]}((0,), 0, (false,), ()) ← FusionTree{Irrep[SU₂]}((0,), 0, (false,), ()): - 1.5420788676203365 + 1.6169351297726544 * Data for fusiontree FusionTree{Irrep[SU₂]}((1/2,), 1/2, (false,), ()) ← FusionTree{Irrep[SU₂]}((1/2,), 1/2, (false,), ()): - 1.1567269847294306
    julia> ε ≈ norm(block(S, Irrep[SU₂](1)))*sqrt(dim(Irrep[SU₂](1)))true
    julia> L, Q = rightorth(t, (1,), (2,3));
    julia> codomain(L), domain(L), domain(Q)(ProductSpace(Rep[SU₂](0=>2, 1/2=>1)), ProductSpace(Rep[SU₂](0=>2, 1/2=>1)), (Rep[SU₂](0=>2, 1/2=>1)' ⊗ Rep[SU₂](0=>1, 1/2=>1, 1=>1)))
    julia> Q*Q'TensorMap(Rep[SU₂](0=>2, 1/2=>1) ← Rep[SU₂](0=>2, 1/2=>1)): + 2.073795142309855
    julia> ε ≈ norm(block(S, Irrep[SU₂](1)))*sqrt(dim(Irrep[SU₂](1)))true
    julia> L, Q = rightorth(t, (1,), (2,3));
    julia> codomain(L), domain(L), domain(Q)(ProductSpace(Rep[SU₂](0=>2, 1/2=>1)), ProductSpace(Rep[SU₂](0=>2, 1/2=>1)), (Rep[SU₂](0=>2, 1/2=>1)' ⊗ Rep[SU₂](0=>1, 1/2=>1, 1=>1)))
    julia> Q*Q'TensorMap(Rep[SU₂](0=>2, 1/2=>1) ← Rep[SU₂](0=>2, 1/2=>1)): * Data for fusiontree FusionTree{Irrep[SU₂]}((0,), 0, (false,), ()) ← FusionTree{Irrep[SU₂]}((0,), 0, (false,), ()): - 0.9999999999999997 4.69897427420827e-17 - 4.69897427420827e-17 0.9999999999999992 + 1.0000000000000004 -2.182341264050547e-16 + -2.182341264050547e-16 0.9999999999999997 * Data for fusiontree FusionTree{Irrep[SU₂]}((1/2,), 1/2, (false,), ()) ← FusionTree{Irrep[SU₂]}((1/2,), 1/2, (false,), ()): - 0.9999999999999998
    julia> P = Q'*Q;
    julia> P ≈ P*Ptrue
    julia> t′ = permute(t, (1,), (2,3));
    julia> t′ ≈ t′ * Ptrue

    Bosonic tensor contractions and tensor networks

    One of the most important operation with tensor maps is to compose them, more generally known as contracting them. As mentioned in the section on category theory, a typical composition of maps in a ribbon category can graphically be represented as a planar arrangement of the morphisms (i.e. tensor maps, boxes with lines eminating from top and bottom, corresponding to source and target, i.e. domain and codomain), where the lines connecting the source and targets of the different morphisms should be thought of as ribbons, that can braid over or underneath each other, and that can twist. Technically, we can embed this diagram in $ℝ × [0,1]$ and attach all the unconnected line endings corresponding objects in the source at some position $(x,0)$ for $x∈ℝ$, and all line endings corresponding to objects in the target at some position $(x,1)$. The resulting morphism is then invariant under what is known as framed three-dimensional isotopy, i.e. three-dimensional rearrangements of the morphism that respect the rules of boxes connected by ribbons whose open endings are kept fixed. Such a two-dimensional diagram cannot easily be encoded in a single line of code.

    However, things simplify when the braiding is symmetric (such that over- and under- crossings become equivalent, i.e. just crossings), and when twists, i.e. self-crossings in this case, are trivial. This amounts to BraidingStyle(I) == Bosonic() in the language of TensorKit.jl, and is true for any subcategory of $\mathbf{Vect}$, i.e. ordinary tensors, possibly with some symmetry constraint. The case of $\mathbf{SVect}$ and its subcategories, and more general categories, are discussed below.

    In the case of triival twists, we can deform the diagram such that we first combine every morphism with a number of coevaluations $η$ so as to represent it as a tensor, i.e. with a trivial domain. We can then rearrange the morphism to be all ligned up horizontally, where the original morphism compositions are now being performed by evaluations $ϵ$. This process will generate a number of crossings and twists, where the latter can be omitted because they act trivially. Similarly, double crossings can also be omitted. As a consequence, the diagram, or the morphism it represents, is completely specified by the tensors it is composed of, and which indices between the different tensors are connect, via the evaluation $ϵ$, and which indices make up the source and target of the resulting morphism. If we also compose the resulting morphisms with coevaluations so that it has a trivial domain, we just have one type of unconnected lines, henceforth called open indices. We sketch such a rearrangement in the following picture

    tensor unitary

    Hence, we can now specify such a tensor diagram, henceforth called a tensor contraction or also tensor network, using a one-dimensional syntax that mimicks abstract index notation and specifies which indices are connected by the evaluation map using Einstein's summation conventation. Indeed, for BraidingStyle(I) == Bosonic(), such a tensor contraction can take the same format as if all tensors were just multi-dimensional arrays. For this, we rely on the interface provided by the package TensorOperations.jl.

    The above picture would be encoded as

    @tensor E[a,b,c,d,e] := A[v,w,d,x]*B[y,z,c,x]*C[v,e,y,b]*D[a,w,z]

    or

    @tensor E[:] := A[1,2,-4,3]*B[4,5,-3,3]*C[1,-5,4,-2]*D[-1,2,5]

    where the latter syntax is known as NCON-style, and labels the unconnected or outgoing indices with negative integers, and the contracted indices with positive integers.

    A number of remarks are in order. TensorOperations.jl accepts both integers and any valid variable name as dummy label for indices, and everything in between [ ] is not resolved in the current context but interpreted as a dummy label. Here, we label the indices of a TensorMap, like A::TensorMap{S,N₁,N₂}, in a linear fashion, where the first position corresponds to the first space in codomain(A), and so forth, up to position N₁. Index N₁+1then corresponds to the first space in domain(A). However, because we have applied the coevaluation $η$, it actually corresponds to the corresponding dual space, in accordance with the interface of space(A, i) that we introduced above, and as indiated by the dotted box around $A$ in the above picture. The same holds for the other tensor maps. Note that our convention also requires that we braid indices that we brought from the domain to the codomain, and so this is only unambiguous for a symmetric braiding, where there is a unique way to permute the indices.

    With the current syntax, we create a new object E because we use the definition operator :=. Furthermore, with the current syntax, it will be a Tensor, i.e. it will have a trivial domain, and correspond to the dotted box in the picture above, rather than the actual morphism E. We can also directly define E with the correct codomain and domain by rather using

    @tensor E[a b c;d e] := A[v,w,d,x]*B[y,z,c,x]*C[v,e,y,b]*D[a,w,z]

    or

    @tensor E[(a,b,c);(d,e)] := A[v,w,d,x]*B[y,z,c,x]*C[v,e,y,b]*D[a,w,z]

    where the latter syntax can also be used when the codomain is empty. When using the assignment operator =, the TensorMap E is assumed to exist and the contents will be written to the currently allocated memory. Note that for existing tensors, both on the left hand side and right hand side, trying to specify the indices in the domain and the codomain seperately using the above syntax, has no effect, as the bipartition of indices are already fixed by the existing object. Hence, if E has been created by the previous line of code, all of the following lines are now equivalent

    @tensor E[(a,b,c);(d,e)] = A[v,w,d,x]*B[y,z,c,x]*C[v,e,y,b]*D[a,w,z]
    + 1.0000000000000002
    julia> P = Q'*Q;
    julia> P ≈ P*Ptrue
    julia> t′ = permute(t, (1,), (2,3));
    julia> t′ ≈ t′ * Ptrue

    Bosonic tensor contractions and tensor networks

    One of the most important operation with tensor maps is to compose them, more generally known as contracting them. As mentioned in the section on category theory, a typical composition of maps in a ribbon category can graphically be represented as a planar arrangement of the morphisms (i.e. tensor maps, boxes with lines eminating from top and bottom, corresponding to source and target, i.e. domain and codomain), where the lines connecting the source and targets of the different morphisms should be thought of as ribbons, that can braid over or underneath each other, and that can twist. Technically, we can embed this diagram in $ℝ × [0,1]$ and attach all the unconnected line endings corresponding objects in the source at some position $(x,0)$ for $x∈ℝ$, and all line endings corresponding to objects in the target at some position $(x,1)$. The resulting morphism is then invariant under what is known as framed three-dimensional isotopy, i.e. three-dimensional rearrangements of the morphism that respect the rules of boxes connected by ribbons whose open endings are kept fixed. Such a two-dimensional diagram cannot easily be encoded in a single line of code.

    However, things simplify when the braiding is symmetric (such that over- and under- crossings become equivalent, i.e. just crossings), and when twists, i.e. self-crossings in this case, are trivial. This amounts to BraidingStyle(I) == Bosonic() in the language of TensorKit.jl, and is true for any subcategory of $\mathbf{Vect}$, i.e. ordinary tensors, possibly with some symmetry constraint. The case of $\mathbf{SVect}$ and its subcategories, and more general categories, are discussed below.

    In the case of triival twists, we can deform the diagram such that we first combine every morphism with a number of coevaluations $η$ so as to represent it as a tensor, i.e. with a trivial domain. We can then rearrange the morphism to be all ligned up horizontally, where the original morphism compositions are now being performed by evaluations $ϵ$. This process will generate a number of crossings and twists, where the latter can be omitted because they act trivially. Similarly, double crossings can also be omitted. As a consequence, the diagram, or the morphism it represents, is completely specified by the tensors it is composed of, and which indices between the different tensors are connect, via the evaluation $ϵ$, and which indices make up the source and target of the resulting morphism. If we also compose the resulting morphisms with coevaluations so that it has a trivial domain, we just have one type of unconnected lines, henceforth called open indices. We sketch such a rearrangement in the following picture

    tensor unitary

    Hence, we can now specify such a tensor diagram, henceforth called a tensor contraction or also tensor network, using a one-dimensional syntax that mimicks abstract index notation and specifies which indices are connected by the evaluation map using Einstein's summation conventation. Indeed, for BraidingStyle(I) == Bosonic(), such a tensor contraction can take the same format as if all tensors were just multi-dimensional arrays. For this, we rely on the interface provided by the package TensorOperations.jl.

    The above picture would be encoded as

    @tensor E[a,b,c,d,e] := A[v,w,d,x]*B[y,z,c,x]*C[v,e,y,b]*D[a,w,z]

    or

    @tensor E[:] := A[1,2,-4,3]*B[4,5,-3,3]*C[1,-5,4,-2]*D[-1,2,5]

    where the latter syntax is known as NCON-style, and labels the unconnected or outgoing indices with negative integers, and the contracted indices with positive integers.

    A number of remarks are in order. TensorOperations.jl accepts both integers and any valid variable name as dummy label for indices, and everything in between [ ] is not resolved in the current context but interpreted as a dummy label. Here, we label the indices of a TensorMap, like A::TensorMap{S,N₁,N₂}, in a linear fashion, where the first position corresponds to the first space in codomain(A), and so forth, up to position N₁. Index N₁+1then corresponds to the first space in domain(A). However, because we have applied the coevaluation $η$, it actually corresponds to the corresponding dual space, in accordance with the interface of space(A, i) that we introduced above, and as indiated by the dotted box around $A$ in the above picture. The same holds for the other tensor maps. Note that our convention also requires that we braid indices that we brought from the domain to the codomain, and so this is only unambiguous for a symmetric braiding, where there is a unique way to permute the indices.

    With the current syntax, we create a new object E because we use the definition operator :=. Furthermore, with the current syntax, it will be a Tensor, i.e. it will have a trivial domain, and correspond to the dotted box in the picture above, rather than the actual morphism E. We can also directly define E with the correct codomain and domain by rather using

    @tensor E[a b c;d e] := A[v,w,d,x]*B[y,z,c,x]*C[v,e,y,b]*D[a,w,z]

    or

    @tensor E[(a,b,c);(d,e)] := A[v,w,d,x]*B[y,z,c,x]*C[v,e,y,b]*D[a,w,z]

    where the latter syntax can also be used when the codomain is empty. When using the assignment operator =, the TensorMap E is assumed to exist and the contents will be written to the currently allocated memory. Note that for existing tensors, both on the left hand side and right hand side, trying to specify the indices in the domain and the codomain seperately using the above syntax, has no effect, as the bipartition of indices are already fixed by the existing object. Hence, if E has been created by the previous line of code, all of the following lines are now equivalent

    @tensor E[(a,b,c);(d,e)] = A[v,w,d,x]*B[y,z,c,x]*C[v,e,y,b]*D[a,w,z]
     @tensor E[a,b,c,d,e] = A[v w d;x]*B[(y,z,c);(x,)]*C[v e y; b]*D[a,w,z]
     @tensor E[a b; c d e] = A[v; w d x]*B[y,z,c,x]*C[v,e,y,b]*D[a w;z]

    and none of those will or can change the partition of the indices of E into its codomain and its domain.

    Two final remarks are in order. Firstly, the order of the tensors appearing on the right hand side is irrelevant, as we can reorder them by using the allowed moves of the Penrose graphical calculus, which yields some crossings and a twist. As the latter is trivial, it can be omitted, and we just use the same rules to evaluate the newly ordered tensor network. For the particular case of matrix matrix multiplication, which also captures more general settings by appropriotely combining spaces into a single line, we indeed find

    tensor contraction reorder

    or thus, the following to lines of code yield the same result

    @tensor C[i,j] := B[i,k]*A[k,j]
     @tensor C[i,j] := A[k,j]*B[i,k]

    Reordering of tensors can be used internally by the @tensor macro to evaluate the contraction in a more efficient manner. In particular, the NCON-style of specifying the contraction gives the user control over the order, and there are other macros, such as @tensoropt, that try to automate this process. There is also an @ncon macro and ncon function, an we recommend reading the manual of TensorOperations.jl to learn more about the possibilities and how they work.

    A final remark involves the use of adjoints of tensors. The current framework is such that the user should not be to worried about the actual bipartition into codomain and domain of a given TensorMap instance. Indeed, for factorizations one just specifies the requested bipartition via the factorize(t, pleft, pright) interface, and for tensor contractions the @contract macro figures out the correct manipulations automatically. However, when wanting to use the adjoint of an instance t::TensorMap{S,N₁,N₂}, the resulting adjoint(t) is a AbstractTensorMap{S,N₂,N₁} and one need to know the values of N₁ and N₂ to know exactly where the ith index of t will end up in adjoint(t), and hence to know and understand the index order of t'. Within the @tensor macro, one can instead use conj() on the whole index expression so as to be able to use the original index ordering of t. Indeed, for matrices of thus, TensorMap{S,1,1} instances, this yields exactly the equivalence one expects, namely equivalence between the following to expressions.

    @tensor C[i,j] := B'[i,k]*A[k,j]
    -@tensor C[i,j] := conj(B[k,i])*A[k,j]

    For e.g. an instance A::TensorMap{S,3,2}, the following two syntaxes have the same effect within an @tensor expression: conj(A[a,b,c,d,e]) and A'[d,e,a,b,c].

    Some examples:

    Fermionic tensor contractions

    TODO

    Anyonic tensor contractions

    TODO

    +@tensor C[i,j] := conj(B[k,i])*A[k,j]

    For e.g. an instance A::TensorMap{S,3,2}, the following two syntaxes have the same effect within an @tensor expression: conj(A[a,b,c,d,e]) and A'[d,e,a,b,c].

    Some examples:

    Fermionic tensor contractions

    TODO

    Anyonic tensor contractions

    TODO

    diff --git a/dev/man/tutorial/index.html b/dev/man/tutorial/index.html index 5f3e4847..7d607c19 100644 --- a/dev/man/tutorial/index.html +++ b/dev/man/tutorial/index.html @@ -1,493 +1,493 @@ Tutorial · TensorKit.jl

    Tutorial

    Before discussing at length all aspects of this package, both its usage and implementation, we start with a short tutorial to sketch the main capabilities. Thereto, we start by loading TensorKit.jl

    julia> using TensorKit

    Cartesian tensors

    The most important objects in TensorKit.jl are tensors, which we now create with random (normally distributed) entries in the following manner

    julia> A = Tensor(randn, ℝ^3 ⊗ ℝ^2 ⊗ ℝ^4)TensorMap((ℝ^3 ⊗ ℝ^2 ⊗ ℝ^4) ← ProductSpace{CartesianSpace, 0}()):
     [:, :, 1] =
    -  0.2210365849069235  -1.7302843974122823
    - -1.354396000596351    0.5625290150311519
    - -1.9015345843488605  -0.4217291675237574
    + -0.6263453764951161   -0.8856201818147352
    + -1.4695568646379336   -1.2414139114562541
    +  0.15060631101099436   0.7952249883922199
     
     [:, :, 2] =
    -  1.4446767974878814  -0.5344349066926972
    -  0.4011412987164587   0.6545701572938563
    - -0.5854057361941739  -0.39705919950306773
    +  1.8472385553363047   -0.8572954041956529
    + -1.338778966753932    -0.963723800253341
    +  0.39637190126323335  -0.4254736775609254
     
     [:, :, 3] =
    - 0.05116384353412213  -0.7203656988574674
    - 1.1079157004350477    0.7491133391309124
    - 0.2794243232604638    1.0171002760576229
    +  0.156942099334832   -0.8571782550886548
    +  0.3829592474389556   0.7188486586656138
    + -1.0121852621242975  -1.4540150011477664
     
     [:, :, 4] =
    -  1.6088525905822937    0.07778621198669779
    - -0.11996279740997542  -1.0473991500145496
    -  0.6605616797500655   -0.8818812373603137

    Note that we entered the tensor size not as plain dimensions, by specifying the vector space associated with these tensor indices, in this case ℝ^n, which can be obtained by typing \bbR+TAB. The tensor then lives in the tensor product of the different spaces, which we can obtain by typing (i.e. \otimes+TAB), although for simplicity also the usual multiplication sign * does the job. Note also that A is printed as an instance of a parametric type TensorMap, which we will discuss below and contains Tensor.

    Briefly sidetracking into the nature of ℝ^n:

    julia> V = ℝ^3ℝ^3
    julia> typeof(V)CartesianSpace
    julia> V == CartesianSpace(3)true
    julia> supertype(CartesianSpace)ElementarySpace{ℝ}
    julia> supertype(ElementarySpace)VectorSpace

    i.e. ℝ^n can also be created without Unicode using the longer syntax CartesianSpace(n). It is subtype of ElementarySpace{ℝ}, with a standard (Euclidean) inner product over the real numbers. Furthermore,

    julia> W = ℝ^3 ⊗ ℝ^2 ⊗ ℝ^4(ℝ^3 ⊗ ℝ^2 ⊗ ℝ^4)
    julia> typeof(W)ProductSpace{CartesianSpace, 3}
    julia> supertype(ProductSpace)CompositeSpace
    julia> supertype(CompositeSpace)VectorSpace

    i.e. the tensor product of a number of CartesianSpaces is some generic parametric ProductSpace type, specifically ProductSpace{CartesianSpace,N} for the tensor product of N instances of CartesianSpace.

    Tensors are itself vectors (but not Vectors), so we can compute linear combinations, provided they live in the same space.

    julia> B = Tensor(randn, ℝ^3 * ℝ^2 * ℝ^4);
    julia> C = 0.5*A + 2.5*BTensorMap((ℝ^3 ⊗ ℝ^2 ⊗ ℝ^4) ← ProductSpace{CartesianSpace, 0}()): + 0.4871612767655687 0.7916663983867505 + -0.47376751248989013 -1.0388733926202567 + 0.12047129978209313 0.5191235432360563

    Note that we entered the tensor size not as plain dimensions, by specifying the vector space associated with these tensor indices, in this case ℝ^n, which can be obtained by typing \bbR+TAB. The tensor then lives in the tensor product of the different spaces, which we can obtain by typing (i.e. \otimes+TAB), although for simplicity also the usual multiplication sign * does the job. Note also that A is printed as an instance of a parametric type TensorMap, which we will discuss below and contains Tensor.

    Briefly sidetracking into the nature of ℝ^n:

    julia> V = ℝ^3ℝ^3
    julia> typeof(V)CartesianSpace
    julia> V == CartesianSpace(3)true
    julia> supertype(CartesianSpace)ElementarySpace{ℝ}
    julia> supertype(ElementarySpace)VectorSpace

    i.e. ℝ^n can also be created without Unicode using the longer syntax CartesianSpace(n). It is subtype of ElementarySpace{ℝ}, with a standard (Euclidean) inner product over the real numbers. Furthermore,

    julia> W = ℝ^3 ⊗ ℝ^2 ⊗ ℝ^4(ℝ^3 ⊗ ℝ^2 ⊗ ℝ^4)
    julia> typeof(W)ProductSpace{CartesianSpace, 3}
    julia> supertype(ProductSpace)CompositeSpace
    julia> supertype(CompositeSpace)VectorSpace

    i.e. the tensor product of a number of CartesianSpaces is some generic parametric ProductSpace type, specifically ProductSpace{CartesianSpace,N} for the tensor product of N instances of CartesianSpace.

    Tensors are itself vectors (but not Vectors), so we can compute linear combinations, provided they live in the same space.

    julia> B = Tensor(randn, ℝ^3 * ℝ^2 * ℝ^4);
    julia> C = 0.5*A + 2.5*BTensorMap((ℝ^3 ⊗ ℝ^2 ⊗ ℝ^4) ← ProductSpace{CartesianSpace, 0}()): [:, :, 1] = - -3.5935088814354623 -3.196363488680346 - 3.8464606941524773 -2.861017574132733 - -4.419954988240105 -2.3037010529814195 + -0.6603268148870567 -0.3366435178056115 + 1.3497015889106017 -3.0392636320725877 + 3.2825938696775614 0.5645444074976373 [:, :, 2] = - 1.9061858135382117 -3.3594732792561794 - -2.07888676894671 3.5555557908440427 - -1.0560612547646673 -0.10816491298265213 + 0.760002157601098 0.7918911740756697 + 2.704246439255191 7.137008963322173 + 2.005159598856597 -2.8298725354132297 [:, :, 3] = - -0.9264008825415807 -5.178257232386987 - 2.059946532144107 -6.276146973486205 - 4.06945838234967 -1.1514094648827222 + 1.2090566834343441 -0.47083959302541156 + -0.1687471972305043 2.609918026528343 + 0.5557101309243442 -1.239799440556535 [:, :, 4] = - -2.142809557219592 1.8897096326759302 - 2.61444065924357 -0.23234662837237607 - -4.307137638727699 4.1297789755970395

    Given that they are behave as vectors, they also have a scalar product and norm, which they inherit from the Euclidean inner product on the individual ℝ^n spaces:

    julia> scalarBA = dot(B,A)-2.010857137078127
    julia> scalarAA = dot(A,A)20.792415039990857
    julia> normA² = norm(A)^220.792415039990857

    If two tensors live on different spaces, these operations have no meaning and are thus not allowed

    julia> B′ = Tensor(randn, ℝ^4 * ℝ^2 * ℝ^3);
    julia> space(B′) == space(A)false
    julia> C′ = 0.5*A + 2.5*B′ERROR: SpaceMismatch("(ℝ^3 ⊗ ℝ^2 ⊗ ℝ^4) ← ProductSpace{CartesianSpace, 0}() ≠ (ℝ^4 ⊗ ℝ^2 ⊗ ℝ^3) ← ProductSpace{CartesianSpace, 0}()")
    julia> scalarBA′ = dot(B′,A)ERROR: SpaceMismatch("(ℝ^4 ⊗ ℝ^2 ⊗ ℝ^3) ← ProductSpace{CartesianSpace, 0}() ≠ (ℝ^3 ⊗ ℝ^2 ⊗ ℝ^4) ← ProductSpace{CartesianSpace, 0}()")

    However, in this particular case, we can reorder the indices of B′ to match space of A, using the routine permute (we deliberately choose not to overload permutedims from Julia Base, for reasons that become clear below):

    julia> space(permute(B′,(3,2,1))) == space(A)true

    We can contract two tensors using Einstein summation convention, which takes the interface from TensorOperations.jl. TensorKit.jl reexports the @tensor macro

    julia> @tensor D[a,b,c,d] := A[a,b,e]*B[d,c,e]TensorMap((ℝ^3 ⊗ ℝ^2 ⊗ ℝ^2 ⊗ ℝ^3) ← ProductSpace{CartesianSpace, 0}()):
    + 1.1667619929856055   -1.7274867851556617
    + 0.34059168917695004   7.512743297688271
    + 4.83357926995861     -2.508081264395529

    Given that they are behave as vectors, they also have a scalar product and norm, which they inherit from the Euclidean inner product on the individual ℝ^n spaces:

    julia> scalarBA = dot(B,A)-8.018445373890012
    julia> scalarAA = dot(A,A)19.752122624059393
    julia> normA² = norm(A)^219.75212262405939

    If two tensors live on different spaces, these operations have no meaning and are thus not allowed

    julia> B′ = Tensor(randn, ℝ^4 * ℝ^2 * ℝ^3);
    julia> space(B′) == space(A)false
    julia> C′ = 0.5*A + 2.5*B′ERROR: SpaceMismatch("(ℝ^3 ⊗ ℝ^2 ⊗ ℝ^4) ← ProductSpace{CartesianSpace, 0}() ≠ (ℝ^4 ⊗ ℝ^2 ⊗ ℝ^3) ← ProductSpace{CartesianSpace, 0}()")
    julia> scalarBA′ = dot(B′,A)ERROR: SpaceMismatch("(ℝ^4 ⊗ ℝ^2 ⊗ ℝ^3) ← ProductSpace{CartesianSpace, 0}() ≠ (ℝ^3 ⊗ ℝ^2 ⊗ ℝ^4) ← ProductSpace{CartesianSpace, 0}()")

    However, in this particular case, we can reorder the indices of B′ to match space of A, using the routine permute (we deliberately choose not to overload permutedims from Julia Base, for reasons that become clear below):

    julia> space(permute(B′,(3,2,1))) == space(A)true

    We can contract two tensors using Einstein summation convention, which takes the interface from TensorOperations.jl. TensorKit.jl reexports the @tensor macro

    julia> @tensor D[a,b,c,d] := A[a,b,e]*B[d,c,e]TensorMap((ℝ^3 ⊗ ℝ^2 ⊗ ℝ^2 ⊗ ℝ^3) ← ProductSpace{CartesianSpace, 0}()):
     [:, :, 1, 1] =
    - -1.5595295042461181  2.4931409956940156
    -  1.9161806567697965  0.42603109608750056
    -  1.6549865955659593  1.0891555268071929
    +  0.21694928795462345   0.08378125724360118
    +  0.28992191134501544   0.17691803817888838
    + -0.4601128358681438   -0.5484376617074957
     
     [:, :, 2, 1] =
    - -0.9005662705626052   3.7204115221871072
    - -1.4572336436398754  -3.5533221778026665
    -  2.447765401928689   -1.728685006728727
    +  0.4588397507384442   -1.1140522133252575
    + -0.3201079084331179    0.34697196817479126
    +  0.11469714038085436  -0.5902853741893431
     
     [:, :, 1, 2] =
    -  0.834647140618531   -2.9943295495002378
    - -2.2774129596468367  -0.24816268447287088
    - -2.0320221647831507  -0.7317841628475998
    +  2.0604379384143994  -1.5889254573761789
    + -3.196542414665528   -2.679131268293149
    +  0.8341343311547335   0.41831744750737077
     
     [:, :, 2, 2] =
    -  1.6390867504646076   3.410137248433252
    - -0.7409934037441455  -1.9767155274983312
    -  0.9677592345159259  -2.791197514703226
    +  7.941955226230259    0.016012949452521016
    + -3.835726423658136   -4.42668439337608
    +  0.5381548167139155  -1.706993426928763
     
     [:, :, 1, 3] =
    - -3.651794144909767   1.2876260443520817
    -  3.7210338385478954  2.1399452516927564
    -  2.031368200149454   3.9411017683950087
    +  1.5284277634076293   -0.608324030540135
    + -3.59490493844129     -3.96744920455324
    +  0.27983264261250734   1.0863103626104782
     
     [:, :, 2, 3] =
    - 2.7746553948176405    2.0496900675212277
    - 0.19334669470480512  -2.859598763751796
    - 2.592844017521057    -1.9489788680234037
    julia> @tensor d = A[a,b,c]*A[a,b,c]20.792415039990857
    julia> d ≈ scalarAA ≈ normA²true

    We hope that the index convention is clear. The := is to create a new tensor D, without the : the result would be written in an existing tensor D, which in this case would yield an error as no tensor D exists. If the contraction yields a scalar, regular assignment with = can be used.

    Finally, we can factorize a tensor, creating a bipartition of a subset of its indices and its complement. With a plain Julia Array, one would apply permutedims and reshape to cast the array into a matrix before applying e.g. the singular value decomposition. With TensorKit.jl, one just specifies which indices go to the left (rows) and right (columns)

    julia> U, S, Vd = tsvd(A, (1,3), (2,));
    julia> @tensor A′[a,b,c] := U[a,c,d]*S[d,e]*Vd[e,b];
    julia> A ≈ A′true
    julia> UTensorMap((ℝ^3 ⊗ ℝ^4) ← ℝ^2): + -2.5471192616365754 0.13772969578708416 + 1.7493164903508143 1.9286301088212978 + -0.33063966479228024 0.2220517169097211
    julia> @tensor d = A[a,b,c]*A[a,b,c]19.752122624059396
    julia> d ≈ scalarAA ≈ normA²true

    We hope that the index convention is clear. The := is to create a new tensor D, without the : the result would be written in an existing tensor D, which in this case would yield an error as no tensor D exists. If the contraction yields a scalar, regular assignment with = can be used.

    Finally, we can factorize a tensor, creating a bipartition of a subset of its indices and its complement. With a plain Julia Array, one would apply permutedims and reshape to cast the array into a matrix before applying e.g. the singular value decomposition. With TensorKit.jl, one just specifies which indices go to the left (rows) and right (columns)

    julia> U, S, Vd = tsvd(A, (1,3), (2,));
    julia> @tensor A′[a,b,c] := U[a,c,d]*S[d,e]*Vd[e,b];
    julia> A ≈ A′true
    julia> UTensorMap((ℝ^3 ⊗ ℝ^4) ← ℝ^2): [:, :, 1] = - -0.04750611054580282 -0.4047159373473946 … -0.4565933886060935 - 0.3788866864070819 -0.11939855292410644 0.04315395466829081 - 0.5425406465839459 0.1693631640438188 -0.17947678569435857 + -0.2823061582690183 0.16928009782574824 … 0.23923708074371874 + -0.5024270964713087 -0.4257440880497985 -0.28408690891167454 + 0.17921166498660382 -0.009854876013374654 0.12098578253819599 [:, :, 2] = - -0.6008127064832698 -0.20026420215479718 … 0.009732487141196589 - 0.20901753759190364 0.22211479863752331 -0.36098362902938097 - -0.12556823427468056 -0.13108279751431037 -0.3120665500798886

    Note that the tsvd routine returns the decomposition of the linear map as three factors, U, S and Vd, each of them a TensorMap, such that Vd is already what is commonly called V'. Furthermore, observe that U is printed differently then A, i.e. as a TensorMap((ℝ^3 ⊗ ℝ^4) ← ProductSpace(ℝ^2)). What this means is that tensors (or more appropriately, TensorMap instances) in TensorKit.jl are always considered to be linear maps between two ProductSpace instances, with

    julia> codomain(U)(ℝ^3 ⊗ ℝ^4)
    julia> domain(U)ProductSpace(ℝ^2)
    julia> codomain(A)(ℝ^3 ⊗ ℝ^2 ⊗ ℝ^4)
    julia> domain(A)ProductSpace{CartesianSpace, 0}()

    Hence, a Tensor instance such as A is just a specific case of TensorMap with an empty domain, i.e. a ProductSpace{CartesianSpace,0} instance. In particular, we can represent a vector v and matrix m as

    julia> v = Tensor(randn, ℝ^3)TensorMap(ℝ^3 ← ProductSpace{CartesianSpace, 0}()):
    - -0.6011413486735957
    - -0.6054302137487861
    -  0.8615661987310759
    julia> m1 = TensorMap(randn, ℝ^4, ℝ^3)TensorMap(ℝ^4 ← ℝ^3): - 0.023257194531194878 -0.21323208583763337 -0.12897258518846827 - -0.14006248508773683 0.32714594000282815 -2.179744271700997 - -2.725448458768963 3.0737556953628236 1.4211164820459505 - -0.49088728917403485 -0.5735013054546806 -1.1871970938163614
    julia> m2 = TensorMap(randn, ℝ^4 → ℝ^2) # alternative syntax for TensorMap(randn, ℝ^2, ℝ^4)TensorMap(ℝ^2 ← ℝ^4): - -0.024146946550839734 0.9012977285654201 … -0.0028448630012551683 - 0.7117119816625458 0.7914432821127432 1.0319307037769332
    julia> w = m1 * v # matrix vector productTensorMap(ℝ^4 ← ProductSpace{CartesianSpace, 0}()): - 0.003997866058582756 - -1.991860671574008 - 1.001821119980093 - -0.3805412222565031
    julia> m3 = m2*m1 # matrix matrix productTensorMap(ℝ^2 ← ℝ^3): - 1.7797274926744606 -1.846966385468707 -2.951489228128627 - -4.7386432363658475 4.1819273961479375 -0.8844979263777336

    Note that for the construction of m1, in accordance with how one specifies the dimensions of a matrix (e.g. randn(4,3)), the first space is the codomain and the second the domain. This is somewhat opposite to the general notation for a function f:domain→codomain, so that we also support this more mathemical notation, as illustrated in the construction of m2. In fact, there is a third syntax which mixes both and reads as TensorMap(randn, codomain←domain).

    This 'matrix vector' or 'matrix matrix' product can be computed between any two TensorMap instances for which the domain of the first matches with the codomain of the second, e.g.

    julia> v′ = v ⊗ vTensorMap((ℝ^3 ⊗ ℝ^3) ← ProductSpace{CartesianSpace, 0}()):
    -  0.36137092108510954   0.3639491352206886  -0.5179230666767821
    -  0.3639491352206886    0.3665457437199009  -0.5216182078564844
    - -0.5179230666767821   -0.5216182078564844   0.7422963147959157
    julia> m1′ = m1 ⊗ m1TensorMap((ℝ^4 ⊗ ℝ^4) ← (ℝ^3 ⊗ ℝ^3)): + -0.06621218276430767 -0.840889123562893 … 0.08221617274612225 + 0.09448171804142282 0.13601669796373803 -0.16029134738428097 + 0.18983236479424898 -0.25256999827905263 0.11690833511383808

    Note that the tsvd routine returns the decomposition of the linear map as three factors, U, S and Vd, each of them a TensorMap, such that Vd is already what is commonly called V'. Furthermore, observe that U is printed differently then A, i.e. as a TensorMap((ℝ^3 ⊗ ℝ^4) ← ProductSpace(ℝ^2)). What this means is that tensors (or more appropriately, TensorMap instances) in TensorKit.jl are always considered to be linear maps between two ProductSpace instances, with

    julia> codomain(U)(ℝ^3 ⊗ ℝ^4)
    julia> domain(U)ProductSpace(ℝ^2)
    julia> codomain(A)(ℝ^3 ⊗ ℝ^2 ⊗ ℝ^4)
    julia> domain(A)ProductSpace{CartesianSpace, 0}()

    Hence, a Tensor instance such as A is just a specific case of TensorMap with an empty domain, i.e. a ProductSpace{CartesianSpace,0} instance. In particular, we can represent a vector v and matrix m as

    julia> v = Tensor(randn, ℝ^3)TensorMap(ℝ^3 ← ProductSpace{CartesianSpace, 0}()):
    + -0.05657148880609898
    + -1.9509798669996792
    +  0.0886700060451214
    julia> m1 = TensorMap(randn, ℝ^4, ℝ^3)TensorMap(ℝ^4 ← ℝ^3): + -0.9976311695007977 -1.2335992011976689 1.365570255053924 + 0.28490191964568523 -0.8601670904082994 1.6971210080640533 + 1.417833955364368 -0.3970822346479611 0.9123743621923551 + 0.6997475227566912 -0.2298820754593008 1.6291837949158101
    julia> m2 = TensorMap(randn, ℝ^4 → ℝ^2) # alternative syntax for TensorMap(randn, ℝ^2, ℝ^4)TensorMap(ℝ^2 ← ℝ^4): + 0.13448826354699522 -1.9292466707883609 … -1.4850758142900071 + -0.5264390562424223 -1.6059459439321193 -3.5337947247096113
    julia> w = m1 * v # matrix vector productTensorMap(ℝ^4 ← ProductSpace{CartesianSpace, 0}()): + 2.584249808792238 + 1.8125350799285553 + 0.7753907078176221 + 0.5533692787982699
    julia> m3 = m2*m1 # matrix matrix productTensorMap(ℝ^2 ← ℝ^3): + -1.856632525435847 1.8723893735380266 -5.595969629323687 + -2.5541206457052943 2.8848852514772663 -9.297463908794215

    Note that for the construction of m1, in accordance with how one specifies the dimensions of a matrix (e.g. randn(4,3)), the first space is the codomain and the second the domain. This is somewhat opposite to the general notation for a function f:domain→codomain, so that we also support this more mathemical notation, as illustrated in the construction of m2. In fact, there is a third syntax which mixes both and reads as TensorMap(randn, codomain←domain).

    This 'matrix vector' or 'matrix matrix' product can be computed between any two TensorMap instances for which the domain of the first matches with the codomain of the second, e.g.

    julia> v′ = v ⊗ vTensorMap((ℝ^3 ⊗ ℝ^3) ← ProductSpace{CartesianSpace, 0}()):
    +  0.0032003333457385824   0.11036983570689683  -0.005016194254418314
    +  0.11036983570689683     3.806322441438086    -0.1729933966007717
    + -0.005016194254418314   -0.1729933966007717    0.007862369972041866
    julia> m1′ = m1 ⊗ m1TensorMap((ℝ^4 ⊗ ℝ^4) ← (ℝ^3 ⊗ ℝ^3)): [:, :, 1, 1] = - 0.0005408970974618409 -0.003257460462208077 … -0.011416661177211442 - -0.003257460462208077 0.019617499728952503 0.06875489361969782 - -0.06338628499033504 0.3817330841137232 1.3378880057086475 - -0.011416661177211442 0.06875489361969782 0.24097033067263252 + 0.9952679503595293 -0.2842270352891472 … -0.6980899394830439 + -0.2842270352891472 0.08116910381779649 0.1993594125006941 + -1.4144753470480962 0.4039436156221432 0.9921257979465375 + -0.6980899394830439 0.1993594125006941 0.48964659560412604 [:, :, 2, 1] = - -0.004959180100618283 0.029865815842860542 … 0.10467292058176095 - 0.007608496766736383 -0.04582087334315977 -0.16059178365227975 - 0.07148693414842137 -0.4305178612451016 -1.508867600879907 - -0.01333803142485372 0.0803260180430438 0.2815245011724183 + 1.23067701378608 -0.3514547804945997 … -0.8632079851127018 + 0.8581295003701301 -0.2450632552733682 -0.6018997906700383 + 0.3961416141398356 -0.11312949090840255 -0.27785731002560193 + 0.2293375237877329 -0.06549384459048907 -0.1608594128288125 [:, :, 3, 1] = - -0.0029995405029193097 0.018064220789686704 … 0.06331100272093447 - -0.050694736555207794 0.3053003995502007 1.0700087566279333 - 0.03305118247442999 -0.19904510607449796 -0.6976080174720776 - -0.027610873757756334 0.16628177524885862 0.582779963098806 + -1.3623354505849488 0.3890535870759109 … 0.9555544031242061 + -1.6931008160593142 0.4835130330684692 1.18755622121116 + -0.9102131019765036 0.2599372072241097 0.6384316997708166 + -1.6253245346536074 0.46415759062715667 1.1400173246076832 [:, :, 1, 2] = - -0.004959180100618283 0.007608496766736383 … -0.01333803142485372 - 0.029865815842860542 -0.04582087334315977 0.0803260180430438 - 0.5811530597062691 -0.8916193979732316 1.5630482490534474 - 0.10467292058176095 -0.16059178365227975 0.2815245011724183 + 1.23067701378608 0.8581295003701301 … 0.2293375237877329 + -0.3514547804945997 -0.2450632552733682 -0.06549384459048907 + -1.7490388347684156 -1.219574108067859 -0.3259346123158306 + -0.8632079851127018 -0.6018997906700383 -0.1608594128288125 [:, :, 2, 2] = - 0.04546792243066784 -0.0697580111601163 … 0.12228887959270725 - -0.0697580111601163 0.10702446606033404 -0.18761862366582055 - -0.6554233382775201 1.0055666962985177 -1.762802903939339 - 0.12228887959270725 -0.18761862366582055 0.32890374735822286 + 1.5217669891955268 1.0611014356242012 … 0.2835823446562557 + 1.0611014356242012 0.7398874234214795 0.1977369959848479 + 0.48984032747151013 0.3415570704299623 0.09128208822889039 + 0.2835823446562557 0.1977369959848479 0.05284576861747567 [:, :, 3, 2] = - 0.02750109335560895 -0.04219285761607628 … 0.07396594597345156 - 0.46479141764743664 -0.7130944887314028 1.2500861853778837 - -0.30302763168489766 0.46491248737243474 -0.8150121576565158 - 0.2531485126148392 -0.3883867092251793 0.6808590831356862 + -1.6845663758138174 -1.174618593037853 … -0.3139201244172828 + -2.093567119883599 -1.4598076395772568 -0.39013769963934536 + -1.125504284393722 -0.784794400490126 -0.20973851197663443 + -2.00975982801233 -1.401370284613084 -0.3745201520799063 [:, :, 1, 3] = - -0.0029995405029193097 -0.050694736555207794 … -0.027610873757756334 - 0.018064220789686704 0.3053003995502007 0.16628177524885862 - 0.35150813352535965 5.940780665817958 3.235644489596794 - 0.06331100272093447 1.0700087566279333 0.582779963098806 + -1.3623354505849488 -1.6931008160593142 … -1.6253245346536074 + 0.3890535870759109 0.4835130330684692 0.46415759062715667 + 1.9361518760510337 2.40623579159542 2.309912103961014 + 0.9555544031242061 1.18755622121116 1.1400173246076832 [:, :, 2, 3] = - 0.02750109335560895 0.46479141764743664 … 0.2531485126148392 - -0.04219285761607628 -0.7130944887314028 -0.3883867092251793 - -0.39643021826872127 -6.70000136957543 -3.649153828636233 - 0.07396594597345156 1.2500861853778837 0.6808590831356862 + -1.6845663758138174 -2.093567119883599 … -2.00975982801233 + -1.174618593037853 -1.4598076395772568 -1.401370284613084 + -0.5422436884455983 -0.6738966023500748 -0.6469199419374154 + -0.3139201244172828 -0.39013769963934536 -0.3745201520799063 [:, :, 3, 3] = - 0.016633927730196705 0.2811272537710326 … 0.15311587831773263 - 0.2811272537710326 4.751285090013311 2.587786064626285 - -0.1832850665434077 -3.0976705111595337 -1.687145357459484 - 0.15311587831773263 2.587786064626285 1.4094369395660145
    julia> w′ = m1′ * v′TensorMap((ℝ^4 ⊗ ℝ^4) ← ProductSpace{CartesianSpace, 0}()): - 1.5982933022370787e-5 -0.007963192172311537 … -0.0015213528363508528 - -0.007963192172311537 3.967508934963258 0.7579850945254322 - 0.004005146652339655 -1.9954880888405726 -0.3812342334796038 - -0.001521352836350839 0.7579850945254322 0.14481162183647336
    julia> w′ ≈ w ⊗ wtrue

    Another example involves checking that U from the singular value decomposition is a unitary, or at least a left isometric tensor

    julia> codomain(U)(ℝ^3 ⊗ ℝ^4)
    julia> domain(U)ProductSpace(ℝ^2)
    julia> space(U)(ℝ^3 ⊗ ℝ^4) ← ℝ^2
    julia> U'*U # should be the identity on the corresponding domain = codomainTensorMap(ℝ^2 ← ℝ^2): - 1.0000000000000002 -1.3146254338634137e-16 - -1.3146254338634137e-16 0.9999999999999999
    julia> U'*U ≈ one(U'*U)true
    julia> P = U*U' # should be a projectorTensorMap((ℝ^3 ⊗ ℝ^4) ← (ℝ^3 ⊗ ℝ^4)): + 1.864782121488039 2.3175379678394017 … 2.2247649303529027 + 2.3175379678394017 2.8802197160123484 2.7649220443491394 + 1.2459112904836755 1.5484096972956873 1.4864255257804329 + 2.2247649303529027 2.7649220443491394 2.6542398376162804
    julia> w′ = m1′ * v′TensorMap((ℝ^4 ⊗ ℝ^4) ← ProductSpace{CartesianSpace, 0}()): + 6.678347074242717 4.684043433734592 … 1.4300444529259277 + 4.684043433734592 3.285283415971614 1.003001229976629 + 2.0038032884169676 1.4054228585700725 0.4290773967719176 + 1.4300444529259277 1.003001229976629 0.30621755871771744
    julia> w′ ≈ w ⊗ wtrue

    Another example involves checking that U from the singular value decomposition is a unitary, or at least a left isometric tensor

    julia> codomain(U)(ℝ^3 ⊗ ℝ^4)
    julia> domain(U)ProductSpace(ℝ^2)
    julia> space(U)(ℝ^3 ⊗ ℝ^4) ← ℝ^2
    julia> U'*U # should be the identity on the corresponding domain = codomainTensorMap(ℝ^2 ← ℝ^2): + 1.0 -1.3795718596145348e-16 + -1.3795718596145348e-16 0.9999999999999998
    julia> U'*U ≈ one(U'*U)true
    julia> P = U*U' # should be a projectorTensorMap((ℝ^3 ⊗ ℝ^4) ← (ℝ^3 ⊗ ℝ^4)): [:, :, 1, 1] = - 0.36323273881094176 0.1395477573676099 … 0.01584357405348784 - -0.14357982527184798 -0.1277772324651754 0.21483347461233462 - 0.04966899475068784 0.07071042515451866 0.1960197925779963 + 0.08408082014302609 0.007888290245274187 … -0.07298181343630232 + 0.1355824226322281 0.11118421543851263 0.09081272385792052 + -0.0631617718917 0.019505303074386804 -0.04189578752475736 [:, :, 2, 1] = - -0.14357982527184798 -0.19520021083989342 … -0.17096289554744604 - 0.18724345215752128 0.0011873661947213534 -0.059101450361080304 - 0.17931546469748993 0.03677084446688228 -0.13322874648122826 + 0.1355824226322281 -0.1644995571175382 … -0.11243126659449461 + 0.2613597823126486 0.2267564573046207 0.12758835890196038 + -0.0721051085273815 -0.018911890621639525 -0.049740835080057146 [:, :, 3, 1] = - 0.04966899475068784 -0.19442802407827764 … -0.2489425635057259 - 0.17931546469748993 -0.0926691311758222 0.06874085136767448 - 0.31011773465458714 0.10834623595549378 -0.05818780568770279 + -0.0631617718917 -0.1292910026752599 … 0.05848136606334528 + -0.0721051085273815 -0.05047793545164615 -0.08134017347697337 + 0.06815314759064754 -0.04971206878798537 0.0438750492471957 [:, :, 1, 2] = - 0.1395477573676099 0.20390074060767768 … 0.18284155248402478 - -0.19520021083989342 0.0038408543286851974 0.05482700524468797 - -0.19442802407827764 -0.04229277982769247 0.13513287422533873 + 0.007888290245274187 0.7357502696462652 … -0.028636609011339013 + -0.1644995571175382 -0.18644496281445028 0.08669699088510055 + -0.1292910026752599 0.2107151301155493 -0.07782646234748178 [:, :, 2, 2] = - -0.1277772324651754 0.0038408543286851974 … 0.056678319295890914 - 0.0011873661947213534 0.06359099821415817 -0.08533232581364979 - -0.0926691311758222 -0.04933714588021431 -0.04788533043712466 + 0.11118421543851263 -0.18644496281445028 … -0.09067100043278674 + 0.2267564573046207 0.19975857063431335 0.0991460221781187 + -0.05047793545164615 -0.03015808196946571 -0.0356074959470931 [:, :, 3, 2] = - 0.07071042515451866 -0.04229277982769247 … -0.07860586261705702 - 0.03677084446688228 -0.04933714588021431 0.054627434253664536 - 0.10834623595549378 0.04586658113911117 0.010509800097500473 + 0.019505303074386804 0.2107151301155493 … -0.02312299037752944 + -0.018911890621639525 -0.03015808196946571 0.04328442659734228 + -0.04971206878798537 0.06388872261191963 -0.03071983788480402 [:, :, 1, 3] = - 0.15041185171675106 0.053324175630043866 … 0.001313572959090328 - -0.055298091889966955 -0.05448285387847712 0.08978341768732226 - 0.026905786484794282 0.03134257716963106 0.07939429746578815 + 0.05851746688849795 0.23411342248376948 … -0.05757053809070536 + 0.039237073027747 0.016142640360931482 0.08751948581280462 + -0.08233102658539161 0.0785483574012436 -0.052143246192619466 [:, :, 2, 3] = - -0.13333738965544112 0.08019577855120634 … 0.1487490830009625 - -0.06975281222328994 0.0931909162055052 -0.10309203016262877 - -0.20493881545711246 -0.08669547644709218 -0.019640987309861708 + -0.06448963615731679 -0.043627962531947166 … 0.057090746450611705 + -0.0949307283637801 -0.07521246409043582 -0.07364981348654195 + 0.05475191166931824 -0.02563982018727999 0.03591184850653224 [:, :, 3, 3] = - -0.20538060936168615 -0.034206485279992746 … 0.04361002683911317 - 0.03953462743240324 0.08799038230893397 -0.12971275185065614 - -0.09158375616259877 -0.06063894558347155 -0.0930417492427478 + 0.13756039152339097 0.017744064024569423 … -0.11954530287774277 + 0.2206529647981392 0.1806068934807973 0.1490887447555642 + -0.10414788102540574 0.03328669388747965 -0.06903002841984258 [:, :, 1, 4] = - 0.01584357405348784 0.18284155248402478 … 0.20857224382474868 - -0.17096289554744604 0.056678319295890914 -0.02321707892145958 - -0.2489425635057259 -0.07860586261705702 0.07891073007046673 + -0.07298181343630232 -0.028636609011339013 … 0.06399387986379682 + -0.11243126659449461 -0.09067100043278674 -0.08114266387179048 + 0.05848136606334528 -0.02312299037752944 0.03855604130111325 [:, :, 2, 4] = - 0.21483347461233462 0.05482700524468797 … -0.02321707892145958 - -0.059101450361080304 -0.08533232581364979 0.13217144423073462 - 0.06874085136767448 0.054627434253664536 0.10490578267265235 + 0.09081272385792052 0.08669699088510055 … -0.08114266387179048 + 0.12758835890196038 0.0991460221781187 0.1063986878612583 + -0.08134017347697337 0.04328442659734228 -0.053109871539386296 [:, :, 3, 4] = - 0.1960197925779963 0.13513287422533873 … 0.07891073007046673 - -0.13322874648122826 -0.04788533043712466 0.10490578267265235 - -0.05818780568770279 0.010509800097500473 0.12959744828194233
    julia> P*P ≈ Ptrue

    Here, the adjoint of a TensorMap results in a new tensor map (actually a simple wrapper of type AdjointTensorMap <: AbstractTensorMap) with domain and codomain interchanged.

    Our original tensor A living in ℝ^4 * ℝ^2 * ℝ^3 is isomorphic to e.g. a linear map ℝ^3 → ℝ^4 * ℝ^2. This is where the full power of permute emerges. It allows to specify a permutation where some indices go to the codomain, and others go to the domain, as in

    julia> A2 = permute(A,(1,2),(3,))TensorMap((ℝ^3 ⊗ ℝ^2) ← ℝ^4):
    + -0.04189578752475736   -0.07782646234748178  …   0.03855604130111325
    + -0.049740835080057146  -0.0356074959470931      -0.053109871539386296
    +  0.0438750492471957    -0.03071983788480402      0.028305118395469113
    julia> P*P ≈ Ptrue

    Here, the adjoint of a TensorMap results in a new tensor map (actually a simple wrapper of type AdjointTensorMap <: AbstractTensorMap) with domain and codomain interchanged.

    Our original tensor A living in ℝ^4 * ℝ^2 * ℝ^3 is isomorphic to e.g. a linear map ℝ^3 → ℝ^4 * ℝ^2. This is where the full power of permute emerges. It allows to specify a permutation where some indices go to the codomain, and others go to the domain, as in

    julia> A2 = permute(A,(1,2),(3,))TensorMap((ℝ^3 ⊗ ℝ^2) ← ℝ^4):
     [:, :, 1] =
    -  0.2210365849069235  -1.7302843974122823
    - -1.354396000596351    0.5625290150311519
    - -1.9015345843488605  -0.4217291675237574
    + -0.6263453764951161   -0.8856201818147352
    + -1.4695568646379336   -1.2414139114562541
    +  0.15060631101099436   0.7952249883922199
     
     [:, :, 2] =
    -  1.4446767974878814  -0.5344349066926972
    -  0.4011412987164587   0.6545701572938563
    - -0.5854057361941739  -0.39705919950306773
    +  1.8472385553363047   -0.8572954041956529
    + -1.338778966753932    -0.963723800253341
    +  0.39637190126323335  -0.4254736775609254
     
     [:, :, 3] =
    - 0.05116384353412213  -0.7203656988574674
    - 1.1079157004350477    0.7491133391309124
    - 0.2794243232604638    1.0171002760576229
    +  0.156942099334832   -0.8571782550886548
    +  0.3829592474389556   0.7188486586656138
    + -1.0121852621242975  -1.4540150011477664
     
     [:, :, 4] =
    -  1.6088525905822937    0.07778621198669779
    - -0.11996279740997542  -1.0473991500145496
    -  0.6605616797500655   -0.8818812373603137
    julia> codomain(A2)(ℝ^3 ⊗ ℝ^2)
    julia> domain(A2)ProductSpace(ℝ^4)

    In fact, tsvd(A, (1,3),(2,)) is a shorthand for tsvd(permute(A,(1,3),(2,))), where tsvd(A::TensorMap) will just compute the singular value decomposition according to the given codomain and domain of A.

    Note, finally, that the @tensor macro treats all indices at the same footing and thus does not distinguish between codomain and domain. The linear numbering is first all indices in the codomain, followed by all indices in the domain. However, when @tensor creates a new tensor (i.e. when using :=), the default syntax always creates a Tensor, i.e. with all indices in the codomain.

    julia> @tensor A′[a,b,c] := U[a,c,d]*S[d,e]*Vd[e,b];
    julia> codomain(A′)(ℝ^3 ⊗ ℝ^2 ⊗ ℝ^4)
    julia> domain(A′)ProductSpace{CartesianSpace, 0}()
    julia> @tensor A2′[(a,b);(c,)] := U[a,c,d]*S[d,e]*Vd[e,b];
    julia> codomain(A2′)(ℝ^3 ⊗ ℝ^2)
    julia> domain(A2′)ProductSpace(ℝ^4)
    julia> @tensor A2′′[a b; c] := U[a,c,d]*S[d,e]*Vd[e,b];
    julia> A2 ≈ A2′ == A2′′true

    As illustrated for A2′ and A2′′, additional syntax is available that enables one to immediately specify the desired codomain and domain indices.

    Complex tensors

    For applications in e.g. quantum physics, we of course want to work with complex tensors. Trying to create a complex-valued tensor with CartesianSpace indices is of course somewhat contrived and prints a (one-time) warning

    julia> A = Tensor(randn, ComplexF64, ℝ^3 ⊗ ℝ^2 ⊗ ℝ^4)┌ Warning: scalartype(data) = ComplexF64 ⊈ ℝ)
    +  0.4871612767655687    0.7916663983867505
    + -0.47376751248989013  -1.0388733926202567
    +  0.12047129978209313   0.5191235432360563
    julia> codomain(A2)(ℝ^3 ⊗ ℝ^2)
    julia> domain(A2)ProductSpace(ℝ^4)

    In fact, tsvd(A, (1,3),(2,)) is a shorthand for tsvd(permute(A,(1,3),(2,))), where tsvd(A::TensorMap) will just compute the singular value decomposition according to the given codomain and domain of A.

    Note, finally, that the @tensor macro treats all indices at the same footing and thus does not distinguish between codomain and domain. The linear numbering is first all indices in the codomain, followed by all indices in the domain. However, when @tensor creates a new tensor (i.e. when using :=), the default syntax always creates a Tensor, i.e. with all indices in the codomain.

    julia> @tensor A′[a,b,c] := U[a,c,d]*S[d,e]*Vd[e,b];
    julia> codomain(A′)(ℝ^3 ⊗ ℝ^2 ⊗ ℝ^4)
    julia> domain(A′)ProductSpace{CartesianSpace, 0}()
    julia> @tensor A2′[(a,b);(c,)] := U[a,c,d]*S[d,e]*Vd[e,b];
    julia> codomain(A2′)(ℝ^3 ⊗ ℝ^2)
    julia> domain(A2′)ProductSpace(ℝ^4)
    julia> @tensor A2′′[a b; c] := U[a,c,d]*S[d,e]*Vd[e,b];
    julia> A2 ≈ A2′ == A2′′true

    As illustrated for A2′ and A2′′, additional syntax is available that enables one to immediately specify the desired codomain and domain indices.

    Complex tensors

    For applications in e.g. quantum physics, we of course want to work with complex tensors. Trying to create a complex-valued tensor with CartesianSpace indices is of course somewhat contrived and prints a (one-time) warning

    julia> A = Tensor(randn, ComplexF64, ℝ^3 ⊗ ℝ^2 ⊗ ℝ^4)┌ Warning: scalartype(data) = ComplexF64 ⊈ ℝ)
     └ @ TensorKit ~/work/TensorKit.jl/TensorKit.jl/src/tensors/tensor.jl:33
     TensorMap((ℝ^3 ⊗ ℝ^2 ⊗ ℝ^4) ← ProductSpace{CartesianSpace, 0}()):
     [:, :, 1] =
    - -0.3393487839330667 - 0.11275502101203562im  …  -0.0575600463403108 - 0.984294359296908im
    - 0.33813138481580945 + 0.1639632832814031im       0.3831231488607673 - 1.367936441156121im
    - -0.3846448035236522 + 0.6651803929306411im      -0.6540157683325921 - 0.28023761303356515im
    + 0.24713635793384592 + 0.5202656899287648im   …  -0.3928401810311721 + 0.5108500932884528im
    +  0.7719160375176704 + 0.26077912520128604im     -0.3845838469637931 + 0.0017304756688756816im
    + 0.03476311286679427 + 0.9684076548014687im      -0.2867920607258084 + 1.0874172076249862im
     
     [:, :, 2] =
    - 0.33259139506701285 - 0.8485826721776121im   …   1.176333308658456 - 0.44250358151731745im
    -  0.6632087771910121 - 0.07441625936977045im     -0.720225149321359 - 0.17422343397308593im
    - -0.8348742336803433 - 0.4310801083542529im      -1.155611167299152 - 0.030546762403582638im
    + -0.23082938500392494 - 0.05054858742794064im  …  0.15784456584024284 - 0.12405394635287982im
    +  -0.5581780598337825 + 0.5457074580654707im       0.1337037220683609 + 0.12041363137392715im
    +  0.16009250257372382 + 0.5821653227024809im      -0.8797879464246959 + 0.4201648954937411im
     
     [:, :, 3] =
    -  0.02285412523883382 - 1.045140982065532im   …   0.27719957373849996 - 0.257271945508919im
    -   0.5957579772248444 - 0.778376332910001im        0.4073398274666202 - 1.1434409791810405im
    - -0.15612365077462098 + 1.5848445138016742im     -0.43901115060359025 + 0.5452146670054137im
    + -0.11992268192299671 - 0.052263127964298293im  …   0.9195380430720581 - 1.0251303715064162im
    +   0.7774238815140149 - 0.45942392060129306im      -0.6508895123617121 + 0.22132183315216436im
    +  -0.3147843434809939 + 0.269726162202291im        0.34510009912749445 - 0.49504398054694276im
     
     [:, :, 4] =
    - -0.48120990242841843 + 1.0076288254054924im   …    0.4100714353669746 - 0.45936453761489815im
    -   0.7327743395647779 - 1.43726000927401im         -0.4107600264913573 - 1.1885765884230397im
    -  -1.3222337430373596 - 0.07845329194170884im     0.022528441585508402 + 0.35277974458693917im

    although most of the above operations will work in the expected way (at your own risk). Indeed, we instead want to work with complex vector spaces

    julia> A = Tensor(randn, ComplexF64, ℂ^3 ⊗ ℂ^2 ⊗ ℂ^4)TensorMap((ℂ^3 ⊗ ℂ^2 ⊗ ℂ^4) ← ProductSpace{ComplexSpace, 0}()):
    +    1.0368223215112873 + 0.07533154835728986im  …   0.6669980535435027 + 0.5454492226201751im
    + -0.034590018479079214 - 0.3088538543477759im       0.7851781075144117 - 0.35439058824245506im
    +    0.2654597625587442 + 0.9646984447927714im      -1.3026314729857769 - 0.602381610214226im

    although most of the above operations will work in the expected way (at your own risk). Indeed, we instead want to work with complex vector spaces

    julia> A = Tensor(randn, ComplexF64, ℂ^3 ⊗ ℂ^2 ⊗ ℂ^4)TensorMap((ℂ^3 ⊗ ℂ^2 ⊗ ℂ^4) ← ProductSpace{ComplexSpace, 0}()):
     [:, :, 1] =
    - 0.5264292347176003 + 0.5117816383763689im   …  -0.29498008384316843 - 0.2248253753432696im
    - 0.9186840936805961 - 0.35583968592972365im      0.13542356998413116 - 0.4810760514756766im
    - 1.2105154783733432 + 0.4140996017633634im      -0.09989946251954755 - 0.4463232646446335im
    + -0.22742374332096316 + 0.9487026786202475im  …  -1.1591977332984702 + 0.45677681177036106im
    +  0.28041701066711877 - 0.8293012986568133im     -1.0190531980358155 - 0.09044741831216423im
    +  -0.7916079924396648 - 1.1815937186221284im     0.13690888196033224 + 0.08764330955843624im
     
     [:, :, 2] =
    -    0.4973180246007411 - 0.21349389077304484im  …  0.051292219083544484 - 1.673167875297341im
    -    0.8681536245876756 - 1.1421788817635974im       0.35377465695547533 + 0.5275070222226576im
    - -0.027749979203459476 - 0.6577665510967985im         0.712782224659383 + 1.6244087391849917im
    +  0.35017371721614066 + 0.3981551500889911im  …  -0.20327368255307093 + 0.03158973483479066im
    + -0.08762777647201796 + 0.5178066224365386im       0.7568712941574708 - 1.2350838378377211im
    +   0.9227773870897051 - 1.6238289734951497im     0.023535914148537652 - 0.07844420608236555im
     
     [:, :, 3] =
    -  1.0858027342788548 - 0.7013898902406186im  …  -0.9820229955585568 + 0.11497965127827975im
    - -0.4203283052883186 - 1.0744639870396426im      -0.386822055667599 + 0.43821793813084864im
    -  0.8310985194474669 - 0.2609058668909562im        1.00867782420442 + 0.32141494771457824im
    +  0.6294758872060258 - 0.1701681092819234im  …    -0.3818321288044495 - 0.4278209491813214im
    + 0.21123461270141347 - 1.9287270814200712im     -0.026220374466250664 - 0.7838660862189499im
    +  0.3357337759346161 + 0.3767690015638773im     -0.041083767514610244 + 0.6115684224372979im
     
     [:, :, 4] =
    -  -1.509319258928805 - 1.1446398588661224im  …    1.4472696662898341 - 0.39521576250017776im
    -  0.5308626319854972 + 1.5297060897344261im     -0.22754748058159302 + 0.505780485307557im
    - -0.1400719353232366 - 0.5447784344383172im      -1.3156407720365366 + 0.46727468453079263im

    where is obtained as \bbC+TAB and we also have the non-Unicode alternative ℂ^n == ComplexSpace(n). Most functionality works exactly the same

    julia> B = Tensor(randn, ℂ^3 * ℂ^2 * ℂ^4);
    julia> C = im*A + (2.5-0.8im)*BTensorMap((ℂ^3 ⊗ ℂ^2 ⊗ ℂ^4) ← ProductSpace{ComplexSpace, 0}()): + -0.6573122474229218 - 1.3701337267515707im … -0.6208016916671303 + 0.6401303730928731im + 0.048084838033900436 + 0.23300364603823953im 0.29154996538960265 - 0.5530140906538717im + -0.05425284740152367 + 0.44120327206502974im 0.48972800688883217 - 0.10107512518616964im

    where is obtained as \bbC+TAB and we also have the non-Unicode alternative ℂ^n == ComplexSpace(n). Most functionality works exactly the same

    julia> B = Tensor(randn, ℂ^3 * ℂ^2 * ℂ^4);
    julia> C = im*A + (2.5-0.8im)*BTensorMap((ℂ^3 ⊗ ℂ^2 ⊗ ℂ^4) ← ProductSpace{ComplexSpace, 0}()): [:, :, 1] = - 4.513555196758257 - 1.0816785525254802im … 2.9395033759413725 - 1.1636770440345614im - -1.303242281157846 + 1.4495903231486182im 0.2823555235198081 + 0.19901413893000908im - -0.7793449506514916 + 1.3273939900175442im -0.6064479027948906 + 0.23698731106110021im + -1.3079513552445483 - 0.11246416680118687im … 1.2324810681091964 - 1.6997602548599287im + 0.41700574480823005 + 0.41235158789866544im -5.249795674007388 + 0.6898245915064416im + 1.494008413021085 - 0.8915806946473309im -1.4982046909539242 + 0.5882885240068884im [:, :, 2] = - -1.6442143932155566 + 1.0917846754770935im … -1.4321169600205936 + 1.0449833663852837im - -2.868393908829474 + 2.1515369175774586im -3.317151247851624 + 1.2464608091567446im - 0.8653691555291132 - 0.09418281262180019im -3.616691089241978 + 1.3503125766776187im + 2.8401953637854405 - 0.6860984472236776im … 0.6683173409188261 - 0.4272439467942283im + 1.727359979785263 - 0.8060810891829944im 0.5841943479163885 + 0.9651559309322972im + 4.52898687652218 - 0.006873141878944611im 0.8795209472962033 - 0.2328086430398905im [:, :, 3] = - 1.3761728705570977 + 0.8698721805775814im … -0.3993633534190641 - 0.8910202108735058im - 1.0206899712266804 - 0.40312062022817063im -0.49230053598458134 - 0.3695156243544045im - -2.2975787747066616 + 1.6498136047587046im -3.678366652305643 + 2.082902369673561im + -0.588341397728774 + 0.872198929449449im … 0.5993099574838788 - 0.4367086114612679im + 0.9277253582299678 + 0.5315551641222466im 0.4883106674201338 + 0.0683573595493705im + 3.623045073714608 - 0.9442067281544992im -0.82256796867277 + 0.026436087280740837im [:, :, 4] = - 1.0423327367741333 - 1.4765809798593683im … 1.8305579451938072 + 0.9879601678278727im - 0.10958146619531961 + 0.0062906140879785966im -2.499643429907272 + 0.41048866169031584im - 0.9214127226588382 - 0.2605949075538033im -1.9802227673738904 - 0.8314973855267453im
    julia> scalarBA = dot(B,A)-2.393821792393743 + 3.0193273933309634im
    julia> scalarAA = dot(A,A)30.114846201178246 + 0.0im
    julia> normA² = norm(A)^230.114846201178246
    julia> U,S,Vd = tsvd(A,(1,3),(2,));
    julia> @tensor A′[a,b,c] := U[a,c,d]*S[d,e]*Vd[e,b];
    julia> A′ ≈ Atrue
    julia> permute(A,(1,3),(2,)) ≈ U*S*Vdtrue

    However, trying the following

    julia> @tensor D[a,b,c,d] := A[a,b,e]*B[d,c,e]ERROR: SpaceMismatch("(ℂ^3 ⊗ ℂ^2) ← ((ℂ^3)' ⊗ (ℂ^2)') ≠ (ℂ^3 ⊗ ℂ^2) ← (ℂ^4)' * ℂ^4 ← ((ℂ^3)' ⊗ (ℂ^2)')")
    julia> @tensor d = A[a,b,c]*A[a,b,c]ERROR: SpaceMismatch("ProductSpace{ComplexSpace, 0}() ← ProductSpace{ComplexSpace, 0}() ≠ ProductSpace{ComplexSpace, 0}() ← ((ℂ^3)' ⊗ (ℂ^2)' ⊗ (ℂ^4)') * (ℂ^3 ⊗ ℂ^2 ⊗ ℂ^4) ← ProductSpace{ComplexSpace, 0}()")

    we obtain SpaceMismatch errors. The reason for this is that, with ComplexSpace, an index in a space ℂ^n can only be contracted with an index in the dual space dual(ℂ^n) == (ℂ^n)'. Because of the complex Euclidean inner product, the dual space is equivalent to the complex conjugate space, but not the the space itself.

    julia> dual(ℂ^3) == conj(ℂ^3) == (ℂ^3)'true
    julia> (ℂ^3)' == ℂ^3false
    julia> @tensor d = conj(A[a,b,c])*A[a,b,c]30.114846201178253 + 2.838956803044093e-17im
    julia> d ≈ normA²true

    This might seem overly strict or puristic, but we believe that it can help to catch errors, e.g. unintended contractions. In particular, contracting two indices both living in ℂ^n would represent an operation that is not invariant under arbitrary unitary basis changes.

    It also makes clear the isomorphism between linear maps ℂ^n → ℂ^m and tensors in ℂ^m ⊗ (ℂ^n)':

    julia> m = TensorMap(randn, ComplexF64, ℂ^3, ℂ^4)TensorMap(ℂ^3 ← ℂ^4):
    - -0.32065024622734645 - 0.9610824814602388im  …  -0.25816971383683773 - 0.5564144910839591im
    - -0.41421820223187217 - 0.49840401179062im          0.955106207931054 + 0.9376416251877253im
    -   0.2510668743523159 + 0.704744119843985im       -1.5092382515411724 - 1.365935054777097im
    julia> m2 = permute(m, (1,2), ())TensorMap((ℂ^3 ⊗ (ℂ^4)') ← ProductSpace{ComplexSpace, 0}()): - -0.32065024622734645 - 0.9610824814602388im … -0.25816971383683773 - 0.5564144910839591im - -0.41421820223187217 - 0.49840401179062im 0.955106207931054 + 0.9376416251877253im - 0.2510668743523159 + 0.704744119843985im -1.5092382515411724 - 1.365935054777097im
    julia> codomain(m2)(ℂ^3 ⊗ (ℂ^4)')
    julia> space(m, 1)ℂ^3
    julia> space(m, 2)(ℂ^4)'

    Hence, spaces become their corresponding dual space if they are 'permuted' from the domain to the codomain or vice versa. Also, spaces in the domain are reported as their dual when probing them with space(A, i). Generalizing matrix vector and matrix matrix multiplication to arbitrary tensor contractions require that the two indices to be contracted have spaces which are each others dual. Knowing this, all the other functionality of tensors with CartesianSpace indices remains the same for tensors with ComplexSpace indices.

    Symmetries

    So far, the functionality that we have illustrated seems to be just a convenience (or inconvenience?) wrapper around dense multidimensional arrays, e.g. Julia's Base Array. More power becomes visible when involving symmetries. With symmetries, we imply that there is some symmetry action defined on every vector space associated with each of the indices of a TensorMap, and the TensorMap is then required to be equivariant, i.e. it acts as an intertwiner between the tensor product representation on the domain and that on the codomain. By Schur's lemma, this means that the tensor is block diagonal in some basis corresponding to the irreducible representations that can be coupled to by combining the different representations on the different spaces in the domain or codomain. For Abelian symmetries, this does not require a basis change and it just imposes that the tensor has some block sparsity. Let's clarify all of this with some examples.

    We start with a simple $ℤ₂$ symmetry:

    julia> V1 = ℤ₂Space(0=>3,1=>2)Rep[ℤ₂](0=>3, 1=>2)
    julia> dim(V1)5
    julia> V2 = ℤ₂Space(0=>1,1=>1)Rep[ℤ₂](0=>1, 1=>1)
    julia> dim(V2)2
    julia> A = Tensor(randn, V1*V1*V2')TensorMap((Rep[ℤ₂](0=>3, 1=>2) ⊗ Rep[ℤ₂](0=>3, 1=>2) ⊗ Rep[ℤ₂](0=>1, 1=>1)') ← ProductSpace{GradedSpace{Z2Irrep, Tuple{Int64, Int64}}, 0}()): + 5.736160508059287 - 2.054440817441391im … 1.6925187291873427 - 1.3672494043967993im + -2.5302144242236184 + 0.7831922870532217im -5.927579744089867 + 2.3653399925075993im + -4.26307900937941 + 1.1687473885390784im -2.420926944367441 + 1.2967686691459877im
    julia> scalarBA = dot(B,A)-0.2916843252398153 - 0.005989657686046396im
    julia> scalarAA = dot(A,A)22.322712407328655 + 0.0im
    julia> normA² = norm(A)^222.32271240732866
    julia> U,S,Vd = tsvd(A,(1,3),(2,));
    julia> @tensor A′[a,b,c] := U[a,c,d]*S[d,e]*Vd[e,b];
    julia> A′ ≈ Atrue
    julia> permute(A,(1,3),(2,)) ≈ U*S*Vdtrue

    However, trying the following

    julia> @tensor D[a,b,c,d] := A[a,b,e]*B[d,c,e]ERROR: SpaceMismatch("(ℂ^3 ⊗ ℂ^2) ← ((ℂ^3)' ⊗ (ℂ^2)') ≠ (ℂ^3 ⊗ ℂ^2) ← (ℂ^4)' * ℂ^4 ← ((ℂ^3)' ⊗ (ℂ^2)')")
    julia> @tensor d = A[a,b,c]*A[a,b,c]ERROR: SpaceMismatch("ProductSpace{ComplexSpace, 0}() ← ProductSpace{ComplexSpace, 0}() ≠ ProductSpace{ComplexSpace, 0}() ← ((ℂ^3)' ⊗ (ℂ^2)' ⊗ (ℂ^4)') * (ℂ^3 ⊗ ℂ^2 ⊗ ℂ^4) ← ProductSpace{ComplexSpace, 0}()")

    we obtain SpaceMismatch errors. The reason for this is that, with ComplexSpace, an index in a space ℂ^n can only be contracted with an index in the dual space dual(ℂ^n) == (ℂ^n)'. Because of the complex Euclidean inner product, the dual space is equivalent to the complex conjugate space, but not the the space itself.

    julia> dual(ℂ^3) == conj(ℂ^3) == (ℂ^3)'true
    julia> (ℂ^3)' == ℂ^3false
    julia> @tensor d = conj(A[a,b,c])*A[a,b,c]22.32271240732866 - 2.8038781022627144e-17im
    julia> d ≈ normA²true

    This might seem overly strict or puristic, but we believe that it can help to catch errors, e.g. unintended contractions. In particular, contracting two indices both living in ℂ^n would represent an operation that is not invariant under arbitrary unitary basis changes.

    It also makes clear the isomorphism between linear maps ℂ^n → ℂ^m and tensors in ℂ^m ⊗ (ℂ^n)':

    julia> m = TensorMap(randn, ComplexF64, ℂ^3, ℂ^4)TensorMap(ℂ^3 ← ℂ^4):
    +  0.1015571874517209 - 0.9493257234973055im  …  0.42269831151466586 - 0.5765320882395841im
    + 0.18087020490579842 - 0.5234634092589058im     -1.5092461222295326 - 0.8142075190481937im
    + 0.13636494094768098 + 0.7232134397665069im      0.7666699258167374 - 0.4902782663521801im
    julia> m2 = permute(m, (1,2), ())TensorMap((ℂ^3 ⊗ (ℂ^4)') ← ProductSpace{ComplexSpace, 0}()): + 0.1015571874517209 - 0.9493257234973055im … 0.42269831151466586 - 0.5765320882395841im + 0.18087020490579842 - 0.5234634092589058im -1.5092461222295326 - 0.8142075190481937im + 0.13636494094768098 + 0.7232134397665069im 0.7666699258167374 - 0.4902782663521801im
    julia> codomain(m2)(ℂ^3 ⊗ (ℂ^4)')
    julia> space(m, 1)ℂ^3
    julia> space(m, 2)(ℂ^4)'

    Hence, spaces become their corresponding dual space if they are 'permuted' from the domain to the codomain or vice versa. Also, spaces in the domain are reported as their dual when probing them with space(A, i). Generalizing matrix vector and matrix matrix multiplication to arbitrary tensor contractions require that the two indices to be contracted have spaces which are each others dual. Knowing this, all the other functionality of tensors with CartesianSpace indices remains the same for tensors with ComplexSpace indices.

    Symmetries

    So far, the functionality that we have illustrated seems to be just a convenience (or inconvenience?) wrapper around dense multidimensional arrays, e.g. Julia's Base Array. More power becomes visible when involving symmetries. With symmetries, we imply that there is some symmetry action defined on every vector space associated with each of the indices of a TensorMap, and the TensorMap is then required to be equivariant, i.e. it acts as an intertwiner between the tensor product representation on the domain and that on the codomain. By Schur's lemma, this means that the tensor is block diagonal in some basis corresponding to the irreducible representations that can be coupled to by combining the different representations on the different spaces in the domain or codomain. For Abelian symmetries, this does not require a basis change and it just imposes that the tensor has some block sparsity. Let's clarify all of this with some examples.

    We start with a simple $ℤ₂$ symmetry:

    julia> V1 = ℤ₂Space(0=>3,1=>2)Rep[ℤ₂](0=>3, 1=>2)
    julia> dim(V1)5
    julia> V2 = ℤ₂Space(0=>1,1=>1)Rep[ℤ₂](0=>1, 1=>1)
    julia> dim(V2)2
    julia> A = Tensor(randn, V1*V1*V2')TensorMap((Rep[ℤ₂](0=>3, 1=>2) ⊗ Rep[ℤ₂](0=>3, 1=>2) ⊗ Rep[ℤ₂](0=>1, 1=>1)') ← ProductSpace{GradedSpace{Z2Irrep, Tuple{Int64, Int64}}, 0}()): * Data for sector (Irrep[ℤ₂](1), Irrep[ℤ₂](1), Irrep[ℤ₂](0)) ← (): [:, :, 1] = - -0.9201678673039971 -1.6157589959132213 - -0.9127648793576008 -0.437168936331648 + -1.160305958017852 -2.9525745115716693 + -0.17042579106637018 1.1975273419178818 * Data for sector (Irrep[ℤ₂](0), Irrep[ℤ₂](1), Irrep[ℤ₂](1)) ← (): [:, :, 1] = - 0.5163566883550336 0.5000247582520584 - 0.1938852195032046 -0.031906428050727555 - -1.6619881960907186 -0.0719598073616838 + -0.2214689040016427 -0.49294965966660964 + -1.4408150274864535 -0.210693788875205 + 0.10698986827536347 -0.5520728604607179 * Data for sector (Irrep[ℤ₂](1), Irrep[ℤ₂](0), Irrep[ℤ₂](1)) ← (): [:, :, 1] = - -0.030685886148252576 1.5560477319386532 -0.06940104927854868 - -0.26541645910742745 -1.4490489419140664 -0.8989213719081749 + -1.7720262199228185 1.0167609381187617 -0.8979404057606747 + -0.7826239871065974 -0.7872783996599764 0.5649899928665778 * Data for sector (Irrep[ℤ₂](0), Irrep[ℤ₂](0), Irrep[ℤ₂](0)) ← (): [:, :, 1] = - 0.4513502606717945 -0.8170310654995396 0.009298198023888813 - 1.0305643926644383 -0.06233889972557998 0.37282083625081364 - 0.38881874443962244 -0.31534545070560244 -0.09516662314676505
    julia> convert(Array, A)5×5×2 Array{Float64, 3}: + 0.10030443922857162 -0.3758584932207784 0.00820766736406824 + 0.5241582937136002 0.4991694135985479 1.0979826150735619 + 0.3115551796468652 0.09077998397319696 -1.0457663965183488
    julia> convert(Array, A)5×5×2 Array{Float64, 3}: [:, :, 1] = - 0.45135 -0.817031 0.0092982 0.0 0.0 - 1.03056 -0.0623389 0.372821 0.0 0.0 - 0.388819 -0.315345 -0.0951666 0.0 0.0 - 0.0 0.0 0.0 -0.920168 -1.61576 - 0.0 0.0 0.0 -0.912765 -0.437169 + 0.100304 -0.375858 0.00820767 0.0 0.0 + 0.524158 0.499169 1.09798 0.0 0.0 + 0.311555 0.09078 -1.04577 0.0 0.0 + 0.0 0.0 0.0 -1.16031 -2.95257 + 0.0 0.0 0.0 -0.170426 1.19753 [:, :, 2] = - 0.0 0.0 0.0 0.516357 0.500025 - 0.0 0.0 0.0 0.193885 -0.0319064 - 0.0 0.0 0.0 -1.66199 -0.0719598 - -0.0306859 1.55605 -0.069401 0.0 0.0 - -0.265416 -1.44905 -0.898921 0.0 0.0

    Here, we create a space 5-dimensional space V1, which has a three-dimensional subspace associated with charge 0 (the trivial irrep of $ℤ₂$) and a two-dimensional subspace with charge 1 (the non-trivial irrep). Similar for V2, where both subspaces are one- dimensional. Representing the tensor as a dense Array, we see that it is zero in those regions where the charges don't add to zero (modulo 2). Of course, the Tensor(Map) type in TensorKit.jl won't store these zero blocks, and only stores the non-zero information, which we can recognize also in the full Array representation.

    From there on, the resulting tensors support all of the same operations as the ones we encountered in the previous examples.

    julia> B = Tensor(randn, V1'*V1*V2);
    julia> @tensor C[a,b] := A[a,c,d]*B[c,b,d]TensorMap((Rep[ℤ₂](0=>3, 1=>2) ⊗ Rep[ℤ₂](0=>3, 1=>2)) ← ProductSpace{GradedSpace{Z2Irrep, Tuple{Int64, Int64}}, 0}()): + 0.0 0.0 0.0 -0.221469 -0.49295 + 0.0 0.0 0.0 -1.44082 -0.210694 + 0.0 0.0 0.0 0.10699 -0.552073 + -1.77203 1.01676 -0.89794 0.0 0.0 + -0.782624 -0.787278 0.56499 0.0 0.0

    Here, we create a space 5-dimensional space V1, which has a three-dimensional subspace associated with charge 0 (the trivial irrep of $ℤ₂$) and a two-dimensional subspace with charge 1 (the non-trivial irrep). Similar for V2, where both subspaces are one- dimensional. Representing the tensor as a dense Array, we see that it is zero in those regions where the charges don't add to zero (modulo 2). Of course, the Tensor(Map) type in TensorKit.jl won't store these zero blocks, and only stores the non-zero information, which we can recognize also in the full Array representation.

    From there on, the resulting tensors support all of the same operations as the ones we encountered in the previous examples.

    julia> B = Tensor(randn, V1'*V1*V2);
    julia> @tensor C[a,b] := A[a,c,d]*B[c,b,d]TensorMap((Rep[ℤ₂](0=>3, 1=>2) ⊗ Rep[ℤ₂](0=>3, 1=>2)) ← ProductSpace{GradedSpace{Z2Irrep, Tuple{Int64, Int64}}, 0}()): * Data for sector (Irrep[ℤ₂](0), Irrep[ℤ₂](0)) ← (): - -2.4773255701550556 -0.9206539736992502 -0.18369977224570797 - -1.8420884294313948 -0.7965964652987731 -0.5170696129135448 - -1.6565896851333208 1.7466375366631726 -0.5457602748192446 + -0.21179636252656361 0.059376308535838536 0.482483312643038 + -3.0682099583902174 -2.4111153818580724 -1.5050588296864924 + 1.758041174320807 0.9660823512890424 1.0279762728203568 * Data for sector (Irrep[ℤ₂](1), Irrep[ℤ₂](1)) ← (): - -2.868963203288887 1.4285315879239562 - 3.476093908782517 1.8773261616489658
    julia> U,S,V = tsvd(A,(1,3),(2,));
    julia> U'*U # should be the identity on the corresponding domain = codomainTensorMap(Rep[ℤ₂](0=>3, 1=>2)' ← Rep[ℤ₂](0=>3, 1=>2)'): + -2.6249910389145636 6.387552862848035 + 1.3023205627074674 -0.8645923145943386
    julia> U,S,V = tsvd(A,(1,3),(2,));
    julia> U'*U # should be the identity on the corresponding domain = codomainTensorMap(Rep[ℤ₂](0=>3, 1=>2)' ← Rep[ℤ₂](0=>3, 1=>2)'): * Data for sector (Irrep[ℤ₂](0),) ← (Irrep[ℤ₂](0),): - 0.9999999999999999 -8.301088443341267e-17 -8.274102359807368e-17 - -8.301088443341267e-17 1.0 -7.628614429821803e-17 - -8.274102359807368e-17 -7.628614429821803e-17 1.0 + 1.0000000000000013 -1.6051388523511438e-16 -1.3944592492205918e-16 + -1.6051388523511438e-16 1.0000000000000013 -3.780263000228819e-16 + -1.3944592492205918e-16 -3.780263000228819e-16 1.0000000000000009 * Data for sector (Irrep[ℤ₂](1),) ← (Irrep[ℤ₂](1),): - 0.999999999999999 2.766195340691415e-16 - 2.766195340691415e-16 0.9999999999999998
    julia> U'*U ≈ one(U'*U)true
    julia> P = U*U' # should be a projectorTensorMap((Rep[ℤ₂](0=>3, 1=>2) ⊗ Rep[ℤ₂](0=>1, 1=>1)') ← (Rep[ℤ₂](0=>3, 1=>2) ⊗ Rep[ℤ₂](0=>1, 1=>1)')): + 0.9999999999999996 2.1233808700718553e-17 + 2.1233808700718553e-17 0.9999999999999996
    julia> U'*U ≈ one(U'*U)true
    julia> P = U*U' # should be a projectorTensorMap((Rep[ℤ₂](0=>3, 1=>2) ⊗ Rep[ℤ₂](0=>1, 1=>1)') ← (Rep[ℤ₂](0=>3, 1=>2) ⊗ Rep[ℤ₂](0=>1, 1=>1)')): * Data for sector (Irrep[ℤ₂](0), Irrep[ℤ₂](0)) ← (Irrep[ℤ₂](0), Irrep[ℤ₂](0)): [:, :, 1, 1] = - 0.24439500689854368 - 0.2980175586887504 - 0.14909581967749905 + 0.07720325641668824 + -0.16215835162756456 + 0.03211078454285867 [:, :, 2, 1] = - 0.2980175586887504 - 0.7234817269210306 - 0.2914764947187793 + -0.16215835162756456 + 0.727033817461309 + -0.3520920799151698 [:, :, 3, 1] = - 0.14909581967749905 - 0.2914764947187793 - 0.19879178867199337 + 0.03211078454285867 + -0.3520920799151698 + 0.47540873937778627 * Data for sector (Irrep[ℤ₂](1), Irrep[ℤ₂](1)) ← (Irrep[ℤ₂](0), Irrep[ℤ₂](0)): [:, :, 1, 1] = - -0.2439279322112434 - 0.11883275077453774 + -0.15623796775072227 + 0.13966451790610557 [:, :, 2, 1] = - 0.06699192320518679 - -0.14763125582787287 + -0.058359817123890816 + -0.21162611779421917 [:, :, 3, 1] = - 0.11250347704054935 - 0.1985660360271051 + -0.0397865715071768 + -0.3504459493356531 * Data for sector (Irrep[ℤ₂](0), Irrep[ℤ₂](0)) ← (Irrep[ℤ₂](1), Irrep[ℤ₂](1)): [:, :, 1, 1] = - -0.2439279322112434 - 0.06699192320518679 - 0.11250347704054935 + -0.15623796775072227 + -0.058359817123890816 + -0.0397865715071768 [:, :, 2, 1] = - 0.11883275077453774 - -0.14763125582787287 - 0.1985660360271051 + 0.13966451790610557 + -0.21162611779421917 + -0.3504459493356531 * Data for sector (Irrep[ℤ₂](1), Irrep[ℤ₂](1)) ← (Irrep[ℤ₂](1), Irrep[ℤ₂](1)): [:, :, 1, 1] = - 0.9158850738858912 - 0.019844875538302845 + 0.9696406152815462 + -0.00620889555932651 [:, :, 2, 1] = - 0.019844875538302845 - 0.9174464036225408 + -0.00620889555932651 + 0.7507135714626737 * Data for sector (Irrep[ℤ₂](0), Irrep[ℤ₂](1)) ← (Irrep[ℤ₂](0), Irrep[ℤ₂](1)): [:, :, 1, 1] = - 0.08879982346161802 - 0.005178160212325329 - -0.08868297783800652 + 0.02403731494722087 + 0.043563870232465086 + 0.01814074021827357 [:, :, 2, 1] = - 0.005178160212325329 - 0.015684003708492083 - -0.11283495836046942 + 0.043563870232465086 + 0.8132975039656875 + -0.16079122834524073 [:, :, 3, 1] = - -0.08868297783800652 - -0.11283495836046942 - 0.8421363565478791 + 0.01814074021827357 + -0.16079122834524073 + 0.06476676487870403 * Data for sector (Irrep[ℤ₂](1), Irrep[ℤ₂](0)) ← (Irrep[ℤ₂](0), Irrep[ℤ₂](1)): [:, :, 1, 1] = - -0.2511662280566023 - -0.09969180907788446 + 0.13986385080105374 + -0.0408745450268453 [:, :, 2, 1] = - 0.036101963846736154 - -0.03709612878966841 + 0.16281409505241443 + 0.31238536226707336 [:, :, 3, 1] = - -0.10436588048308028 - 0.31851856795594913 + 0.1294655967721087 + -0.13276966795866516 * Data for sector (Irrep[ℤ₂](0), Irrep[ℤ₂](1)) ← (Irrep[ℤ₂](1), Irrep[ℤ₂](0)): [:, :, 1, 1] = - -0.2511662280566023 - 0.036101963846736154 - -0.10436588048308028 + 0.13986385080105374 + 0.16281409505241443 + 0.1294655967721087 [:, :, 2, 1] = - -0.09969180907788446 - -0.03709612878966841 - 0.31851856795594913 + -0.0408745450268453 + 0.31238536226707336 + -0.13276966795866516 * Data for sector (Irrep[ℤ₂](1), Irrep[ℤ₂](0)) ← (Irrep[ℤ₂](1), Irrep[ℤ₂](0)): [:, :, 1, 1] = - 0.8778394379958192 - 0.17876603753571849 + 0.8250081207861304 + -0.2855486900107816 [:, :, 2, 1] = - 0.17876603753571849 - 0.17554037828619046
    julia> P*P ≈ Ptrue

    We also support other abelian symmetries, e.g.

    julia> V = U₁Space(0=>2,1=>1,-1=>1)Rep[U₁](0=>2, 1=>1, -1=>1)
    julia> dim(V)4
    julia> A = TensorMap(randn, V*V, V)TensorMap((Rep[U₁](0=>2, 1=>1, -1=>1) ⊗ Rep[U₁](0=>2, 1=>1, -1=>1)) ← Rep[U₁](0=>2, 1=>1, -1=>1)): + -0.2855486900107816 + 0.2728902954222564
    julia> P*P ≈ Ptrue

    We also support other abelian symmetries, e.g.

    julia> V = U₁Space(0=>2,1=>1,-1=>1)Rep[U₁](0=>2, 1=>1, -1=>1)
    julia> dim(V)4
    julia> A = TensorMap(randn, V*V, V)TensorMap((Rep[U₁](0=>2, 1=>1, -1=>1) ⊗ Rep[U₁](0=>2, 1=>1, -1=>1)) ← Rep[U₁](0=>2, 1=>1, -1=>1)): * Data for sector (Irrep[U₁](0), Irrep[U₁](0)) ← (Irrep[U₁](0),): [:, :, 1] = - -0.7390001706067019 -1.4282930827313847 - 0.1612884986016238 1.324113836743095 + -1.2824317073651037 -0.4535168667092357 + 0.6343125335834302 0.13275511778865678 [:, :, 2] = - -0.7458751386276218 -0.037150518131181635 - -1.4578151314312702 0.3663617411775646 + -0.09450447058253475 2.260637863914944 + 0.05164875306661292 -0.18933554948374548 * Data for sector (Irrep[U₁](-1), Irrep[U₁](1)) ← (Irrep[U₁](0),): [:, :, 1] = - 1.0277416686663485 + 1.404727647793804 [:, :, 2] = - -0.008761241615484661 + 0.21352862457838512 * Data for sector (Irrep[U₁](1), Irrep[U₁](-1)) ← (Irrep[U₁](0),): [:, :, 1] = - -1.2521595758421407 + -0.2474212020234898 [:, :, 2] = - -0.9525003351767227 + -1.0361898855562237 * Data for sector (Irrep[U₁](1), Irrep[U₁](0)) ← (Irrep[U₁](1),): [:, :, 1] = - 1.8430941875241866 0.7457318407850286 + 0.24224092631355212 -0.5033928147706093 * Data for sector (Irrep[U₁](0), Irrep[U₁](1)) ← (Irrep[U₁](1),): [:, :, 1] = - 0.3182438760733253 - -0.8158885082062156 + -1.2019130169132863 + -1.2706853731625811 * Data for sector (Irrep[U₁](-1), Irrep[U₁](0)) ← (Irrep[U₁](-1),): [:, :, 1] = - 1.270427266535907 -1.05144341390221 + -1.0544889108931064 -1.3884529686127847 * Data for sector (Irrep[U₁](0), Irrep[U₁](-1)) ← (Irrep[U₁](-1),): [:, :, 1] = - 0.0994757935303114 - 0.15361586184109638
    julia> dim(A)20
    julia> convert(Array, A)4×4×4 Array{Float64, 3}: + -0.9352605783233429 + -0.4770132724375018
    julia> dim(A)20
    julia> convert(Array, A)4×4×4 Array{Float64, 3}: [:, :, 1] = - -0.739 -1.42829 0.0 0.0 - 0.161288 1.32411 0.0 0.0 - 0.0 0.0 0.0 -1.25216 - 0.0 0.0 1.02774 0.0 + -1.28243 -0.453517 0.0 0.0 + 0.634313 0.132755 0.0 0.0 + 0.0 0.0 0.0 -0.247421 + 0.0 0.0 1.40473 0.0 [:, :, 2] = - -0.745875 -0.0371505 0.0 0.0 - -1.45782 0.366362 0.0 0.0 - 0.0 0.0 0.0 -0.9525 - 0.0 0.0 -0.00876124 0.0 + -0.0945045 2.26064 0.0 0.0 + 0.0516488 -0.189336 0.0 0.0 + 0.0 0.0 0.0 -1.03619 + 0.0 0.0 0.213529 0.0 [:, :, 3] = - 0.0 0.0 0.318244 0.0 - 0.0 0.0 -0.815889 0.0 - 1.84309 0.745732 0.0 0.0 - 0.0 0.0 0.0 0.0 + 0.0 0.0 -1.20191 0.0 + 0.0 0.0 -1.27069 0.0 + 0.242241 -0.503393 0.0 0.0 + 0.0 0.0 0.0 0.0 [:, :, 4] = - 0.0 0.0 0.0 0.0994758 - 0.0 0.0 0.0 0.153616 - 0.0 0.0 0.0 0.0 - 1.27043 -1.05144 0.0 0.0
    julia> V = GradedSpace[Irrep[U₁×ℤ₂]]((0,0)=>2,(1,1)=>1,(-1,0)=>1)ERROR: MethodError: Cannot `convert` an object of type + 0.0 0.0 0.0 -0.935261 + 0.0 0.0 0.0 -0.477013 + 0.0 0.0 0.0 0.0 + -1.05449 -1.38845 0.0 0.0
    julia> V = GradedSpace[Irrep[U₁×ℤ₂]]((0,0)=>2,(1,1)=>1,(-1,0)=>1)ERROR: MethodError: Cannot `convert` an object of type Type{TensorKit.ProductSector{Tuple{U1Irrep, Z2Irrep}}} to an object of type GradedSpace @@ -500,61 +500,61 @@ @ Base Base.jl:64
    julia> dim(V)4
    julia> A = TensorMap(randn, V*V, V)TensorMap((Rep[U₁](0=>2, 1=>1, -1=>1) ⊗ Rep[U₁](0=>2, 1=>1, -1=>1)) ← Rep[U₁](0=>2, 1=>1, -1=>1)): * Data for sector (Irrep[U₁](0), Irrep[U₁](0)) ← (Irrep[U₁](0),): [:, :, 1] = - 1.5592070037936203 -0.6046233595478492 - -0.5681136955259156 0.4234313993682341 + 0.1373056155097603 0.8919366718547397 + 0.5346298992196507 -0.23013890877813228 [:, :, 2] = - -0.9524907699267443 -0.8848153709751899 - 1.7878486315343927 -1.6528619870844365 + 0.7315141669286511 -0.5153921727706144 + -0.7303799919828469 1.4420371321555656 * Data for sector (Irrep[U₁](-1), Irrep[U₁](1)) ← (Irrep[U₁](0),): [:, :, 1] = - 0.04690148526212282 + 0.39146038167009334 [:, :, 2] = - 1.5700843310216899 + 0.5855557698323619 * Data for sector (Irrep[U₁](1), Irrep[U₁](-1)) ← (Irrep[U₁](0),): [:, :, 1] = - 0.12251357710690203 + 1.1361657542452377 [:, :, 2] = - 1.242381681260966 + -0.004268133874441151 * Data for sector (Irrep[U₁](1), Irrep[U₁](0)) ← (Irrep[U₁](1),): [:, :, 1] = - -0.044495797692934234 -0.9869187239089 + -0.11059761515199369 0.9596197457230806 * Data for sector (Irrep[U₁](0), Irrep[U₁](1)) ← (Irrep[U₁](1),): [:, :, 1] = - 0.6727926146316807 - 1.676841331000959 + -0.9242242339499496 + -0.35831700970219416 * Data for sector (Irrep[U₁](-1), Irrep[U₁](0)) ← (Irrep[U₁](-1),): [:, :, 1] = - -1.2268035929166208 0.017579469726206585 + 1.6005193227199288 2.2320005200406263 * Data for sector (Irrep[U₁](0), Irrep[U₁](-1)) ← (Irrep[U₁](-1),): [:, :, 1] = - 1.4215561956129894 - 0.8036596962712947
    julia> dim(A)20
    julia> convert(Array, A)4×4×4 Array{Float64, 3}: + 0.4401415224656061 + -0.7144541323108412
    julia> dim(A)20
    julia> convert(Array, A)4×4×4 Array{Float64, 3}: [:, :, 1] = - 1.55921 -0.604623 0.0 0.0 - -0.568114 0.423431 0.0 0.0 - 0.0 0.0 0.0 0.122514 - 0.0 0.0 0.0469015 0.0 + 0.137306 0.891937 0.0 0.0 + 0.53463 -0.230139 0.0 0.0 + 0.0 0.0 0.0 1.13617 + 0.0 0.0 0.39146 0.0 [:, :, 2] = - -0.952491 -0.884815 0.0 0.0 - 1.78785 -1.65286 0.0 0.0 - 0.0 0.0 0.0 1.24238 - 0.0 0.0 1.57008 0.0 + 0.731514 -0.515392 0.0 0.0 + -0.73038 1.44204 0.0 0.0 + 0.0 0.0 0.0 -0.00426813 + 0.0 0.0 0.585556 0.0 [:, :, 3] = - 0.0 0.0 0.672793 0.0 - 0.0 0.0 1.67684 0.0 - -0.0444958 -0.986919 0.0 0.0 - 0.0 0.0 0.0 0.0 + 0.0 0.0 -0.924224 0.0 + 0.0 0.0 -0.358317 0.0 + -0.110598 0.95962 0.0 0.0 + 0.0 0.0 0.0 0.0 [:, :, 4] = - 0.0 0.0 0.0 1.42156 - 0.0 0.0 0.0 0.80366 - 0.0 0.0 0.0 0.0 - -1.2268 0.0175795 0.0 0.0

    Here, the dim of a TensorMap returns the number of linearly independent components, i.e. the number of non-zero entries in the case of an abelian symmetry. Also note that we can use × (obtained as \times+TAB) to combine different symmetries. The general space associated with symmetries is a GradedSpace. Although this is actually an abstract type, it is the access point for users to construct spaces with arbitrary symmetries, and ℤ₂Space (also Z2Space as non-Unicode alternative) and U₁Space (or U1Space) are just convenient synonyms, e.g.

    julia> GradedSpace[Irrep[U₁]](0=>3,1=>2,-1=>1) == U1Space(-1=>1,1=>2,0=>3)ERROR: MethodError: Cannot `convert` an object of type
    + 0.0      0.0    0.0   0.440142
    + 0.0      0.0    0.0  -0.714454
    + 0.0      0.0    0.0   0.0
    + 1.60052  2.232  0.0   0.0

    Here, the dim of a TensorMap returns the number of linearly independent components, i.e. the number of non-zero entries in the case of an abelian symmetry. Also note that we can use × (obtained as \times+TAB) to combine different symmetries. The general space associated with symmetries is a GradedSpace. Although this is actually an abstract type, it is the access point for users to construct spaces with arbitrary symmetries, and ℤ₂Space (also Z2Space as non-Unicode alternative) and U₁Space (or U1Space) are just convenient synonyms, e.g.

    julia> GradedSpace[Irrep[U₁]](0=>3,1=>2,-1=>1) == U1Space(-1=>1,1=>2,0=>3)ERROR: MethodError: Cannot `convert` an object of type
       Type{U1Irrep} to an object of type
       GradedSpace
     
    @@ -581,109 +581,109 @@
        @ Base Base.jl:64

    Note that now V has a two-dimensional subspace with spin zero, and two one-dimensional subspaces with spin 1/2 and spin 1. However, a subspace with spin j has an additional 2j+1 dimensional degeneracy on which the irreducible representation acts. This brings the total dimension to 2*1 + 1*2 + 1*3. Creating a tensor with SU₂ symmetry yields

    julia> A = TensorMap(randn, V*V, V)TensorMap((Rep[SU₂](0=>2, 1/2=>1, 1=>1) ⊗ Rep[SU₂](0=>2, 1/2=>1, 1=>1)) ← Rep[SU₂](0=>2, 1/2=>1, 1=>1)):
     * Data for fusiontree FusionTree{Irrep[SU₂]}((0, 0), 0, (false, false), ()) ← FusionTree{Irrep[SU₂]}((0,), 0, (false,), ()):
     [:, :, 1] =
    - 0.5880281359585298  -0.1354936338022071
    - 0.6007364019171977  -0.30719953869711336
    + -0.4139395402505528  -1.3927440947034493
    +  1.175930652037399    0.05902170737081042
     
     [:, :, 2] =
    - -0.39698482635893767  0.6351211103383761
    -  0.7994520007637228   1.6945704339778553
    + 1.8587133088946028  -0.47605913410085404
    + 0.212546195759631   -0.8139038212168653
     * Data for fusiontree FusionTree{Irrep[SU₂]}((1, 1), 0, (false, false), ()) ← FusionTree{Irrep[SU₂]}((0,), 0, (false,), ()):
     [:, :, 1] =
    - 0.48093790439684353
    + 0.27791312424912135
     
     [:, :, 2] =
    - -1.1344265968135219
    + 1.5106150427698062
     * Data for fusiontree FusionTree{Irrep[SU₂]}((1/2, 1/2), 0, (false, false), ()) ← FusionTree{Irrep[SU₂]}((0,), 0, (false,), ()):
     [:, :, 1] =
    - 0.3687359507236529
    + -1.7600043165415549
     
     [:, :, 2] =
    - -0.6850257620858446
    + -0.01680204236197071
     * Data for fusiontree FusionTree{Irrep[SU₂]}((0, 1/2), 1/2, (false, false), ()) ← FusionTree{Irrep[SU₂]}((1/2,), 1/2, (false,), ()):
     [:, :, 1] =
    -  0.9419811182203497
    - -2.8040047205749086
    +  0.41373643887678974
    + -0.006697204318366773
     * Data for fusiontree FusionTree{Irrep[SU₂]}((1/2, 1), 1/2, (false, false), ()) ← FusionTree{Irrep[SU₂]}((1/2,), 1/2, (false,), ()):
     [:, :, 1] =
    - -1.0894431015597332
    + 1.9437029537227313
     * Data for fusiontree FusionTree{Irrep[SU₂]}((1/2, 0), 1/2, (false, false), ()) ← FusionTree{Irrep[SU₂]}((1/2,), 1/2, (false,), ()):
     [:, :, 1] =
    - 0.6294069682733889  0.8481513797591721
    + -0.013077402281789939  0.5524600717342965
     * Data for fusiontree FusionTree{Irrep[SU₂]}((1, 1/2), 1/2, (false, false), ()) ← FusionTree{Irrep[SU₂]}((1/2,), 1/2, (false,), ()):
     [:, :, 1] =
    - 1.9419702595954367
    + -1.9364969088808162
     * Data for fusiontree FusionTree{Irrep[SU₂]}((1, 1), 1, (false, false), ()) ← FusionTree{Irrep[SU₂]}((1,), 1, (false,), ()):
     [:, :, 1] =
    - 3.925381190660305e-5
    + 0.06721365156287816
     * Data for fusiontree FusionTree{Irrep[SU₂]}((1, 0), 1, (false, false), ()) ← FusionTree{Irrep[SU₂]}((1,), 1, (false,), ()):
     [:, :, 1] =
    - -0.024018542733816657  -0.4269561113390705
    + -0.15631792595153543  1.19635465187691
     * Data for fusiontree FusionTree{Irrep[SU₂]}((1/2, 1/2), 1, (false, false), ()) ← FusionTree{Irrep[SU₂]}((1,), 1, (false,), ()):
     [:, :, 1] =
    - 0.5738221500974018
    + 1.0421580207466858
     * Data for fusiontree FusionTree{Irrep[SU₂]}((0, 1), 1, (false, false), ()) ← FusionTree{Irrep[SU₂]}((1,), 1, (false,), ()):
     [:, :, 1] =
    - -2.14258623264577
    -  0.6363480135484346
    julia> dim(A)24
    julia> convert(Array, A)7×7×7 Array{Float64, 3}: + 0.825737142599899 + -0.023905573306575847
    julia> dim(A)24
    julia> convert(Array, A)7×7×7 Array{Float64, 3}: [:, :, 1] = - 0.588028 -0.135494 0.0 0.0 0.0 0.0 0.0 - 0.600736 -0.3072 0.0 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 0.260736 0.0 0.0 0.0 - 0.0 0.0 -0.260736 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 0.0 0.0 0.27767 - 0.0 0.0 0.0 0.0 0.0 -0.27767 0.0 - 0.0 0.0 0.0 0.0 0.27767 0.0 0.0 + -0.41394 -1.39274 0.0 0.0 0.0 0.0 0.0 + 1.17593 0.0590217 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 -1.24451 0.0 0.0 0.0 + 0.0 0.0 1.24451 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 0.0 0.0 0.160453 + 0.0 0.0 0.0 0.0 0.0 -0.160453 0.0 + 0.0 0.0 0.0 0.0 0.160453 0.0 0.0 [:, :, 2] = - -0.396985 0.635121 0.0 0.0 0.0 0.0 0.0 - 0.799452 1.69457 0.0 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 -0.484386 0.0 0.0 0.0 - 0.0 0.0 0.484386 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 0.0 0.0 -0.654962 - 0.0 0.0 0.0 0.0 0.0 0.654962 0.0 - 0.0 0.0 0.0 0.0 -0.654962 0.0 0.0 + 1.85871 -0.476059 0.0 0.0 0.0 0.0 0.0 + 0.212546 -0.813904 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 -0.0118808 0.0 0.0 0.0 + 0.0 0.0 0.0118808 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 0.0 0.0 0.872154 + 0.0 0.0 0.0 0.0 0.0 -0.872154 0.0 + 0.0 0.0 0.0 0.0 0.872154 0.0 0.0 [:, :, 3] = - 0.0 0.0 0.941981 0.0 0.0 0.0 0.0 - 0.0 0.0 -2.804 0.0 0.0 0.0 0.0 - 0.629407 0.848151 0.0 0.0 0.0 -0.62899 0.0 - 0.0 0.0 0.0 0.0 0.889527 0.0 0.0 - 0.0 0.0 0.0 1.58561 0.0 0.0 0.0 - 0.0 0.0 -1.1212 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 0.413736 0.0 0.0 0.0 0.0 + 0.0 0.0 -0.0066972 0.0 0.0 0.0 0.0 + -0.0130774 0.55246 0.0 0.0 0.0 1.1222 0.0 + 0.0 0.0 0.0 0.0 -1.58703 0.0 0.0 + 0.0 0.0 0.0 -1.58114 0.0 0.0 0.0 + 0.0 0.0 1.11804 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 0.0 0.0 0.0 [:, :, 4] = - 0.0 0.0 0.0 0.941981 0.0 0.0 0.0 - 0.0 0.0 0.0 -2.804 0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 0.0 0.0 -0.889527 - 0.629407 0.848151 0.0 0.0 0.0 0.62899 0.0 - 0.0 0.0 0.0 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 1.1212 0.0 0.0 0.0 - 0.0 0.0 -1.58561 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.413736 0.0 0.0 0.0 + 0.0 0.0 0.0 -0.0066972 0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 0.0 0.0 1.58703 + -0.0130774 0.55246 0.0 0.0 0.0 -1.1222 0.0 + 0.0 0.0 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 -1.11804 0.0 0.0 0.0 + 0.0 0.0 1.58114 0.0 0.0 0.0 0.0 [:, :, 5] = - 0.0 0.0 0.0 0.0 -2.14259 0.0 0.0 - 0.0 0.0 0.0 0.0 0.636348 0.0 0.0 - 0.0 0.0 0.573822 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 0.0 0.0 0.0 - -0.0240185 -0.426956 0.0 0.0 0.0 2.77566e-5 0.0 - 0.0 0.0 0.0 0.0 -2.77566e-5 0.0 0.0 - 0.0 0.0 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 0.825737 0.0 0.0 + 0.0 0.0 0.0 0.0 -0.0239056 0.0 0.0 + 0.0 0.0 1.04216 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 0.0 0.0 0.0 + -0.156318 1.19635 0.0 0.0 0.0 0.0475272 0.0 + 0.0 0.0 0.0 0.0 -0.0475272 0.0 0.0 + 0.0 0.0 0.0 0.0 0.0 0.0 0.0 [:, :, 6] = - 0.0 0.0 0.0 … 0.0 -2.14259 0.0 - 0.0 0.0 0.0 0.0 0.636348 0.0 - 0.0 0.0 0.0 0.0 0.0 0.0 - 0.0 0.0 0.405754 0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 0.0 2.77566e-5 - -0.0240185 -0.426956 0.0 … 0.0 0.0 0.0 - 0.0 0.0 0.0 -2.77566e-5 0.0 0.0 + 0.0 0.0 0.0 0.0 0.0 0.825737 0.0 + 0.0 0.0 0.0 0.0 0.0 -0.0239056 0.0 + 0.0 0.0 0.0 0.736917 0.0 0.0 0.0 + 0.0 0.0 0.736917 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 0.0 0.0 0.0475272 + -0.156318 1.19635 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 -0.0475272 0.0 0.0 [:, :, 7] = - 0.0 0.0 0.0 0.0 0.0 0.0 -2.14259 - 0.0 0.0 0.0 0.0 0.0 0.0 0.636348 - 0.0 0.0 0.0 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 0.573822 0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 0.0 0.0 0.0 - 0.0 0.0 0.0 0.0 0.0 0.0 2.77566e-5 - -0.0240185 -0.426956 0.0 0.0 0.0 -2.77566e-5 0.0
    julia> norm(A) ≈ norm(convert(Array, A))true

    In this case, the full Array representation of the tensor has again many zeros, but it is less obvious to recognize the dense blocks, as there are additional zeros and the numbers in the original tensor data do not match with those in the Array. The reason is of course that the original tensor data now needs to be transformed with a construction known as fusion trees, which are made up out of the Clebsch-Gordan coefficients of the group. Indeed, note that the non-zero blocks are also no longer labeled by a list of sectors, but by pair of fusion trees. This will be explained further in the manual. However, the Clebsch-Gordan coefficients of the group are only needed to actually convert a tensor to an Array. For working with tensors with SU₂Space indices, e.g. contracting or factorizing them, the Clebsch-Gordan coefficients are never needed explicitly. Instead, recoupling relations are used to symbolically manipulate the basis of fusion trees, and this only requires what is known as the topological data of the group (or its representation theory).

    In fact, this formalism extends beyond the case of group representations on vector spaces, and can also deal with super vector spaces (to describe fermions) and more general (unitary) fusion categories. Preliminary support for these generalizations is present in TensorKit.jl and will be extended in the near future.

    All of these concepts will be explained throughout the remainder of this manual, including several details regarding their implementation. However, to just use tensors and their manipulations (contractions, factorizations, ...) in higher level algorithms (e.g. tensoer network algorithms), one does not need to know or understand most of these details, and one can immediately refer to the general interface of the TensorMap type, discussed on the last page. Adhering to this interface should yield code and algorithms that are oblivious to the underlying symmetries and can thus work with arbitrary symmetric tensors.

    + 0.0 0.0 0.0 0.0 0.0 0.0 0.825737 + 0.0 0.0 0.0 0.0 0.0 0.0 -0.0239056 + 0.0 0.0 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 1.04216 0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 0.0 0.0 0.0 + 0.0 0.0 0.0 0.0 0.0 0.0 0.0475272 + -0.156318 1.19635 0.0 0.0 0.0 -0.0475272 0.0
    julia> norm(A) ≈ norm(convert(Array, A))true

    In this case, the full Array representation of the tensor has again many zeros, but it is less obvious to recognize the dense blocks, as there are additional zeros and the numbers in the original tensor data do not match with those in the Array. The reason is of course that the original tensor data now needs to be transformed with a construction known as fusion trees, which are made up out of the Clebsch-Gordan coefficients of the group. Indeed, note that the non-zero blocks are also no longer labeled by a list of sectors, but by pair of fusion trees. This will be explained further in the manual. However, the Clebsch-Gordan coefficients of the group are only needed to actually convert a tensor to an Array. For working with tensors with SU₂Space indices, e.g. contracting or factorizing them, the Clebsch-Gordan coefficients are never needed explicitly. Instead, recoupling relations are used to symbolically manipulate the basis of fusion trees, and this only requires what is known as the topological data of the group (or its representation theory).

    In fact, this formalism extends beyond the case of group representations on vector spaces, and can also deal with super vector spaces (to describe fermions) and more general (unitary) fusion categories. Preliminary support for these generalizations is present in TensorKit.jl and will be extended in the near future.

    All of these concepts will be explained throughout the remainder of this manual, including several details regarding their implementation. However, to just use tensors and their manipulations (contractions, factorizations, ...) in higher level algorithms (e.g. tensoer network algorithms), one does not need to know or understand most of these details, and one can immediately refer to the general interface of the TensorMap type, discussed on the last page. Adhering to this interface should yield code and algorithms that are oblivious to the underlying symmetries and can thus work with arbitrary symmetric tensors.

    diff --git a/dev/search/index.html b/dev/search/index.html index 4259a1b9..109972b4 100644 --- a/dev/search/index.html +++ b/dev/search/index.html @@ -1,2 +1,2 @@ -Search · TensorKit.jl

    Loading search...

      +Search · TensorKit.jl

      Loading search...