diff --git a/previews/PR1514/AdvancedUsageGuide.html b/previews/PR1514/AdvancedUsageGuide.html index 141e47c0d0..7123aac1b0 100644 --- a/previews/PR1514/AdvancedUsageGuide.html +++ b/previews/PR1514/AdvancedUsageGuide.html @@ -461,4 +461,4 @@ myscale2! (generic function with 1 method) julia> @btime myscale2!(A, 2) setup = (A = random_itensor(i)); - 3.549 μs (2 allocations: 112 bytes)

A very efficient function is written for the Tensor type. Then, the ITensor version just wraps the Tensor function by calling it after converting the ITensor to a Tensor (without any copying) with the tensor function. This is the basis for the design of all performance critical ITensors.jl functions.

+ 3.549 μs (2 allocations: 112 bytes)

A very efficient function is written for the Tensor type. Then, the ITensor version just wraps the Tensor function by calling it after converting the ITensor to a Tensor (without any copying) with the tensor function. This is the basis for the design of all performance critical ITensors.jl functions.

diff --git a/previews/PR1514/CodeTiming.html b/previews/PR1514/CodeTiming.html index 5636ebe16f..bf09cb03df 100644 --- a/previews/PR1514/CodeTiming.html +++ b/previews/PR1514/CodeTiming.html @@ -1,2 +1,2 @@ -Timing and profiling · ITensors.jl

Timing and Profiling your code

It is very important to time and profile your code to make sure your code is running as fast as possible. Here are some tips on timing and profiling your code.

If you are concerned about the performance of your code, a good place to start is Julia's performance tips.

Timing and benchmarking

Julia has many nice timing tools available. Tools like @time and TimerOutputs can be used to measure the time of specific lines of code. For microbenchmarking, we recommend the BenchmarkTools package. For profiling your code, see the Julia documentation on profiling.

+Timing and profiling · ITensors.jl

Timing and Profiling your code

It is very important to time and profile your code to make sure your code is running as fast as possible. Here are some tips on timing and profiling your code.

If you are concerned about the performance of your code, a good place to start is Julia's performance tips.

Timing and benchmarking

Julia has many nice timing tools available. Tools like @time and TimerOutputs can be used to measure the time of specific lines of code. For microbenchmarking, we recommend the BenchmarkTools package. For profiling your code, see the Julia documentation on profiling.

diff --git a/previews/PR1514/ContractionSequenceOptimization.html b/previews/PR1514/ContractionSequenceOptimization.html index 0d99357c87..16b94f30f1 100644 --- a/previews/PR1514/ContractionSequenceOptimization.html +++ b/previews/PR1514/ContractionSequenceOptimization.html @@ -1,15 +1,15 @@ -Contraction sequence optimization · ITensors.jl

Contraction sequence optimization

When contracting a tensor network, the sequence of contraction makes a big difference in the computational cost. However, the complexity of determining the optimal sequence grows exponentially with the number of tensors, but there are many heuristic algorithms available for computing optimal sequences for small networks[1][2][3][4][5][6]. ITensors.jl provides some functionality for helping you find the optimal contraction sequence for small tensor network, as we will show below.

The algorithm in ITensors.jl currently uses a modified version of[1] with simplifications for outer product contractions similar to those used in TensorOperations.jl.

Functions

ITensors.ContractionSequenceOptimization.contraction_costFunction
contraction_cost(A; sequence)

Return the cost of contracting the collection of ITensors according to the specified sequence, where the cost is measured in the number of floating point operations that would need to be performed to contract dense tensors of the dimensions specified by the indices of the tensors (so for now, sparsity is ignored in computing the costs). Pairwise costs are returned in a vector (contracting N tensors requires N-1 pairwise contractions). You can use sum(contraction_cost(A; sequence)) to get the total cost of the contraction.

If no sequence is specified, left associative contraction is used, in other words the sequence is equivalent to [[[[1, 2], 3], 4], …].

source
NDTensors.contractFunction
contract(ψ::MPS, A::MPO; kwargs...) -> MPS
+Contraction sequence optimization · ITensors.jl

Contraction sequence optimization

When contracting a tensor network, the sequence of contraction makes a big difference in the computational cost. However, the complexity of determining the optimal sequence grows exponentially with the number of tensors, but there are many heuristic algorithms available for computing optimal sequences for small networks[1][2][3][4][5][6]. ITensors.jl provides some functionality for helping you find the optimal contraction sequence for small tensor network, as we will show below.

The algorithm in ITensors.jl currently uses a modified version of[1] with simplifications for outer product contractions similar to those used in TensorOperations.jl.

Functions

ITensors.ContractionSequenceOptimization.contraction_costFunction
contraction_cost(A; sequence)

Return the cost of contracting the collection of ITensors according to the specified sequence, where the cost is measured in the number of floating point operations that would need to be performed to contract dense tensors of the dimensions specified by the indices of the tensors (so for now, sparsity is ignored in computing the costs). Pairwise costs are returned in a vector (contracting N tensors requires N-1 pairwise contractions). You can use sum(contraction_cost(A; sequence)) to get the total cost of the contraction.

If no sequence is specified, left associative contraction is used, in other words the sequence is equivalent to [[[[1, 2], 3], 4], …].

source
NDTensors.contractFunction
contract(ψ::MPS, A::MPO; kwargs...) -> MPS
 *(::MPS, ::MPO; kwargs...) -> MPS
 
 contract(A::MPO, ψ::MPS; kwargs...) -> MPS
-*(::MPO, ::MPS; kwargs...) -> MPS

Contract the MPO A with the MPS ψ, returning an MPS with the unique site indices of the MPO.

For example, for an MPO with site indices with prime levels of 1 and 0, such as -s'-A-s-, and an MPS with site indices with prime levels of 0, such as -s-x, the result is an MPS y with site indices with prime levels of 1, -s'-y = -s'-A-s-x.

Since it is common to contract an MPO with prime levels of 1 and 0 with an MPS with prime level of 0 and want a resulting MPS with prime levels of 0, we provide a convenience function apply:

apply(A, x; kwargs...) = replaceprime(contract(A, x; kwargs...), 2 => 1)`.

Choose the method with the method keyword, for example "densitymatrix" and "naive".

Keywords

  • cutoff::Float64=1e-13: the cutoff value for truncating the density matrix eigenvalues. Note that the default is somewhat arbitrary and subject to change, in general you should set a cutoff value.
  • maxdim::Int=maxlinkdim(A) * maxlinkdim(ψ)): the maximal bond dimension of the results MPS.
  • mindim::Int=1: the minimal bond dimension of the resulting MPS.
  • normalize::Bool=false: whether or not to normalize the resulting MPS.
  • method::String="densitymatrix": the algorithm to use for the contraction. Currently the options are "densitymatrix", where the network formed by the MPO and MPS is squared and contracted down to a density matrix which is diagonalized iteratively at each site, and "naive", where the MPO and MPS tensor are contracted exactly at each site and then a truncation of the resulting MPS is performed.

See also apply.

source
contract(A::MPO, B::MPO; kwargs...) -> MPO
+*(::MPO, ::MPS; kwargs...) -> MPS

Contract the MPO A with the MPS ψ, returning an MPS with the unique site indices of the MPO.

For example, for an MPO with site indices with prime levels of 1 and 0, such as -s'-A-s-, and an MPS with site indices with prime levels of 0, such as -s-x, the result is an MPS y with site indices with prime levels of 1, -s'-y = -s'-A-s-x.

Since it is common to contract an MPO with prime levels of 1 and 0 with an MPS with prime level of 0 and want a resulting MPS with prime levels of 0, we provide a convenience function apply:

apply(A, x; kwargs...) = replaceprime(contract(A, x; kwargs...), 2 => 1)`.

Choose the method with the method keyword, for example "densitymatrix" and "naive".

Keywords

  • cutoff::Float64=1e-13: the cutoff value for truncating the density matrix eigenvalues. Note that the default is somewhat arbitrary and subject to change, in general you should set a cutoff value.
  • maxdim::Int=maxlinkdim(A) * maxlinkdim(ψ)): the maximal bond dimension of the results MPS.
  • mindim::Int=1: the minimal bond dimension of the resulting MPS.
  • normalize::Bool=false: whether or not to normalize the resulting MPS.
  • method::String="densitymatrix": the algorithm to use for the contraction. Currently the options are "densitymatrix", where the network formed by the MPO and MPS is squared and contracted down to a density matrix which is diagonalized iteratively at each site, and "naive", where the MPO and MPS tensor are contracted exactly at each site and then a truncation of the resulting MPS is performed.

See also apply.

source
contract(A::MPO, B::MPO; kwargs...) -> MPO
 *(::MPO, ::MPO; kwargs...) -> MPO

Contract the MPO A with the MPO B, returning an MPO with the site indices that are not shared between A and B.

If you are contracting two MPOs with the same sets of indices, likely you want to call something like:

C = contract(A', B; cutoff=1e-12)
 C = replaceprime(C, 2 => 1)

That is because if MPO A has the index structure -s'-A-s- and MPO B has the Index structure -s'-B-s-, if we only want to contract over on set of the indices, we would do (-s'-A-s-)'-s'-B-s- = -s''-A-s'-s'-B-s- = -s''-C-s-, and then map the prime levels back to pairs of primed and unprimed indices with: replaceprime(-s''-C-s-, 2 => 1) = -s'-C-s-.

Since this is a common use case, you can use the convenience function:

C = apply(A, B; cutoff=1e-12)

which is the same as the code above.

If you are contracting MPOs that have diverging norms, such as MPOs representing sums of local operators, the truncation can become numerically unstable (see https://arxiv.org/abs/1909.06341 for a more numerically stable alternative). For now, you can use the following options to contract MPOs like that:

C = contract(A, B; alg="naive", truncate=false)
 # Bring the indices back to pairs of primed and unprimed
-C = apply(A, B; alg="naive", truncate=false)

Keywords

  • cutoff::Float64=1e-14: the cutoff value for truncating the density matrix eigenvalues. Note that the default is somewhat arbitrary and subject to change, in general you should set a cutoff value.
  • maxdim::Int=maxlinkdim(A) * maxlinkdim(B)): the maximal bond dimension of the results MPS.
  • mindim::Int=1: the minimal bond dimension of the resulting MPS.
  • alg="zipup": Either "zipup" or "naive". "zipup" contracts pairs of site tensors and truncates with SVDs in a sweep across the sites, while "naive" first contracts pairs of tensor exactly and then truncates at the end if truncate=true.
  • truncate=true: Enable or disable truncation. If truncate=false, ignore other truncation parameters like cutoff and maxdim. This is most relevant for the "naive" version, if you just want to contract the tensors pairwise exactly. This can be useful if you are contracting MPOs that have diverging norms, such as MPOs originating from sums of local operators.

See also apply for details about the arguments available.

source
*(As::ITensor...; sequence = default_sequence(), kwargs...)
+C = apply(A, B; alg="naive", truncate=false)

Keywords

  • cutoff::Float64=1e-14: the cutoff value for truncating the density matrix eigenvalues. Note that the default is somewhat arbitrary and subject to change, in general you should set a cutoff value.
  • maxdim::Int=maxlinkdim(A) * maxlinkdim(B)): the maximal bond dimension of the results MPS.
  • mindim::Int=1: the minimal bond dimension of the resulting MPS.
  • alg="zipup": Either "zipup" or "naive". "zipup" contracts pairs of site tensors and truncates with SVDs in a sweep across the sites, while "naive" first contracts pairs of tensor exactly and then truncates at the end if truncate=true.
  • truncate=true: Enable or disable truncation. If truncate=false, ignore other truncation parameters like cutoff and maxdim. This is most relevant for the "naive" version, if you just want to contract the tensors pairwise exactly. This can be useful if you are contracting MPOs that have diverging norms, such as MPOs originating from sums of local operators.

See also apply for details about the arguments available.

source
*(As::ITensor...; sequence = default_sequence(), kwargs...)
 *(As::Vector{<: ITensor}; sequence = default_sequence(), kwargs...)
-contract(As::ITensor...; sequence = default_sequence(), kwargs...)

Contract the set of ITensors according to the contraction sequence.

The default sequence is "automatic" if ITensors.using_contraction_sequence_optimization() is true, otherwise it is "left_associative" (the ITensors are contracted from left to right).

You can change the default with ITensors.enable_contraction_sequence_optimization() and ITensors.disable_contraction_sequence_optimization().

For a custom sequence, the sequence should be provided as a binary tree where the leaves are integers n specifying the ITensor As[n] and branches are accessed by indexing with 1 or 2, i.e. sequence = Any[Any[1, 3], Any[2, 4]].

source

Examples

In the following example we show how to compute the contraction sequence cost of a

using ITensors
+contract(As::ITensor...; sequence = default_sequence(), kwargs...)

Contract the set of ITensors according to the contraction sequence.

The default sequence is "automatic" if ITensors.using_contraction_sequence_optimization() is true, otherwise it is "left_associative" (the ITensors are contracted from left to right).

You can change the default with ITensors.enable_contraction_sequence_optimization() and ITensors.disable_contraction_sequence_optimization().

For a custom sequence, the sequence should be provided as a binary tree where the leaves are integers n specifying the ITensor As[n] and branches are accessed by indexing with 1 or 2, i.e. sequence = Any[Any[1, 3], Any[2, 4]].

source

Examples

In the following example we show how to compute the contraction sequence cost of a

using ITensors
 using Symbolics
 
 using ITensors: contraction_cost
@@ -103,4 +103,4 @@
 # Fix d to a certain value (such as 4 for a Hubbard site)
 @variables d
 var_sub = Dict(d => 4)
-display(substitute.(sum.(getindex.(sequence_costs, :symbolic_cost)), (var_sub,)))

A future direction will be to allow optimizing over contraction sequences with the dimensions specified symbolically, so that the optimal sequence in limits of certain dimensions can be found. In addition, we plan to implement more algorithms that work for larger networks, as well as algorithms like[2] which take an optimal sequence for a closed network and generate optimal sequences for environments of each tensor in the network, which is helpful for computing gradients of tensor networks.

+display(substitute.(sum.(getindex.(sequence_costs, :symbolic_cost)), (var_sub,)))

A future direction will be to allow optimizing over contraction sequences with the dimensions specified symbolically, so that the optimal sequence in limits of certain dimensions can be found. In addition, we plan to implement more algorithms that work for larger networks, as well as algorithms like[2] which take an optimal sequence for a closed network and generate optimal sequences for environments of each tensor in the network, which is helpful for computing gradients of tensor networks.

diff --git a/previews/PR1514/DMRG.html b/previews/PR1514/DMRG.html index 5ff88ad802..ca01b96e43 100644 --- a/previews/PR1514/DMRG.html +++ b/previews/PR1514/DMRG.html @@ -2,4 +2,4 @@ DMRG · ITensors.jl

DMRG

ITensors.ITensorMPS.dmrgFunction
dmrg(H::MPO, psi0::MPS; kwargs...)
 dmrg(H::MPO, psi0::MPS, sweeps::Sweeps; kwargs...)

Use the density matrix renormalization group (DMRG) algorithm to optimize a matrix product state (MPS) such that it is the eigenvector of lowest eigenvalue of a Hermitian matrix H, represented as a matrix product operator (MPO).

dmrg(Hs::Vector{MPO}, psi0::MPS; kwargs...)
 dmrg(Hs::Vector{MPO}, psi0::MPS, sweeps::Sweeps; kwargs...)

Use the density matrix renormalization group (DMRG) algorithm to optimize a matrix product state (MPS) such that it is the eigenvector of lowest eigenvalue of a Hermitian matrix H. This version of dmrg accepts a representation of H as a Vector of MPOs, Hs = [H1, H2, H3, ...] such that H is defined as H = H1 + H2 + H3 + ... Note that this sum of MPOs is not actually computed; rather the set of MPOs [H1,H2,H3,..] is efficiently looped over at each step of the DMRG algorithm when optimizing the MPS.

dmrg(H::MPO, Ms::Vector{MPS}, psi0::MPS; weight=1.0, kwargs...)
-dmrg(H::MPO, Ms::Vector{MPS}, psi0::MPS, sweeps::Sweeps; weight=1.0, kwargs...)

Use the density matrix renormalization group (DMRG) algorithm to optimize a matrix product state (MPS) such that it is the eigenvector of lowest eigenvalue of a Hermitian matrix H, subject to the constraint that the MPS is orthogonal to each of the MPS provided in the Vector Ms. The orthogonality constraint is approximately enforced by adding to H terms of the form w|M1><M1| + w|M2><M2| + ... where Ms=[M1, M2, ...] and w is the "weight" parameter, which can be adjusted through the optional weight keyword argument.

Note

dmrg will report the energy of the operator H + w|M1><M1| + w|M2><M2| + ..., not the operator H. If you want the expectation value of the MPS eigenstate with respect to just H, you can compute it yourself with an observer or after DMRG is run with inner(psi', H, psi).

The MPS psi0 is used to initialize the MPS to be optimized.

The number of sweeps of thd DMRG algorithm is controlled by passing the nsweeps keyword argument. The keyword arguments maxdim, cutoff, noise, and mindim can also be passed to control the cost versus accuracy of the algorithm - see below for details.

Alternatively the number of sweeps and accuracy parameters can be passed through a Sweeps object, though this interface is no longer preferred.

Returns:

  • energy::Number - eigenvalue of the optimized MPS
  • psi::MPS - optimized MPS

Keyword arguments:

  • nsweeps::Int - number of "sweeps" of DMRG to perform

Optional keyword arguments:

  • maxdim - integer or array of integers specifying the maximum size allowed for the bond dimension or rank of the MPS being optimized.
  • cutoff - float or array of floats specifying the truncation error cutoff or threshold to use for truncating the bond dimension or rank of the MPS.
  • eigsolve_krylovdim::Int = 3 - maximum dimension of Krylov space used to locally solve the eigenvalue problem. Try setting to a higher value if convergence is slow or the Hamiltonian is close to a critical point. [krylovkit]
  • eigsolve_tol::Number = 1e-14 - Krylov eigensolver tolerance. [krylovkit]
  • eigsolve_maxiter::Int = 1 - number of times the Krylov subspace can be rebuilt. [krylovkit]
  • eigsolve_verbosity::Int = 0 - verbosity level of the Krylov solver. Warning: enabling this will lead to a lot of outputs to the terminal. [krylovkit]
  • ishermitian=true - boolean specifying if dmrg should assume the MPO (or more general linear operator) represents a Hermitian matrix. [krylovkit]
  • noise - float or array of floats specifying strength of the "noise term" to use to aid convergence.
  • mindim - integer or array of integers specifying the minimum size of the bond dimension or rank, if possible.
  • outputlevel::Int = 1 - larger outputlevel values make DMRG print more information and 0 means no output.
  • observer - object implementing the Observer interface which can perform measurements and stop DMRG early.
  • write_when_maxdim_exceeds::Int - when the allowed maxdim exceeds this value, begin saving tensors to disk to free RAM memory in large calculations
  • write_path::String = tempdir() - path to use to save files to disk (to save RAM) when maxdim exceeds the write_when_maxdim_exceeds option, if set
source
  • krylovkitThe dmrg function in ITensors.jl currently uses the eigsolve function in KrylovKit.jl as the internal the eigensolver. See the KrylovKit.jl documention on the eigsolve function for more details: KrylovKit.eigsolve.
+dmrg(H::MPO, Ms::Vector{MPS}, psi0::MPS, sweeps::Sweeps; weight=1.0, kwargs...)

Use the density matrix renormalization group (DMRG) algorithm to optimize a matrix product state (MPS) such that it is the eigenvector of lowest eigenvalue of a Hermitian matrix H, subject to the constraint that the MPS is orthogonal to each of the MPS provided in the Vector Ms. The orthogonality constraint is approximately enforced by adding to H terms of the form w|M1><M1| + w|M2><M2| + ... where Ms=[M1, M2, ...] and w is the "weight" parameter, which can be adjusted through the optional weight keyword argument.

Note

dmrg will report the energy of the operator H + w|M1><M1| + w|M2><M2| + ..., not the operator H. If you want the expectation value of the MPS eigenstate with respect to just H, you can compute it yourself with an observer or after DMRG is run with inner(psi', H, psi).

The MPS psi0 is used to initialize the MPS to be optimized.

The number of sweeps of thd DMRG algorithm is controlled by passing the nsweeps keyword argument. The keyword arguments maxdim, cutoff, noise, and mindim can also be passed to control the cost versus accuracy of the algorithm - see below for details.

Alternatively the number of sweeps and accuracy parameters can be passed through a Sweeps object, though this interface is no longer preferred.

Returns:

Keyword arguments:

Optional keyword arguments:

source
diff --git a/previews/PR1514/DMRGObserver.html b/previews/PR1514/DMRGObserver.html index f2c7f08a6e..97f44745f1 100644 --- a/previews/PR1514/DMRGObserver.html +++ b/previews/PR1514/DMRGObserver.html @@ -7,8 +7,8 @@ println("Total Sz after sweep $sw = ", sum(Szs)/N) end

Constructors

ITensors.ITensorMPS.DMRGObserverMethod
DMRGObserver(;energy_tol=0.0,
               minsweeps=2,
-              energy_type=Float64)

Construct a DMRGObserver by providing the energy tolerance used for early stopping, and minimum number of sweeps that must be done.

Optional keyword arguments:

  • energy_tol: if the energy from one sweep to the next no longer changes by more than this amount, stop after the current sweep
  • minsweeps: do at least this many sweeps
  • energy_type: type to use when storing energies at each step
source
ITensors.ITensorMPS.DMRGObserverMethod
DMRGObserver(ops::Vector{String},
+              energy_type=Float64)

Construct a DMRGObserver by providing the energy tolerance used for early stopping, and minimum number of sweeps that must be done.

Optional keyword arguments:

  • energy_tol: if the energy from one sweep to the next no longer changes by more than this amount, stop after the current sweep
  • minsweeps: do at least this many sweeps
  • energy_type: type to use when storing energies at each step
source
ITensors.ITensorMPS.DMRGObserverMethod
DMRGObserver(ops::Vector{String},
              sites::Vector{<:Index};
              energy_tol=0.0,
              minsweeps=2,
-             energy_type=Float64)

Construct a DMRGObserver, provide an array of ops of operator names which are strings recognized by the op function. Each of these operators will be measured on every site during every step of DMRG and the results recorded inside the DMRGOberver for later analysis. The array sites is the basis of sites used to define the MPS and MPO for the DMRG calculation.

Optionally, one can provide an energy tolerance used for early stopping, and minimum number of sweeps that must be done.

Optional keyword arguments:

  • energy_tol: if the energy from one sweep to the next no longer changes by more than this amount, stop after the current sweep
  • minsweeps: do at least this many sweeps
  • energy_type: type to use when storing energies at each step
source

Methods

ITensors.ITensorMPS.measurementsMethod
measurements(o::DMRGObserver)

After using a DMRGObserver object o within a DMRG calculation, retrieve a dictionary of measurement results, with the keys being operator names and values being DMRGMeasurement objects.

source
ITensors.ITensorMPS.DMRGMeasurementType

A DMRGMeasurement object is an alias for Vector{Vector{Float64}}, in other words an array of arrays of real numbers.

Given a DMRGMeasurement M,the result for the measurement on sweep n and site i as M[n][i].

source
ITensors.ITensorMPS.energiesMethod
energies(o::DMRGObserver)

After using a DMRGObserver object o within a DMRG calculation, retrieve an array of the energy after each sweep.

source
+ energy_type=Float64)

Construct a DMRGObserver, provide an array of ops of operator names which are strings recognized by the op function. Each of these operators will be measured on every site during every step of DMRG and the results recorded inside the DMRGOberver for later analysis. The array sites is the basis of sites used to define the MPS and MPO for the DMRG calculation.

Optionally, one can provide an energy tolerance used for early stopping, and minimum number of sweeps that must be done.

Optional keyword arguments:

source

Methods

ITensors.ITensorMPS.measurementsMethod
measurements(o::DMRGObserver)

After using a DMRGObserver object o within a DMRG calculation, retrieve a dictionary of measurement results, with the keys being operator names and values being DMRGMeasurement objects.

source
ITensors.ITensorMPS.DMRGMeasurementType

A DMRGMeasurement object is an alias for Vector{Vector{Float64}}, in other words an array of arrays of real numbers.

Given a DMRGMeasurement M,the result for the measurement on sweep n and site i as M[n][i].

source
ITensors.ITensorMPS.energiesMethod
energies(o::DMRGObserver)

After using a DMRGObserver object o within a DMRG calculation, retrieve an array of the energy after each sweep.

source
diff --git a/previews/PR1514/DeveloperGuide.html b/previews/PR1514/DeveloperGuide.html index 6fc372622f..171780ac74 100644 --- a/previews/PR1514/DeveloperGuide.html +++ b/previews/PR1514/DeveloperGuide.html @@ -29,4 +29,4 @@ # Call fA like this: fA(my_callback, psi; callback_args = (; a, b)) -
  • External (non-ITensor) Functions: Though it requires judgment in each case, if the keyword arguments an external (non-ITensor) function accepts are small in number, not expected to change, and known ahead of time, try to list them explicitly if possible (rather than forwarding with kwargs...). Possible exceptions could be if you want to make use of defaults defined for keyword arguments of an external function.

  • +
  • External (non-ITensor) Functions: Though it requires judgment in each case, if the keyword arguments an external (non-ITensor) function accepts are small in number, not expected to change, and known ahead of time, try to list them explicitly if possible (rather than forwarding with kwargs...). Possible exceptions could be if you want to make use of defaults defined for keyword arguments of an external function.

  • diff --git a/previews/PR1514/Einsum.html b/previews/PR1514/Einsum.html index 32e401cdc1..009d544372 100644 --- a/previews/PR1514/Einsum.html +++ b/previews/PR1514/Einsum.html @@ -1,23 +1,23 @@ -ITensor indices and Einstein notation · ITensors.jl

    ITensor Index identity: dimension labels and Einstein notation

    Many tensor contraction libraries use Einstein notation, such as NumPy's einsum function, ncon, and various Julia packages such as TensorOperations.jl, Tullio.jl, OMEinsum.jl, and Einsum.jl, among others.

    ITensor also uses Einstein notation, however the labels are stored inside the tensor and carried around with them during various operations. In addition, the labels that determine if tensor indices match with each other, and therefore automatically contract when doing * or match when adding or subtracting, are more sophisticated than simple characters or strings. ITensor indices are given a unique random ID number when they are constructed, and additionally users can add additional information like prime levels and tags which uniquely determine an Index. This is in contrast to simpler implementations of the same idea, such as the NamedDims.jl package, which only allow symbols as the metadata for uniquely identifying a tensor/array dimension.

    Index identity

    Here is an illustration of how the different types of Index metadata (random ID, prime level, and tags) work for Index identity:

    julia> i = Index(2)(dim=2|id=709)
    julia> j = Index(2)(dim=2|id=813)
    julia> i == jfalse
    julia> id(i)0xe4cb64c289a7d6dd
    julia> id(j)0x543f8cf918cfebdd
    julia> ip = i'(dim=2|id=709)'
    julia> ip == ifalse
    julia> plev(i) == 0true
    julia> plev(ip) == 1true
    julia> noprime(ip) == itrue
    julia> ix = addtags(i, "x")(dim=2|id=709|"x")
    julia> ix == ifalse
    julia> removetags(ix, "x") == itrue
    julia> ixyz = addtags(ix, "y,z")(dim=2|id=709|"x,y,z")
    julia> ixyz == addtags(i, "z,y,x")true

    The different metadata that are stored inside of ITensor indices that determine their identity are useful in different contexts. The random ID is particularly useful in the case when a new Index needs to be generated internally by ITensor, such as when performing a matrix factorization. In the case of a matrix factorization, we want to make sure that the new Index will not accidentally clash with an existing one, for example:

    julia> i = Index(2, "i")(dim=2|id=51|"i")
    julia> j = Index(2, "j")(dim=2|id=751|"j")
    julia> A = random_itensor(i, j)ITensor ord=2 (dim=2|id=51|"i") (dim=2|id=751|"j") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> U, S, V = svd(A, i; lefttags="i", righttags="j");
    julia> inds(U)((dim=2|id=51|"i"), (dim=2|id=962|"i"))
    julia> inds(S)((dim=2|id=962|"i"), (dim=2|id=65|"j"))
    julia> inds(V)((dim=2|id=751|"j"), (dim=2|id=65|"j"))
    julia> norm(U * S * V - A)2.7128146828510983e-15

    You can see that it would have been a problem here if there wasn't a new ID assigned to the Index, since it would have clashed with the original index. In this case, it could be avoided by giving the new indices different tags (with the keyword arguments lefttags and righttags), but in more complicated examples where it is not practical to do that (such as a case where many new indices are being introduced, for example for a tensor train (TT)/matrix product state (MPS)), it is convenient to not force users to come up with unique prime levels or tags themselves. It can also help to avoid accidental contractions in more complicated tensor network algorithms where there are many indices that can potentially have the same prime levels or tags.

    In contrast, using multiple indices with the same Index ID but different prime levels and tags can be useful in situations where there is a more fundamental relationship between the spaces. For example, in the case of an ITensor corresponding to a Hermitian operator, it is helpful to make the bra space and ket spaces the same up to a prime level:

    i = Index(2, "i")
    +ITensor indices and Einstein notation · ITensors.jl

    ITensor Index identity: dimension labels and Einstein notation

    Many tensor contraction libraries use Einstein notation, such as NumPy's einsum function, ncon, and various Julia packages such as TensorOperations.jl, Tullio.jl, OMEinsum.jl, and Einsum.jl, among others.

    ITensor also uses Einstein notation, however the labels are stored inside the tensor and carried around with them during various operations. In addition, the labels that determine if tensor indices match with each other, and therefore automatically contract when doing * or match when adding or subtracting, are more sophisticated than simple characters or strings. ITensor indices are given a unique random ID number when they are constructed, and additionally users can add additional information like prime levels and tags which uniquely determine an Index. This is in contrast to simpler implementations of the same idea, such as the NamedDims.jl package, which only allow symbols as the metadata for uniquely identifying a tensor/array dimension.

    Index identity

    Here is an illustration of how the different types of Index metadata (random ID, prime level, and tags) work for Index identity:

    julia> i = Index(2)(dim=2|id=464)
    julia> j = Index(2)(dim=2|id=833)
    julia> i == jfalse
    julia> id(i)0x0b1e7ab3995994c0
    julia> id(j)0x1f5abfdd7f972e01
    julia> ip = i'(dim=2|id=464)'
    julia> ip == ifalse
    julia> plev(i) == 0true
    julia> plev(ip) == 1true
    julia> noprime(ip) == itrue
    julia> ix = addtags(i, "x")(dim=2|id=464|"x")
    julia> ix == ifalse
    julia> removetags(ix, "x") == itrue
    julia> ixyz = addtags(ix, "y,z")(dim=2|id=464|"x,y,z")
    julia> ixyz == addtags(i, "z,y,x")true

    The different metadata that are stored inside of ITensor indices that determine their identity are useful in different contexts. The random ID is particularly useful in the case when a new Index needs to be generated internally by ITensor, such as when performing a matrix factorization. In the case of a matrix factorization, we want to make sure that the new Index will not accidentally clash with an existing one, for example:

    julia> i = Index(2, "i")(dim=2|id=789|"i")
    julia> j = Index(2, "j")(dim=2|id=573|"j")
    julia> A = random_itensor(i, j)ITensor ord=2 (dim=2|id=789|"i") (dim=2|id=573|"j") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> U, S, V = svd(A, i; lefttags="i", righttags="j");
    julia> inds(U)((dim=2|id=789|"i"), (dim=2|id=580|"i"))
    julia> inds(S)((dim=2|id=580|"i"), (dim=2|id=983|"j"))
    julia> inds(V)((dim=2|id=573|"j"), (dim=2|id=983|"j"))
    julia> norm(U * S * V - A)2.7128146828510983e-15

    You can see that it would have been a problem here if there wasn't a new ID assigned to the Index, since it would have clashed with the original index. In this case, it could be avoided by giving the new indices different tags (with the keyword arguments lefttags and righttags), but in more complicated examples where it is not practical to do that (such as a case where many new indices are being introduced, for example for a tensor train (TT)/matrix product state (MPS)), it is convenient to not force users to come up with unique prime levels or tags themselves. It can also help to avoid accidental contractions in more complicated tensor network algorithms where there are many indices that can potentially have the same prime levels or tags.

    In contrast, using multiple indices with the same Index ID but different prime levels and tags can be useful in situations where there is a more fundamental relationship between the spaces. For example, in the case of an ITensor corresponding to a Hermitian operator, it is helpful to make the bra space and ket spaces the same up to a prime level:

    i = Index(2, "i")
     j = Index(3, "j")
     A = random_itensor(i', j', dag(i), dag(j))
     H = 0.5 * (A + swapprime(dag(A), 0 => 1))
     v = random_itensor(i, j)
     Hv = noprime(H * v)
     vH = dag(v)' * H
    -norm(Hv - dag(vH))

    Note that we have added dag in a few places, which is superfluous in this case since the tensors are real and dense but become important when the tensors are complex and/or have symmetries. You can see that in this case, it is very useful to relate the bra and ket spaces by prime levels, since it makes it much easier to perform operations that map from one space to another. We could have created A from 4 entirely different indices with different ID numbers, but it would make the operations a bit more cumbersome, as shown below:

    julia> i = Index(2, "i")(dim=2|id=298|"i")
    julia> j = Index(3, "j")(dim=3|id=750|"j")
    julia> ip = Index(2, "i")(dim=2|id=240|"i")
    julia> jp = Index(3, "jp")(dim=3|id=430|"jp")
    julia> A = random_itensor(ip, jp, dag(i), dag(j))ITensor ord=4 (dim=2|id=240|"i") (dim=3|id=430|"jp") (dim=2|id=298|"i") (dim=3|id=750|"j") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> H = 0.5 * (A + swapinds(dag(A), (i, j), (ip, jp)))ITensor ord=4 (dim=2|id=240|"i") (dim=3|id=430|"jp") (dim=2|id=298|"i") (dim=3|id=750|"j") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> v = random_itensor(i, j)ITensor ord=2 (dim=2|id=298|"i") (dim=3|id=750|"j") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> Hv = replaceinds(H * v, (ip, jp) => (i, j))ITensor ord=2 (dim=2|id=298|"i") (dim=3|id=750|"j") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> vH = replaceinds(dag(v), (i, j) => (ip, jp)) * HITensor ord=2 (dim=2|id=298|"i") (dim=3|id=750|"j") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> norm(Hv - dag(vH))0.0

    Relationship to other Einstein notation-based libraries

    Here we show examples of different ways to perform the contraction "ab,bc,cd->ad" in ITensor.

    julia> da, dc = 2, 3;
    julia> db, dd = da, dc;
    julia> tags = ("a", "b", "c", "d");
    julia> dims = (da, db, dc, dd);
    julia> a, b, c, d = Index.(dims, tags);
    julia> Aab = random_itensor(a, b)ITensor ord=2 (dim=2|id=924|"a") (dim=2|id=552|"b") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> Bbc = random_itensor(b, c)ITensor ord=2 (dim=2|id=552|"b") (dim=3|id=789|"c") +norm(Hv - dag(vH))

    Note that we have added dag in a few places, which is superfluous in this case since the tensors are real and dense but become important when the tensors are complex and/or have symmetries. You can see that in this case, it is very useful to relate the bra and ket spaces by prime levels, since it makes it much easier to perform operations that map from one space to another. We could have created A from 4 entirely different indices with different ID numbers, but it would make the operations a bit more cumbersome, as shown below:

    julia> i = Index(2, "i")(dim=2|id=165|"i")
    julia> j = Index(3, "j")(dim=3|id=697|"j")
    julia> ip = Index(2, "i")(dim=2|id=308|"i")
    julia> jp = Index(3, "jp")(dim=3|id=168|"jp")
    julia> A = random_itensor(ip, jp, dag(i), dag(j))ITensor ord=4 (dim=2|id=308|"i") (dim=3|id=168|"jp") (dim=2|id=165|"i") (dim=3|id=697|"j") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> H = 0.5 * (A + swapinds(dag(A), (i, j), (ip, jp)))ITensor ord=4 (dim=2|id=308|"i") (dim=3|id=168|"jp") (dim=2|id=165|"i") (dim=3|id=697|"j") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> v = random_itensor(i, j)ITensor ord=2 (dim=2|id=165|"i") (dim=3|id=697|"j") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> Hv = replaceinds(H * v, (ip, jp) => (i, j))ITensor ord=2 (dim=2|id=165|"i") (dim=3|id=697|"j") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> vH = replaceinds(dag(v), (i, j) => (ip, jp)) * HITensor ord=2 (dim=2|id=165|"i") (dim=3|id=697|"j") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> norm(Hv - dag(vH))0.0

    Relationship to other Einstein notation-based libraries

    Here we show examples of different ways to perform the contraction "ab,bc,cd->ad" in ITensor.

    julia> da, dc = 2, 3;
    julia> db, dd = da, dc;
    julia> tags = ("a", "b", "c", "d");
    julia> dims = (da, db, dc, dd);
    julia> a, b, c, d = Index.(dims, tags);
    julia> Aab = random_itensor(a, b)ITensor ord=2 (dim=2|id=467|"a") (dim=2|id=396|"b") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> Bbc = random_itensor(b, c)ITensor ord=2 (dim=2|id=396|"b") (dim=3|id=410|"c") NDTensors.Dense{Float64, Vector{Float64}}
    julia> Ccd = random_itensor(c, d) - # "ab,bc,cd->ad"ITensor ord=2 (dim=3|id=789|"c") (dim=3|id=766|"d") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> out1 = Aab * Bbc * CcdITensor ord=2 (dim=2|id=924|"a") (dim=3|id=766|"d") + # "ab,bc,cd->ad"ITensor ord=2 (dim=3|id=410|"c") (dim=3|id=722|"d") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> out1 = Aab * Bbc * CcdITensor ord=2 (dim=2|id=467|"a") (dim=3|id=722|"d") NDTensors.Dense{Float64, Vector{Float64}}
    julia> @show hassameinds(out1, (a, d)) # @@ -25,9 +25,9 @@ # # "ba,bc,dc->ad"hassameinds(out1, (a, d)) = true -true
    julia> Aba = replaceinds(Aab, (a, b) => (b, a))ITensor ord=2 (dim=2|id=552|"b") (dim=2|id=924|"a") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> Cdc = replaceinds(Ccd, (c, d) => (d, c))ITensor ord=2 (dim=3|id=766|"d") (dim=3|id=789|"c") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> out2 = Aba * Bbc * CdcITensor ord=2 (dim=2|id=924|"a") (dim=3|id=766|"d") +true
    julia> Aba = replaceinds(Aab, (a, b) => (b, a))ITensor ord=2 (dim=2|id=396|"b") (dim=2|id=467|"a") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> Cdc = replaceinds(Ccd, (c, d) => (d, c))ITensor ord=2 (dim=3|id=722|"d") (dim=3|id=410|"c") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> out2 = Aba * Bbc * CdcITensor ord=2 (dim=2|id=467|"a") (dim=3|id=722|"d") NDTensors.Dense{Float64, Vector{Float64}}
    julia> @show hassameinds(out2, (a, d)) # @@ -38,9 +38,9 @@ # since it doesn't check if the indices # are compatible in dimension, # so is not recommended in general.hassameinds(out2, (a, d)) = true -true
    julia> using ITensors: setinds
    julia> Aba = setinds(Aab, (b, a))ITensor ord=2 (dim=2|id=552|"b") (dim=2|id=924|"a") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> Cdc = setinds(Ccd, (d, c))ITensor ord=2 (dim=3|id=766|"d") (dim=3|id=789|"c") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> out2 = Aba * Bbc * CdcITensor ord=2 (dim=2|id=924|"a") (dim=3|id=766|"d") +true
    julia> using ITensors: setinds
    julia> Aba = setinds(Aab, (b, a))ITensor ord=2 (dim=2|id=396|"b") (dim=2|id=467|"a") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> Cdc = setinds(Ccd, (d, c))ITensor ord=2 (dim=3|id=722|"d") (dim=3|id=410|"c") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> out2 = Aba * Bbc * CdcITensor ord=2 (dim=2|id=467|"a") (dim=3|id=722|"d") NDTensors.Dense{Float64, Vector{Float64}}
    julia> @show hassameinds(out2, (a, d)) # @@ -48,14 +48,14 @@ # the indices were made with these # prime levels in the first place) #hassameinds(out2, (a, d)) = true -true
    julia> a = Index(da, "a")(dim=2|id=965|"a")
    julia> c = Index(dc, "c")(dim=3|id=414|"c")
    julia> b, d = a', c'((dim=2|id=965|"a")', (dim=3|id=414|"c")')
    julia> Aab = random_itensor(a, b)ITensor ord=2 (dim=2|id=965|"a") (dim=2|id=965|"a")' -NDTensors.Dense{Float64, Vector{Float64}}
    julia> Bbc = random_itensor(b, c)ITensor ord=2 (dim=2|id=965|"a")' (dim=3|id=414|"c") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> Ccd = random_itensor(c, d)ITensor ord=2 (dim=3|id=414|"c") (dim=3|id=414|"c")' -NDTensors.Dense{Float64, Vector{Float64}}
    julia> out1 = Aab * Bbc * CcdITensor ord=2 (dim=2|id=965|"a") (dim=3|id=414|"c")' +true
    julia> a = Index(da, "a")(dim=2|id=76|"a")
    julia> c = Index(dc, "c")(dim=3|id=798|"c")
    julia> b, d = a', c'((dim=2|id=76|"a")', (dim=3|id=798|"c")')
    julia> Aab = random_itensor(a, b)ITensor ord=2 (dim=2|id=76|"a") (dim=2|id=76|"a")' +NDTensors.Dense{Float64, Vector{Float64}}
    julia> Bbc = random_itensor(b, c)ITensor ord=2 (dim=2|id=76|"a")' (dim=3|id=798|"c") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> Ccd = random_itensor(c, d)ITensor ord=2 (dim=3|id=798|"c") (dim=3|id=798|"c")' +NDTensors.Dense{Float64, Vector{Float64}}
    julia> out1 = Aab * Bbc * CcdITensor ord=2 (dim=2|id=76|"a") (dim=3|id=798|"c")' NDTensors.Dense{Float64, Vector{Float64}}
    julia> @show hassameinds(out1, (a, d))hassameinds(out1, (a, d)) = true -true
    julia> Aba = swapprime(Aab, 0 => 1)ITensor ord=2 (dim=2|id=965|"a")' (dim=2|id=965|"a") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> Cdc = swapprime(Ccd, 0 => 1)ITensor ord=2 (dim=3|id=414|"c")' (dim=3|id=414|"c") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> out2 = Aba * Bbc * CdcITensor ord=2 (dim=2|id=965|"a") (dim=3|id=414|"c")' +true
    julia> Aba = swapprime(Aab, 0 => 1)ITensor ord=2 (dim=2|id=76|"a")' (dim=2|id=76|"a") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> Cdc = swapprime(Ccd, 0 => 1)ITensor ord=2 (dim=3|id=798|"c")' (dim=3|id=798|"c") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> out2 = Aba * Bbc * CdcITensor ord=2 (dim=2|id=76|"a") (dim=3|id=798|"c")' NDTensors.Dense{Float64, Vector{Float64}}
    julia> @show hassameinds(out2, (a, d)) # @@ -63,14 +63,14 @@ # the indices were made with these # tags in the first place) #hassameinds(out2, (a, d)) = true -true
    julia> a = Index(da, "a")(dim=2|id=148|"a")
    julia> c = Index(dc, "c")(dim=3|id=970|"c")
    julia> b, d = settags(a, "b"), settags(c, "d")((dim=2|id=148|"b"), (dim=3|id=970|"d"))
    julia> Aab = random_itensor(a, b)ITensor ord=2 (dim=2|id=148|"a") (dim=2|id=148|"b") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> Bbc = random_itensor(b, c)ITensor ord=2 (dim=2|id=148|"b") (dim=3|id=970|"c") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> Ccd = random_itensor(c, d)ITensor ord=2 (dim=3|id=970|"c") (dim=3|id=970|"d") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> out1 = Aab * Bbc * CcdITensor ord=2 (dim=2|id=148|"a") (dim=3|id=970|"d") +true
    julia> a = Index(da, "a")(dim=2|id=243|"a")
    julia> c = Index(dc, "c")(dim=3|id=139|"c")
    julia> b, d = settags(a, "b"), settags(c, "d")((dim=2|id=243|"b"), (dim=3|id=139|"d"))
    julia> Aab = random_itensor(a, b)ITensor ord=2 (dim=2|id=243|"a") (dim=2|id=243|"b") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> Bbc = random_itensor(b, c)ITensor ord=2 (dim=2|id=243|"b") (dim=3|id=139|"c") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> Ccd = random_itensor(c, d)ITensor ord=2 (dim=3|id=139|"c") (dim=3|id=139|"d") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> out1 = Aab * Bbc * CcdITensor ord=2 (dim=2|id=243|"a") (dim=3|id=139|"d") NDTensors.Dense{Float64, Vector{Float64}}
    julia> @show hassameinds(out1, (a, d))hassameinds(out1, (a, d)) = true -true
    julia> Aba = swaptags(Aab, "a", "b")ITensor ord=2 (dim=2|id=148|"b") (dim=2|id=148|"a") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> Cdc = swaptags(Ccd, "c", "d")ITensor ord=2 (dim=3|id=970|"d") (dim=3|id=970|"c") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> out2 = Aba * Bbc * CdcITensor ord=2 (dim=2|id=148|"a") (dim=3|id=970|"d") +true
    julia> Aba = swaptags(Aab, "a", "b")ITensor ord=2 (dim=2|id=243|"b") (dim=2|id=243|"a") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> Cdc = swaptags(Ccd, "c", "d")ITensor ord=2 (dim=3|id=139|"d") (dim=3|id=139|"c") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> out2 = Aba * Bbc * CdcITensor ord=2 (dim=2|id=243|"a") (dim=3|id=139|"d") NDTensors.Dense{Float64, Vector{Float64}}
    julia> @show hassameinds(out2, (a, d)) # @@ -83,14 +83,14 @@ -1.26482 0.972584 0.0521089
    julia> C = randn(dc, dd)3×3 Matrix{Float64}: 0.073576 -0.572066 0.237091 -0.355748 -0.244972 0.662303 - 1.91204 1.69045 0.713253
    julia> tags = ("a", "b", "c", "d")("a", "b", "c", "d")
    julia> dims = (da, db, dc, dd)(2, 2, 3, 3)
    julia> a, b, c, d = Index.(dims, tags)((dim=2|id=536|"a"), (dim=2|id=767|"b"), (dim=3|id=268|"c"), (dim=3|id=765|"d"))
    julia> Aab = itensor(A, a, b)ITensor ord=2 (dim=2|id=536|"a") (dim=2|id=767|"b") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> Bbc = itensor(B, b, c)ITensor ord=2 (dim=2|id=767|"b") (dim=3|id=268|"c") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> Ccd = itensor(C, c, d)ITensor ord=2 (dim=3|id=268|"c") (dim=3|id=765|"d") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> out1 = Aab * Bbc * CcdITensor ord=2 (dim=2|id=536|"a") (dim=3|id=765|"d") + 1.91204 1.69045 0.713253
    julia> tags = ("a", "b", "c", "d")("a", "b", "c", "d")
    julia> dims = (da, db, dc, dd)(2, 2, 3, 3)
    julia> a, b, c, d = Index.(dims, tags)((dim=2|id=60|"a"), (dim=2|id=762|"b"), (dim=3|id=608|"c"), (dim=3|id=447|"d"))
    julia> Aab = itensor(A, a, b)ITensor ord=2 (dim=2|id=60|"a") (dim=2|id=762|"b") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> Bbc = itensor(B, b, c)ITensor ord=2 (dim=2|id=762|"b") (dim=3|id=608|"c") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> Ccd = itensor(C, c, d)ITensor ord=2 (dim=3|id=608|"c") (dim=3|id=447|"d") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> out1 = Aab * Bbc * CcdITensor ord=2 (dim=2|id=60|"a") (dim=3|id=447|"d") NDTensors.Dense{Float64, Vector{Float64}}
    julia> @show hassameinds(out1, (a, d))hassameinds(out1, (a, d)) = true -true
    julia> Aba = itensor(A, b, a)ITensor ord=2 (dim=2|id=767|"b") (dim=2|id=536|"a") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> Cdc = itensor(C, d, c)ITensor ord=2 (dim=3|id=765|"d") (dim=3|id=268|"c") -NDTensors.Dense{Float64, Vector{Float64}}
    julia> out2 = Aba * Bbc * CdcITensor ord=2 (dim=2|id=536|"a") (dim=3|id=765|"d") +true
    julia> Aba = itensor(A, b, a)ITensor ord=2 (dim=2|id=762|"b") (dim=2|id=60|"a") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> Cdc = itensor(C, d, c)ITensor ord=2 (dim=3|id=447|"d") (dim=3|id=608|"c") +NDTensors.Dense{Float64, Vector{Float64}}
    julia> out2 = Aba * Bbc * CdcITensor ord=2 (dim=2|id=60|"a") (dim=3|id=447|"d") NDTensors.Dense{Float64, Vector{Float64}}
    julia> @show hassameinds(out2, (a, d)) # @@ -103,4 +103,4 @@ # #out2 = A[b, a] * B[b, c] * C[d, c] #@show hassameinds(out2, (a, d))hassameinds(out2, (a, d)) = true -true
    +true
    diff --git a/previews/PR1514/HDF5FileFormats.html b/previews/PR1514/HDF5FileFormats.html index 4a2ab3bb1c..768b31eb12 100644 --- a/previews/PR1514/HDF5FileFormats.html +++ b/previews/PR1514/HDF5FileFormats.html @@ -1,2 +1,2 @@ -HDF5 File Formats · ITensors.jl

    HDF5 File Formats

    This page lists the formats for the HDF5 representations of various types in the ITensors module.

    HDF5 is a portable file format which has a directory structure similar to a file system. In addition to containing "groups" (= directories) and "datasets" (= files), groups can have "attributes" appended to them, which are similar to 'tags' or 'keywords'. Unless otherwise specified, integers are 64 bit and are signed (H5T_STD_I64LE) unless explicitly stated. (For example, the "id" field of the Index type is stored as an unsigned 64 bit integer (H5T_STD_U64LE).)

    Each type in ITensor which is writeable to HDF5 is written to its own group, with the name of the group either specified by the user or specified to some default value when it is a subgroup of another ITensor type (for example, the Index type saves its TagSet in a subgroup named "tags").

    Each group corresponding to an ITensors type always carries the following attributes:

    • "type" –- a string such as Index or TagSet specifying the information necessary to determine the type of the object saved to the HDF5 group
    • "version" –- an integer specifying the file format version used to store the data. This version is in general different from the release version of ITensors.jl. The purpose of the version number is to aid in maintaining backwards compatibility, while allowing the format to be occasionally changed.

    The C++ version of ITensor uses exactly the same file formats listed below, for the purpose of interoperability with the Julia version of ITensor, even though conventions such as the "type" field values are Julia-centric.

    TagSet

    HDF5 file format for the ITensors.TagSet type.

    Attributes:

    • "version" = 1
    • "type" = "TagSet"

    Datasets and Subgroups:

    • "tags" [string] = a comma separated string of the tags in the TagSet

    QN

    HDF5 file format for the ITensors.QN type.

    Attributes:

    • "version" = 1
    • "type" = "QN"

    Datasets and Subgroups:

    • "names" [group] = array of strings (length 4) of names of quantum numbers
    • "vals" [group] = array of integers (length 4) of quantum number values
    • "mods" [group] = array of integers (length 4) of moduli of quantum numbers

    QNBlocks

    HDF5 file format for the ITensors.QNBlocks type. (Note: QNBlocks is equivalent to Vector{Pair{QN, Int64}}.)

    Attributes:

    • "version" = 1
    • "type" = "QNBlocks"

    Datasets and Subgroups:

    • "length" [integer] = the number of blocks (length of Vector)
    • "dims" [group] = array of (integer) dimensions of each block
    • "QN[n]" [group] = these groups "QN[1]", "QN[2]", etc. correspond to the QN of each block

    Index

    HDF5 file format for the ITensors.Index type.

    Attributes:

    • "version" = 1
    • "type" = "Index"
    • "space_type" = "Int" if the Index is a regular, dense Index or "QNBlocks" if the Index is a QNIndex (carries QN subspace information)

    Datasets and Subgroups:

    • "id" [unsigned integer] = id number of the Index
    • "dim" [integer] = dimension of the Index
    • "dir" [integer] = arrow direction of the Index, +1 for ITensors.Out and -1 for ITensors.In
    • "plev" [integer] = prime level of the Index
    • "tags" [group] = the TagSet of the Index

    Optional Datasets and Subgroups:

    • "space" [group] = if the "space_type" attribute is "QNBlocks", this group is present and represents a QNBlocks object

    IndexSet

    HDF5 file format for types in the Union type ITensors.Indices which includes IndexSet and tuples of Index objects.

    Attributes:

    • "version" = 1
    • "type" = "IndexSet"

    Datasets and Subgroups:

    • "length" [integer] = number of indices
    • "index_n" [group] = for n=1 to n=length each of these groups contains an Index

    ITensor

    HDF5 file format for the ITensors.ITensor type.

    Attributes:

    • "version" = 1
    • "type" = "ITensor"

    Datasets and Subgroups:

    • "inds" [group] = indices of the ITensor
    • "storage" [group] = storage of the ITensor (note that some earlier versions of ITensors.jl may call this group "store")

    NDTensors.Dense

    HDF5 file format for objects which are subtypes of ITensors.NDTensors.Dense.

    Attributes:

    • "version" = 1
    • "type" = "Dense{Float64}" or "Dense{ComplexF64}"

    Datasets and Subgroups:

    • "data" = array of either real or complex values (in the same dataset format used by the HDF5.jl library for storing Vector{Float64} or Vector{ComplexF64})

    NDTensors.BlockSparse

    HDF5 file format for objects which are subtypes of ITensors.NDTensors.BlockSparse.

    Attributes:

    • "version" = 1
    • "type" = "BlockSparse{Float64}" or "BlockSparse{ComplexF64}"

    Datasets and Subgroups:

    • "ndims" [integer] = number of dimensions (order) of the tensor
    • "offsets" = block offset data flattened into an array of integers
    • "data" = array of either real or complex values (in the same dataset format used by the HDF5.jl library for storing Vector{Float64} or Vector{ComplexF64})

    MPS

    HDF5 file format for ITensors.MPS

    Attributes:

    • "version" = 1
    • "type" = "MPS"

    Datasets and Subgroups:

    • "length" [integer] = number of tensors of the MPS
    • "rlim" [integer] = right orthogonality limit
    • "llim" [integer] = left orthogonality limit
    • "MPS[n]" [group,ITensor] = each of these groups, where n=1,...,length, stores the nth ITensor of the MPS

    MPO

    HDF5 file format for ITensors.MPO

    Attributes:

    • "version" = 1
    • "type" = "MPO"

    Datasets and Subgroups:

    • "length" [integer] = number of tensors of the MPO
    • "rlim" [integer] = right orthogonality limit
    • "llim" [integer] = left orthogonality limit
    • "MPO[n]" [group,ITensor] = each of these groups, where n=1,...,length, stores the nth ITensor of the MPO
    +HDF5 File Formats · ITensors.jl

    HDF5 File Formats

    This page lists the formats for the HDF5 representations of various types in the ITensors module.

    HDF5 is a portable file format which has a directory structure similar to a file system. In addition to containing "groups" (= directories) and "datasets" (= files), groups can have "attributes" appended to them, which are similar to 'tags' or 'keywords'. Unless otherwise specified, integers are 64 bit and are signed (H5T_STD_I64LE) unless explicitly stated. (For example, the "id" field of the Index type is stored as an unsigned 64 bit integer (H5T_STD_U64LE).)

    Each type in ITensor which is writeable to HDF5 is written to its own group, with the name of the group either specified by the user or specified to some default value when it is a subgroup of another ITensor type (for example, the Index type saves its TagSet in a subgroup named "tags").

    Each group corresponding to an ITensors type always carries the following attributes:

    • "type" –- a string such as Index or TagSet specifying the information necessary to determine the type of the object saved to the HDF5 group
    • "version" –- an integer specifying the file format version used to store the data. This version is in general different from the release version of ITensors.jl. The purpose of the version number is to aid in maintaining backwards compatibility, while allowing the format to be occasionally changed.

    The C++ version of ITensor uses exactly the same file formats listed below, for the purpose of interoperability with the Julia version of ITensor, even though conventions such as the "type" field values are Julia-centric.

    TagSet

    HDF5 file format for the ITensors.TagSet type.

    Attributes:

    • "version" = 1
    • "type" = "TagSet"

    Datasets and Subgroups:

    • "tags" [string] = a comma separated string of the tags in the TagSet

    QN

    HDF5 file format for the ITensors.QN type.

    Attributes:

    • "version" = 1
    • "type" = "QN"

    Datasets and Subgroups:

    • "names" [group] = array of strings (length 4) of names of quantum numbers
    • "vals" [group] = array of integers (length 4) of quantum number values
    • "mods" [group] = array of integers (length 4) of moduli of quantum numbers

    QNBlocks

    HDF5 file format for the ITensors.QNBlocks type. (Note: QNBlocks is equivalent to Vector{Pair{QN, Int64}}.)

    Attributes:

    • "version" = 1
    • "type" = "QNBlocks"

    Datasets and Subgroups:

    • "length" [integer] = the number of blocks (length of Vector)
    • "dims" [group] = array of (integer) dimensions of each block
    • "QN[n]" [group] = these groups "QN[1]", "QN[2]", etc. correspond to the QN of each block

    Index

    HDF5 file format for the ITensors.Index type.

    Attributes:

    • "version" = 1
    • "type" = "Index"
    • "space_type" = "Int" if the Index is a regular, dense Index or "QNBlocks" if the Index is a QNIndex (carries QN subspace information)

    Datasets and Subgroups:

    • "id" [unsigned integer] = id number of the Index
    • "dim" [integer] = dimension of the Index
    • "dir" [integer] = arrow direction of the Index, +1 for ITensors.Out and -1 for ITensors.In
    • "plev" [integer] = prime level of the Index
    • "tags" [group] = the TagSet of the Index

    Optional Datasets and Subgroups:

    • "space" [group] = if the "space_type" attribute is "QNBlocks", this group is present and represents a QNBlocks object

    IndexSet

    HDF5 file format for types in the Union type ITensors.Indices which includes IndexSet and tuples of Index objects.

    Attributes:

    • "version" = 1
    • "type" = "IndexSet"

    Datasets and Subgroups:

    • "length" [integer] = number of indices
    • "index_n" [group] = for n=1 to n=length each of these groups contains an Index

    ITensor

    HDF5 file format for the ITensors.ITensor type.

    Attributes:

    • "version" = 1
    • "type" = "ITensor"

    Datasets and Subgroups:

    • "inds" [group] = indices of the ITensor
    • "storage" [group] = storage of the ITensor (note that some earlier versions of ITensors.jl may call this group "store")

    NDTensors.Dense

    HDF5 file format for objects which are subtypes of ITensors.NDTensors.Dense.

    Attributes:

    • "version" = 1
    • "type" = "Dense{Float64}" or "Dense{ComplexF64}"

    Datasets and Subgroups:

    • "data" = array of either real or complex values (in the same dataset format used by the HDF5.jl library for storing Vector{Float64} or Vector{ComplexF64})

    NDTensors.BlockSparse

    HDF5 file format for objects which are subtypes of ITensors.NDTensors.BlockSparse.

    Attributes:

    • "version" = 1
    • "type" = "BlockSparse{Float64}" or "BlockSparse{ComplexF64}"

    Datasets and Subgroups:

    • "ndims" [integer] = number of dimensions (order) of the tensor
    • "offsets" = block offset data flattened into an array of integers
    • "data" = array of either real or complex values (in the same dataset format used by the HDF5.jl library for storing Vector{Float64} or Vector{ComplexF64})

    MPS

    HDF5 file format for ITensors.MPS

    Attributes:

    • "version" = 1
    • "type" = "MPS"

    Datasets and Subgroups:

    • "length" [integer] = number of tensors of the MPS
    • "rlim" [integer] = right orthogonality limit
    • "llim" [integer] = left orthogonality limit
    • "MPS[n]" [group,ITensor] = each of these groups, where n=1,...,length, stores the nth ITensor of the MPS

    MPO

    HDF5 file format for ITensors.MPO

    Attributes:

    • "version" = 1
    • "type" = "MPO"

    Datasets and Subgroups:

    • "length" [integer] = number of tensors of the MPO
    • "rlim" [integer] = right orthogonality limit
    • "llim" [integer] = left orthogonality limit
    • "MPO[n]" [group,ITensor] = each of these groups, where n=1,...,length, stores the nth ITensor of the MPO
    diff --git a/previews/PR1514/ITensorType.html b/previews/PR1514/ITensorType.html index 33074c0895..8fdc92632e 100644 --- a/previews/PR1514/ITensorType.html +++ b/previews/PR1514/ITensorType.html @@ -60,20 +60,20 @@ NDTensors.Dense{Float64,Array{Float64,1}} 2×2 -0.3674957028513448 1.6904886171664615 - 1.2579101497658178 -1.3559959053693322source

    Dense Constructors

    ITensors.ITensorMethod
    ITensor([::Type{ElT} = Float64, ]inds)
    +  1.2579101497658178  -1.3559959053693322
    source

    Dense Constructors

    ITensors.ITensorMethod
    ITensor([::Type{ElT} = Float64, ]inds)
     ITensor([::Type{ElT} = Float64, ]inds::Index...)

    Construct an ITensor filled with zeros having indices inds and element type ElT. If the element type is not specified, it defaults to Float64.

    The storage will have NDTensors.Dense type.

    Examples

    i = Index(2,"index_i")
     j = Index(4,"index_j")
     k = Index(3,"index_k")
     
     A = ITensor(i,j)
    -B = ITensor(ComplexF64,k,j)
    source
    ITensors.ITensorMethod
    ITensor([::Type{ElT} = Float64, ]::UndefInitializer, inds)
    +B = ITensor(ComplexF64,k,j)
    source
    ITensors.ITensorMethod
    ITensor([::Type{ElT} = Float64, ]::UndefInitializer, inds)
     ITensor([::Type{ElT} = Float64, ]::UndefInitializer, inds::Index...)

    Construct an ITensor filled with undefined elements having indices inds and element type ElT. If the element type is not specified, it defaults to Float64. One purpose for using this constructor is that initializing the elements in an undefined way is faster than initializing them to a set value such as zero.

    The storage will have NDTensors.Dense type.

    Examples

    i = Index(2,"index_i")
     j = Index(4,"index_j")
     k = Index(3,"index_k")
     
     A = ITensor(undef,i,j)
    -B = ITensor(ComplexF64,undef,k,j)
    source
    ITensors.ITensorMethod
    ITensor([ElT::Type, ]x::Number, inds)
    -ITensor([ElT::Type, ]x::Number, inds::Index...)

    Construct an ITensor with all elements set to x and indices inds.

    If x isa Int or x isa Complex{Int} then the elements will be set to float(x) unless specified otherwise by the first input.

    The storage will have NDTensors.Dense type.

    Examples

    ```julia i = Index(2,"indexi"); j = Index(4,"indexj"); k = Index(3,"index_k");

    A = ITensor(1.0, i, j) A = ITensor(1, i, j) # same as above B = ITensor(2.0+3.0im, j, k) ```

    !!! warning In future versions this may not automatically convert integer inputs with float, and in that case the particular element type should not be relied on.

    source
    ITensors.ITensorMethod
    ITensor([ElT::Type, ]A::AbstractArray, inds)
    +B = ITensor(ComplexF64,undef,k,j)
    source
    ITensors.ITensorMethod
    ITensor([ElT::Type, ]x::Number, inds)
    +ITensor([ElT::Type, ]x::Number, inds::Index...)

    Construct an ITensor with all elements set to x and indices inds.

    If x isa Int or x isa Complex{Int} then the elements will be set to float(x) unless specified otherwise by the first input.

    The storage will have NDTensors.Dense type.

    Examples

    ```julia i = Index(2,"indexi"); j = Index(4,"indexj"); k = Index(3,"index_k");

    A = ITensor(1.0, i, j) A = ITensor(1, i, j) # same as above B = ITensor(2.0+3.0im, j, k) ```

    !!! warning In future versions this may not automatically convert integer inputs with float, and in that case the particular element type should not be relied on.

    source
    ITensors.ITensorMethod
    ITensor([ElT::Type, ]A::AbstractArray, inds)
     ITensor([ElT::Type, ]A::AbstractArray, inds::Index...)
     
     itensor([ElT::Type, ]A::AbstractArray, inds)
    @@ -87,13 +87,13 @@
     T = ITensor(M, i, j)
     T[i => 1, j => 1] = 3.3
     M[1, 1] == 3.3
    -T[i => 1, j => 1] == 3.3
    Warning

    In future versions this may not automatically convert Int/Complex{Int} inputs to floating point versions with float (once tensor operations using Int/Complex{Int} are natively as fast as floating point operations), and in that case the particular element type should not be relied on. To avoid extra conversions (and therefore allocations) it is best practice to directly construct with itensor([0. 1; 1 0], i', dag(i)) if you want a floating point element type. The conversion is done as a performance optimization since often tensors are passed to BLAS/LAPACK and need to be converted to floating point types compatible with those libraries, but future projects in Julia may allow for efficient operations with more general element types (for example see https://github.com/JuliaLinearAlgebra/Octavian.jl).

    source
    ITensors.random_itensorMethod
    random_itensor([rng=Random.default_rng()], [ElT=Float64], inds)
    +T[i => 1, j => 1] == 3.3
    Warning

    In future versions this may not automatically convert Int/Complex{Int} inputs to floating point versions with float (once tensor operations using Int/Complex{Int} are natively as fast as floating point operations), and in that case the particular element type should not be relied on. To avoid extra conversions (and therefore allocations) it is best practice to directly construct with itensor([0. 1; 1 0], i', dag(i)) if you want a floating point element type. The conversion is done as a performance optimization since often tensors are passed to BLAS/LAPACK and need to be converted to floating point types compatible with those libraries, but future projects in Julia may allow for efficient operations with more general element types (for example see https://github.com/JuliaLinearAlgebra/Octavian.jl).

    source
    ITensors.random_itensorMethod
    random_itensor([rng=Random.default_rng()], [ElT=Float64], inds)
     random_itensor([rng=Random.default_rng()], [ElT=Float64], inds::Index...)

    Construct an ITensor with type ElT and indices inds, whose elements are normally distributed random numbers. If the element type is not specified, it defaults to Float64.

    Examples

    i = Index(2,"index_i")
     j = Index(4,"index_j")
     k = Index(3,"index_k")
     
     A = random_itensor(i,j)
    -B = random_itensor(ComplexF64,undef,k,j)
    source
    ITensors.onehotFunction
    onehot(ivs...)
    +B = random_itensor(ComplexF64,undef,k,j)
    source
    ITensors.onehotFunction
    onehot(ivs...)
     setelt(ivs...)
     onehot(::Type, ivs...)
     setelt(::Type, ivs...)

    Create an ITensor with all zeros except the specified value, which is set to 1.

    Examples

    i = Index(2,"i")
    @@ -105,7 +105,7 @@
     
     j = Index(3,"j")
     B = onehot(i=>1,j=>3)
    -# B[i=>1,j=>3] == 1, all other element zero
    source

    Dense View Constructors

    ITensors.itensorMethod
    itensor(args...; kwargs...)

    Like the ITensor constructor, but with attempt to make a view of the input data when possible.

    source

    QN BlockSparse Constructors

    ITensors.ITensorMethod
    ITensor([::Type{ElT} = Float64, ][flux::QN = QN(), ]inds)
    +# B[i=>1,j=>3] == 1, all other element zero
    source

    Dense View Constructors

    ITensors.itensorMethod
    itensor(args...; kwargs...)

    Like the ITensor constructor, but with attempt to make a view of the input data when possible.

    source

    QN BlockSparse Constructors

    ITensors.ITensorMethod
    ITensor([::Type{ElT} = Float64, ][flux::QN = QN(), ]inds)
     ITensor([::Type{ElT} = Float64, ][flux::QN = QN(), ]inds::Index...)

    Construct an ITensor with BlockSparse storage filled with zero(ElT) where the nonzero blocks are determined by flux.

    If ElT is not specified it defaults to Float64.

    If flux is not specified, the ITensor will be empty (it will contain no blocks, and have an undefined flux). The flux will be set by the first element that is set.

    Examples

    julia> i
     (dim=3|id=212|"i") <Out>
      1: QN(0) => 1
    @@ -209,7 +209,7 @@
      2  0
     
     julia> flux(A)
    -QN(-1)
    source
    ITensors.ITensorMethod
    ITensor([ElT::Type, ]A::AbstractArray, inds)
    +QN(-1)
    source
    ITensors.ITensorMethod
    ITensor([ElT::Type, ]A::AbstractArray, inds)
     ITensor([ElT::Type, ]A::AbstractArray, inds::Index...)
     
     itensor([ElT::Type, ]A::AbstractArray, inds)
    @@ -223,7 +223,7 @@
     T = ITensor(M, i, j)
     T[i => 1, j => 1] = 3.3
     M[1, 1] == 3.3
    -T[i => 1, j => 1] == 3.3
    Warning

    In future versions this may not automatically convert Int/Complex{Int} inputs to floating point versions with float (once tensor operations using Int/Complex{Int} are natively as fast as floating point operations), and in that case the particular element type should not be relied on. To avoid extra conversions (and therefore allocations) it is best practice to directly construct with itensor([0. 1; 1 0], i', dag(i)) if you want a floating point element type. The conversion is done as a performance optimization since often tensors are passed to BLAS/LAPACK and need to be converted to floating point types compatible with those libraries, but future projects in Julia may allow for efficient operations with more general element types (for example see https://github.com/JuliaLinearAlgebra/Octavian.jl).

    source
    ITensor([ElT::Type, ]::AbstractArray, inds; tol=0.0, checkflux=true)

    Create a block sparse ITensor from the input Array, and collection of QN indices. Zeros are dropped and nonzero blocks are determined from the zero values of the array.

    Optionally, you can set a tolerance such that elements less than or equal to the tolerance are dropped.

    By default, this will check that the flux of the nonzero blocks are consistent with each other. You can disable this check by setting checkflux=false.

    Examples

    julia> i = Index([QN(0)=>1, QN(1)=>2], "i");
    +T[i => 1, j => 1] == 3.3
    Warning

    In future versions this may not automatically convert Int/Complex{Int} inputs to floating point versions with float (once tensor operations using Int/Complex{Int} are natively as fast as floating point operations), and in that case the particular element type should not be relied on. To avoid extra conversions (and therefore allocations) it is best practice to directly construct with itensor([0. 1; 1 0], i', dag(i)) if you want a floating point element type. The conversion is done as a performance optimization since often tensors are passed to BLAS/LAPACK and need to be converted to floating point types compatible with those libraries, but future projects in Julia may allow for efficient operations with more general element types (for example see https://github.com/JuliaLinearAlgebra/Octavian.jl).

    source
    ITensor([ElT::Type, ]::AbstractArray, inds; tol=0.0, checkflux=true)

    Create a block sparse ITensor from the input Array, and collection of QN indices. Zeros are dropped and nonzero blocks are determined from the zero values of the array.

    Optionally, you can set a tolerance such that elements less than or equal to the tolerance are dropped.

    By default, this will check that the flux of the nonzero blocks are consistent with each other. You can disable this check by setting checkflux=false.

    Examples

    julia> i = Index([QN(0)=>1, QN(1)=>2], "i");
     
     julia> A = [1e-9 0.0 0.0;
                 0.0 2.0 3.0;
    @@ -242,17 +242,17 @@
     Block: (2, 2)
      [2:3, 2:3]
      2.0  3.0
    - 0.0  4.0
    source
    ITensors.ITensorMethod
    ITensor([::Type{ElT} = Float64,] ::UndefInitializer, flux::QN, inds)
    + 0.0  4.0
    source
    ITensors.ITensorMethod
    ITensor([::Type{ElT} = Float64,] ::UndefInitializer, flux::QN, inds)
     ITensor([::Type{ElT} = Float64,] ::UndefInitializer, flux::QN, inds::Index...)

    Construct an ITensor with indices inds and BlockSparse storage with undefined elements of type ElT, where the nonzero (allocated) blocks are determined by the provided QN flux. One purpose for using this constructor is that initializing the elements in an undefined way is faster than initializing them to a set value such as zero.

    The storage will have NDTensors.BlockSparse type.

    Examples

    i = Index([QN(0)=>1, QN(1)=>2], "i")
     A = ITensor(undef,QN(0),i',dag(i))
     B = ITensor(Float64,undef,QN(0),i',dag(i))
    -C = ITensor(ComplexF64,undef,QN(0),i',dag(i))
    source

    Diagonal constructors

    ITensors.diag_itensorMethod
    diag_itensor([::Type{ElT} = Float64, ]inds)
    -diag_itensor([::Type{ElT} = Float64, ]inds::Index...)

    Make a sparse ITensor of element type ElT with only elements along the diagonal stored. Defaults to having zero(T) along the diagonal.

    The storage will have NDTensors.Diag type.

    source
    ITensors.diag_itensorMethod
    diag_itensor([ElT::Type, ]v::AbstractVector, inds...)
    -diagitensor([ElT::Type, ]v::AbstractVector, inds...)

    Make a sparse ITensor with non-zero elements only along the diagonal. In general, the diagonal elements will be those stored in v and the ITensor will have element type eltype(v), unless specified explicitly by ElT. The storage will have NDTensors.Diag type.

    In the case when eltype(v) isa Union{Int, Complex{Int}}, by default it will be converted to float(v). Note that this behavior is subject to change in the future.

    The version diag_itensor will never output an ITensor whose storage data is an alias of the input vector data.

    The version diagitensor might output an ITensor whose storage data is an alias of the input vector data in order to minimize operations.

    source
    ITensors.diag_itensorMethod
    diag_itensor([ElT::Type, ]x::Number, inds...)
    -diagitensor([ElT::Type, ]x::Number, inds...)

    Make a sparse ITensor with non-zero elements only along the diagonal. In general, the diagonal elements will be set to the value x and the ITensor will have element type eltype(x), unless specified explicitly by ElT. The storage will have NDTensors.Diag type.

    In the case when x isa Union{Int, Complex{Int}}, by default it will be converted to float(x). Note that this behavior is subject to change in the future.

    source
    ITensors.deltaMethod
    delta([::Type{ElT} = Float64, ]inds)
    -delta([::Type{ElT} = Float64, ]inds::Index...)

    Make a uniform diagonal ITensor with all diagonal elements one(ElT). Only a single diagonal element is stored.

    This function has an alias δ.

    source

    QN Diagonal constructors

    ITensors.diag_itensorMethod
    diag_itensor([::Type{ElT} = Float64, ][flux::QN = QN(), ]is)
    -diag_itensor([::Type{ElT} = Float64, ][flux::QN = QN(), ]is::Index...)

    Make an ITensor with storage type NDTensors.DiagBlockSparse with elements zero(ElT). The ITensor only has diagonal blocks consistent with the specified flux.

    If the element type is not specified, it defaults to Float64. If theflux is not specified, it defaults to QN().

    source
    ITensors.deltaMethod
    delta([::Type{ElT} = Float64, ][flux::QN = QN(), ]is)
    -delta([::Type{ElT} = Float64, ][flux::QN = QN(), ]is::Index...)

    Make an ITensor with storage type NDTensors.DiagBlockSparse with uniform elements one(ElT). The ITensor only has diagonal blocks consistent with the specified flux.

    If the element type is not specified, it defaults to Float64. If theflux is not specified, it defaults to QN().

    source

    Convert to Array

    Core.ArrayMethod
    Array{ElT, N}(T::ITensor, i:Index...)
    +C = ITensor(ComplexF64,undef,QN(0),i',dag(i))
    source

    Diagonal constructors

    ITensors.diag_itensorMethod
    diag_itensor([::Type{ElT} = Float64, ]inds)
    +diag_itensor([::Type{ElT} = Float64, ]inds::Index...)

    Make a sparse ITensor of element type ElT with only elements along the diagonal stored. Defaults to having zero(T) along the diagonal.

    The storage will have NDTensors.Diag type.

    source
    ITensors.diag_itensorMethod
    diag_itensor([ElT::Type, ]v::AbstractVector, inds...)
    +diagitensor([ElT::Type, ]v::AbstractVector, inds...)

    Make a sparse ITensor with non-zero elements only along the diagonal. In general, the diagonal elements will be those stored in v and the ITensor will have element type eltype(v), unless specified explicitly by ElT. The storage will have NDTensors.Diag type.

    In the case when eltype(v) isa Union{Int, Complex{Int}}, by default it will be converted to float(v). Note that this behavior is subject to change in the future.

    The version diag_itensor will never output an ITensor whose storage data is an alias of the input vector data.

    The version diagitensor might output an ITensor whose storage data is an alias of the input vector data in order to minimize operations.

    source
    ITensors.diag_itensorMethod
    diag_itensor([ElT::Type, ]x::Number, inds...)
    +diagitensor([ElT::Type, ]x::Number, inds...)

    Make a sparse ITensor with non-zero elements only along the diagonal. In general, the diagonal elements will be set to the value x and the ITensor will have element type eltype(x), unless specified explicitly by ElT. The storage will have NDTensors.Diag type.

    In the case when x isa Union{Int, Complex{Int}}, by default it will be converted to float(x). Note that this behavior is subject to change in the future.

    source
    ITensors.deltaMethod
    delta([::Type{ElT} = Float64, ]inds)
    +delta([::Type{ElT} = Float64, ]inds::Index...)

    Make a uniform diagonal ITensor with all diagonal elements one(ElT). Only a single diagonal element is stored.

    This function has an alias δ.

    source

    QN Diagonal constructors

    ITensors.diag_itensorMethod
    diag_itensor([::Type{ElT} = Float64, ][flux::QN = QN(), ]is)
    +diag_itensor([::Type{ElT} = Float64, ][flux::QN = QN(), ]is::Index...)

    Make an ITensor with storage type NDTensors.DiagBlockSparse with elements zero(ElT). The ITensor only has diagonal blocks consistent with the specified flux.

    If the element type is not specified, it defaults to Float64. If theflux is not specified, it defaults to QN().

    source
    ITensors.deltaMethod
    delta([::Type{ElT} = Float64, ][flux::QN = QN(), ]is)
    +delta([::Type{ElT} = Float64, ][flux::QN = QN(), ]is::Index...)

    Make an ITensor with storage type NDTensors.DiagBlockSparse with uniform elements one(ElT). The ITensor only has diagonal blocks consistent with the specified flux.

    If the element type is not specified, it defaults to Float64. If theflux is not specified, it defaults to QN().

    source

    Convert to Array

    Core.ArrayMethod
    Array{ElT, N}(T::ITensor, i:Index...)
     Array{ElT}(T::ITensor, i:Index...)
     Array(T::ITensor, i:Index...)
     
    @@ -260,9 +260,9 @@
     Matrix(T::ITensor, row_i:Index, col_i::Index)
     
     Vector{ElT}(T::ITensor)
    -Vector(T::ITensor)

    Given an ITensor T with indices i..., returns an Array with a copy of the ITensor's elements. The order in which the indices are provided indicates the order of the data in the resulting Array.

    source
    NDTensors.arrayMethod
    array(T::ITensor, inds...)

    Convert an ITensor T to an Array.

    The ordering of the elements in the Array are specified by the input indices inds. This tries to avoid copying of possible (i.e. may return a view of the original data), for example if the ITensor's storage is Dense and the indices are already in the specified ordering so that no permutation is required.

    Warning

    Note that in the future we may return specialized AbstractArray types for certain storage types, for example a LinearAlgebra.Diagonal type for an ITensor with Diag storage. The specific storage type shouldn't be relied upon.

    See also matrix, vector.

    source
    NDTensors.matrixMethod
    matrix(T::ITensor, inds...)

    Convert an ITensor T to a Matrix.

    The ordering of the elements in the Matrix are specified by the input indices inds. This tries to avoid copying of possible (i.e. may return a view of the original data), for example if the ITensor's storage is Dense and the indices are already in the specified ordering so that no permutation is required.

    Warning

    Note that in the future we may return specialized AbstractArray types for certain storage types, for example a LinearAlgebra.Diagonal type for an ITensor with Diag storage. The specific storage type shouldn't be relied upon.

    See also array, vector.

    source
    NDTensors.vectorMethod
    vector(T::ITensor, inds...)

    Convert an ITensor T to an Vector.

    The ordering of the elements in the Array are specified by the input indices inds. This tries to avoid copying of possible (i.e. may return a view of the original data), for example if the ITensor's storage is Dense and the indices are already in the specified ordering so that no permutation is required.

    Warning

    Note that in the future we may return specialized AbstractArray types for certain storage types, for example a LinearAlgebra.Diagonal type for an ITensor with Diag storage. The specific storage type shouldn't be relied upon.

    See also array, matrix.

    source
    NDTensors.arrayMethod
    array(T::ITensor)

    Given an ITensor T, returns an Array with a copy of the ITensor's elements, or a view in the case the the ITensor's storage is Dense.

    The ordering of the elements in the Array, in terms of which Index is treated as the row versus column, depends on the internal layout of the ITensor.

    Warning

    This method is intended for developer use only and not recommended for use in ITensor applications unless you know what you are doing (for example you are certain of the memory ordering of the ITensor because you permuted the indices into a certain order).

    See also matrix, vector.

    source
    NDTensors.matrixMethod
    matrix(T::ITensor)

    Given an ITensor T with two indices, returns a Matrix with a copy of the ITensor's elements, or a view in the case the ITensor's storage is Dense.

    The ordering of the elements in the Matrix, in terms of which Index is treated as the row versus column, depends on the internal layout of the ITensor.

    Warning

    This method is intended for developer use only and not recommended for use in ITensor applications unless you know what you are doing (for example you are certain of the memory ordering of the ITensor because you permuted the indices into a certain order).

    See also array, vector.

    source
    NDTensors.vectorMethod
    vector(T::ITensor)

    Given an ITensor T with one index, returns a Vector with a copy of the ITensor's elements, or a view in the case the ITensor's storage is Dense.

    See also array, matrix.

    source

    Getting and setting elements

    Base.getindexMethod
    getindex(T::ITensor, ivs...)

    Get the specified element of the ITensor, using a list of IndexVals or Pair{<:Index, Int}.

    Example

    i = Index(2; tags = "i")
    +Vector(T::ITensor)

    Given an ITensor T with indices i..., returns an Array with a copy of the ITensor's elements. The order in which the indices are provided indicates the order of the data in the resulting Array.

    source
    NDTensors.arrayMethod
    array(T::ITensor, inds...)

    Convert an ITensor T to an Array.

    The ordering of the elements in the Array are specified by the input indices inds. This tries to avoid copying of possible (i.e. may return a view of the original data), for example if the ITensor's storage is Dense and the indices are already in the specified ordering so that no permutation is required.

    Warning

    Note that in the future we may return specialized AbstractArray types for certain storage types, for example a LinearAlgebra.Diagonal type for an ITensor with Diag storage. The specific storage type shouldn't be relied upon.

    See also matrix, vector.

    source
    NDTensors.matrixMethod
    matrix(T::ITensor, inds...)

    Convert an ITensor T to a Matrix.

    The ordering of the elements in the Matrix are specified by the input indices inds. This tries to avoid copying of possible (i.e. may return a view of the original data), for example if the ITensor's storage is Dense and the indices are already in the specified ordering so that no permutation is required.

    Warning

    Note that in the future we may return specialized AbstractArray types for certain storage types, for example a LinearAlgebra.Diagonal type for an ITensor with Diag storage. The specific storage type shouldn't be relied upon.

    See also array, vector.

    source
    NDTensors.vectorMethod
    vector(T::ITensor, inds...)

    Convert an ITensor T to an Vector.

    The ordering of the elements in the Array are specified by the input indices inds. This tries to avoid copying of possible (i.e. may return a view of the original data), for example if the ITensor's storage is Dense and the indices are already in the specified ordering so that no permutation is required.

    Warning

    Note that in the future we may return specialized AbstractArray types for certain storage types, for example a LinearAlgebra.Diagonal type for an ITensor with Diag storage. The specific storage type shouldn't be relied upon.

    See also array, matrix.

    source
    NDTensors.arrayMethod
    array(T::ITensor)

    Given an ITensor T, returns an Array with a copy of the ITensor's elements, or a view in the case the the ITensor's storage is Dense.

    The ordering of the elements in the Array, in terms of which Index is treated as the row versus column, depends on the internal layout of the ITensor.

    Warning

    This method is intended for developer use only and not recommended for use in ITensor applications unless you know what you are doing (for example you are certain of the memory ordering of the ITensor because you permuted the indices into a certain order).

    See also matrix, vector.

    source
    NDTensors.matrixMethod
    matrix(T::ITensor)

    Given an ITensor T with two indices, returns a Matrix with a copy of the ITensor's elements, or a view in the case the ITensor's storage is Dense.

    The ordering of the elements in the Matrix, in terms of which Index is treated as the row versus column, depends on the internal layout of the ITensor.

    Warning

    This method is intended for developer use only and not recommended for use in ITensor applications unless you know what you are doing (for example you are certain of the memory ordering of the ITensor because you permuted the indices into a certain order).

    See also array, vector.

    source
    NDTensors.vectorMethod
    vector(T::ITensor)

    Given an ITensor T with one index, returns a Vector with a copy of the ITensor's elements, or a view in the case the ITensor's storage is Dense.

    See also array, matrix.

    source

    Getting and setting elements

    Base.getindexMethod
    getindex(T::ITensor, ivs...)

    Get the specified element of the ITensor, using a list of IndexVals or Pair{<:Index, Int}.

    Example

    i = Index(2; tags = "i")
     A = ITensor(2.0, i, i')
    -A[i => 1, i' => 2] # 2.0, same as: A[i' => 2, i => 1]
    source
    Base.setindex!Method
    setindex!(T::ITensor, x::Number, ivs...)
    +A[i => 1, i' => 2] # 2.0, same as: A[i' => 2, i => 1]
    source
    Base.setindex!Method
    setindex!(T::ITensor, x::Number, ivs...)
     
     setindex!(T::ITensor, x::Number, I::Integer...)
     
    @@ -273,41 +273,41 @@
     
     # Some simple slicing is also supported
     A[i => 2, i' => :] = [2.0 3.0]
    -A[2, :] = [2.0 3.0]
    source

    Properties

    NDTensors.indsMethod
    inds(T::ITensor)

    Return the indices of the ITensor as a Tuple.

    source
    NDTensors.indMethod
    ind(T::ITensor, i::Int)

    Get the Index of the ITensor along dimension i.

    source
    ITensors.dirMethod
    dir(A::ITensor, i::Index)

    Return the direction of the Index i in the ITensor A.

    source

    Priming and tagging

    ITensors.primeMethod
    prime[!](A::ITensor, plinc::Int = 1; <keyword arguments>) -> ITensor
    +A[2, :] = [2.0 3.0]
    source

    Properties

    NDTensors.indsMethod
    inds(T::ITensor)

    Return the indices of the ITensor as a Tuple.

    source
    NDTensors.indMethod
    ind(T::ITensor, i::Int)

    Get the Index of the ITensor along dimension i.

    source
    ITensors.dirMethod
    dir(A::ITensor, i::Index)

    Return the direction of the Index i in the ITensor A.

    source

    Priming and tagging

    ITensors.primeMethod
    prime[!](A::ITensor, plinc::Int = 1; <keyword arguments>) -> ITensor
     
    -prime(inds, plinc::Int = 1; <keyword arguments>) -> IndexSet

    Increase the prime level of the indices of an ITensor or collection of indices.

    Optionally, only modify the indices with the specified keyword arguments.

    Arguments

    • tags = nothing: if specified, only modify Index i if hastags(i, tags) == true.
    • plev = nothing: if specified, only modify Index i if hasplev(i, plev) == true.

    The ITensor functions come in two versions, f and f!. The latter modifies the ITensor in-place. In both versions, the ITensor storage is not modified or copied (so it returns an ITensor with a view of the original storage).

    source
    ITensors.setprimeMethod
    setprime[!](A::ITensor, plev::Int; <keyword arguments>) -> ITensor
    +prime(inds, plinc::Int = 1; <keyword arguments>) -> IndexSet

    Increase the prime level of the indices of an ITensor or collection of indices.

    Optionally, only modify the indices with the specified keyword arguments.

    Arguments

    • tags = nothing: if specified, only modify Index i if hastags(i, tags) == true.
    • plev = nothing: if specified, only modify Index i if hasplev(i, plev) == true.

    The ITensor functions come in two versions, f and f!. The latter modifies the ITensor in-place. In both versions, the ITensor storage is not modified or copied (so it returns an ITensor with a view of the original storage).

    source
    ITensors.setprimeMethod
    setprime[!](A::ITensor, plev::Int; <keyword arguments>) -> ITensor
     
    -setprime(inds, plev::Int; <keyword arguments>) -> IndexSet

    Set the prime level of the indices of an ITensor or collection of indices.

    Optionally, only modify the indices with the specified keyword arguments.

    Arguments

    • tags = nothing: if specified, only modify Index i if hastags(i, tags) == true.
    • plev = nothing: if specified, only modify Index i if hasplev(i, plev) == true.

    The ITensor functions come in two versions, f and f!. The latter modifies the ITensor in-place. In both versions, the ITensor storage is not modified or copied (so it returns an ITensor with a view of the original storage).

    source
    ITensors.noprimeMethod
    noprime[!](A::ITensor; <keyword arguments>) -> ITensor
    +setprime(inds, plev::Int; <keyword arguments>) -> IndexSet

    Set the prime level of the indices of an ITensor or collection of indices.

    Optionally, only modify the indices with the specified keyword arguments.

    Arguments

    • tags = nothing: if specified, only modify Index i if hastags(i, tags) == true.
    • plev = nothing: if specified, only modify Index i if hasplev(i, plev) == true.

    The ITensor functions come in two versions, f and f!. The latter modifies the ITensor in-place. In both versions, the ITensor storage is not modified or copied (so it returns an ITensor with a view of the original storage).

    source
    ITensors.noprimeMethod
    noprime[!](A::ITensor; <keyword arguments>) -> ITensor
     
    -noprime(inds; <keyword arguments>) -> IndexSet

    Set the prime level of the indices of an ITensor or collection of indices to zero.

    Optionally, only modify the indices with the specified keyword arguments.

    Arguments

    • tags = nothing: if specified, only modify Index i if hastags(i, tags) == true.
    • plev = nothing: if specified, only modify Index i if hasplev(i, plev) == true.

    The ITensor functions come in two versions, f and f!. The latter modifies the ITensor in-place. In both versions, the ITensor storage is not modified or copied (so it returns an ITensor with a view of the original storage).

    source
    ITensors.mapprimeMethod
    replaceprime[!](A::ITensor, plold::Int, plnew::Int; <keyword arguments>) -> ITensor
    +noprime(inds; <keyword arguments>) -> IndexSet

    Set the prime level of the indices of an ITensor or collection of indices to zero.

    Optionally, only modify the indices with the specified keyword arguments.

    Arguments

    • tags = nothing: if specified, only modify Index i if hastags(i, tags) == true.
    • plev = nothing: if specified, only modify Index i if hasplev(i, plev) == true.

    The ITensor functions come in two versions, f and f!. The latter modifies the ITensor in-place. In both versions, the ITensor storage is not modified or copied (so it returns an ITensor with a view of the original storage).

    source
    ITensors.mapprimeMethod
    replaceprime[!](A::ITensor, plold::Int, plnew::Int; <keyword arguments>) -> ITensor
     replaceprime[!](A::ITensor, plold => plnew; <keyword arguments>) -> ITensor
     mapprime[!](A::ITensor, <arguments>; <keyword arguments>) -> ITensor
     
     replaceprime(inds, plold::Int, plnew::Int; <keyword arguments>)
     replaceprime(inds::IndexSet, plold => plnew; <keyword arguments>)
    -mapprime(inds, <arguments>; <keyword arguments>)

    Set the prime level of the indices of an ITensor or collection of indices with prime level plold to plnew.

    Optionally, only modify the indices with the specified keyword arguments.

    Arguments

    • tags = nothing: if specified, only modify Index i if hastags(i, tags) == true.
    • plev = nothing: if specified, only modify Index i if hasplev(i, plev) == true.

    The ITensor functions come in two versions, f and f!. The latter modifies the ITensor in-place. In both versions, the ITensor storage is not modified or copied (so it returns an ITensor with a view of the original storage).

    source
    ITensors.swapprimeMethod
    swapprime[!](A::ITensor, pl1::Int, pl2::Int; <keyword arguments>) -> ITensor
    +mapprime(inds, <arguments>; <keyword arguments>)

    Set the prime level of the indices of an ITensor or collection of indices with prime level plold to plnew.

    Optionally, only modify the indices with the specified keyword arguments.

    Arguments

    • tags = nothing: if specified, only modify Index i if hastags(i, tags) == true.
    • plev = nothing: if specified, only modify Index i if hasplev(i, plev) == true.

    The ITensor functions come in two versions, f and f!. The latter modifies the ITensor in-place. In both versions, the ITensor storage is not modified or copied (so it returns an ITensor with a view of the original storage).

    source
    ITensors.swapprimeMethod
    swapprime[!](A::ITensor, pl1::Int, pl2::Int; <keyword arguments>) -> ITensor
     swapprime[!](A::ITensor, pl1 => pl2; <keyword arguments>) -> ITensor
     
     swapprime(inds, pl1::Int, pl2::Int; <keyword arguments>)
    -swapprime(inds, pl1 => pl2; <keyword arguments>)

    Set the prime level of the indices of an ITensor or collection of indices with prime level pl1 to pl2, and those with prime level pl2 to pl1.

    Optionally, only modify the indices with the specified keyword arguments.

    Arguments

    • tags = nothing: if specified, only modify Index i if hastags(i, tags) == true.
    • plev = nothing: if specified, only modify Index i if hasplev(i, plev) == true.

    The ITensor functions come in two versions, f and f!. The latter modifies the ITensor in-place. In both versions, the ITensor storage is not modified or copied (so it returns an ITensor with a view of the original storage).

    source
    ITensors.TagSets.addtagsMethod
    addtags[!](A::ITensor, ts::String; <keyword arguments>) -> ITensor
    +swapprime(inds, pl1 => pl2; <keyword arguments>)

    Set the prime level of the indices of an ITensor or collection of indices with prime level pl1 to pl2, and those with prime level pl2 to pl1.

    Optionally, only modify the indices with the specified keyword arguments.

    Arguments

    • tags = nothing: if specified, only modify Index i if hastags(i, tags) == true.
    • plev = nothing: if specified, only modify Index i if hasplev(i, plev) == true.

    The ITensor functions come in two versions, f and f!. The latter modifies the ITensor in-place. In both versions, the ITensor storage is not modified or copied (so it returns an ITensor with a view of the original storage).

    source
    ITensors.TagSets.addtagsMethod
    addtags[!](A::ITensor, ts::String; <keyword arguments>) -> ITensor
     
    -addtags(inds, ts::String; <keyword arguments>)

    Add the tags ts to the indices of an ITensor or collection of indices.

    Optionally, only modify the indices with the specified keyword arguments.

    Arguments

    • tags = nothing: if specified, only modify Index i if hastags(i, tags) == true.
    • plev = nothing: if specified, only modify Index i if hasplev(i, plev) == true.

    The ITensor functions come in two versions, f and f!. The latter modifies the ITensor in-place. In both versions, the ITensor storage is not modified or copied (so it returns an ITensor with a view of the original storage).

    source
    ITensors.TagSets.removetagsMethod
    removetags[!](A::ITensor, ts::String; <keyword arguments>) -> ITensor
    +addtags(inds, ts::String; <keyword arguments>)

    Add the tags ts to the indices of an ITensor or collection of indices.

    Optionally, only modify the indices with the specified keyword arguments.

    Arguments

    • tags = nothing: if specified, only modify Index i if hastags(i, tags) == true.
    • plev = nothing: if specified, only modify Index i if hasplev(i, plev) == true.

    The ITensor functions come in two versions, f and f!. The latter modifies the ITensor in-place. In both versions, the ITensor storage is not modified or copied (so it returns an ITensor with a view of the original storage).

    source
    ITensors.TagSets.removetagsMethod
    removetags[!](A::ITensor, ts::String; <keyword arguments>) -> ITensor
     
    -removetags(inds, ts::String; <keyword arguments>)

    Remove the tags ts from the indices of an ITensor or collection of indices.

    Optionally, only modify the indices with the specified keyword arguments.

    Arguments

    • tags = nothing: if specified, only modify Index i if hastags(i, tags) == true.
    • plev = nothing: if specified, only modify Index i if hasplev(i, plev) == true.

    The ITensor functions come in two versions, f and f!. The latter modifies the ITensor in-place. In both versions, the ITensor storage is not modified or copied (so it returns an ITensor with a view of the original storage).

    source
    ITensors.TagSets.replacetagsMethod
    replacetags[!](A::ITensor, tsold::String, tsnew::String; <keyword arguments>) -> ITensor
    +removetags(inds, ts::String; <keyword arguments>)

    Remove the tags ts from the indices of an ITensor or collection of indices.

    Optionally, only modify the indices with the specified keyword arguments.

    Arguments

    • tags = nothing: if specified, only modify Index i if hastags(i, tags) == true.
    • plev = nothing: if specified, only modify Index i if hasplev(i, plev) == true.

    The ITensor functions come in two versions, f and f!. The latter modifies the ITensor in-place. In both versions, the ITensor storage is not modified or copied (so it returns an ITensor with a view of the original storage).

    source
    ITensors.TagSets.replacetagsMethod
    replacetags[!](A::ITensor, tsold::String, tsnew::String; <keyword arguments>) -> ITensor
     
    -replacetags(is::IndexSet, tsold::String, tsnew::String; <keyword arguments>) -> IndexSet

    Replace the tags tsold with tsnew for the indices of an ITensor.

    Optionally, only modify the indices with the specified keyword arguments.

    Arguments

    • tags = nothing: if specified, only modify Index i if hastags(i, tags) == true.
    • plev = nothing: if specified, only modify Index i if hasplev(i, plev) == true.

    The ITensor functions come in two versions, f and f!. The latter modifies the ITensor in-place. In both versions, the ITensor storage is not modified or copied (so it returns an ITensor with a view of the original storage).

    source
    ITensors.settagsMethod
    settags[!](A::ITensor, ts::String; <keyword arguments>) -> ITensor
    +replacetags(is::IndexSet, tsold::String, tsnew::String; <keyword arguments>) -> IndexSet

    Replace the tags tsold with tsnew for the indices of an ITensor.

    Optionally, only modify the indices with the specified keyword arguments.

    Arguments

    • tags = nothing: if specified, only modify Index i if hastags(i, tags) == true.
    • plev = nothing: if specified, only modify Index i if hasplev(i, plev) == true.

    The ITensor functions come in two versions, f and f!. The latter modifies the ITensor in-place. In both versions, the ITensor storage is not modified or copied (so it returns an ITensor with a view of the original storage).

    source
    ITensors.settagsMethod
    settags[!](A::ITensor, ts::String; <keyword arguments>) -> ITensor
     
    -settags(is::IndexSet, ts::String; <keyword arguments>) -> IndexSet

    Set the tags of the indices of an ITensor or IndexSet to ts.

    Optionally, only modify the indices with the specified keyword arguments.

    Arguments

    • tags = nothing: if specified, only modify Index i if hastags(i, tags) == true.
    • plev = nothing: if specified, only modify Index i if hasplev(i, plev) == true.

    The ITensor functions come in two versions, f and f!. The latter modifies the ITensor in-place. In both versions, the ITensor storage is not modified or copied (so it returns an ITensor with a view of the original storage).

    source
    ITensors.swaptagsMethod
    swaptags[!](A::ITensor, ts1::String, ts2::String; <keyword arguments>) -> ITensor
    +settags(is::IndexSet, ts::String; <keyword arguments>) -> IndexSet

    Set the tags of the indices of an ITensor or IndexSet to ts.

    Optionally, only modify the indices with the specified keyword arguments.

    Arguments

    • tags = nothing: if specified, only modify Index i if hastags(i, tags) == true.
    • plev = nothing: if specified, only modify Index i if hasplev(i, plev) == true.

    The ITensor functions come in two versions, f and f!. The latter modifies the ITensor in-place. In both versions, the ITensor storage is not modified or copied (so it returns an ITensor with a view of the original storage).

    source
    ITensors.swaptagsMethod
    swaptags[!](A::ITensor, ts1::String, ts2::String; <keyword arguments>) -> ITensor
     
    -swaptags(is::IndexSet, ts1::String, ts2::String; <keyword arguments>) -> IndexSet

    Swap the tags ts1 with ts2 for the indices of an ITensor.

    Optionally, only modify the indices with the specified keyword arguments.

    Arguments

    • tags = nothing: if specified, only modify Index i if hastags(i, tags) == true.
    • plev = nothing: if specified, only modify Index i if hasplev(i, plev) == true.

    The ITensor functions come in two versions, f and f!. The latter modifies the ITensor in-place. In both versions, the ITensor storage is not modified or copied (so it returns an ITensor with a view of the original storage).

    source

    Index collections set operations

    ITensors.commonindsFunction
    commoninds(A, B; kwargs...)

    Return a Vector with indices that are common between the indices of A and B (the set intersection, similar to Base.intersect).

    Optional keyword arguments:

    • tags::String - a tag name or comma separated list of tag names that the returned indices must all have
    • plev::Int - common prime level that the returned indices must all have
    • inds - Index or collection of indices. Returned indices must come from this set of indices.
    source
    ITensors.commonindFunction
    commonind(A, B; kwargs...)

    Return the first Index common between the indices of A and B.

    See also commoninds.

    Optional keyword arguments:

    • tags::String - a tag name or comma separated list of tag names that the returned indices must all have
    • plev::Int - common prime level that the returned indices must all have
    • inds - Index or collection of indices. Returned indices must come from this set of indices.
    source
    ITensors.uniqueindsFunction
    uniqueinds(A, B; kwargs...)

    Return Vector with indices that are unique to the set of indices of A and not in B (the set difference, similar to Base.setdiff).

    Optional keyword arguments:

    • tags::String - a tag name or comma separated list of tag names that the returned indices must all have
    • plev::Int - common prime level that the returned indices must all have
    • inds - Index or collection of indices. Returned indices must come from this set of indices.
    source
    ITensors.uniqueindFunction
    uniqueind(A, B; kwargs...)

    Return the first Index unique to the set of indices of A and not in B.

    See also uniqueinds.

    Optional keyword arguments:

    • tags::String - a tag name or comma separated list of tag names that the returned indices must all have
    • plev::Int - common prime level that the returned indices must all have
    • inds - Index or collection of indices. Returned indices must come from this set of indices.
    source
    ITensors.noncommonindsFunction
    noncommoninds(A, B; kwargs...)

    Return a Vector with indices that are not common between the indices of A and B (the symmetric set difference, similar to Base.symdiff).

    Optional keyword arguments:

    • tags::String - a tag name or comma separated list of tag names that the returned indices must all have
    • plev::Int - common prime level that the returned indices must all have
    • inds - Index or collection of indices. Returned indices must come from this set of indices.
    source
    ITensors.noncommonindFunction
    noncommonind(A, B; kwargs...)

    Return the first Index not common between the indices of A and B.

    See also noncommoninds.

    Optional keyword arguments:

    • tags::String - a tag name or comma separated list of tag names that the returned indices must all have
    • plev::Int - common prime level that the returned indices must all have
    • inds - Index or collection of indices. Returned indices must come from this set of indices.
    source
    ITensors.unionindsFunction
    unioninds(A, B; kwargs...)

    Return a Vector with indices that are the union of the indices of A and B (the set union, similar to Base.union).

    Optional keyword arguments:

    • tags::String - a tag name or comma separated list of tag names that the returned indices must all have
    • plev::Int - common prime level that the returned indices must all have
    • inds - Index or collection of indices. Returned indices must come from this set of indices.
    source
    ITensors.unionindFunction
    unionind(A, B; kwargs...)

    Return the first Index in the union of the indices of A and B.

    See also unioninds.

    Optional keyword arguments:

    • tags::String - a tag name or comma separated list of tag names that the returned indices must all have
    • plev::Int - common prime level that the returned indices must all have
    • inds - Index or collection of indices. Returned indices must come from this set of indices.
    source
    ITensors.hascommonindsFunction
    hascommoninds(A, B; kwargs...)
    +swaptags(is::IndexSet, ts1::String, ts2::String; <keyword arguments>) -> IndexSet

    Swap the tags ts1 with ts2 for the indices of an ITensor.

    Optionally, only modify the indices with the specified keyword arguments.

    Arguments

    • tags = nothing: if specified, only modify Index i if hastags(i, tags) == true.
    • plev = nothing: if specified, only modify Index i if hasplev(i, plev) == true.

    The ITensor functions come in two versions, f and f!. The latter modifies the ITensor in-place. In both versions, the ITensor storage is not modified or copied (so it returns an ITensor with a view of the original storage).

    source

    Index collections set operations

    ITensors.commonindsFunction
    commoninds(A, B; kwargs...)

    Return a Vector with indices that are common between the indices of A and B (the set intersection, similar to Base.intersect).

    Optional keyword arguments:

    • tags::String - a tag name or comma separated list of tag names that the returned indices must all have
    • plev::Int - common prime level that the returned indices must all have
    • inds - Index or collection of indices. Returned indices must come from this set of indices.
    source
    ITensors.commonindFunction
    commonind(A, B; kwargs...)

    Return the first Index common between the indices of A and B.

    See also commoninds.

    Optional keyword arguments:

    • tags::String - a tag name or comma separated list of tag names that the returned indices must all have
    • plev::Int - common prime level that the returned indices must all have
    • inds - Index or collection of indices. Returned indices must come from this set of indices.
    source
    ITensors.uniqueindsFunction
    uniqueinds(A, B; kwargs...)

    Return Vector with indices that are unique to the set of indices of A and not in B (the set difference, similar to Base.setdiff).

    Optional keyword arguments:

    • tags::String - a tag name or comma separated list of tag names that the returned indices must all have
    • plev::Int - common prime level that the returned indices must all have
    • inds - Index or collection of indices. Returned indices must come from this set of indices.
    source
    ITensors.uniqueindFunction
    uniqueind(A, B; kwargs...)

    Return the first Index unique to the set of indices of A and not in B.

    See also uniqueinds.

    Optional keyword arguments:

    • tags::String - a tag name or comma separated list of tag names that the returned indices must all have
    • plev::Int - common prime level that the returned indices must all have
    • inds - Index or collection of indices. Returned indices must come from this set of indices.
    source
    ITensors.noncommonindsFunction
    noncommoninds(A, B; kwargs...)

    Return a Vector with indices that are not common between the indices of A and B (the symmetric set difference, similar to Base.symdiff).

    Optional keyword arguments:

    • tags::String - a tag name or comma separated list of tag names that the returned indices must all have
    • plev::Int - common prime level that the returned indices must all have
    • inds - Index or collection of indices. Returned indices must come from this set of indices.
    source
    ITensors.noncommonindFunction
    noncommonind(A, B; kwargs...)

    Return the first Index not common between the indices of A and B.

    See also noncommoninds.

    Optional keyword arguments:

    • tags::String - a tag name or comma separated list of tag names that the returned indices must all have
    • plev::Int - common prime level that the returned indices must all have
    • inds - Index or collection of indices. Returned indices must come from this set of indices.
    source
    ITensors.unionindsFunction
    unioninds(A, B; kwargs...)

    Return a Vector with indices that are the union of the indices of A and B (the set union, similar to Base.union).

    Optional keyword arguments:

    • tags::String - a tag name or comma separated list of tag names that the returned indices must all have
    • plev::Int - common prime level that the returned indices must all have
    • inds - Index or collection of indices. Returned indices must come from this set of indices.
    source
    ITensors.unionindFunction
    unionind(A, B; kwargs...)

    Return the first Index in the union of the indices of A and B.

    See also unioninds.

    Optional keyword arguments:

    • tags::String - a tag name or comma separated list of tag names that the returned indices must all have
    • plev::Int - common prime level that the returned indices must all have
    • inds - Index or collection of indices. Returned indices must come from this set of indices.
    source
    ITensors.hascommonindsFunction
    hascommoninds(A, B; kwargs...)
     
    -hascommoninds(B; kwargs...) -> f::Function

    Check if the ITensors or sets of indices A and B have common indices.

    If only one ITensor or set of indices B is passed, return a function f such that f(A) = hascommoninds(A, B; kwargs...)

    source

    Index Manipulations

    ITensors.replaceindMethod
    replaceind[!](A::ITensor, i1::Index, i2::Index) -> ITensor

    Replace the Index i1 with the Index i2 in the ITensor.

    The indices must have the same space (i.e. the same dimension and QNs, if applicable).

    source
    ITensors.replaceindsMethod
    replaceinds(A::ITensor, inds1, inds2) -> ITensor
    +hascommoninds(B; kwargs...) -> f::Function

    Check if the ITensors or sets of indices A and B have common indices.

    If only one ITensor or set of indices B is passed, return a function f such that f(A) = hascommoninds(A, B; kwargs...)

    source

    Index Manipulations

    ITensors.replaceindMethod
    replaceind[!](A::ITensor, i1::Index, i2::Index) -> ITensor

    Replace the Index i1 with the Index i2 in the ITensor.

    The indices must have the same space (i.e. the same dimension and QNs, if applicable).

    source
    ITensors.replaceindsMethod
    replaceinds(A::ITensor, inds1, inds2) -> ITensor
     
    -replaceinds!(A::ITensor, inds1, inds2)

    Replace the Index inds1[n] with the Index inds2[n] in the ITensor, where n runs from 1 to length(inds1) == length(inds2).

    The indices must have the same space (i.e. the same dimension and QNs, if applicable).

    The storage of the ITensor is not modified or copied (the output ITensor is a view of the input ITensor).

    source
    ITensors.swapindMethod
    swapind(A::ITensor, i1::Index, i2::Index) -> ITensor
    +replaceinds!(A::ITensor, inds1, inds2)

    Replace the Index inds1[n] with the Index inds2[n] in the ITensor, where n runs from 1 to length(inds1) == length(inds2).

    The indices must have the same space (i.e. the same dimension and QNs, if applicable).

    The storage of the ITensor is not modified or copied (the output ITensor is a view of the input ITensor).

    source
    ITensors.swapindMethod
    swapind(A::ITensor, i1::Index, i2::Index) -> ITensor
     
    -swapind!(A::ITensor, i1::Index, i2::Index)

    Swap the Index i1 with the Index i2 in the ITensor.

    The indices must have the same space (i.e. the same dimension and QNs, if applicable).

    source
    ITensors.swapindsMethod
    swapinds(A::ITensor, inds1, inds2) -> ITensor
    +swapind!(A::ITensor, i1::Index, i2::Index)

    Swap the Index i1 with the Index i2 in the ITensor.

    The indices must have the same space (i.e. the same dimension and QNs, if applicable).

    source
    ITensors.swapindsMethod
    swapinds(A::ITensor, inds1, inds2) -> ITensor
     
    -swapinds!(A::ITensor, inds1, inds2)

    Swap the Index inds1[n] with the Index inds2[n] in the ITensor, where n runs from 1 to length(inds1) == length(inds2).

    The indices must have the same space (i.e. the same dimension and QNs, if applicable).

    The storage of the ITensor is not modified or copied (the output ITensor is a view of the input ITensor).

    source

    Math operations

    Base.:*Method
    A::ITensor * B::ITensor
    +swapinds!(A::ITensor, inds1, inds2)

    Swap the Index inds1[n] with the Index inds2[n] in the ITensor, where n runs from 1 to length(inds1) == length(inds2).

    The indices must have the same space (i.e. the same dimension and QNs, if applicable).

    The storage of the ITensor is not modified or copied (the output ITensor is a view of the input ITensor).

    source

    Math operations

    Base.:*Method
    A::ITensor * B::ITensor
     contract(A::ITensor, B::ITensor)

    Contract ITensors A and B to obtain a new ITensor. This contraction * operator finds all matching indices common to A and B and sums over them, such that the result will have only the unique indices of A and B. To prevent indices from matching, their prime level or tags can be modified such that they no longer compare equal - for more information see the documentation on Index objects.

    Examples

    i = Index(2,"index_i"); j = Index(4,"index_j"); k = Index(3,"index_k")
     
     A = random_itensor(i,j)
    @@ -324,7 +324,7 @@
     
     A = random_itensor(i,j,k)
     B = random_itensor(k,i,j)
    -C = A * B # inner product of A and B, all indices contracted
    source
    ITensors.dagMethod
    dag(T::ITensor; allow_alias = true)

    Complex conjugate the elements of the ITensor T and dagger the indices.

    By default, an alias of the ITensor is returned (i.e. the output ITensor may share data with the input ITensor). If allow_alias = false, an alias is never returned.

    source
    ITensors.directsumMethod
    directsum(A::Pair{ITensor}, B::Pair{ITensor}, ...; tags)
    +C = A * B # inner product of A and B, all indices contracted
    source
    ITensors.dagMethod
    dag(T::ITensor; allow_alias = true)

    Complex conjugate the elements of the ITensor T and dagger the indices.

    By default, an alias of the ITensor is returned (i.e. the output ITensor may share data with the input ITensor). If allow_alias = false, an alias is never returned.

    source
    ITensors.directsumMethod
    directsum(A::Pair{ITensor}, B::Pair{ITensor}, ...; tags)
     
     directsum(output_inds, A::Pair{ITensor}, B::Pair{ITensor}, ...; tags)

    Given a list of pairs of ITensors and indices, perform a partial direct sum of the tensors over the specified indices. Indices that are not specified to be summed must match between the tensors.

    (Note: Pair{ITensor} in Julia is short for Pair{ITensor,<:Any} which means any pair T => x where T is an ITensor.)

    If all indices are specified then the operation is equivalent to creating a block diagonal tensor.

    Returns the ITensor representing the partial direct sum as well as the new direct summed indices. The tags of the direct summed indices are specified by the keyword arguments.

    Optionally, pass the new direct summed indices of the output tensor as the first argument (either a single Index or a collection), which must be proper direct sums of the input indices that are specified to be direct summed.

    See Section 2.3 of https://arxiv.org/abs/1405.7786 for a definition of a partial direct sum of tensors.

    Examples

    x = Index(2, "x")
     i1 = Index(3, "i1")
    @@ -350,18 +350,18 @@
     S, s = directsum(A1 => (i1, j1), A2 => (i2, j2); tags = ["sum_i", "sum_j"])
     length(s) == 2
     dim(s[1]) == dim(i1) + dim(i2)
    -dim(s[2]) == dim(j1) + dim(j2)
    source
    Base.expMethod
    exp(A::ITensor, Linds=Rinds', Rinds=inds(A,plev=0); ishermitian = false)

    Compute the exponential of the tensor A by treating it as a matrix $A_{lr}$ with the left index l running over all indices in Linds and r running over all indices in Rinds.

    Only accepts index lists Linds,Rinds such that: (1) length(Linds) + length(Rinds) == length(inds(A)) (2) length(Linds) == length(Rinds) (3) For each pair of indices (Linds[n],Rinds[n]), Linds[n] and Rinds[n] represent the same Hilbert space (the same QN structure in the QN case, or just the same length in the dense case), and appear in A with opposite directions.

    When ishermitian=true the exponential of Hermitian(A_{lr}) is computed internally.

    source
    LinearAlgebra.nullspaceMethod
    nullspace(T::ITensor, left_inds...; tags="n", atol=1E-12, kwargs...)

    Viewing the ITensor T as a matrix with the provided left_inds viewed as the row space and remaining indices viewed as the right indices or column space, the nullspace function computes the right null space. That is, it will return a tensor N acting on the right indices of T such that T*N is zero. The returned tensor N will also have a new index with the label "n" which indexes through the 'vectors' in the null space.

    For example, if T has the indices i,j,k, calling N = nullspace(T,i,k) returns N with index j such that

           ___       ___
    +dim(s[2]) == dim(j1) + dim(j2)
    source
    Base.expMethod
    exp(A::ITensor, Linds=Rinds', Rinds=inds(A,plev=0); ishermitian = false)

    Compute the exponential of the tensor A by treating it as a matrix $A_{lr}$ with the left index l running over all indices in Linds and r running over all indices in Rinds.

    Only accepts index lists Linds,Rinds such that: (1) length(Linds) + length(Rinds) == length(inds(A)) (2) length(Linds) == length(Rinds) (3) For each pair of indices (Linds[n],Rinds[n]), Linds[n] and Rinds[n] represent the same Hilbert space (the same QN structure in the QN case, or just the same length in the dense case), and appear in A with opposite directions.

    When ishermitian=true the exponential of Hermitian(A_{lr}) is computed internally.

    source
    LinearAlgebra.nullspaceMethod
    nullspace(T::ITensor, left_inds...; tags="n", atol=1E-12, kwargs...)

    Viewing the ITensor T as a matrix with the provided left_inds viewed as the row space and remaining indices viewed as the right indices or column space, the nullspace function computes the right null space. That is, it will return a tensor N acting on the right indices of T such that T*N is zero. The returned tensor N will also have a new index with the label "n" which indexes through the 'vectors' in the null space.

    For example, if T has the indices i,j,k, calling N = nullspace(T,i,k) returns N with index j such that

           ___       ___
       i --|   |     |   |
           | T |--j--| N |--n  ≈ 0
       k --|   |     |   |
    -       ---       ---

    The index n can be obtained by calling n = uniqueindex(N,T)

    Note that the implementation of this function is subject to change in the future, in which case the precise atol value that gives a certain null space size may change in future versions of ITensor.

    Keyword arguments:

    • atol::Float64=1E-12 - singular values of T†*T below this value define the null space
    • tags::String="n" - choose the tags of the index selecting elements of the null space
    source

    Decompositions

    LinearAlgebra.svdMethod
    svd(A::ITensor, inds::Index...; <keyword arguments>)

    Singular value decomposition (SVD) of an ITensor A, computed by treating the "left indices" provided collectively as a row index, and the remaining "right indices" as a column index (matricization of a tensor).

    The first three return arguments are U, S, and V, such that A ≈ U * S * V.

    Whether or not the SVD performs a trunction depends on the keyword arguments provided.

    If the left or right set of indices are empty, all input indices are put on V or U respectively. To specify an empty set of left indices, you must explicitly use svd(A, ()) (svd(A) is currently undefined).

    Examples

    Computing the SVD of an order-three ITensor, such that the indices i and k end up on U and j ends up on V

    i = Index(2)
    +       ---       ---

    The index n can be obtained by calling n = uniqueindex(N,T)

    Note that the implementation of this function is subject to change in the future, in which case the precise atol value that gives a certain null space size may change in future versions of ITensor.

    Keyword arguments:

    • atol::Float64=1E-12 - singular values of T†*T below this value define the null space
    • tags::String="n" - choose the tags of the index selecting elements of the null space
    source

    Decompositions

    LinearAlgebra.svdMethod
    svd(A::ITensor, inds::Index...; <keyword arguments>)

    Singular value decomposition (SVD) of an ITensor A, computed by treating the "left indices" provided collectively as a row index, and the remaining "right indices" as a column index (matricization of a tensor).

    The first three return arguments are U, S, and V, such that A ≈ U * S * V.

    Whether or not the SVD performs a trunction depends on the keyword arguments provided.

    If the left or right set of indices are empty, all input indices are put on V or U respectively. To specify an empty set of left indices, you must explicitly use svd(A, ()) (svd(A) is currently undefined).

    Examples

    Computing the SVD of an order-three ITensor, such that the indices i and k end up on U and j ends up on V

    i = Index(2)
     j = Index(5)
     k = Index(2)
     A = random_itensor(i, j, k)
     U, S, V = svd(A, i, k);
     @show norm(A - U * S * V) <= 10 * eps() * norm(A)

    The following code will truncate the last 2 singular values, since the total number of singular values is 4. The norm of the difference with the original tensor will be the sqrt root of the sum of the squares of the singular values that get truncated.

    trunc, Strunc, Vtrunc = svd(A, i, k; maxdim=2);
     @show norm(A - Utrunc * Strunc * Vtrunc) ≈ sqrt(S[3, 3]^2 + S[4, 4]^2)

    Alternatively we can specify that we want to truncate the weights of the singular values up to a certain cutoff, so the total error will be no larger than the cutoff.

    Utrunc2, Strunc2, Vtrunc2 = svd(A, i, k; cutoff=1e-10);
    -@show norm(A - Utrunc2 * Strunc2 * Vtrunc2) <= 1e-10

    Keywords

    • maxdim::Int: the maximum number of singular values to keep.
    • mindim::Int: the minimum number of singular values to keep.
    • cutoff::Float64: set the desired truncation error of the SVD, by default defined as the sum of the squares of the smallest singular values.
    • lefttags::String = "Link,u": set the tags of the Index shared by U and S.
    • righttags::String = "Link,v": set the tags of the Index shared by S and V.
    • alg::String = "divide_and_conquer". Options:
    • "divide_and_conquer" - A divide-and-conquer algorithm. LAPACK's gesdd. Fast, but may lead to some innacurate singular values for very ill-conditioned matrices. Also may sometimes fail to converge, leading to errors (in which case "qr_iteration" or "recursive" can be tried).
      • "qr_iteration" - Typically slower but more accurate for very ill-conditioned matrices compared to "divide_and_conquer". LAPACK's gesvd.
      • "recursive" - ITensor's custom svd. Very reliable, but may be slow if high precision is needed. To get an svd of a matrix A, an eigendecomposition of $A^{\dagger} A$ is used to compute U and then a qr of $A^{\dagger} U$ is used to compute V. This is performed recursively to compute small singular values.
    • use_absolute_cutoff::Bool = false: set if all probability weights below the cutoff value should be discarded, rather than the sum of discarded weights.
    • use_relative_cutoff::Bool = true: set if the singular values should be normalized for the sake of truncation.
    • min_blockdim::Int = 0: for SVD of block-sparse or QN ITensors, require that the number of singular values kept be greater than or equal to this value when possible

    See also: factorize, eigen

    source
    LinearAlgebra.eigenMethod
    eigen(A::ITensor[, Linds, Rinds]; <keyword arguments>)

    Eigendecomposition of an ITensor A, computed by treating the "left indices" Linds provided collectively as a row index, and remaining "right indices" Rinds as a column index (matricization of a tensor).

    If no indices are provided, pairs of primed and unprimed indices are searched for, with Linds taken to be the primed indices and Rinds taken to be the unprimed indices.

    The return arguments are the eigenvalues D and eigenvectors U as tensors, such that A * U ∼ U * D (more precisely they are approximately equal up to proper replacements of indices, see the example for details).

    Whether or not eigen performs a trunction depends on the keyword arguments provided. Note that truncation is only well defined for positive semidefinite matrices.

    Arguments

    - `maxdim::Int`: the maximum number of singular values to keep.
    +@show norm(A - Utrunc2 * Strunc2 * Vtrunc2) <= 1e-10

    Keywords

    • maxdim::Int: the maximum number of singular values to keep.
    • mindim::Int: the minimum number of singular values to keep.
    • cutoff::Float64: set the desired truncation error of the SVD, by default defined as the sum of the squares of the smallest singular values.
    • lefttags::String = "Link,u": set the tags of the Index shared by U and S.
    • righttags::String = "Link,v": set the tags of the Index shared by S and V.
    • alg::String = "divide_and_conquer". Options:
    • "divide_and_conquer" - A divide-and-conquer algorithm. LAPACK's gesdd. Fast, but may lead to some innacurate singular values for very ill-conditioned matrices. Also may sometimes fail to converge, leading to errors (in which case "qr_iteration" or "recursive" can be tried).
      • "qr_iteration" - Typically slower but more accurate for very ill-conditioned matrices compared to "divide_and_conquer". LAPACK's gesvd.
      • "recursive" - ITensor's custom svd. Very reliable, but may be slow if high precision is needed. To get an svd of a matrix A, an eigendecomposition of $A^{\dagger} A$ is used to compute U and then a qr of $A^{\dagger} U$ is used to compute V. This is performed recursively to compute small singular values.
    • use_absolute_cutoff::Bool = false: set if all probability weights below the cutoff value should be discarded, rather than the sum of discarded weights.
    • use_relative_cutoff::Bool = true: set if the singular values should be normalized for the sake of truncation.
    • min_blockdim::Int = 0: for SVD of block-sparse or QN ITensors, require that the number of singular values kept be greater than or equal to this value when possible

    See also: factorize, eigen

    source
    LinearAlgebra.eigenMethod
    eigen(A::ITensor[, Linds, Rinds]; <keyword arguments>)

    Eigendecomposition of an ITensor A, computed by treating the "left indices" Linds provided collectively as a row index, and remaining "right indices" Rinds as a column index (matricization of a tensor).

    If no indices are provided, pairs of primed and unprimed indices are searched for, with Linds taken to be the primed indices and Rinds taken to be the unprimed indices.

    The return arguments are the eigenvalues D and eigenvectors U as tensors, such that A * U ∼ U * D (more precisely they are approximately equal up to proper replacements of indices, see the example for details).

    Whether or not eigen performs a trunction depends on the keyword arguments provided. Note that truncation is only well defined for positive semidefinite matrices.

    Arguments

    - `maxdim::Int`: the maximum number of singular values to keep.
     - `mindim::Int`: the minimum number of singular values to keep.
     - `cutoff::Float64`: set the desired truncation error of the eigenvalues,
        by default defined as the sum of the squares of the smallest eigenvalues.
    @@ -393,7 +393,7 @@
     D, U = eigen(A, Linds, Rinds)
     dl, dr = uniqueind(D, U), commonind(D, U)
     Ul = replaceinds(U, (Rinds..., dr) => (Linds..., dl))
    -A * U ≈ Ul * D # true

    See also: svd, factorize

    source
    LinearAlgebra.factorizeMethod
    factorize(A::ITensor, Linds::Index...; <keyword arguments>)

    Perform a factorization of A into ITensors L and R such that A ≈ L * R.

    Arguments

    • ortho::String = "left": Choose orthogonality properties of the factorization.
      • "left": the left factor L is an orthogonal basis such that L * dag(prime(L, commonind(L,R))) ≈ I.
      • "right": the right factor R forms an orthogonal basis.
      • "none", neither of the factors form an orthogonal basis, and in general are made as symmetrically as possible (depending on the decomposition used).
    • which_decomp::Union{String, Nothing} = nothing: choose what kind of decomposition is used.
      • nothing: choose the decomposition automatically based on the other arguments. For example, when nothing is chosen and ortho = "left" or "right", and a cutoff is provided, svd or eigen is used depending on the provided cutoff (eigen is only used when the cutoff is greater than 1e-12, since it has a lower precision). When no truncation is requested qr is used for dense ITensors and svd for block-sparse ITensors (in the future qr will be used also for block-sparse ITensors in this case).
      • "svd": L = U and R = S * V for ortho = "left", L = U * S and R = V for ortho = "right", and L = U * sqrt.(S) and R = sqrt.(S) * V for ortho = "none". To control which svd algorithm is choose, use the svd_alg keyword argument. See the documentation for svd for the supported algorithms, which are the same as those accepted by the alg keyword argument.
      • "eigen": L = U and $R = U^{\dagger} A$ where U is determined from the eigendecompositon $A A^{\dagger} = U D U^{\dagger}$ for ortho = "left" (and vice versa for ortho = "right"). "eigen" is not supported for ortho = "none".
      • "qr": L=Q and R an upper-triangular matrix when ortho = "left", and R = Q and L a lower-triangular matrix when ortho = "right" (currently supported for dense ITensors only). In the future, other decompositions like QR (for block-sparse ITensors), polar, cholesky, LU, etc. are expected to be supported.

    For truncation arguments, see: svd

    source

    Memory operations

    ITensors.permuteMethod
    permute(T::ITensor, inds...; allow_alias = false)

    Return a new ITensor T with indices permuted according to the input indices inds. The storage of the ITensor is permuted accordingly.

    If called with allow_alias = true, it avoids copying data if possible. Therefore, it may return an alias of the input ITensor (an ITensor that shares the same data), such as if the permutation turns out to be trivial.

    By default, allow_alias = false, and it never returns an alias of the input ITensor.

    Examples

    i = Index(2, "index_i"); j = Index(4, "index_j"); k = Index(3, "index_k");
    +A * U ≈ Ul * D # true

    See also: svd, factorize

    source
    LinearAlgebra.factorizeMethod
    factorize(A::ITensor, Linds::Index...; <keyword arguments>)

    Perform a factorization of A into ITensors L and R such that A ≈ L * R.

    Arguments

    • ortho::String = "left": Choose orthogonality properties of the factorization.
      • "left": the left factor L is an orthogonal basis such that L * dag(prime(L, commonind(L,R))) ≈ I.
      • "right": the right factor R forms an orthogonal basis.
      • "none", neither of the factors form an orthogonal basis, and in general are made as symmetrically as possible (depending on the decomposition used).
    • which_decomp::Union{String, Nothing} = nothing: choose what kind of decomposition is used.
      • nothing: choose the decomposition automatically based on the other arguments. For example, when nothing is chosen and ortho = "left" or "right", and a cutoff is provided, svd or eigen is used depending on the provided cutoff (eigen is only used when the cutoff is greater than 1e-12, since it has a lower precision). When no truncation is requested qr is used for dense ITensors and svd for block-sparse ITensors (in the future qr will be used also for block-sparse ITensors in this case).
      • "svd": L = U and R = S * V for ortho = "left", L = U * S and R = V for ortho = "right", and L = U * sqrt.(S) and R = sqrt.(S) * V for ortho = "none". To control which svd algorithm is choose, use the svd_alg keyword argument. See the documentation for svd for the supported algorithms, which are the same as those accepted by the alg keyword argument.
      • "eigen": L = U and $R = U^{\dagger} A$ where U is determined from the eigendecompositon $A A^{\dagger} = U D U^{\dagger}$ for ortho = "left" (and vice versa for ortho = "right"). "eigen" is not supported for ortho = "none".
      • "qr": L=Q and R an upper-triangular matrix when ortho = "left", and R = Q and L a lower-triangular matrix when ortho = "right" (currently supported for dense ITensors only). In the future, other decompositions like QR (for block-sparse ITensors), polar, cholesky, LU, etc. are expected to be supported.

    For truncation arguments, see: svd

    source

    Memory operations

    ITensors.permuteMethod
    permute(T::ITensor, inds...; allow_alias = false)

    Return a new ITensor T with indices permuted according to the input indices inds. The storage of the ITensor is permuted accordingly.

    If called with allow_alias = true, it avoids copying data if possible. Therefore, it may return an alias of the input ITensor (an ITensor that shares the same data), such as if the permutation turns out to be trivial.

    By default, allow_alias = false, and it never returns an alias of the input ITensor.

    Examples

    i = Index(2, "index_i"); j = Index(4, "index_j"); k = Index(3, "index_k");
     T = random_itensor(i, j, k)
     
     pT_1 = permute(T, k, i, j)
    @@ -409,4 +409,4 @@
     
     pT_alias = permute(T, i, j, k; allow_alias = true)
     pT_alias[1, 1, 1] = 12
    -T[1, 1, 1] == pT_alias[1, 1, 1]
    source
    NDTensors.denseMethod
    dense(T::ITensor)

    Make a new ITensor where the storage is the closest Dense storage, avoiding allocating new data if possible. For example, an ITensor with Diag storage will become Dense storage, filled with zeros except for the diagonal values.

    source
    NDTensors.denseblocksMethod
    denseblocks(T::ITensor)

    Make a new ITensor where any blocks which have a sparse format, such as diagonal sparsity, are made dense while still preserving the outer block-sparse structure. This method avoids allocating new data if possible.

    For example, an ITensor with DiagBlockSparse storage will have BlockSparse storage afterwards.

    source
    +T[1, 1, 1] == pT_alias[1, 1, 1]source
    NDTensors.denseMethod
    dense(T::ITensor)

    Make a new ITensor where the storage is the closest Dense storage, avoiding allocating new data if possible. For example, an ITensor with Diag storage will become Dense storage, filled with zeros except for the diagonal values.

    source
    NDTensors.denseblocksMethod
    denseblocks(T::ITensor)

    Make a new ITensor where any blocks which have a sparse format, such as diagonal sparsity, are made dense while still preserving the outer block-sparse structure. This method avoids allocating new data if possible.

    For example, an ITensor with DiagBlockSparse storage will have BlockSparse storage afterwards.

    source
    diff --git a/previews/PR1514/IncludedSiteTypes.html b/previews/PR1514/IncludedSiteTypes.html index ce09323880..8bfbd8223b 100644 --- a/previews/PR1514/IncludedSiteTypes.html +++ b/previews/PR1514/IncludedSiteTypes.html @@ -13,4 +13,4 @@ sites = siteinds("Electron",N)

    Available keyword arguments for enabling and customizing quantum numbers (QN) subspaces:

    For example:

    sites = siteinds("Electron",N; conserve_nfparity=true)

    "Electron" States

    The available state names for "Electron" sites are:

    "Electron" Operators

    Operators associated with "Electron" sites can be made using the op function, for example

    Cup = op("Cup",s)
     Cup4 = op("Cup",sites[4])

    Single-fermion operators:

    Non-fermionic single particle operators (these do not have Jordan-Wigner string attached, so will commute within systems such as OpSum or the apply function):

    "tJ" SiteType

    "tJ" sites are similar to electron sites, but cannot be doubly occupied The states of site indices with the "tJ" SiteType correspond to $|0\rangle$, $|\!\uparrow\rangle$, $|\!\downarrow\rangle$.

    Making a single "tJ" site or collection of N "tJ" sites

    s = siteind("tJ")
     sites = siteinds("tJ",N)

    Available keyword arguments for enabling and customizing quantum numbers (QN) subspaces:

    For example:

    sites = siteinds("tJ",N; conserve_nfparity=true)

    "tJ" States

    The available state names for "tJ" sites are:

    "tJ" Operators

    Operators associated with "tJ" sites can be made using the op function, for example

    Cup = op("Cup",s)
    -Cup4 = op("Cup",sites[4])

    Single-fermion operators:

    Non-fermionic single particle operators (these do not have Jordan-Wigner string attached, so will commute within systems such as OpSum or the apply function):

    +Cup4 = op("Cup",sites[4])

    Single-fermion operators:

    Non-fermionic single particle operators (these do not have Jordan-Wigner string attached, so will commute within systems such as OpSum or the apply function):

    diff --git a/previews/PR1514/IndexSetType.html b/previews/PR1514/IndexSetType.html index dfb55f69a3..8a41a396bf 100644 --- a/previews/PR1514/IndexSetType.html +++ b/previews/PR1514/IndexSetType.html @@ -1,9 +1,9 @@ -Index collections · ITensors.jl

    Index collections

    Collections of Index are used throughout ITensors.jl to represent the dimensions of tensors. In general, collections that are recognized and returned by ITensors.jl functions are either Vector of Index or Tuple of Index, depending on the context. For example internally an ITensor has a static number of indices so stores a Tuple of Index, while set operations like commoninds((i, j, k), (j, k, l)) will return a Vector [j, k] since the operation is inherently dynamic, i.e. the number of indices in the intersection can't in general be known before running the code. Vector of Index and Tuple of Index can usually be used interchangeably, but one or the other may be faster depending on the operation being performed.

    Priming and tagging

    Documentation for priming and tagging collections of Index can be found in the ITensor Priming and tagging section.

    Set operations

    Documentation for set operations involving Index collections can be found in the ITensor Index collections set operations section.

    Subsets

    ITensors.getfirstMethod
    getfirst(f::Function, is::Indices)

    Get the first Index matching the pattern function, return nothing if not found.

    source
    ITensors.getfirstMethod
    getfirst(is::Indices)

    Return the first Index in the Indices. If the Indices is empty, return nothing.

    source

    Iterating

    ITensors.eachvalMethod
    eachval(is::Index...)
    -eachval(is::Tuple{Vararg{Index}})

    Create an iterator whose values correspond to a Cartesian indexing over the dimensions of the provided Index objects.

    source
    ITensors.eachindvalMethod
    eachindval(is::Index...)
    +Index collections · ITensors.jl

    Index collections

    Collections of Index are used throughout ITensors.jl to represent the dimensions of tensors. In general, collections that are recognized and returned by ITensors.jl functions are either Vector of Index or Tuple of Index, depending on the context. For example internally an ITensor has a static number of indices so stores a Tuple of Index, while set operations like commoninds((i, j, k), (j, k, l)) will return a Vector [j, k] since the operation is inherently dynamic, i.e. the number of indices in the intersection can't in general be known before running the code. Vector of Index and Tuple of Index can usually be used interchangeably, but one or the other may be faster depending on the operation being performed.

    Priming and tagging

    Documentation for priming and tagging collections of Index can be found in the ITensor Priming and tagging section.

    Set operations

    Documentation for set operations involving Index collections can be found in the ITensor Index collections set operations section.

    Subsets

    ITensors.getfirstMethod
    getfirst(f::Function, is::Indices)

    Get the first Index matching the pattern function, return nothing if not found.

    source
    ITensors.getfirstMethod
    getfirst(is::Indices)

    Return the first Index in the Indices. If the Indices is empty, return nothing.

    source

    Iterating

    ITensors.eachvalMethod
    eachval(is::Index...)
    +eachval(is::Tuple{Vararg{Index}})

    Create an iterator whose values correspond to a Cartesian indexing over the dimensions of the provided Index objects.

    source
    ITensors.eachindvalMethod
    eachindval(is::Index...)
     eachindval(is::Tuple{Vararg{Index}})

    Create an iterator whose values are Index=>value pairs corresponding to a Cartesian indexing over the dimensions of the provided Index objects.

    Example

    i = Index(3; tags="i")
     j = Index(2; tags="j")
     T = random_itensor(j, i)
     for iv in eachindval(i, j)
       @show T[iv...]
    -end
    source
    ITensors.dirMethod
    dir(is::Indices, i::Index)

    Return the direction of the Index i in the Indices is.

    source
    +end
    source
    ITensors.dirMethod
    dir(is::Indices, i::Index)

    Return the direction of the Index i in the Indices is.

    source
    diff --git a/previews/PR1514/IndexType.html b/previews/PR1514/IndexType.html index 5b0f72b234..24ca775ed6 100644 --- a/previews/PR1514/IndexType.html +++ b/previews/PR1514/IndexType.html @@ -1,5 +1,5 @@ -Index · ITensors.jl

    Index

    Description

    ITensors.IndexType

    An Index represents a single tensor index with fixed dimension dim. Copies of an Index compare equal unless their tags are different.

    An Index carries a TagSet, a set of tags which are small strings that specify properties of the Index to help distinguish it from other Indices. There is a special tag which is referred to as the integer tag or prime level which can be incremented or decremented with special priming functions.

    Internally, an Index has a fixed id number, which is how the ITensor library knows two indices are copies of a single original Index. Index objects must have the same id, as well as the tags to compare equal.

    source
    ITensors.QNIndexType

    A QN Index is an Index with QN block storage instead of just an integer dimension. The QN block storage is a vector of pairs of QNs and block dimensions. The total dimension of a QN Index is the sum of the dimensions of the blocks of the Index.

    source

    Constructors

    ITensors.IndexMethod
    Index(dim::Int; tags::Union{AbstractString, TagSet} = "",
    +Index · ITensors.jl

    Index

    Description

    ITensors.IndexType

    An Index represents a single tensor index with fixed dimension dim. Copies of an Index compare equal unless their tags are different.

    An Index carries a TagSet, a set of tags which are small strings that specify properties of the Index to help distinguish it from other Indices. There is a special tag which is referred to as the integer tag or prime level which can be incremented or decremented with special priming functions.

    Internally, an Index has a fixed id number, which is how the ITensor library knows two indices are copies of a single original Index. Index objects must have the same id, as well as the tags to compare equal.

    source
    ITensors.QNIndexType

    A QN Index is an Index with QN block storage instead of just an integer dimension. The QN block storage is a vector of pairs of QNs and block dimensions. The total dimension of a QN Index is the sum of the dimensions of the blocks of the Index.

    source

    Constructors

    ITensors.IndexMethod
    Index(dim::Int; tags::Union{AbstractString, TagSet} = "",
                     plev::Int = 0)

    Create an Index with a unique id, a TagSet given by tags, and a prime level plev.

    Examples

    julia> i = Index(2; tags="l", plev=1)
     (dim=2|id=818|"l")'
     
    @@ -10,7 +10,7 @@
     1
     
     julia> tags(i)
    -"l"
    source
    ITensors.IndexMethod
    Index(dim::Integer, tags::Union{AbstractString, TagSet}; plev::Int = 0)

    Create an Index with a unique id and a tagset given by tags.

    Examples

    julia> i = Index(2, "l,tag")
    +"l"
    source
    ITensors.IndexMethod
    Index(dim::Integer, tags::Union{AbstractString, TagSet}; plev::Int = 0)

    Create an Index with a unique id and a tagset given by tags.

    Examples

    julia> i = Index(2, "l,tag")
     (dim=2|id=58|"l,tag")
     
     julia> dim(i)
    @@ -20,12 +20,12 @@
     0
     
     julia> tags(i)
    -"l,tag"
    source
    ITensors.IndexMethod
    Index(qnblocks::Pair{QN, Int64}...; dir::Arrow = Out,
                                         tags = "",
    -                                    plev::Integer = 0)

    Construct a QN Index from a list of pairs of QN and block dimensions.

    Example

    Index(QN("Sz", -1) => 1, QN("Sz", 1) => 1; tags = "i")
    source
    ITensors.IndexMethod
    Index(qnblocks::Vector{Pair{QN, Int64}}; dir::Arrow = Out,
    +                                    plev::Integer = 0)

    Construct a QN Index from a list of pairs of QN and block dimensions.

    Example

    Index(QN("Sz", -1) => 1, QN("Sz", 1) => 1; tags = "i")
    source
    ITensors.IndexMethod
    Index(qnblocks::Vector{Pair{QN, Int64}}; dir::Arrow = Out,
                                              tags = "",
    -                                         plev::Integer = 0)

    Construct a QN Index from a Vector of pairs of QN and block dimensions.

    Note: in the future, this may enforce that all blocks have the same QNs (which would allow for some optimizations, for example when constructing random QN ITensors).

    Example

    Index([QN("Sz", -1) => 1, QN("Sz", 1) => 1]; tags = "i")
    source
    ITensors.IndexMethod
    Index(qnblocks::Vector{Pair{QN, Int64}}, tags; dir::Arrow = Out,
    -                                               plev::Integer = 0)

    Construct a QN Index from a Vector of pairs of QN and block dimensions.

    Example

    Index([QN("Sz", -1) => 1, QN("Sz", 1) => 1], "i"; dir = In)
    source

    Properties

    ITensors.idMethod
    id(i::Index)

    Obtain the id of an Index, which is a unique 64 digit integer.

    source
    ITensors.hasidMethod
    hasid(i::Index, id::ITensors.IDType)

    Check if an Index i has the provided id.

    Examples

    julia> i = Index(2)
    +                                         plev::Integer = 0)

    Construct a QN Index from a Vector of pairs of QN and block dimensions.

    Note: in the future, this may enforce that all blocks have the same QNs (which would allow for some optimizations, for example when constructing random QN ITensors).

    Example

    Index([QN("Sz", -1) => 1, QN("Sz", 1) => 1]; tags = "i")
    source
    ITensors.IndexMethod
    Index(qnblocks::Vector{Pair{QN, Int64}}, tags; dir::Arrow = Out,
    +                                               plev::Integer = 0)

    Construct a QN Index from a Vector of pairs of QN and block dimensions.

    Example

    Index([QN("Sz", -1) => 1, QN("Sz", 1) => 1], "i"; dir = In)
    source

    Properties

    ITensors.idMethod
    id(i::Index)

    Obtain the id of an Index, which is a unique 64 digit integer.

    source
    ITensors.hasidMethod
    hasid(i::Index, id::ITensors.IDType)

    Check if an Index i has the provided id.

    Examples

    julia> i = Index(2)
     (dim=2|id=321)
     
     julia> hasid(i, id(i))
    @@ -35,23 +35,23 @@
     (dim=2|id=17)
     
     julia> hasid(i, id(j))
    -false
    source
    ITensors.TagSets.set_strict_tags!Method
    set_strict_tags!(enable::Bool) -> Bool
    -

    Enable or disable checking for overflow of the number of tags of a TagSet or the number of characters of a tag. If enabled (set to true), an error will be thrown if overflow occurs, otherwise the overflow will be ignored and the extra tags or tag characters will be dropped. This could cause unexpected bugs if tags are being used to distinguish Index objects that have the same ids and prime levels, but that is generally discouraged and should only be used if you know what you are doing.

    See also ITensors.using_strict_tags.

    source
    ITensors.TagSets.hastagsMethod
    hastags(i::Index, ts::Union{AbstractString,TagSet})

    Check if an Index i has the provided tags, which can be a string of comma-separated tags or a TagSet object.

    Examples

    julia> i = Index(2, "SpinHalf,Site,n=3")
    +false
    source
    ITensors.TagSets.set_strict_tags!Method
    set_strict_tags!(enable::Bool) -> Bool
    +

    Enable or disable checking for overflow of the number of tags of a TagSet or the number of characters of a tag. If enabled (set to true), an error will be thrown if overflow occurs, otherwise the overflow will be ignored and the extra tags or tag characters will be dropped. This could cause unexpected bugs if tags are being used to distinguish Index objects that have the same ids and prime levels, but that is generally discouraged and should only be used if you know what you are doing.

    See also ITensors.using_strict_tags.

    source
    ITensors.TagSets.hastagsMethod
    hastags(i::Index, ts::Union{AbstractString,TagSet})

    Check if an Index i has the provided tags, which can be a string of comma-separated tags or a TagSet object.

    Examples

    julia> i = Index(2, "SpinHalf,Site,n=3")
     (dim=2|id=861|"Site,SpinHalf,n=3")
     
     julia> hastags(i, "SpinHalf,Site")
     true
     
     julia> hastags(i, "Link")
    -false
    source
    ITensors.hasplevMethod
    hasplev(i::Index, plev::Int)

    Check if an Index i has the provided prime level.

    Examples

    julia> i = Index(2; plev=2)
    +false
    source
    ITensors.hasplevMethod
    hasplev(i::Index, plev::Int)

    Check if an Index i has the provided prime level.

    Examples

    julia> i = Index(2; plev=2)
     (dim=2|id=543)''
     
     julia> hasplev(i, 2)
     true
     
     julia> hasplev(i, 1)
    -false
    source
    NDTensors.dimMethod
    dim(i::Index)

    Obtain the dimension of an Index.

    For a QN Index, this is the sum of the block dimensions.

    source
    Base.:==Method
    ==(i1::Index, i1::Index)

    Compare indices for equality. First the id's are compared, then the prime levels are compared, and finally the tags are compared.

    source
    ITensors.dirMethod
    dir(i::Index)

    Return the direction of an Index (ITensors.In, ITensors.Out, or ITensors.Neither).

    source

    Priming and tagging methods

    ITensors.primeMethod
    prime(i::Index, plinc::Int = 1)

    Return a copy of Index i with its prime level incremented by the amount plinc

    source
    Base.:^Method
    ^(i::Index, pl::Int)

    Prime an Index using the notation i^3.

    source
    ITensors.setprimeMethod
    setprime(i::Index, plev::Int)

    Return a copy of Index i with its prime level set to plev

    source
    ITensors.settagsMethod
    settags(i::Index, ts)

    Return a copy of Index i with tags replaced by the ones given The ts argument can be a comma-separated string of tags or a TagSet.

    Examples

    julia> i = Index(2, "SpinHalf,Site,n=3")
    +false
    source
    NDTensors.dimMethod
    dim(i::Index)

    Obtain the dimension of an Index.

    For a QN Index, this is the sum of the block dimensions.

    source
    Base.:==Method
    ==(i1::Index, i1::Index)

    Compare indices for equality. First the id's are compared, then the prime levels are compared, and finally the tags are compared.

    source
    ITensors.dirMethod
    dir(i::Index)

    Return the direction of an Index (ITensors.In, ITensors.Out, or ITensors.Neither).

    source

    Priming and tagging methods

    ITensors.primeMethod
    prime(i::Index, plinc::Int = 1)

    Return a copy of Index i with its prime level incremented by the amount plinc

    source
    Base.:^Method
    ^(i::Index, pl::Int)

    Prime an Index using the notation i^3.

    source
    ITensors.setprimeMethod
    setprime(i::Index, plev::Int)

    Return a copy of Index i with its prime level set to plev

    source
    ITensors.settagsMethod
    settags(i::Index, ts)

    Return a copy of Index i with tags replaced by the ones given The ts argument can be a comma-separated string of tags or a TagSet.

    Examples

    julia> i = Index(2, "SpinHalf,Site,n=3")
     (dim=2|id=543|"Site,SpinHalf,n=3")
     
     julia> hastags(i, "Link")
    @@ -64,7 +64,7 @@
     true
     
     julia> hastags(j, "n=4,Link")
    -true
    source
    ITensors.TagSets.addtagsMethod
    addtags(i::Index,ts)

    Return a copy of Index i with the specified tags added to the existing ones. The ts argument can be a comma-separated string of tags or a TagSet.

    source
    ITensors.TagSets.removetagsMethod
    removetags(i::Index, ts)

    Return a copy of Index i with the specified tags removed. The ts argument can be a comma-separated string of tags or a TagSet.

    source
    ITensors.TagSets.addtagsMethod
    addtags(i::Index,ts)

    Return a copy of Index i with the specified tags added to the existing ones. The ts argument can be a comma-separated string of tags or a TagSet.

    source
    ITensors.TagSets.removetagsMethod
    removetags(i::Index, ts)

    Return a copy of Index i with the specified tags removed. The ts argument can be a comma-separated string of tags or a TagSet.

    source
    ITensors.TagSets.replacetagsMethod
    replacetags(i::Index, tsold, tsnew)
     
     replacetags(i::Index, tsold => tsnew)

    If the tag set of i contains the tags specified by tsold, replaces these with the tags specified by tsnew, preserving any other tags. The arguments tsold and tsnew can be comma-separated strings of tags, or TagSet objects.

    Examples

    julia> i = Index(2; tags="l,x", plev=1)
     (dim=2|id=83|"l,x")'
    @@ -73,4 +73,4 @@
     (dim=2|id=83|"m,x")'
     
     julia> replacetags(i, "l" => "m")
    -(dim=2|id=83|"m,x")'
    source

    Methods

    NDTensors.simMethod
    sim(i::Index; tags = tags(i), plev = plev(i), dir = dir(i))

    Produces an Index with the same properties (dimension or QN structure) but with a new id.

    source

    Iterating

    ITensors.eachvalMethod
    eachval(i::Index)

    Create an iterator whose values range over the dimension of the provided Index.

    source
    ITensors.eachindvalMethod
    eachindval(i::Index)

    Create an iterator whose values are Pairs of the form i=>n with n from 1:dim(i). This iterator is useful for accessing elements of an ITensor in a loop without needing to know the ordering of the indices. See also eachindval(is::Index...).

    source
    +(dim=2|id=83|"m,x")'
    source

    Methods

    NDTensors.simMethod
    sim(i::Index; tags = tags(i), plev = plev(i), dir = dir(i))

    Produces an Index with the same properties (dimension or QN structure) but with a new id.

    source

    Iterating

    ITensors.eachvalMethod
    eachval(i::Index)

    Create an iterator whose values range over the dimension of the provided Index.

    source
    ITensors.eachindvalMethod
    eachindval(i::Index)

    Create an iterator whose values are Pairs of the form i=>n with n from 1:dim(i). This iterator is useful for accessing elements of an ITensor in a loop without needing to know the ordering of the indices. See also eachindval(is::Index...).

    source
    diff --git a/previews/PR1514/MPSandMPO.html b/previews/PR1514/MPSandMPO.html index 172108d0fa..e0d8857a0e 100644 --- a/previews/PR1514/MPSandMPO.html +++ b/previews/PR1514/MPSandMPO.html @@ -1,9 +1,9 @@ -MPS and MPO · ITensors.jl

    MPS and MPO

    Types

    MPS Constructors

    ITensors.ITensorMPS.MPSMethod
    MPS([::Type{ElT} = Float64, ]sites; linkdims=1)

    Construct an MPS filled with Empty ITensors of type ElT from a collection of indices.

    Optionally specify the link dimension with the keyword argument linkdims, which by default is 1.

    In the future we may generalize linkdims to allow specifying each individual link dimension as a vector, and additionally allow specifying quantum numbers.

    source
    ITensors.ITensorMPS.random_mpsMethod
    random_mps(sites::Vector{<:Index}; linkdims=1)
    -random_mps(eltype::Type{<:Number}, sites::Vector{<:Index}; linkdims=1)

    Construct a random MPS with link dimension linkdims which by default has element type Float64.

    linkdims can also accept a Vector{Int} with length(linkdims) == length(sites) - 1 for constructing an MPS with non-uniform bond dimension.

    source
    ITensors.ITensorMPS.random_mpsMethod
    random_mps(eltype::Type{<:Number}, sites::Vector{<:Index}; linkdims=1)

    Construct a random MPS with link dimension linkdims of type eltype.

    linkdims can also accept a Vector{Int} with length(linkdims) == length(sites) - 1 for constructing an MPS with non-uniform bond dimension.

    source
    ITensors.ITensorMPS.random_mpsMethod
    random_mps(sites::Vector{<:Index}, state; linkdims=1)

    Construct a real, random MPS with link dimension linkdims, made by randomizing an initial product state specified by state. This version of random_mps is necessary when creating QN-conserving random MPS (consisting of QNITensors). The initial state array provided determines the total QN of the resulting random MPS.

    source
    ITensors.ITensorMPS.MPSMethod
    MPS(sites::Vector{<:Index},states)

    Construct a product state MPS having site indices sites, and which corresponds to the initial state given by the array states. The states array may consist of either an array of integers or strings, as recognized by the state function defined for the relevant Index tag type.

    Examples

    N = 10
    +MPS and MPO · ITensors.jl

    MPS and MPO

    Types

    MPS Constructors

    ITensors.ITensorMPS.MPSMethod
    MPS([::Type{ElT} = Float64, ]sites; linkdims=1)

    Construct an MPS filled with Empty ITensors of type ElT from a collection of indices.

    Optionally specify the link dimension with the keyword argument linkdims, which by default is 1.

    In the future we may generalize linkdims to allow specifying each individual link dimension as a vector, and additionally allow specifying quantum numbers.

    source
    ITensors.ITensorMPS.random_mpsMethod
    random_mps(sites::Vector{<:Index}; linkdims=1)
    +random_mps(eltype::Type{<:Number}, sites::Vector{<:Index}; linkdims=1)

    Construct a random MPS with link dimension linkdims which by default has element type Float64.

    linkdims can also accept a Vector{Int} with length(linkdims) == length(sites) - 1 for constructing an MPS with non-uniform bond dimension.

    source
    ITensors.ITensorMPS.random_mpsMethod
    random_mps(eltype::Type{<:Number}, sites::Vector{<:Index}; linkdims=1)

    Construct a random MPS with link dimension linkdims of type eltype.

    linkdims can also accept a Vector{Int} with length(linkdims) == length(sites) - 1 for constructing an MPS with non-uniform bond dimension.

    source
    ITensors.ITensorMPS.random_mpsMethod
    random_mps(sites::Vector{<:Index}, state; linkdims=1)

    Construct a real, random MPS with link dimension linkdims, made by randomizing an initial product state specified by state. This version of random_mps is necessary when creating QN-conserving random MPS (consisting of QNITensors). The initial state array provided determines the total QN of the resulting random MPS.

    source
    ITensors.ITensorMPS.MPSMethod
    MPS(sites::Vector{<:Index},states)

    Construct a product state MPS having site indices sites, and which corresponds to the initial state given by the array states. The states array may consist of either an array of integers or strings, as recognized by the state function defined for the relevant Index tag type.

    Examples

    N = 10
     sites = siteinds("S=1/2", N)
     states = [isodd(n) ? "Up" : "Dn" for n in 1:N]
    -psi = MPS(sites, states)
    source
    ITensors.ITensorMPS.MPSMethod
    MPS(::Type{T},
         sites::Vector{<:Index},
         states::Union{Vector{String},
                       Vector{Int},
    @@ -12,7 +12,7 @@
     sites = siteinds("S=1/2", N)
     states = [isodd(n) ? "Up" : "Dn" for n in 1:N]
     psi = MPS(ComplexF64, sites, states)
    -phi = MPS(sites, "Up")
    source
    ITensors.ITensorMPS.MPSMethod
    MPS(ivals::Vector{<:Pair{<:Index}})

    Construct a product state MPS with element type Float64 and nonzero values determined from the input IndexVals.

    source
    ITensors.ITensorMPS.MPSMethod
    MPS(::Type{T<:Number}, ivals::Vector{<:Pair{<:Index}})

    Construct a product state MPS with element type T and nonzero values determined from the input IndexVals.

    source

    MPO Constructors

    ITensors.ITensorMPS.MPOMethod
    MPO([::Type{ElT} = Float64}, ]sites, ops::Vector{String})

    Make an MPO with pairs of sites s[i] and s[i]' and operators ops on each site.

    source
    ITensors.ITensorMPS.MPOMethod
    MPO([::Type{ElT} = Float64, ]sites, op::String)

    Make an MPO with pairs of sites s[i] and s[i]' and operator op on every site.

    source

    Copying behavior

    ITensors.ITensorMPS.MPSMethod
    MPS(ivals::Vector{<:Pair{<:Index}})

    Construct a product state MPS with element type Float64 and nonzero values determined from the input IndexVals.

    source
    ITensors.ITensorMPS.MPSMethod
    MPS(::Type{T<:Number}, ivals::Vector{<:Pair{<:Index}})

    Construct a product state MPS with element type T and nonzero values determined from the input IndexVals.

    source

    MPO Constructors

    ITensors.ITensorMPS.MPOMethod
    MPO([::Type{ElT} = Float64}, ]sites, ops::Vector{String})

    Make an MPO with pairs of sites s[i] and s[i]' and operators ops on each site.

    source
    ITensors.ITensorMPS.MPOMethod
    MPO([::Type{ElT} = Float64, ]sites, op::String)

    Make an MPO with pairs of sites s[i] and s[i]' and operator op on every site.

    source

    Copying behavior

    Base.copyMethod
    copy(::MPS)
     copy(::MPO)

    Make a shallow copy of an MPS or MPO. By shallow copy, it means that a new MPS/MPO is returned, but the data of the tensors are still shared between the returned MPS/MPO and the original MPS/MPO.

    Therefore, replacing an entire tensor of the returned MPS/MPO will not modify the input MPS/MPO, but modifying the data of the returned MPS/MPO will modify the input MPS/MPO.

    Use deepcopy for an alternative that copies the ITensors as well.

    Examples

    julia> using ITensors, ITensorMPS
     
     julia> s = siteinds("S=1/2", 3);
    @@ -40,7 +40,7 @@
     3.0000000000000004
     
     julia> norm(M3)
    -3.0000000000000004
    source
    Base.deepcopyMethod
    deepcopy(::MPS)
     deepcopy(::MPO)

    Make a deep copy of an MPS or MPO. By deep copy, it means that a new MPS/MPO is returned that doesn't share any data with the input MPS/MPO.

    Therefore, modifying the resulting MPS/MPO will note modify the original MPS/MPO.

    Use copy for an alternative that performs a shallow copy that avoids copying the ITensor data.

    Examples

    julia> using ITensors, ITensorMPS
     
     julia> s = siteinds("S=1/2", 3);
    @@ -68,19 +68,19 @@
     3.0
     
     julia> norm(M3)
    -3.0
    source

    Properties

    Base.eltypeMethod
    eltype(m::MPS)
    -eltype(m::MPO)

    The element type of the MPS/MPO. Always returns ITensor.

    For the element type of the ITensors of the MPS/MPO, use promote_itensor_eltype.

    source

    Properties

    Base.eltypeMethod
    eltype(m::MPS)
    +eltype(m::MPO)

    The element type of the MPS/MPO. Always returns ITensor.

    For the element type of the ITensors of the MPS/MPO, use promote_itensor_eltype.

    source
    ITensors.fluxMethod
    flux(M::MPS)
     
     flux(M::MPO)
     
     totalqn(M::MPS)
     
    -totalqn(M::MPO)

    For an MPS or MPO which conserves quantum numbers, compute the total QN flux. For a tensor network such as an MPS or MPO, the flux is the sum of fluxes of each of the tensors in the network. The name totalqn is an alias for flux.

    source
    ITensors.hasqnsMethod
    hasqns(M::MPS)
    +totalqn(M::MPO)

    For an MPS or MPO which conserves quantum numbers, compute the total QN flux. For a tensor network such as an MPS or MPO, the flux is the sum of fluxes of each of the tensors in the network. The name totalqn is an alias for flux.

    source
    ITensors.hasqnsMethod
    hasqns(M::MPS)
     
    -hasqns(M::MPO)

    Return true if the MPS or MPO has tensors which carry quantum numbers.

    source
    ITensors.ITensorMPS.maxlinkdimMethod
    maxlinkdim(M::MPS)
    -maxlinkdim(M::MPO)

    Get the maximum link dimension of the MPS or MPO.

    The minimum this will return is 1, even if there are no link indices.

    source

    Obtaining and finding indices

    ITensors.SiteTypes.siteindsMethod
    siteinds(commoninds, A::MPO, B::MPS, j::Integer; kwargs...)
    -siteinds(commonind, A::MPO, B::MPO, j::Integer; kwargs...)

    Get the site index (or indices) of the jth MPO tensor of A that is shared with MPS/MPO B.

    source
    ITensors.SiteTypes.siteindsMethod
    siteinds(uniqueinds, A::MPO, B::MPS, j::Integer; kwargs...)
    -siteinds(uniqueind, A::MPO, B::MPS, j::Integer; kwargs...)

    Get the site index (or indices) of MPO A that is unique to A (not shared with MPS/MPO B).

    source
    ITensors.ITensorMPS.findsiteFunction
    findsite(M::Union{MPS, MPO}, is)

    Return the first site of the MPS or MPO that has at least one Index in common with the Index or collection of indices is.

    To find all sites with common indices with is, use the findsites function.

    Examples

    s = siteinds("S=1/2", 5)
    +hasqns(M::MPO)

    Return true if the MPS or MPO has tensors which carry quantum numbers.

    source
    ITensors.ITensorMPS.maxlinkdimMethod
    maxlinkdim(M::MPS)
    +maxlinkdim(M::MPO)

    Get the maximum link dimension of the MPS or MPO.

    The minimum this will return is 1, even if there are no link indices.

    source

    Obtaining and finding indices

    ITensors.SiteTypes.siteindsMethod
    siteinds(commoninds, A::MPO, B::MPS, j::Integer; kwargs...)
    +siteinds(commonind, A::MPO, B::MPO, j::Integer; kwargs...)

    Get the site index (or indices) of the jth MPO tensor of A that is shared with MPS/MPO B.

    source
    ITensors.SiteTypes.siteindsMethod
    siteinds(uniqueinds, A::MPO, B::MPS, j::Integer; kwargs...)
    +siteinds(uniqueind, A::MPO, B::MPS, j::Integer; kwargs...)

    Get the site index (or indices) of MPO A that is unique to A (not shared with MPS/MPO B).

    source
    ITensors.ITensorMPS.findsiteFunction
    findsite(M::Union{MPS, MPO}, is)

    Return the first site of the MPS or MPO that has at least one Index in common with the Index or collection of indices is.

    To find all sites with common indices with is, use the findsites function.

    Examples

    s = siteinds("S=1/2", 5)
     ψ = random_mps(s)
     findsite(ψ, s[3]) == 3
     findsite(ψ, (s[3], s[4])) == 3
    @@ -89,7 +89,7 @@
     findsite(M, s[4]) == 4
     findsite(M, s[4]') == 4
     findsite(M, (s[4]', s[4])) == 4
    -findsite(M, (s[4]', s[3])) == 3
    source
    ITensors.ITensorMPS.findsitesFunction
    findsites(M::Union{MPS, MPO}, is)

    Return the sites of the MPS or MPO that have indices in common with the collection of site indices is.

    Examples

    s = siteinds("S=1/2", 5)
    +findsite(M, (s[4]', s[3])) == 3
    source
    ITensors.ITensorMPS.findsitesFunction
    findsites(M::Union{MPS, MPO}, is)

    Return the sites of the MPS or MPO that have indices in common with the collection of site indices is.

    Examples

    s = siteinds("S=1/2", 5)
     ψ = random_mps(s)
     findsites(ψ, s[3]) == [3]
     findsites(ψ, (s[4], s[1])) == [1, 4]
    @@ -98,38 +98,38 @@
     findsites(M, s[4]) == [4]
     findsites(M, s[4]') == [4]
     findsites(M, (s[4]', s[4])) == [4]
    -findsites(M, (s[4]', s[3])) == [3, 4]
    source
    ITensors.ITensorMPS.firstsiteindsFunction
    firstsiteinds(M::MPO; kwargs...)

    Get a Vector of the first site Index found on each site of M.

    By default, it finds the first site Index with prime level 0.

    source
    ITensors.ITensorMPS.linkindMethod
    linkind(M::MPS, j::Integer)
    -linkind(M::MPO, j::Integer)

    Get the link or bond Index connecting the MPS or MPO tensor on site j to site j+1.

    If there is no link Index, return nothing.

    source
    ITensors.SiteTypes.siteindMethod
    siteind(::typeof(first), M::Union{MPS,MPO}, j::Integer; kwargs...)

    Return the first site Index found on the MPS or MPO (the first Index unique to the jth MPS/MPO tensor).

    You can choose different filters, like prime level and tags, with the kwargs.

    source
    ITensors.SiteTypes.siteindsMethod
    siteinds(M::MPS)
    -siteinds(::typeof(first), M::MPS)

    Get a vector of the first site Index found on each tensor of the MPS.

    siteinds(::typeof(only), M::MPS)

    Get a vector of the only site Index found on each tensor of the MPS. Errors if more than one is found.

    siteinds(::typeof(all), M::MPS)

    Get a vector of the all site Indices found on each tensor of the MPS. Returns a Vector of IndexSets.

    source
    ITensors.SiteTypes.siteindMethod
    siteind(M::MPO, j::Int; plev = 0, kwargs...)

    Get the first site Index of the MPO found, by default with prime level 0.

    source
    ITensors.SiteTypes.siteindsMethod
    siteinds(M::Union{MPS, MPO}}, j::Integer; kwargs...)

    Return the site Indices found of the MPO or MPO at the site j as an IndexSet.

    Optionally filter prime tags and prime levels with keyword arguments like plev and tags.

    source

    Priming and tagging

    ITensors.primeMethod
    prime[!](M::MPS, args...; kwargs...)
    -prime[!](M::MPO, args...; kwargs...)

    Apply prime to all ITensors of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors. Alternatively apply the function in-place.

    source
    ITensors.primeMethod
    prime[!](siteinds, M::MPS, args...; kwargs...)
    -prime[!](siteinds, M::MPO, args...; kwargs...)

    Apply prime to all site indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.primeMethod
    prime[!](linkinds, M::MPS, args...; kwargs...)
    -prime[!](linkinds, M::MPO, args...; kwargs...)

    Apply prime to all link indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.primeMethod
    prime[!](siteinds, commoninds, M1::MPO, M2::MPS, args...; kwargs...)
    -prime[!](siteinds, commoninds, M1::MPO, M2::MPO, args...; kwargs...)

    Apply prime to the site indices that are shared by M1 and M2.

    Returns new MPSs/MPOs. The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.primeMethod
    prime[!](siteinds, uniqueinds, M1::MPO, M2::MPS, args...; kwargs...)

    Apply prime to the site indices of M1 that are not shared with M2. Returns new MPSs/MPOs.

    The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.swapprimeMethod
    swapprime[!](M::MPS, args...; kwargs...)
    -swapprime[!](M::MPO, args...; kwargs...)

    Apply swapprime to all ITensors of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors. Alternatively apply the function in-place.

    source
    ITensors.setprimeMethod
    setprime[!](M::MPS, args...; kwargs...)
    -setprime[!](M::MPO, args...; kwargs...)

    Apply setprime to all ITensors of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors. Alternatively apply the function in-place.

    source
    ITensors.setprimeMethod
    setprime[!](siteinds, M::MPS, args...; kwargs...)
    -setprime[!](siteinds, M::MPO, args...; kwargs...)

    Apply setprime to all site indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.setprimeMethod
    setprime[!](linkinds, M::MPS, args...; kwargs...)
    -setprime[!](linkinds, M::MPO, args...; kwargs...)

    Apply setprime to all link indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.setprimeMethod
    setprime[!](siteinds, commoninds, M1::MPO, M2::MPS, args...; kwargs...)
    -setprime[!](siteinds, commoninds, M1::MPO, M2::MPO, args...; kwargs...)

    Apply setprime to the site indices that are shared by M1 and M2.

    Returns new MPSs/MPOs. The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.setprimeMethod
    setprime[!](siteinds, uniqueinds, M1::MPO, M2::MPS, args...; kwargs...)

    Apply setprime to the site indices of M1 that are not shared with M2. Returns new MPSs/MPOs.

    The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.noprimeMethod
    noprime[!](M::MPS, args...; kwargs...)
    -noprime[!](M::MPO, args...; kwargs...)

    Apply noprime to all ITensors of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors. Alternatively apply the function in-place.

    source
    ITensors.noprimeMethod
    noprime[!](siteinds, M::MPS, args...; kwargs...)
    -noprime[!](siteinds, M::MPO, args...; kwargs...)

    Apply noprime to all site indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.noprimeMethod
    noprime[!](linkinds, M::MPS, args...; kwargs...)
    -noprime[!](linkinds, M::MPO, args...; kwargs...)

    Apply noprime to all link indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.noprimeMethod
    noprime[!](siteinds, commoninds, M1::MPO, M2::MPS, args...; kwargs...)
    -noprime[!](siteinds, commoninds, M1::MPO, M2::MPO, args...; kwargs...)

    Apply noprime to the site indices that are shared by M1 and M2.

    Returns new MPSs/MPOs. The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.noprimeMethod
    noprime[!](siteinds, uniqueinds, M1::MPO, M2::MPS, args...; kwargs...)

    Apply noprime to the site indices of M1 that are not shared with M2. Returns new MPSs/MPOs.

    The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.addtagsMethod
    addtags[!](M::MPS, args...; kwargs...)
    -addtags[!](M::MPO, args...; kwargs...)

    Apply addtags to all ITensors of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors. Alternatively apply the function in-place.

    source
    ITensors.TagSets.addtagsMethod
    addtags[!](siteinds, M::MPS, args...; kwargs...)
    -addtags[!](siteinds, M::MPO, args...; kwargs...)

    Apply addtags to all site indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.addtagsMethod
    addtags[!](linkinds, M::MPS, args...; kwargs...)
    -addtags[!](linkinds, M::MPO, args...; kwargs...)

    Apply addtags to all link indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.addtagsMethod
    addtags[!](siteinds, commoninds, M1::MPO, M2::MPS, args...; kwargs...)
    -addtags[!](siteinds, commoninds, M1::MPO, M2::MPO, args...; kwargs...)

    Apply addtags to the site indices that are shared by M1 and M2.

    Returns new MPSs/MPOs. The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.addtagsMethod
    addtags[!](siteinds, uniqueinds, M1::MPO, M2::MPS, args...; kwargs...)

    Apply addtags to the site indices of M1 that are not shared with M2. Returns new MPSs/MPOs.

    The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.removetagsMethod
    removetags[!](M::MPS, args...; kwargs...)
    -removetags[!](M::MPO, args...; kwargs...)

    Apply removetags to all ITensors of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors. Alternatively apply the function in-place.

    source
    ITensors.TagSets.removetagsMethod
    removetags[!](siteinds, M::MPS, args...; kwargs...)
    -removetags[!](siteinds, M::MPO, args...; kwargs...)

    Apply removetags to all site indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.removetagsMethod
    removetags[!](linkinds, M::MPS, args...; kwargs...)
    -removetags[!](linkinds, M::MPO, args...; kwargs...)

    Apply removetags to all link indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.removetagsMethod
    removetags[!](siteinds, commoninds, M1::MPO, M2::MPS, args...; kwargs...)
    -removetags[!](siteinds, commoninds, M1::MPO, M2::MPO, args...; kwargs...)

    Apply removetags to the site indices that are shared by M1 and M2.

    Returns new MPSs/MPOs. The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.removetagsMethod
    removetags[!](siteinds, uniqueinds, M1::MPO, M2::MPS, args...; kwargs...)

    Apply removetags to the site indices of M1 that are not shared with M2. Returns new MPSs/MPOs.

    The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.replacetagsMethod
    replacetags[!](M::MPS, args...; kwargs...)
    -replacetags[!](M::MPO, args...; kwargs...)

    Apply replacetags to all ITensors of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors. Alternatively apply the function in-place.

    source
    ITensors.TagSets.replacetagsMethod
    replacetags[!](siteinds, M::MPS, args...; kwargs...)
    -replacetags[!](siteinds, M::MPO, args...; kwargs...)

    Apply replacetags to all site indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.replacetagsMethod
    replacetags[!](linkinds, M::MPS, args...; kwargs...)
    -replacetags[!](linkinds, M::MPO, args...; kwargs...)

    Apply replacetags to all link indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.replacetagsMethod
    replacetags[!](siteinds, commoninds, M1::MPO, M2::MPS, args...; kwargs...)
    -replacetags[!](siteinds, commoninds, M1::MPO, M2::MPO, args...; kwargs...)

    Apply replacetags to the site indices that are shared by M1 and M2.

    Returns new MPSs/MPOs. The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.replacetagsMethod
    replacetags[!](siteinds, uniqueinds, M1::MPO, M2::MPS, args...; kwargs...)

    Apply replacetags to the site indices of M1 that are not shared with M2. Returns new MPSs/MPOs.

    The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.settagsMethod
    settags[!](M::MPS, args...; kwargs...)
    -settags[!](M::MPO, args...; kwargs...)

    Apply settags to all ITensors of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors. Alternatively apply the function in-place.

    source
    ITensors.settagsMethod
    settags[!](siteinds, M::MPS, args...; kwargs...)
    -settags[!](siteinds, M::MPO, args...; kwargs...)

    Apply settags to all site indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.settagsMethod
    settags[!](linkinds, M::MPS, args...; kwargs...)
    -settags[!](linkinds, M::MPO, args...; kwargs...)

    Apply settags to all link indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.settagsMethod
    settags[!](siteinds, commoninds, M1::MPO, M2::MPS, args...; kwargs...)
    -settags[!](siteinds, commoninds, M1::MPO, M2::MPO, args...; kwargs...)

    Apply settags to the site indices that are shared by M1 and M2.

    Returns new MPSs/MPOs. The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.settagsMethod
    settags[!](siteinds, uniqueinds, M1::MPO, M2::MPS, args...; kwargs...)

    Apply settags to the site indices of M1 that are not shared with M2. Returns new MPSs/MPOs.

    The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source

    Operations

    ITensors.ITensorMPS.firstsiteindsFunction
    firstsiteinds(M::MPO; kwargs...)

    Get a Vector of the first site Index found on each site of M.

    By default, it finds the first site Index with prime level 0.

    source
    ITensors.ITensorMPS.linkindMethod
    linkind(M::MPS, j::Integer)
    +linkind(M::MPO, j::Integer)

    Get the link or bond Index connecting the MPS or MPO tensor on site j to site j+1.

    If there is no link Index, return nothing.

    source
    ITensors.SiteTypes.siteindMethod
    siteind(::typeof(first), M::Union{MPS,MPO}, j::Integer; kwargs...)

    Return the first site Index found on the MPS or MPO (the first Index unique to the jth MPS/MPO tensor).

    You can choose different filters, like prime level and tags, with the kwargs.

    source
    ITensors.SiteTypes.siteindsMethod
    siteinds(M::MPS)
    +siteinds(::typeof(first), M::MPS)

    Get a vector of the first site Index found on each tensor of the MPS.

    siteinds(::typeof(only), M::MPS)

    Get a vector of the only site Index found on each tensor of the MPS. Errors if more than one is found.

    siteinds(::typeof(all), M::MPS)

    Get a vector of the all site Indices found on each tensor of the MPS. Returns a Vector of IndexSets.

    source
    ITensors.SiteTypes.siteindMethod
    siteind(M::MPO, j::Int; plev = 0, kwargs...)

    Get the first site Index of the MPO found, by default with prime level 0.

    source
    ITensors.SiteTypes.siteindsMethod
    siteinds(M::Union{MPS, MPO}}, j::Integer; kwargs...)

    Return the site Indices found of the MPO or MPO at the site j as an IndexSet.

    Optionally filter prime tags and prime levels with keyword arguments like plev and tags.

    source

    Priming and tagging

    ITensors.primeMethod
    prime[!](M::MPS, args...; kwargs...)
    +prime[!](M::MPO, args...; kwargs...)

    Apply prime to all ITensors of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors. Alternatively apply the function in-place.

    source
    ITensors.primeMethod
    prime[!](siteinds, M::MPS, args...; kwargs...)
    +prime[!](siteinds, M::MPO, args...; kwargs...)

    Apply prime to all site indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.primeMethod
    prime[!](linkinds, M::MPS, args...; kwargs...)
    +prime[!](linkinds, M::MPO, args...; kwargs...)

    Apply prime to all link indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.primeMethod
    prime[!](siteinds, commoninds, M1::MPO, M2::MPS, args...; kwargs...)
    +prime[!](siteinds, commoninds, M1::MPO, M2::MPO, args...; kwargs...)

    Apply prime to the site indices that are shared by M1 and M2.

    Returns new MPSs/MPOs. The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.primeMethod
    prime[!](siteinds, uniqueinds, M1::MPO, M2::MPS, args...; kwargs...)

    Apply prime to the site indices of M1 that are not shared with M2. Returns new MPSs/MPOs.

    The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.swapprimeMethod
    swapprime[!](M::MPS, args...; kwargs...)
    +swapprime[!](M::MPO, args...; kwargs...)

    Apply swapprime to all ITensors of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors. Alternatively apply the function in-place.

    source
    ITensors.setprimeMethod
    setprime[!](M::MPS, args...; kwargs...)
    +setprime[!](M::MPO, args...; kwargs...)

    Apply setprime to all ITensors of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors. Alternatively apply the function in-place.

    source
    ITensors.setprimeMethod
    setprime[!](siteinds, M::MPS, args...; kwargs...)
    +setprime[!](siteinds, M::MPO, args...; kwargs...)

    Apply setprime to all site indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.setprimeMethod
    setprime[!](linkinds, M::MPS, args...; kwargs...)
    +setprime[!](linkinds, M::MPO, args...; kwargs...)

    Apply setprime to all link indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.setprimeMethod
    setprime[!](siteinds, commoninds, M1::MPO, M2::MPS, args...; kwargs...)
    +setprime[!](siteinds, commoninds, M1::MPO, M2::MPO, args...; kwargs...)

    Apply setprime to the site indices that are shared by M1 and M2.

    Returns new MPSs/MPOs. The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.setprimeMethod
    setprime[!](siteinds, uniqueinds, M1::MPO, M2::MPS, args...; kwargs...)

    Apply setprime to the site indices of M1 that are not shared with M2. Returns new MPSs/MPOs.

    The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.noprimeMethod
    noprime[!](M::MPS, args...; kwargs...)
    +noprime[!](M::MPO, args...; kwargs...)

    Apply noprime to all ITensors of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors. Alternatively apply the function in-place.

    source
    ITensors.noprimeMethod
    noprime[!](siteinds, M::MPS, args...; kwargs...)
    +noprime[!](siteinds, M::MPO, args...; kwargs...)

    Apply noprime to all site indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.noprimeMethod
    noprime[!](linkinds, M::MPS, args...; kwargs...)
    +noprime[!](linkinds, M::MPO, args...; kwargs...)

    Apply noprime to all link indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.noprimeMethod
    noprime[!](siteinds, commoninds, M1::MPO, M2::MPS, args...; kwargs...)
    +noprime[!](siteinds, commoninds, M1::MPO, M2::MPO, args...; kwargs...)

    Apply noprime to the site indices that are shared by M1 and M2.

    Returns new MPSs/MPOs. The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.noprimeMethod
    noprime[!](siteinds, uniqueinds, M1::MPO, M2::MPS, args...; kwargs...)

    Apply noprime to the site indices of M1 that are not shared with M2. Returns new MPSs/MPOs.

    The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.addtagsMethod
    addtags[!](M::MPS, args...; kwargs...)
    +addtags[!](M::MPO, args...; kwargs...)

    Apply addtags to all ITensors of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors. Alternatively apply the function in-place.

    source
    ITensors.TagSets.addtagsMethod
    addtags[!](siteinds, M::MPS, args...; kwargs...)
    +addtags[!](siteinds, M::MPO, args...; kwargs...)

    Apply addtags to all site indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.addtagsMethod
    addtags[!](linkinds, M::MPS, args...; kwargs...)
    +addtags[!](linkinds, M::MPO, args...; kwargs...)

    Apply addtags to all link indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.addtagsMethod
    addtags[!](siteinds, commoninds, M1::MPO, M2::MPS, args...; kwargs...)
    +addtags[!](siteinds, commoninds, M1::MPO, M2::MPO, args...; kwargs...)

    Apply addtags to the site indices that are shared by M1 and M2.

    Returns new MPSs/MPOs. The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.addtagsMethod
    addtags[!](siteinds, uniqueinds, M1::MPO, M2::MPS, args...; kwargs...)

    Apply addtags to the site indices of M1 that are not shared with M2. Returns new MPSs/MPOs.

    The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.removetagsMethod
    removetags[!](M::MPS, args...; kwargs...)
    +removetags[!](M::MPO, args...; kwargs...)

    Apply removetags to all ITensors of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors. Alternatively apply the function in-place.

    source
    ITensors.TagSets.removetagsMethod
    removetags[!](siteinds, M::MPS, args...; kwargs...)
    +removetags[!](siteinds, M::MPO, args...; kwargs...)

    Apply removetags to all site indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.removetagsMethod
    removetags[!](linkinds, M::MPS, args...; kwargs...)
    +removetags[!](linkinds, M::MPO, args...; kwargs...)

    Apply removetags to all link indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.removetagsMethod
    removetags[!](siteinds, commoninds, M1::MPO, M2::MPS, args...; kwargs...)
    +removetags[!](siteinds, commoninds, M1::MPO, M2::MPO, args...; kwargs...)

    Apply removetags to the site indices that are shared by M1 and M2.

    Returns new MPSs/MPOs. The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.removetagsMethod
    removetags[!](siteinds, uniqueinds, M1::MPO, M2::MPS, args...; kwargs...)

    Apply removetags to the site indices of M1 that are not shared with M2. Returns new MPSs/MPOs.

    The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.replacetagsMethod
    replacetags[!](M::MPS, args...; kwargs...)
    +replacetags[!](M::MPO, args...; kwargs...)

    Apply replacetags to all ITensors of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors. Alternatively apply the function in-place.

    source
    ITensors.TagSets.replacetagsMethod
    replacetags[!](siteinds, M::MPS, args...; kwargs...)
    +replacetags[!](siteinds, M::MPO, args...; kwargs...)

    Apply replacetags to all site indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.replacetagsMethod
    replacetags[!](linkinds, M::MPS, args...; kwargs...)
    +replacetags[!](linkinds, M::MPO, args...; kwargs...)

    Apply replacetags to all link indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.replacetagsMethod
    replacetags[!](siteinds, commoninds, M1::MPO, M2::MPS, args...; kwargs...)
    +replacetags[!](siteinds, commoninds, M1::MPO, M2::MPO, args...; kwargs...)

    Apply replacetags to the site indices that are shared by M1 and M2.

    Returns new MPSs/MPOs. The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.TagSets.replacetagsMethod
    replacetags[!](siteinds, uniqueinds, M1::MPO, M2::MPS, args...; kwargs...)

    Apply replacetags to the site indices of M1 that are not shared with M2. Returns new MPSs/MPOs.

    The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.settagsMethod
    settags[!](M::MPS, args...; kwargs...)
    +settags[!](M::MPO, args...; kwargs...)

    Apply settags to all ITensors of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors. Alternatively apply the function in-place.

    source
    ITensors.settagsMethod
    settags[!](siteinds, M::MPS, args...; kwargs...)
    +settags[!](siteinds, M::MPO, args...; kwargs...)

    Apply settags to all site indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.settagsMethod
    settags[!](linkinds, M::MPS, args...; kwargs...)
    +settags[!](linkinds, M::MPO, args...; kwargs...)

    Apply settags to all link indices of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors.

    source
    ITensors.settagsMethod
    settags[!](siteinds, commoninds, M1::MPO, M2::MPS, args...; kwargs...)
    +settags[!](siteinds, commoninds, M1::MPO, M2::MPO, args...; kwargs...)

    Apply settags to the site indices that are shared by M1 and M2.

    Returns new MPSs/MPOs. The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source
    ITensors.settagsMethod
    settags[!](siteinds, uniqueinds, M1::MPO, M2::MPS, args...; kwargs...)

    Apply settags to the site indices of M1 that are not shared with M2. Returns new MPSs/MPOs.

    The ITensors of the MPSs/MPOs will be a view of the storage of the original ITensors.

    source

    Operations

    ITensors.ITensorMPS.expectMethod
    expect(psi::MPS, op::AbstractString...; kwargs...)
     expect(psi::MPS, op::Matrix{<:Number}...; kwargs...)
     expect(psi::MPS, ops; kwargs...)

    Given an MPS psi and a single operator name, returns a vector of the expected value of the operator on each site of the MPS.

    If multiple operator names are provided, returns a tuple of expectation value vectors.

    If a container of operator names is provided, returns the same type of container with names replaced by vectors of expectation values.

    Optional Keyword Arguments

    • sites = 1:length(psi): compute expected values only for sites in the given range

    Examples

    N = 10
     
    @@ -144,7 +144,7 @@
     s = siteinds("Electron", N)
     psi = random_mps(s; linkdims=8)
     dens = expect(psi, "Ntot")
    -updens, dndens = expect(psi, "Nup", "Ndn") # pass more than one operator
    source
    ITensors.ITensorMPS.correlation_matrixMethod
    correlation_matrix(psi::MPS,
                        Op1::AbstractString,
                        Op2::AbstractString;
                        kwargs...)
    @@ -162,14 +162,14 @@
     
     s = siteinds("Electron", N; conserve_qns=true)
     psi = random_mps(s, n -> isodd(n) ? "Up" : "Dn"; linkdims=m)
    -Cuu = correlation_matrix(psi, "Cdagup", "Cup"; sites=2:8)
    source
    ITensors.dagMethod
    dag[!](M::MPS, args...; kwargs...)
    -dag[!](M::MPO, args...; kwargs...)

    Apply dag to all ITensors of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors. Alternatively apply the function in-place.

    source
    NDTensors.denseMethod
    dense(::MPS/MPO)

    Given an MPS (or MPO), return a new MPS (or MPO) having called dense on each ITensor to convert each tensor to use dense storage and remove any QN or other sparse structure information, if it is not dense already.

    source
    ITensors.ITensorMPS.movesiteMethod
    movesite(::Union{MPS, MPO}, n1n2::Pair{Int, Int})

    Create a new MPS/MPO where the site at n1 is moved to n2, for a pair n1n2 = n1 => n2.

    This is done with a series a pairwise swaps, and can introduce a lot of entanglement into your state, so use with caution.

    source
    ITensors.dagMethod
    dag[!](M::MPS, args...; kwargs...)
    +dag[!](M::MPO, args...; kwargs...)

    Apply dag to all ITensors of an MPS/MPO, returning a new MPS/MPO.

    The ITensors of the MPS/MPO will be a view of the storage of the original ITensors. Alternatively apply the function in-place.

    source
    NDTensors.denseMethod
    dense(::MPS/MPO)

    Given an MPS (or MPO), return a new MPS (or MPO) having called dense on each ITensor to convert each tensor to use dense storage and remove any QN or other sparse structure information, if it is not dense already.

    source
    ITensors.ITensorMPS.movesiteMethod
    movesite(::Union{MPS, MPO}, n1n2::Pair{Int, Int})

    Create a new MPS/MPO where the site at n1 is moved to n2, for a pair n1n2 = n1 => n2.

    This is done with a series a pairwise swaps, and can introduce a lot of entanglement into your state, so use with caution.

    source
    ITensors.ITensorMPS.orthogonalize!Function
    orthogonalize!(M::MPS, j::Int; kwargs...)
     orthogonalize(M::MPS, j::Int; kwargs...)
     
     orthogonalize!(M::MPO, j::Int; kwargs...)
    -orthogonalize(M::MPO, j::Int; kwargs...)

    Move the orthogonality center of the MPS to site j. No observable property of the MPS will be changed, and no truncation of the bond indices is performed. Afterward, tensors 1,2,...,j-1 will be left-orthogonal and tensors j+1,j+2,...,N will be right-orthogonal.

    Either modify in-place with orthogonalize! or out-of-place with orthogonalize.

    source
    ITensors.ITensorMPS.replacebond!Method
    replacebond!(M::MPS, b::Int, phi::ITensor; kwargs...)

    Factorize the ITensor phi and replace the ITensors b and b+1 of MPS M with the factors. Choose the orthogonality with ortho="left"/"right".

    source
    ITensors.ITensorMPS.sampleMethod
    sample(m::MPS)

    Given a normalized MPS m with orthocenter(m)==1, returns a Vector{Int} of length(m) corresponding to one sample of the probability distribution defined by squaring the components of the tensor that the MPS represents

    source
    ITensors.ITensorMPS.sample!Method
    sample!(m::MPS)

    Given a normalized MPS m, returns a Vector{Int} of length(m) corresponding to one sample of the probability distribution defined by squaring the components of the tensor that the MPS represents. If the MPS does not have an orthogonality center, orthogonalize!(m,1) will be called before computing the sample.

    source
    ITensors.ITensorMPS.sampleMethod
    sample(M::MPO)

    Given a normalized MPO M, returns a Vector{Int} of length(M) corresponding to one sample of the probability distribution defined by the MPO, treating the MPO as a density matrix.

    The MPO M should have an (approximately) positive spectrum.

    source
    NDTensors.truncate!Function
    truncate!(M::MPS; kwargs...)
    -truncate!(M::MPO; kwargs...)

    Perform a truncation of all bonds of an MPS/MPO, using the truncation parameters (cutoff,maxdim, etc.) provided as keyword arguments.

    Keyword arguments:

    • site_range=1:N - only truncate the MPS bonds between these sites
    source

    Gate evolution

    ITensors.productMethod
    apply(o::ITensor, ψ::Union{MPS, MPO}, [ns::Vector{Int}]; kwargs...)
    -product([...])

    Get the product of the operator o with the MPS/MPO ψ, where the operator is applied to the sites ns. If ns are not specified, the sites are determined by the common indices between o and the site indices of ψ.

    If ns are non-contiguous, the sites of the MPS are moved to be contiguous. By default, the sites are moved back to their original locations. You can leave them where they are by setting the keyword argument move_sites_back to false.

    Keywords

    • cutoff::Real: singular value truncation cutoff.
    • maxdim::Int: maximum MPS/MPO dimension.
    • apply_dag::Bool = false: apply the gate and the dagger of the gate (only relevant for MPO evolution).
    • move_sites_back::Bool = true: after the ITensors are applied to the MPS or MPO, move the sites of the MPS or MPO back to their original locations.
    source
    ITensors.productMethod
    apply(As::Vector{<:ITensor}, M::Union{MPS, MPO}; kwargs...)
    +orthogonalize(M::MPO, j::Int; kwargs...)

    Move the orthogonality center of the MPS to site j. No observable property of the MPS will be changed, and no truncation of the bond indices is performed. Afterward, tensors 1,2,...,j-1 will be left-orthogonal and tensors j+1,j+2,...,N will be right-orthogonal.

    Either modify in-place with orthogonalize! or out-of-place with orthogonalize.

    source
    ITensors.ITensorMPS.replacebond!Method
    replacebond!(M::MPS, b::Int, phi::ITensor; kwargs...)

    Factorize the ITensor phi and replace the ITensors b and b+1 of MPS M with the factors. Choose the orthogonality with ortho="left"/"right".

    source
    ITensors.ITensorMPS.sampleMethod
    sample(m::MPS)

    Given a normalized MPS m with orthocenter(m)==1, returns a Vector{Int} of length(m) corresponding to one sample of the probability distribution defined by squaring the components of the tensor that the MPS represents

    source
    ITensors.ITensorMPS.sample!Method
    sample!(m::MPS)

    Given a normalized MPS m, returns a Vector{Int} of length(m) corresponding to one sample of the probability distribution defined by squaring the components of the tensor that the MPS represents. If the MPS does not have an orthogonality center, orthogonalize!(m,1) will be called before computing the sample.

    source
    ITensors.ITensorMPS.sampleMethod
    sample(M::MPO)

    Given a normalized MPO M, returns a Vector{Int} of length(M) corresponding to one sample of the probability distribution defined by the MPO, treating the MPO as a density matrix.

    The MPO M should have an (approximately) positive spectrum.

    source
    NDTensors.truncate!Function
    truncate!(M::MPS; kwargs...)
    +truncate!(M::MPO; kwargs...)

    Perform a truncation of all bonds of an MPS/MPO, using the truncation parameters (cutoff,maxdim, etc.) provided as keyword arguments.

    Keyword arguments:

    • site_range=1:N - only truncate the MPS bonds between these sites
    source

    Gate evolution

    ITensors.productMethod
    apply(o::ITensor, ψ::Union{MPS, MPO}, [ns::Vector{Int}]; kwargs...)
    +product([...])

    Get the product of the operator o with the MPS/MPO ψ, where the operator is applied to the sites ns. If ns are not specified, the sites are determined by the common indices between o and the site indices of ψ.

    If ns are non-contiguous, the sites of the MPS are moved to be contiguous. By default, the sites are moved back to their original locations. You can leave them where they are by setting the keyword argument move_sites_back to false.

    Keywords

    • cutoff::Real: singular value truncation cutoff.
    • maxdim::Int: maximum MPS/MPO dimension.
    • apply_dag::Bool = false: apply the gate and the dagger of the gate (only relevant for MPO evolution).
    • move_sites_back::Bool = true: after the ITensors are applied to the MPS or MPO, move the sites of the MPS or MPO back to their original locations.
    source
    ITensors.productMethod
    apply(As::Vector{<:ITensor}, M::Union{MPS, MPO}; kwargs...)
     product([...])

    Apply the ITensors As to the MPS or MPO M, treating them as gates or matrices from pairs of prime or unprimed indices.

    Keywords

    • cutoff::Real: singular value truncation cutoff.
    • maxdim::Int: maximum MPS/MPO dimension.
    • apply_dag::Bool = false: apply the gate and the dagger of the gate (only relevant for MPO evolution).
    • move_sites_back::Bool = true: after the ITensor is applied to the MPS or MPO, move the sites of the MPS or MPO back to their original locations.

    Examples

    Apply one-site gates to an MPS:

    N = 3
     
     ITensors.op(::OpName"σx", ::SiteType"S=1/2", s::Index) =
    @@ -232,19 +232,19 @@
           ("expS⋅S", (2, 3), (τ = τ,))]
     ψ0 = MPS(s, n -> n == 1 ? "↓" : "↑")
     expτH = ops(os, s)
    -ψτ = apply(expτH, ψ0)
    source

    Algebra Operations

    ITensors.innerMethod
    inner(A::MPS, B::MPS)
    -inner(A::MPO, B::MPO)

    Compute the inner product ⟨A|B⟩. If A and B are MPOs, computes the Frobenius inner product.

    Use loginner to avoid underflow/overflow for taking overlaps of large MPS or MPO.

    ITensors 0.3

    Before ITensors 0.3, inner had a keyword argument make_inds_match that default to true. When true, the function attempted to make the site indices match before contracting. So for example, the inputs could have different site indices, as long as they have the same dimensions or QN blocks. This behavior was fragile since it only worked for MPS with single site indices per tensor, and as of ITensors 0.3 has been deprecated. As of ITensors 0.3 you will need to make sure the MPS or MPO you input have compatible site indices to contract over, such as by making sure the prime levels match properly.

    Same as dot.

    See also loginner, logdot.

    source
    ITensors.ITensorMPS.loginnerMethod
    loginner(A::MPS, B::MPS)
    -loginner(A::MPO, B::MPO)

    Compute the logarithm of the inner product ⟨A|B⟩. If A and B are MPOs, computes the logarithm of the Frobenius inner product.

    This is useful for larger MPS/MPO, where in the limit of large numbers of sites the inner product can diverge or approach zero.

    ITensors 0.3

    Before ITensors 0.3, inner had a keyword argument make_inds_match that default to true. When true, the function attempted to make the site indices match before contracting. So for example, the inputs could have different site indices, as long as they have the same dimensions or QN blocks. This behavior was fragile since it only worked for MPS with single site indices per tensor, and as of ITensors 0.3 has been deprecated. As of ITensors 0.3 you will need to make sure the MPS or MPO you input have compatible site indices to contract over, such as by making sure the prime levels match properly.

    Same as logdot.

    See also inner, dot.

    source
    ITensors.innerMethod
    inner(y::MPS, A::MPO, x::MPS)

    Compute ⟨y|A|x⟩ = ⟨y|Ax⟩ efficiently and exactly without making any intermediate MPOs. In general it is more efficient and accurate than inner(y, apply(A, x)).

    This is helpful for computing the expectation value of an operator A, which would be:

    inner(x', A, x)

    assuming x is normalized.

    If you want to compute ⟨By|Ax⟩ you can use inner(B::MPO, y::MPS, A::MPO, x::MPS).

    This is helpful for computing the variance of an operator A, which would be:

    inner(A, x, A, x) - inner(x', A, x) ^ 2

    assuming x is normalized.

    ITensors 0.3

    Before ITensors 0.3, inner had a keyword argument make_inds_match that default to true. When true, the function attempted to make the site indices match before contracting. So for example, the inputs could have different site indices, as long as they have the same dimensions or QN blocks. This behavior was fragile since it only worked for MPS with single site indices per tensor, and as of ITensors 0.3 has been deprecated. As of ITensors 0.3 you will need to make sure the MPS or MPO you input have compatible site indices to contract over, such as by making sure the prime levels match properly.

    Same as dot.

    source
    ITensors.innerMethod
    inner(B::MPO, y::MPS, A::MPO, x::MPS)

    Compute ⟨By|A|x⟩ = ⟨By|Ax⟩ efficiently and exactly without making any intermediate MPOs. In general it is more efficient and accurate than inner(apply(B, y), apply(A, x)).

    This is helpful for computing the variance of an operator A, which would be:

    inner(A, x, A, x) - inner(x, A, x) ^ 2
    ITensors 0.3

    Before ITensors 0.3, inner had a keyword argument make_inds_match that default to true. When true, the function attempted to make the site indices match before contracting. So for example, the inputs could have different site indices, as long as they have the same dimensions or QN blocks. This behavior was fragile since it only worked for MPS with single site indices per tensor, and as of ITensors 0.3 has been deprecated. As of ITensors 0.3 you will need to make sure the MPS or MPO you input have compatible site indices to contract over, such as by making sure the prime levels match properly.

    Same as dot.

    source
    LinearAlgebra.normMethod
    norm(A::MPS)
    -norm(A::MPO)

    Compute the norm of the MPS or MPO.

    If the MPS or MPO has a well defined orthogonality center, this reduces to the norm of the orthogonality center tensor. Otherwise, it computes the norm with the full inner product of the MPS/MPO with itself.

    See also lognorm.

    source
    LinearAlgebra.normalizeMethod
    normalize(A::MPS; (lognorm!)=[])
    -normalize(A::MPO; (lognorm!)=[])

    Return a new MPS or MPO A that is the same as the original MPS or MPO but with norm(A) ≈ 1.

    In practice, this evenly spreads lognorm(A) over the tensors within the range of the orthogonality center to avoid numerical overflow in the case of diverging norms.

    See also normalize!, norm, lognorm.

    source

    Algebra Operations

    ITensors.innerMethod
    inner(A::MPS, B::MPS)
    +inner(A::MPO, B::MPO)

    Compute the inner product ⟨A|B⟩. If A and B are MPOs, computes the Frobenius inner product.

    Use loginner to avoid underflow/overflow for taking overlaps of large MPS or MPO.

    ITensors 0.3

    Before ITensors 0.3, inner had a keyword argument make_inds_match that default to true. When true, the function attempted to make the site indices match before contracting. So for example, the inputs could have different site indices, as long as they have the same dimensions or QN blocks. This behavior was fragile since it only worked for MPS with single site indices per tensor, and as of ITensors 0.3 has been deprecated. As of ITensors 0.3 you will need to make sure the MPS or MPO you input have compatible site indices to contract over, such as by making sure the prime levels match properly.

    Same as dot.

    See also loginner, logdot.

    source
    ITensors.ITensorMPS.loginnerMethod
    loginner(A::MPS, B::MPS)
    +loginner(A::MPO, B::MPO)

    Compute the logarithm of the inner product ⟨A|B⟩. If A and B are MPOs, computes the logarithm of the Frobenius inner product.

    This is useful for larger MPS/MPO, where in the limit of large numbers of sites the inner product can diverge or approach zero.

    ITensors 0.3

    Before ITensors 0.3, inner had a keyword argument make_inds_match that default to true. When true, the function attempted to make the site indices match before contracting. So for example, the inputs could have different site indices, as long as they have the same dimensions or QN blocks. This behavior was fragile since it only worked for MPS with single site indices per tensor, and as of ITensors 0.3 has been deprecated. As of ITensors 0.3 you will need to make sure the MPS or MPO you input have compatible site indices to contract over, such as by making sure the prime levels match properly.

    Same as logdot.

    See also inner, dot.

    source
    ITensors.innerMethod
    inner(y::MPS, A::MPO, x::MPS)

    Compute ⟨y|A|x⟩ = ⟨y|Ax⟩ efficiently and exactly without making any intermediate MPOs. In general it is more efficient and accurate than inner(y, apply(A, x)).

    This is helpful for computing the expectation value of an operator A, which would be:

    inner(x', A, x)

    assuming x is normalized.

    If you want to compute ⟨By|Ax⟩ you can use inner(B::MPO, y::MPS, A::MPO, x::MPS).

    This is helpful for computing the variance of an operator A, which would be:

    inner(A, x, A, x) - inner(x', A, x) ^ 2

    assuming x is normalized.

    ITensors 0.3

    Before ITensors 0.3, inner had a keyword argument make_inds_match that default to true. When true, the function attempted to make the site indices match before contracting. So for example, the inputs could have different site indices, as long as they have the same dimensions or QN blocks. This behavior was fragile since it only worked for MPS with single site indices per tensor, and as of ITensors 0.3 has been deprecated. As of ITensors 0.3 you will need to make sure the MPS or MPO you input have compatible site indices to contract over, such as by making sure the prime levels match properly.

    Same as dot.

    source
    ITensors.innerMethod
    inner(B::MPO, y::MPS, A::MPO, x::MPS)

    Compute ⟨By|A|x⟩ = ⟨By|Ax⟩ efficiently and exactly without making any intermediate MPOs. In general it is more efficient and accurate than inner(apply(B, y), apply(A, x)).

    This is helpful for computing the variance of an operator A, which would be:

    inner(A, x, A, x) - inner(x, A, x) ^ 2
    ITensors 0.3

    Before ITensors 0.3, inner had a keyword argument make_inds_match that default to true. When true, the function attempted to make the site indices match before contracting. So for example, the inputs could have different site indices, as long as they have the same dimensions or QN blocks. This behavior was fragile since it only worked for MPS with single site indices per tensor, and as of ITensors 0.3 has been deprecated. As of ITensors 0.3 you will need to make sure the MPS or MPO you input have compatible site indices to contract over, such as by making sure the prime levels match properly.

    Same as dot.

    source
    LinearAlgebra.normMethod
    norm(A::MPS)
    +norm(A::MPO)

    Compute the norm of the MPS or MPO.

    If the MPS or MPO has a well defined orthogonality center, this reduces to the norm of the orthogonality center tensor. Otherwise, it computes the norm with the full inner product of the MPS/MPO with itself.

    See also lognorm.

    source
    LinearAlgebra.normalizeMethod
    normalize(A::MPS; (lognorm!)=[])
    +normalize(A::MPO; (lognorm!)=[])

    Return a new MPS or MPO A that is the same as the original MPS or MPO but with norm(A) ≈ 1.

    In practice, this evenly spreads lognorm(A) over the tensors within the range of the orthogonality center to avoid numerical overflow in the case of diverging norms.

    See also normalize!, norm, lognorm.

    source
    LinearAlgebra.normalize!Method
    normalize!(A::MPS; (lognorm!)=[])
     normalize!(A::MPO; (lognorm!)=[])

    Change the MPS or MPO A in-place such that norm(A) ≈ 1. This modifies the data of the tensors within the orthogonality center.

    In practice, this evenly spreads lognorm(A) over the tensors within the range of the orthogonality center to avoid numerical overflow in the case of diverging norms.

    If the norm of the input MPS or MPO is 0, normalizing is ill-defined. In this case, we just return the original MPS or MPO. You can check for this case as follows:

    s = siteinds("S=1/2", 4)
     ψ = 0 * random_mps(s)
     lognorm_ψ = []
     normalize!(ψ; (lognorm!)=lognorm_ψ)
    -lognorm_ψ[1] == -Inf # There was an infinite norm

    See also normalize, norm, lognorm.

    source
    ITensors.ITensorMPS.lognormMethod
    lognorm(A::MPS)
    -lognorm(A::MPO)

    Compute the logarithm of the norm of the MPS or MPO.

    This is useful for larger MPS/MPO that are not gauged, where in the limit of large numbers of sites the norm can diverge or approach zero.

    See also norm, logdot.

    source
    ITensors.ITensorMPS.lognormMethod
    lognorm(A::MPS)
    +lognorm(A::MPO)

    Compute the logarithm of the norm of the MPS or MPO.

    This is useful for larger MPS/MPO that are not gauged, where in the limit of large numbers of sites the norm can diverge or approach zero.

    See also norm, logdot.

    source
    Base.:+Method
    +(A::MPS/MPO...; kwargs...)
     add(A::MPS/MPO...; kwargs...)

    Add arbitrary numbers of MPS/MPO with each other, optionally truncating the results.

    A cutoff of 1e-15 is used by default, and in general users should set their own cutoff for their particular application.

    Keywords

    • cutoff::Real: singular value truncation cutoff
    • maxdim::Int: maximum MPS/MPO bond dimension
    • alg = "densitymatrix": "densitymatrix" or "directsum". "densitymatrix" adds the MPS/MPO by adding up and diagoanlizing local density matrices site by site in a single sweep through the system, truncating the density matrix with cutoff and maxdim. "directsum" performs a direct sum of each tensors on each site of the input MPS/MPO being summed. It doesn't perform any truncation, and therefore ignores cutoff and maxdim. The bond dimension of the output is the sum of the bond dimensions of the inputs. You can truncate the resulting MPS/MPO with the truncate! function.

    Examples

    N = 10
     
     s = siteinds("S=1/2", N; conserve_qns = true)
    @@ -280,18 +280,18 @@
     @show inner(ψ, ψ)
     @show inner(ψ₁, ψ₁) + 2 * inner(ψ₁, ψ₂) + inner(ψ₁, ψ₃) +
           2 * inner(ψ₂, ψ₁) + 4 * inner(ψ₂, ψ₂) + 2 * inner(ψ₂, ψ₃) +
    -      inner(ψ₃, ψ₁) + 2 * inner(ψ₃, ψ₂) + inner(ψ₃, ψ₃)
    source
    NDTensors.contractMethod
    contract(ψ::MPS, A::MPO; kwargs...) -> MPS
    +      inner(ψ₃, ψ₁) + 2 * inner(ψ₃, ψ₂) + inner(ψ₃, ψ₃)
    source
    NDTensors.contractMethod
    contract(ψ::MPS, A::MPO; kwargs...) -> MPS
     *(::MPS, ::MPO; kwargs...) -> MPS
     
     contract(A::MPO, ψ::MPS; kwargs...) -> MPS
    -*(::MPO, ::MPS; kwargs...) -> MPS

    Contract the MPO A with the MPS ψ, returning an MPS with the unique site indices of the MPO.

    For example, for an MPO with site indices with prime levels of 1 and 0, such as -s'-A-s-, and an MPS with site indices with prime levels of 0, such as -s-x, the result is an MPS y with site indices with prime levels of 1, -s'-y = -s'-A-s-x.

    Since it is common to contract an MPO with prime levels of 1 and 0 with an MPS with prime level of 0 and want a resulting MPS with prime levels of 0, we provide a convenience function apply:

    apply(A, x; kwargs...) = replaceprime(contract(A, x; kwargs...), 2 => 1)`.

    Choose the method with the method keyword, for example "densitymatrix" and "naive".

    Keywords

    • cutoff::Float64=1e-13: the cutoff value for truncating the density matrix eigenvalues. Note that the default is somewhat arbitrary and subject to change, in general you should set a cutoff value.
    • maxdim::Int=maxlinkdim(A) * maxlinkdim(ψ)): the maximal bond dimension of the results MPS.
    • mindim::Int=1: the minimal bond dimension of the resulting MPS.
    • normalize::Bool=false: whether or not to normalize the resulting MPS.
    • method::String="densitymatrix": the algorithm to use for the contraction. Currently the options are "densitymatrix", where the network formed by the MPO and MPS is squared and contracted down to a density matrix which is diagonalized iteratively at each site, and "naive", where the MPO and MPS tensor are contracted exactly at each site and then a truncation of the resulting MPS is performed.

    See also apply.

    source
    ITensors.applyMethod
    apply(A::MPO, x::MPS; kwargs...)

    Contract the MPO A with the MPS x and then map the prime level of the resulting MPS back to 0.

    Equivalent to replaceprime(contract(A, x; kwargs...), 2 => 1).

    See also contract for details about the arguments available.

    source
    NDTensors.contractMethod
    contract(A::MPO, B::MPO; kwargs...) -> MPO
    +*(::MPO, ::MPS; kwargs...) -> MPS

    Contract the MPO A with the MPS ψ, returning an MPS with the unique site indices of the MPO.

    For example, for an MPO with site indices with prime levels of 1 and 0, such as -s'-A-s-, and an MPS with site indices with prime levels of 0, such as -s-x, the result is an MPS y with site indices with prime levels of 1, -s'-y = -s'-A-s-x.

    Since it is common to contract an MPO with prime levels of 1 and 0 with an MPS with prime level of 0 and want a resulting MPS with prime levels of 0, we provide a convenience function apply:

    apply(A, x; kwargs...) = replaceprime(contract(A, x; kwargs...), 2 => 1)`.

    Choose the method with the method keyword, for example "densitymatrix" and "naive".

    Keywords

    • cutoff::Float64=1e-13: the cutoff value for truncating the density matrix eigenvalues. Note that the default is somewhat arbitrary and subject to change, in general you should set a cutoff value.
    • maxdim::Int=maxlinkdim(A) * maxlinkdim(ψ)): the maximal bond dimension of the results MPS.
    • mindim::Int=1: the minimal bond dimension of the resulting MPS.
    • normalize::Bool=false: whether or not to normalize the resulting MPS.
    • method::String="densitymatrix": the algorithm to use for the contraction. Currently the options are "densitymatrix", where the network formed by the MPO and MPS is squared and contracted down to a density matrix which is diagonalized iteratively at each site, and "naive", where the MPO and MPS tensor are contracted exactly at each site and then a truncation of the resulting MPS is performed.

    See also apply.

    source
    ITensors.applyMethod
    apply(A::MPO, x::MPS; kwargs...)

    Contract the MPO A with the MPS x and then map the prime level of the resulting MPS back to 0.

    Equivalent to replaceprime(contract(A, x; kwargs...), 2 => 1).

    See also contract for details about the arguments available.

    source
    NDTensors.contractMethod
    contract(A::MPO, B::MPO; kwargs...) -> MPO
     *(::MPO, ::MPO; kwargs...) -> MPO

    Contract the MPO A with the MPO B, returning an MPO with the site indices that are not shared between A and B.

    If you are contracting two MPOs with the same sets of indices, likely you want to call something like:

    C = contract(A', B; cutoff=1e-12)
     C = replaceprime(C, 2 => 1)

    That is because if MPO A has the index structure -s'-A-s- and MPO B has the Index structure -s'-B-s-, if we only want to contract over on set of the indices, we would do (-s'-A-s-)'-s'-B-s- = -s''-A-s'-s'-B-s- = -s''-C-s-, and then map the prime levels back to pairs of primed and unprimed indices with: replaceprime(-s''-C-s-, 2 => 1) = -s'-C-s-.

    Since this is a common use case, you can use the convenience function:

    C = apply(A, B; cutoff=1e-12)

    which is the same as the code above.

    If you are contracting MPOs that have diverging norms, such as MPOs representing sums of local operators, the truncation can become numerically unstable (see https://arxiv.org/abs/1909.06341 for a more numerically stable alternative). For now, you can use the following options to contract MPOs like that:

    C = contract(A, B; alg="naive", truncate=false)
     # Bring the indices back to pairs of primed and unprimed
    -C = apply(A, B; alg="naive", truncate=false)

    Keywords

    • cutoff::Float64=1e-14: the cutoff value for truncating the density matrix eigenvalues. Note that the default is somewhat arbitrary and subject to change, in general you should set a cutoff value.
    • maxdim::Int=maxlinkdim(A) * maxlinkdim(B)): the maximal bond dimension of the results MPS.
    • mindim::Int=1: the minimal bond dimension of the resulting MPS.
    • alg="zipup": Either "zipup" or "naive". "zipup" contracts pairs of site tensors and truncates with SVDs in a sweep across the sites, while "naive" first contracts pairs of tensor exactly and then truncates at the end if truncate=true.
    • truncate=true: Enable or disable truncation. If truncate=false, ignore other truncation parameters like cutoff and maxdim. This is most relevant for the "naive" version, if you just want to contract the tensors pairwise exactly. This can be useful if you are contracting MPOs that have diverging norms, such as MPOs originating from sums of local operators.

    See also apply for details about the arguments available.

    source
    ITensors.applyMethod
    apply(A::MPO, B::MPO; kwargs...)

    Contract the MPO A' with the MPO B and then map the prime level of the resulting MPO back to having pairs of indices with prime levels of 1 and 0.

    Equivalent to replaceprime(contract(A', B; kwargs...), 2 => 1).

    See also contract for details about the arguments available.

    source
    ITensors.ITensorMPS.error_contractMethod
    error_contract(y::MPS, A::MPO, x::MPS;
    +C = apply(A, B; alg="naive", truncate=false)

    Keywords

    • cutoff::Float64=1e-14: the cutoff value for truncating the density matrix eigenvalues. Note that the default is somewhat arbitrary and subject to change, in general you should set a cutoff value.
    • maxdim::Int=maxlinkdim(A) * maxlinkdim(B)): the maximal bond dimension of the results MPS.
    • mindim::Int=1: the minimal bond dimension of the resulting MPS.
    • alg="zipup": Either "zipup" or "naive". "zipup" contracts pairs of site tensors and truncates with SVDs in a sweep across the sites, while "naive" first contracts pairs of tensor exactly and then truncates at the end if truncate=true.
    • truncate=true: Enable or disable truncation. If truncate=false, ignore other truncation parameters like cutoff and maxdim. This is most relevant for the "naive" version, if you just want to contract the tensors pairwise exactly. This can be useful if you are contracting MPOs that have diverging norms, such as MPOs originating from sums of local operators.

    See also apply for details about the arguments available.

    source
    ITensors.applyMethod
    apply(A::MPO, B::MPO; kwargs...)

    Contract the MPO A' with the MPO B and then map the prime level of the resulting MPO back to having pairs of indices with prime levels of 1 and 0.

    Equivalent to replaceprime(contract(A', B; kwargs...), 2 => 1).

    See also contract for details about the arguments available.

    source
    ITensors.ITensorMPS.error_contractMethod
    error_contract(y::MPS, A::MPO, x::MPS;
                    make_inds_match::Bool = true)
     error_contract(y::MPS, x::MPS, x::MPO;
    -               make_inds_match::Bool = true)

    Compute the distance between A|x> and an approximation MPS y: | |y> - A|x> |/| A|x> | = √(1 + (<y|y> - 2*real(<y|A|x>))/<Ax|A|x>).

    If make_inds_match = true, the function attempts match the site indices of y with the site indices of A that are not common with x.

    source
    NDTensors.outerMethod
    outer(x::MPS, y::MPS; <keyword argument>) -> MPO

    Compute the outer product of MPS x and MPS y, returning an MPO approximation. Note that y will be conjugated.

    In Dirac notation, this is the operation |x⟩⟨y|.

    If you want an outer product of an MPS with itself, you should call outer(x', x; kwargs...) so that the resulting MPO has site indices with indices coming in pairs of prime levels of 1 and 0. If not, the site indices won't be unique which would not be an outer product.

    For example:

    s = siteinds("S=1/2", 5)
    +               make_inds_match::Bool = true)

    Compute the distance between A|x> and an approximation MPS y: | |y> - A|x> |/| A|x> | = √(1 + (<y|y> - 2*real(<y|A|x>))/<Ax|A|x>).

    If make_inds_match = true, the function attempts match the site indices of y with the site indices of A that are not common with x.

    source
    NDTensors.outerMethod
    outer(x::MPS, y::MPS; <keyword argument>) -> MPO

    Compute the outer product of MPS x and MPS y, returning an MPO approximation. Note that y will be conjugated.

    In Dirac notation, this is the operation |x⟩⟨y|.

    If you want an outer product of an MPS with itself, you should call outer(x', x; kwargs...) so that the resulting MPO has site indices with indices coming in pairs of prime levels of 1 and 0. If not, the site indices won't be unique which would not be an outer product.

    For example:

    s = siteinds("S=1/2", 5)
     x = random_mps(s)
     y = random_mps(s)
     outer(x, y) # Incorrect! Site indices must be unique.
    @@ -302,4 +302,4 @@
     y = convert(MPS, Y)
     outer(x, y) # Incorrect! Site indices must be unique.
     outer(x', y) # Incorrect! Site indices must be unique.
    -outer(addtags(x, "Out"), addtags(y, "In")) # This performs a proper outer product.

    The keyword arguments determine the truncation, and accept the same arguments as contract(::MPO, ::MPO; kwargs...).

    See also apply, contract.

    source
    ITensors.ITensorMPS.projectorMethod
    projector(x::MPS; <keyword argument>) -> MPO

    Computes the projector onto the state x. In Dirac notation, this is the operation |x⟩⟨x|/|⟨x|x⟩|².

    Use keyword arguments to control the level of truncation, which are the same as those accepted by contract(::MPO, ::MPO; kw...).

    Keywords

    • normalize::Bool=true: whether or not to normalize the input MPS before forming the projector. If normalize==false and the input MPS is not already normalized, this function will not output a proper project, and simply outputs outer(x, x) = |x⟩⟨x|, i.e. the projector scaled by norm(x)^2.
    • truncation keyword arguments accepted by contract(::MPO, ::MPO; kw...).

    See also outer, contract.

    source
    +outer(addtags(x, "Out"), addtags(y, "In")) # This performs a proper outer product.

    The keyword arguments determine the truncation, and accept the same arguments as contract(::MPO, ::MPO; kwargs...).

    See also apply, contract.

    source
    ITensors.ITensorMPS.projectorMethod
    projector(x::MPS; <keyword argument>) -> MPO

    Computes the projector onto the state x. In Dirac notation, this is the operation |x⟩⟨x|/|⟨x|x⟩|².

    Use keyword arguments to control the level of truncation, which are the same as those accepted by contract(::MPO, ::MPO; kw...).

    Keywords

    • normalize::Bool=true: whether or not to normalize the input MPS before forming the projector. If normalize==false and the input MPS is not already normalized, this function will not output a proper project, and simply outputs outer(x, x) = |x⟩⟨x|, i.e. the projector scaled by norm(x)^2.
    • truncation keyword arguments accepted by contract(::MPO, ::MPO; kw...).

    See also outer, contract.

    source
    diff --git a/previews/PR1514/Multithreading.html b/previews/PR1514/Multithreading.html index 9aebf9e27c..8843b557c8 100644 --- a/previews/PR1514/Multithreading.html +++ b/previews/PR1514/Multithreading.html @@ -36,7 +36,7 @@ $ julia -t 4 -$ JULIA_NUM_THREADS=4 julia

    In addition, we have found that it is best to disable BLAS and Strided multithreading when using block sparse multithreading. You can do that with the commands using LinearAlgebra; BLAS.set_num_threads(1) and ITensors.Strided.disable_threads().

    See also: ITensors.enable_threaded_blocksparse, ITensors.disable_threaded_blocksparse, ITensors.using_threaded_blocksparse.

    source
    enable_threaded_blocksparse(enable::Bool)

    enable_threaded_blocksparse(true) enables threaded block sparse operations (equivalent to enable_threaded_blocksparse()).

    enable_threaded_blocksparse(false) disables threaded block sparse operations (equivalent to enable_threaded_blocksparse()).

    source

    Here is a simple example of using block sparse multithreading to speed up a sparse tensor contraction:

    using BenchmarkTools
    +$ JULIA_NUM_THREADS=4 julia

    In addition, we have found that it is best to disable BLAS and Strided multithreading when using block sparse multithreading. You can do that with the commands using LinearAlgebra; BLAS.set_num_threads(1) and ITensors.Strided.disable_threads().

    See also: ITensors.enable_threaded_blocksparse, ITensors.disable_threaded_blocksparse, ITensors.using_threaded_blocksparse.

    source
    enable_threaded_blocksparse(enable::Bool)

    enable_threaded_blocksparse(true) enables threaded block sparse operations (equivalent to enable_threaded_blocksparse()).

    enable_threaded_blocksparse(false) disables threaded block sparse operations (equivalent to enable_threaded_blocksparse()).

    source

    Here is a simple example of using block sparse multithreading to speed up a sparse tensor contraction:

    using BenchmarkTools
     using ITensors, ITensorMPS
     using LinearAlgebra
     using Strided
    @@ -92,4 +92,4 @@
     Threaded contract:
       5.934 ms (446 allocations: 7.37 MiB)
     
    -C_contract ≈ C_threaded_contract = true

    In addition, we plan to add more threading to other parts of the code beyond contraction (such as SVD) and improve composibility with other forms of threading like BLAS and Strided, so stay tuned!

    +C_contract ≈ C_threaded_contract = true

    In addition, we plan to add more threading to other parts of the code beyond contraction (such as SVD) and improve composibility with other forms of threading like BLAS and Strided, so stay tuned!

    diff --git a/previews/PR1514/Observer.html b/previews/PR1514/Observer.html index 68a23853c3..734553e6c8 100644 --- a/previews/PR1514/Observer.html +++ b/previews/PR1514/Observer.html @@ -84,4 +84,4 @@ energy, psi = dmrg(H,psi0; nsweeps, cutoff, maxdim, observer=obs, outputlevel=1) return -end +end diff --git a/previews/PR1514/OpSum.html b/previews/PR1514/OpSum.html index 261396f84d..5240bf4e1e 100644 --- a/previews/PR1514/OpSum.html +++ b/previews/PR1514/OpSum.html @@ -1,5 +1,5 @@ -OpSum · ITensors.jl

    OpSum

    Description

    ITensors.Ops.OpSumType

    An OpSum represents a sum of operator terms.

    Often it is used to create matrix product operator (MPO) approximation of the sum of the terms in the OpSum oject. Each term is a product of local operators specified by names such as "Sz" or "N", times an optional coefficient which can be real or complex.

    Which local operator names are available is determined by the function op associated with the TagType defined by special Index tags, such as "S=1/2", "S=1", "Fermion", and "Electron".

    source

    Methods

    ITensors.ITensorMPS.add!Function
    add!(opsum::OpSum,
    +OpSum · ITensors.jl

    OpSum

    Description

    ITensors.Ops.OpSumType

    An OpSum represents a sum of operator terms.

    Often it is used to create matrix product operator (MPO) approximation of the sum of the terms in the OpSum oject. Each term is a product of local operators specified by names such as "Sz" or "N", times an optional coefficient which can be real or complex.

    Which local operator names are available is determined by the function op associated with the TagType defined by special Index tags, such as "S=1/2", "S=1", "Fermion", and "Electron".

    source

    Methods

    ITensors.ITensorMPS.add!Function
    add!(opsum::OpSum,
          op1::String, i1::Int)
     
     add!(opsum::OpSum,
    @@ -25,7 +25,7 @@
     
     opsum += (0.5,"S+",4,"S-",5)
     
    -opsum .+= (0.5,"S+",5,"S-",6)
    source
    ITensors.ITensorMPS.MPOMethod
    MPO(os::OpSum, sites::Vector{<:Index}; splitblocks=true, kwargs...)
     MPO(eltype::Type{<:Number}, os::OpSum, sites::Vector{<:Index}; splitblocks=true, kwargs...)

    Convert an OpSum object os to an MPO, with indices given by sites. The resulting MPO will have the indices sites[1], sites[1]', sites[2], sites[2]' etc. The conversion is done by an algorithm that compresses the MPO resulting from adding the OpSum terms together, often achieving the minimum possible bond dimension.

    Optionally specify the desired element type of the output MPO by passing the type as the first argument.

    The keyword argument splitblocks controls the sparsity of the resulting MPO. With the default splitblocks=true, the link indices of the MPO are split into blocks of dimension 1, potentially making the MPO more sparse.

    With the splitblocks=false, the blocks of the link dimensions are packed as much as possible according to common quantum numbers, making larger blocks. Before ITensors 0.3.19, this was the default output, but we have found that in general MPOs output with splitblocks=true lead to better performance in algorithms like DMRG.

    Examples

    os = OpSum()
     os += "Sz",1,"Sz",2
     os += "Sz",2,"Sz",3
    @@ -34,4 +34,4 @@
     sites = siteinds("S=1/2",4)
     H = MPO(os,sites)
     H = MPO(Float32,os,sites)
    -H = MPO(os,sites; splitblocks=false)
    source
    +H = MPO(os,sites; splitblocks=false)
    source
    diff --git a/previews/PR1514/ProjMPO.html b/previews/PR1514/ProjMPO.html index 7b394cde03..5d134326c0 100644 --- a/previews/PR1514/ProjMPO.html +++ b/previews/PR1514/ProjMPO.html @@ -3,8 +3,8 @@ | | | | | | | | | | | o--o--o--o--o--o--o--o--o--o--o H | | | | | | | | | | | -o--o--o- -o--o--o--o--o--o |psi>source

    Methods

    ITensors.productMethod
    product(P::ProjMPO,v::ITensor)::ITensor
    +o--o--o-      -o--o--o--o--o--o |psi>
    source

    Methods

    ITensors.productMethod
    product(P::ProjMPO,v::ITensor)::ITensor
     
    -(P::ProjMPO)(v::ITensor)

    Efficiently multiply the ProjMPO P by an ITensor v in the sense that the ProjMPO is a generalized square matrix or linear operator and v is a generalized vector in the space where it acts. The returned ITensor will have the same indices as v. The operator overload P(v) is shorthand for product(P,v).

    source
    ITensors.ITensorMPS.position!Method
    position!(P::ProjMPO, psi::MPS, pos::Int)

    Given an MPS psi, shift the projection of the MPO represented by the ProjMPO P such that the set of unprojected sites begins with site pos. This operation efficiently reuses previous projections of the MPO on sites that have already been projected. The MPS psi must have compatible bond indices with the previous projected MPO tensors for this operation to succeed.

    source
    ITensors.ITensorMPS.noisetermMethod
    noiseterm(P::ProjMPO,
    +(P::ProjMPO)(v::ITensor)

    Efficiently multiply the ProjMPO P by an ITensor v in the sense that the ProjMPO is a generalized square matrix or linear operator and v is a generalized vector in the space where it acts. The returned ITensor will have the same indices as v. The operator overload P(v) is shorthand for product(P,v).

    source
    ITensors.ITensorMPS.position!Method
    position!(P::ProjMPO, psi::MPS, pos::Int)

    Given an MPS psi, shift the projection of the MPO represented by the ProjMPO P such that the set of unprojected sites begins with site pos. This operation efficiently reuses previous projections of the MPO on sites that have already been projected. The MPS psi must have compatible bond indices with the previous projected MPO tensors for this operation to succeed.

    source
    ITensors.ITensorMPS.noisetermMethod
    noiseterm(P::ProjMPO,
               phi::ITensor,
    -          ortho::String)

    Return a "noise term" or density matrix perturbation ITensor as proposed in Phys. Rev. B 72, 180403 for aiding convergence of DMRG calculations. The ITensor phi is the contracted product of MPS tensors acted on by the ProjMPO P, and ortho is a String which can take the values "left" or "right" depending on the sweeping direction of the DMRG calculation.

    source

    Properties

    Base.lengthMethod
    length(P::ProjMPO)

    The length of a ProjMPO is the same as the length of the MPO used to construct it

    source
    Base.eltypeMethod
    eltype(P::ProjMPO)

    Deduce the element type (such as Float64 or ComplexF64) of the tensors in the ProjMPO P.

    source
    Base.sizeMethod
    size(P::ProjMPO)

    The size of a ProjMPO are its dimensions (d,d) when viewed as a matrix or linear operator acting on a space of dimension d.

    For example, if a ProjMPO maps from a space with indices (a,s1,s2,b) to the space (a',s1',s2',b') then the size is (d,d) where d = dim(a)*dim(s1)*dim(s1)*dim(b)

    source
    + ortho::String)

    Return a "noise term" or density matrix perturbation ITensor as proposed in Phys. Rev. B 72, 180403 for aiding convergence of DMRG calculations. The ITensor phi is the contracted product of MPS tensors acted on by the ProjMPO P, and ortho is a String which can take the values "left" or "right" depending on the sweeping direction of the DMRG calculation.

    source

    Properties

    Base.lengthMethod
    length(P::ProjMPO)

    The length of a ProjMPO is the same as the length of the MPO used to construct it

    source
    Base.eltypeMethod
    eltype(P::ProjMPO)

    Deduce the element type (such as Float64 or ComplexF64) of the tensors in the ProjMPO P.

    source
    Base.sizeMethod
    size(P::ProjMPO)

    The size of a ProjMPO are its dimensions (d,d) when viewed as a matrix or linear operator acting on a space of dimension d.

    For example, if a ProjMPO maps from a space with indices (a,s1,s2,b) to the space (a',s1',s2',b') then the size is (d,d) where d = dim(a)*dim(s1)*dim(s1)*dim(b)

    source
    diff --git a/previews/PR1514/ProjMPOSum.html b/previews/PR1514/ProjMPOSum.html index cd9c06c870..08a903dd48 100644 --- a/previews/PR1514/ProjMPOSum.html +++ b/previews/PR1514/ProjMPOSum.html @@ -3,8 +3,8 @@ | | | | | | | | | | | Σⱼ o--o--o--o--o--o--o--o--o--o--o Hⱼ | | | | | | | | | | | - o--o--o- -o--o--o--o--o--o |psi>source

    Methods

    ITensors.productMethod
    product(P::ProjMPOSum,v::ITensor)
    +     o--o--o-      -o--o--o--o--o--o |psi>
    source

    Methods

    ITensors.productMethod
    product(P::ProjMPOSum,v::ITensor)
     
    -(P::ProjMPOSum)(v::ITensor)

    Efficiently multiply the ProjMPOSum P by an ITensor v in the sense that the ProjMPOSum is a generalized square matrix or linear operator and v is a generalized vector in the space where it acts. The returned ITensor will have the same indices as v. The operator overload P(v) is shorthand for product(P,v).

    source
    ITensors.ITensorMPS.position!Method
    position!(P::ProjMPOSum, psi::MPS, pos::Int)

    Given an MPS psi, shift the projection of the MPO represented by the ProjMPOSum P such that the set of unprojected sites begins with site pos. This operation efficiently reuses previous projections of the MPOs on sites that have already been projected. The MPS psi must have compatible bond indices with the previous projected MPO tensors for this operation to succeed.

    source
    ITensors.ITensorMPS.noisetermMethod
    noiseterm(P::ProjMPOSum,
    +(P::ProjMPOSum)(v::ITensor)

    Efficiently multiply the ProjMPOSum P by an ITensor v in the sense that the ProjMPOSum is a generalized square matrix or linear operator and v is a generalized vector in the space where it acts. The returned ITensor will have the same indices as v. The operator overload P(v) is shorthand for product(P,v).

    source
    ITensors.ITensorMPS.position!Method
    position!(P::ProjMPOSum, psi::MPS, pos::Int)

    Given an MPS psi, shift the projection of the MPO represented by the ProjMPOSum P such that the set of unprojected sites begins with site pos. This operation efficiently reuses previous projections of the MPOs on sites that have already been projected. The MPS psi must have compatible bond indices with the previous projected MPO tensors for this operation to succeed.

    source
    ITensors.ITensorMPS.noisetermMethod
    noiseterm(P::ProjMPOSum,
               phi::ITensor,
    -          ortho::String)

    Return a "noise term" or density matrix perturbation ITensor as proposed in Phys. Rev. B 72, 180403 for aiding convergence of DMRG calculations. The ITensor phi is the contracted product of MPS tensors acted on by the ProjMPOSum P, and ortho is a String which can take the values "left" or "right" depending on the sweeping direction of the DMRG calculation.

    source

    Properties

    Base.eltypeMethod
    eltype(P::ProjMPOSum)

    Deduce the element type (such as Float64 or ComplexF64) of the tensors in the ProjMPOSum P.

    source
    Base.sizeMethod
    size(P::ProjMPOSum)

    The size of a ProjMPOSum are its dimensions (d,d) when viewed as a matrix or linear operator acting on a space of dimension d.

    For example, if a ProjMPOSum maps from a space with indices (a,s1,s2,b) to the space (a',s1',s2',b') then the size is (d,d) where d = dim(a)*dim(s1)*dim(s1)*dim(b)

    source
    + ortho::String)

    Return a "noise term" or density matrix perturbation ITensor as proposed in Phys. Rev. B 72, 180403 for aiding convergence of DMRG calculations. The ITensor phi is the contracted product of MPS tensors acted on by the ProjMPOSum P, and ortho is a String which can take the values "left" or "right" depending on the sweeping direction of the DMRG calculation.

    source

    Properties

    Base.eltypeMethod
    eltype(P::ProjMPOSum)

    Deduce the element type (such as Float64 or ComplexF64) of the tensors in the ProjMPOSum P.

    source
    Base.sizeMethod
    size(P::ProjMPOSum)

    The size of a ProjMPOSum are its dimensions (d,d) when viewed as a matrix or linear operator acting on a space of dimension d.

    For example, if a ProjMPOSum maps from a space with indices (a,s1,s2,b) to the space (a',s1',s2',b') then the size is (d,d) where d = dim(a)*dim(s1)*dim(s1)*dim(b)

    source
    diff --git a/previews/PR1514/QN.html b/previews/PR1514/QN.html index 7f6f0937cc..ea582accf8 100644 --- a/previews/PR1514/QN.html +++ b/previews/PR1514/QN.html @@ -1,4 +1,4 @@ -QN · ITensors.jl

    QN

    Description

    ITensors.QuantumNumbers.QNType

    A QN object stores a collection of up to four named values such as ("Sz",1) or ("N",0). These values can include a third integer "m" which makes them obey addition modulo m, for example ("P",1,2) for a value obeying addition mod 2. (The default is regular integer addition).

    Adding or subtracting pairs of QN objects performs addition and subtraction element-wise on each of the named values. If a name is missing from the collection, its value is treated as zero.

    source

    Constructors

    ITensors.QuantumNumbers.QNMethod
    QN(qvs...)

    Construct a QN from a set of up to four named value tuples.

    Examples

    q = QN(("Sz",1))
    +QN · ITensors.jl

    QN

    Description

    ITensors.QuantumNumbers.QNType

    A QN object stores a collection of up to four named values such as ("Sz",1) or ("N",0). These values can include a third integer "m" which makes them obey addition modulo m, for example ("P",1,2) for a value obeying addition mod 2. (The default is regular integer addition).

    Adding or subtracting pairs of QN objects performs addition and subtraction element-wise on each of the named values. If a name is missing from the collection, its value is treated as zero.

    source

    Constructors

    ITensors.QuantumNumbers.QNMethod
    QN(qvs...)

    Construct a QN from a set of up to four named value tuples.

    Examples

    q = QN(("Sz",1))
     q = QN(("N",1),("Sz",-1))
    -q = QN(("P",0,2),("Sz",0)).
    source
    ITensors.QuantumNumbers.QNType
    QN(name,val::Int,modulus::Int=1)

    Construct a QN with a single named value by providing the name, value, and optional modulus.

    source
    ITensors.QuantumNumbers.QNType
    QN(val::Int,modulus::Int=1)

    Construct a QN with a single unnamed value (equivalent to the name being the empty string) with optional modulus.

    source

    Properties

    ITensors.valMethod
    val(q::QN,name)

    Get the value within the QN q corresponding to the string name

    source
    Base.zeroMethod
    zero(q::QN)

    Returns a QN object containing the same names as q, but with all values set to zero.

    source
    +q = QN(("P",0,2),("Sz",0)).
    source
    ITensors.QuantumNumbers.QNType
    QN(name,val::Int,modulus::Int=1)

    Construct a QN with a single named value by providing the name, value, and optional modulus.

    source
    ITensors.QuantumNumbers.QNType
    QN(val::Int,modulus::Int=1)

    Construct a QN with a single unnamed value (equivalent to the name being the empty string) with optional modulus.

    source

    Properties

    ITensors.valMethod
    val(q::QN,name)

    Get the value within the QN q corresponding to the string name

    source
    Base.zeroMethod
    zero(q::QN)

    Returns a QN object containing the same names as q, but with all values set to zero.

    source
    diff --git a/previews/PR1514/QNTricks.html b/previews/PR1514/QNTricks.html index 9a046b2308..19f6b4bd9b 100644 --- a/previews/PR1514/QNTricks.html +++ b/previews/PR1514/QNTricks.html @@ -2,71 +2,71 @@ Symmetric (QN conserving) tensors: background and usage · ITensors.jl

    Symmetric (QN Conserving) Tensors: Background and Usage

    Here is a collection of background material and example codes for understanding how symmetric tensors (tensors with conserved quantum numbers) work in ITensors.jl

    Combiners and Symmetric Tensors

    In ITensors.jl, combiners are special sparse tensors that represent the action of taking the tensor product of one or more indices. It generalizes the idea of reshaping and permuting. For dense ITensors, a combiner is just the action of permuting and reshaping the data of the tensor. For symmetric tensors (quantum number conserving tensors represented as block sparse tensors), the combiner also fuses symmetry sectors together. They can be used for various purposes. Generally they are used internally in the library, for example in order to reshape a high order ITensor into an order 2 ITensor to perform a matrix decomposition like an SVD or eigendecomposition.

    For example:

    julia> using ITensors
            
            # This is a short code showing how a combiner
    -       # can be used to "flip" the direction of an Index
    julia> i = Index([QN(0) => 2, QN(1) => 3], "i")(dim=5|id=777|"i") <Out> + # can be used to "flip" the direction of an Index
    julia> i = Index([QN(0) => 2, QN(1) => 3], "i")(dim=5|id=546|"i") <Out> 1: QN(0) => 2 - 2: QN(1) => 3
    julia> j = Index([QN(0) => 2, QN(1) => 3], "j")(dim=5|id=721|"j") <Out> + 2: QN(1) => 3
    julia> j = Index([QN(0) => 2, QN(1) => 3], "j")(dim=5|id=49|"j") <Out> 1: QN(0) => 2 2: QN(1) => 3
    julia> A = random_itensor(i, dag(j))ITensor ord=2 -(dim=5|id=777|"i") <Out> +(dim=5|id=546|"i") <Out> 1: QN(0) => 2 2: QN(1) => 3 -(dim=5|id=721|"j") <In> +(dim=5|id=49|"j") <In> 1: QN(0) => 2 2: QN(1) => 3 NDTensors.BlockSparse{Float64, Vector{Float64}, 2}
    julia> C = combiner(i, dag(j); tags = "c", dir = dir(i))ITensor ord=3 -(dim=25|id=249|"c") <Out> +(dim=25|id=745|"c") <Out> 1: QN(-1) => 6 2: QN(0) => 13 3: QN(1) => 6 -(dim=5|id=777|"i") <In> +(dim=5|id=546|"i") <In> 1: QN(0) => 2 2: QN(1) => 3 -(dim=5|id=721|"j") <Out> +(dim=5|id=49|"j") <Out> 1: QN(0) => 2 2: QN(1) => 3 -NDTensors.Combiner
    julia> inds(A)((dim=5|id=777|"i") <Out> +NDTensors.Combiner
    julia> inds(A)((dim=5|id=546|"i") <Out> 1: QN(0) => 2 - 2: QN(1) => 3, (dim=5|id=721|"j") <In> + 2: QN(1) => 3, (dim=5|id=49|"j") <In> 1: QN(0) => 2 - 2: QN(1) => 3)
    julia> inds(A * C)((dim=25|id=249|"c") <Out> + 2: QN(1) => 3)
    julia> inds(A * C)((dim=25|id=745|"c") <Out> 1: QN(-1) => 6 2: QN(0) => 13 3: QN(1) => 6,)

    You can see that the combiner reshapes the indices of A into a single Index that contains the tensor product of the two input spaces. The spaces have size QN(-1) => 2 * 3, QN(0) => 2 * 2 + 3 * 3, and QN(0) => 2 * 3 (determined from all of the combinations of combining the sectors of the different indices, where the QNs are added and the block dimensions are multiplied). The ordering of the sectors is determined internally by ITensors.jl.

    You can also use a combiner on a single Index, which can be helpful for changing the direction of an Index or combining multiple sectors of the same symmetry into a single sector:

    julia> using ITensors
            
            # This is a short code showing how a combiner
    -       # can be used to "flip" the direction of an Index
    julia> i = Index([QN(0) => 2, QN(1) => 3], "i")(dim=5|id=131|"i") <Out> + # can be used to "flip" the direction of an Index
    julia> i = Index([QN(0) => 2, QN(1) => 3], "i")(dim=5|id=725|"i") <Out> 1: QN(0) => 2 - 2: QN(1) => 3
    julia> j = dag(Index([QN(0) => 2, QN(1) => 3], "j"))(dim=5|id=640|"j") <In> + 2: QN(1) => 3
    julia> j = dag(Index([QN(0) => 2, QN(1) => 3], "j"))(dim=5|id=736|"j") <In> 1: QN(0) => 2 2: QN(1) => 3
    julia> A = random_itensor(i, j)ITensor ord=2 -(dim=5|id=131|"i") <Out> +(dim=5|id=725|"i") <Out> 1: QN(0) => 2 2: QN(1) => 3 -(dim=5|id=640|"j") <In> +(dim=5|id=736|"j") <In> 1: QN(0) => 2 2: QN(1) => 3 NDTensors.BlockSparse{Float64, Vector{Float64}, 2}
    julia> C = combiner(j; tags = "jflip", dir = -dir(j))ITensor ord=2 -(dim=5|id=921|"jflip") <Out> +(dim=5|id=175|"jflip") <Out> 1: QN(-1) => 3 2: QN(0) => 2 -(dim=5|id=640|"j") <Out> +(dim=5|id=736|"j") <Out> 1: QN(0) => 2 2: QN(1) => 3 -NDTensors.Combiner
    julia> inds(A)((dim=5|id=131|"i") <Out> +NDTensors.Combiner
    julia> inds(A)((dim=5|id=725|"i") <Out> 1: QN(0) => 2 - 2: QN(1) => 3, (dim=5|id=640|"j") <In> + 2: QN(1) => 3, (dim=5|id=736|"j") <In> 1: QN(0) => 2 - 2: QN(1) => 3)
    julia> inds(A * C)((dim=5|id=921|"jflip") <Out> + 2: QN(1) => 3)
    julia> inds(A * C)((dim=5|id=175|"jflip") <Out> 1: QN(-1) => 3 - 2: QN(0) => 2, (dim=5|id=131|"i") <Out> + 2: QN(0) => 2, (dim=5|id=725|"i") <Out> 1: QN(0) => 2 - 2: QN(1) => 3)

    Unless you are writing very specialized custom code with symmetric tensors, this is generally not needed.

    Block Sparsity and Quantum Numbers

    In general, not all blocks that are allowed according to the flux will actually exist in the tensor (which helps in many cases for efficiency). Usually this would happen when the tensor is first constructed and not all blocks are explicitly set:

    julia> using ITensors
    julia> i = Index([QN(0) => 1, QN(1) => 1])(dim=2|id=101) <Out> + 2: QN(1) => 3)

    Unless you are writing very specialized custom code with symmetric tensors, this is generally not needed.

    Block Sparsity and Quantum Numbers

    In general, not all blocks that are allowed according to the flux will actually exist in the tensor (which helps in many cases for efficiency). Usually this would happen when the tensor is first constructed and not all blocks are explicitly set:

    julia> using ITensors
    julia> i = Index([QN(0) => 1, QN(1) => 1])(dim=2|id=57) <Out> 1: QN(0) => 1 2: QN(1) => 1
    julia> A = ITensor(i', dag(i));
    julia> A[2, 2] = 1.0;
    julia> @show A;A = ITensor ord=2 -Dim 1: (dim=2|id=101)' <Out> +Dim 1: (dim=2|id=57)' <Out> 1: QN(0) => 1 2: QN(1) => 1 -Dim 2: (dim=2|id=101) <In> +Dim 2: (dim=2|id=57) <In> 1: QN(0) => 1 2: QN(1) => 1 NDTensors.BlockSparse{Float64, Vector{Float64}, 2} @@ -74,31 +74,31 @@ Block(2, 2) [2:2, 2:2] 1.0
    julia> D, U = eigen(A; ishermitian=true);
    julia> @show D;D = ITensor ord=2 -Dim 1: (dim=1|id=488|"Link,eigen")' <Out> +Dim 1: (dim=1|id=163|"Link,eigen")' <Out> 1: QN(1) => 1 -Dim 2: (dim=1|id=488|"Link,eigen") <In> +Dim 2: (dim=1|id=163|"Link,eigen") <In> 1: QN(1) => 1 NDTensors.DiagBlockSparse{Float64, Vector{Float64}, 2} 1×1 Block(1, 1) [1:1, 1:1] 1.0
    julia> @show U;U = ITensor ord=2 -Dim 1: (dim=2|id=101) <Out> +Dim 1: (dim=2|id=57) <Out> 1: QN(0) => 1 2: QN(1) => 1 -Dim 2: (dim=1|id=488|"Link,eigen") <In> +Dim 2: (dim=1|id=163|"Link,eigen") <In> 1: QN(1) => 1 NDTensors.BlockSparse{Float64, Vector{Float64}, 2} 2×1 Block(2, 1) [2:2, 1:1] - 1.0

    If we had set A[1, 1] = 0.0 as well, then all of the allowed blocks (according to the flux QN(0) would exist and would be included in the eigendecomposition:

    julia> using ITensors
    julia> i = Index([QN(0) => 1, QN(1) => 1])(dim=2|id=287) <Out> + 1.0

    If we had set A[1, 1] = 0.0 as well, then all of the allowed blocks (according to the flux QN(0) would exist and would be included in the eigendecomposition:

    julia> using ITensors
    julia> i = Index([QN(0) => 1, QN(1) => 1])(dim=2|id=340) <Out> 1: QN(0) => 1 2: QN(1) => 1
    julia> A = ITensor(i', dag(i));
    julia> A[2, 2] = 1.0;
    julia> A[1, 1] = 0.0;
    julia> @show A;A = ITensor ord=2 -Dim 1: (dim=2|id=287)' <Out> +Dim 1: (dim=2|id=340)' <Out> 1: QN(0) => 1 2: QN(1) => 1 -Dim 2: (dim=2|id=287) <In> +Dim 2: (dim=2|id=340) <In> 1: QN(0) => 1 2: QN(1) => 1 NDTensors.BlockSparse{Float64, Vector{Float64}, 2} @@ -110,10 +110,10 @@ Block(1, 1) [1:1, 1:1] 0.0
    julia> D, U = eigen(A; ishermitian=true);
    julia> @show D;D = ITensor ord=2 -Dim 1: (dim=2|id=385|"Link,eigen")' <Out> +Dim 1: (dim=2|id=780|"Link,eigen")' <Out> 1: QN(0) => 1 2: QN(1) => 1 -Dim 2: (dim=2|id=385|"Link,eigen") <In> +Dim 2: (dim=2|id=780|"Link,eigen") <In> 1: QN(0) => 1 2: QN(1) => 1 NDTensors.DiagBlockSparse{Float64, Vector{Float64}, 2} @@ -125,10 +125,10 @@ Block(2, 2) [2:2, 2:2] 1.0
    julia> @show U;U = ITensor ord=2 -Dim 1: (dim=2|id=287) <Out> +Dim 1: (dim=2|id=340) <Out> 1: QN(0) => 1 2: QN(1) => 1 -Dim 2: (dim=2|id=385|"Link,eigen") <In> +Dim 2: (dim=2|id=780|"Link,eigen") <In> 1: QN(0) => 1 2: QN(1) => 1 NDTensors.BlockSparse{Float64, Vector{Float64}, 2} @@ -139,24 +139,24 @@ Block(2, 2) [2:2, 2:2] - 1.0

    "Missing" blocks can also occur with tensor contractions, since the final blocks of the output tensor are made from combinations of contractions of blocks from the input tensors, and there is no guarantee that all flux-consistent blocks will end up in the result:

    julia> using ITensors
    julia> i = Index([QN(0) => 1, QN(1) => 1])(dim=2|id=844) <Out> + 1.0

    "Missing" blocks can also occur with tensor contractions, since the final blocks of the output tensor are made from combinations of contractions of blocks from the input tensors, and there is no guarantee that all flux-consistent blocks will end up in the result:

    julia> using ITensors
    julia> i = Index([QN(0) => 1, QN(1) => 1])(dim=2|id=663) <Out> 1: QN(0) => 1 - 2: QN(1) => 1
    julia> j = Index([QN(0) => 1])(dim=1|id=868) <Out> + 2: QN(1) => 1
    julia> j = Index([QN(0) => 1])(dim=1|id=183) <Out> 1: QN(0) => 1
    julia> A = ITensor(i, dag(j));
    julia> A[2, 1] = 1.0;
    julia> @show A;A = ITensor ord=2 -Dim 1: (dim=2|id=844) <Out> +Dim 1: (dim=2|id=663) <Out> 1: QN(0) => 1 2: QN(1) => 1 -Dim 2: (dim=1|id=868) <In> +Dim 2: (dim=1|id=183) <In> 1: QN(0) => 1 NDTensors.BlockSparse{Float64, Vector{Float64}, 2} 2×1 Block(2, 1) [2:2, 1:1] 1.0
    julia> A2 = prime(A, i) * dag(A);
    julia> @show A2;A2 = ITensor ord=2 -Dim 1: (dim=2|id=844)' <Out> +Dim 1: (dim=2|id=663)' <Out> 1: QN(0) => 1 2: QN(1) => 1 -Dim 2: (dim=2|id=844) <In> +Dim 2: (dim=2|id=663) <In> 1: QN(0) => 1 2: QN(1) => 1 NDTensors.BlockSparse{Float64, Vector{Float64}, 2} @@ -164,22 +164,22 @@ Block(2, 2) [2:2, 2:2] 1.0
    julia> D, U = eigen(A2; ishermitian=true);
    julia> @show D;D = ITensor ord=2 -Dim 1: (dim=1|id=942|"Link,eigen")' <Out> +Dim 1: (dim=1|id=378|"Link,eigen")' <Out> 1: QN(1) => 1 -Dim 2: (dim=1|id=942|"Link,eigen") <In> +Dim 2: (dim=1|id=378|"Link,eigen") <In> 1: QN(1) => 1 NDTensors.DiagBlockSparse{Float64, Vector{Float64}, 2} 1×1 Block(1, 1) [1:1, 1:1] 1.0
    julia> @show U;U = ITensor ord=2 -Dim 1: (dim=2|id=844) <Out> +Dim 1: (dim=2|id=663) <Out> 1: QN(0) => 1 2: QN(1) => 1 -Dim 2: (dim=1|id=942|"Link,eigen") <In> +Dim 2: (dim=1|id=378|"Link,eigen") <In> 1: QN(1) => 1 NDTensors.BlockSparse{Float64, Vector{Float64}, 2} 2×1 Block(2, 1) [2:2, 1:1] - 1.0
    + 1.0 diff --git a/previews/PR1514/RunningOnGPUs.html b/previews/PR1514/RunningOnGPUs.html index f7ad40f5bd..6c7ab75205 100644 --- a/previews/PR1514/RunningOnGPUs.html +++ b/previews/PR1514/RunningOnGPUs.html @@ -26,4 +26,4 @@ Bmtl = mtl(B) # Perform tensor operations on Apple GPU -Amtl * Bmtl

    Note that we highly recommend using these new package extensions as opposed to ITensorGPU.jl, which is ITensor's previous CUDA backend. The package extensions are better integrated into the main library so are more reliable and better supported right now. We plan to deprecate ITensorGPU.jl in the future.

    GPU backends

    ITensor currently provides package extensions for the following GPU backends:

    Our goal is to support all GPU backends which are supported by the JuliaGPU organization.

    Notice that cuTENSOR.jl is an extension of CUDA.jl that provides new functionality for accelerated binary tensor contractions. If the cuTENSOR.jl library is loaded then ITensors with CuArray data are contracted using cuTENSOR and if the cuTENSOR.jl library is not loaded but CUDA.jl is loaded then binary tensor contractions are mapped to a matrix multiplication and performed using cuBLAS.

    Some important caveats to keep in mind related to the ITensor GPU backends are:

    The table below summarizes each backend's current capabilities.

    CUDAcuTENSORROCmMetaloneAPI
    Contractions (dense)✓ (cuBLAS)N/A[oneapi]
    QR (dense)✓ (cuSOLVER)✓ (cuSOLVER)On CPU[linalg]On CPU[linalg]N/A[oneapi]
    SVD (dense)✓ (cuSOLVER)✓ (cuSOLVER)On CPU[linalg]On CPU[linalg]N/A[oneapi]
    Eigendecomposition (dense)✓ (cuSOLVER)✓ (cuSOLVER)On CPU[linalg]On CPU[linalg]N/A[oneapi]
    Double precision (Float64)N/A[metal]N/A[oneapi]
    Block sparse[blocksparse][blocksparse][blocksparse][blocksparse]N/A[oneapi]
    +Amtl * Bmtl

    Note that we highly recommend using these new package extensions as opposed to ITensorGPU.jl, which is ITensor's previous CUDA backend. The package extensions are better integrated into the main library so are more reliable and better supported right now. We plan to deprecate ITensorGPU.jl in the future.

    GPU backends

    ITensor currently provides package extensions for the following GPU backends:

    Our goal is to support all GPU backends which are supported by the JuliaGPU organization.

    Notice that cuTENSOR.jl is an extension of CUDA.jl that provides new functionality for accelerated binary tensor contractions. If the cuTENSOR.jl library is loaded then ITensors with CuArray data are contracted using cuTENSOR and if the cuTENSOR.jl library is not loaded but CUDA.jl is loaded then binary tensor contractions are mapped to a matrix multiplication and performed using cuBLAS.

    Some important caveats to keep in mind related to the ITensor GPU backends are:

    The table below summarizes each backend's current capabilities.

    CUDAcuTENSORROCmMetaloneAPI
    Contractions (dense)✓ (cuBLAS)N/A[oneapi]
    QR (dense)✓ (cuSOLVER)✓ (cuSOLVER)On CPU[linalg]On CPU[linalg]N/A[oneapi]
    SVD (dense)✓ (cuSOLVER)✓ (cuSOLVER)On CPU[linalg]On CPU[linalg]N/A[oneapi]
    Eigendecomposition (dense)✓ (cuSOLVER)✓ (cuSOLVER)On CPU[linalg]On CPU[linalg]N/A[oneapi]
    Double precision (Float64)N/A[metal]N/A[oneapi]
    Block sparse[blocksparse][blocksparse][blocksparse][blocksparse]N/A[oneapi]
    diff --git a/previews/PR1514/SiteType.html b/previews/PR1514/SiteType.html index 6f1cd45470..081e70ceb6 100644 --- a/previews/PR1514/SiteType.html +++ b/previews/PR1514/SiteType.html @@ -41,8 +41,8 @@ 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

    Many operators are available, for example:

    You can view the source code for the internal SiteType definitions and operators that are defined here.

    source

    Methods

    ITensors.SiteTypes.opFunction
    op(opname::String, s::Index; kwargs...)

    Return an ITensor corresponding to the operator named opname for the Index s. The operator is constructed by calling an overload of either the op or op! methods which take a SiteType argument that corresponds to one of the tags of the Index s and an OpName"opname" argument that corresponds to the input operator name.

    Operator names can be combined using the "*" symbol, for example "S+*S-" or "Sz*Sz*Sz". The result is an ITensor made by forming each operator then contracting them together in a way corresponding to the usual operator product or matrix multiplication.

    The op system is used by the OpSum system to convert operator names into ITensors, and can be used directly such as for applying operators to MPS.

    Example

    s = Index(2, "Site,S=1/2")
    -Sz = op("Sz", s)

    To see all of the operator names defined for the site types included with ITensor, please view the source code for each site type. Note that some site types such as "S=1/2" and "Qubit" are aliases for each other and share operator definitions.

    source
    op(X::AbstractArray, s::Index...)
    + 0.0  0.0  0.0  1.0

    Many operators are available, for example:

    • SiteType"S=1/2": "Sz", "Sx", "Sy", "S+", "S-", ...
    • SiteType"Electron": "Nup", "Ndn", "Nupdn", "Ntot", "Cup", "Cdagup", "Cdn", "Cdagdn", "Sz", "Sx", "Sy", "S+", "S-", ...
    • ...

    You can view the source code for the internal SiteType definitions and operators that are defined here.

    source

    Methods

    ITensors.SiteTypes.opFunction
    op(opname::String, s::Index; kwargs...)

    Return an ITensor corresponding to the operator named opname for the Index s. The operator is constructed by calling an overload of either the op or op! methods which take a SiteType argument that corresponds to one of the tags of the Index s and an OpName"opname" argument that corresponds to the input operator name.

    Operator names can be combined using the "*" symbol, for example "S+*S-" or "Sz*Sz*Sz". The result is an ITensor made by forming each operator then contracting them together in a way corresponding to the usual operator product or matrix multiplication.

    The op system is used by the OpSum system to convert operator names into ITensors, and can be used directly such as for applying operators to MPS.

    Example

    s = Index(2, "Site,S=1/2")
    +Sz = op("Sz", s)

    To see all of the operator names defined for the site types included with ITensor, please view the source code for each site type. Note that some site types such as "S=1/2" and "Qubit" are aliases for each other and share operator definitions.

    source
    op(X::AbstractArray, s::Index...)
     op(M::Matrix, s::Index...)

    Given a matrix M and a set of indices s,t,... return an operator ITensor with matrix elements given by M and indices s, s', t, t'

    Example

    julia> s = siteind("S=1/2")
     (dim=2|id=575|"S=1/2,Site")
     
    @@ -59,57 +59,57 @@
      0.5   0.0
      0.0  -0.5
     ITensor ord=2 (dim=2|id=575|"S=1/2,Site")' (dim=2|id=575|"S=1/2,Site")
    -NDTensors.Dense{Float64, Vector{Float64}}
    source
    op(opname::String,sites::Vector{<:Index},n::Int; kwargs...)

    Return an ITensor corresponding to the operator named opname for the n'th Index in the array sites.

    Example

    s = siteinds("S=1/2", 4)
    -Sz2 = op("Sz", s, 2)
    source
    ITensors.SiteTypes.stateFunction
    state(s::Index, name::String; kwargs...)

    Return an ITensor corresponding to the state named name for the Index s. The returned ITensor will have s as its only index.

    The terminology here is based on the idea of a single-site state or wavefunction in physics.

    The state function is implemented for various Index tags by overloading either the state or state! methods which take a SiteType argument corresponding to one of the tags of the Index s and an StateName"name" argument that corresponds to the input state name.

    The state system is used by the MPS type to construct product-state MPS and for other purposes.

    Example

    s = Index(2, "Site,S=1/2")
    +NDTensors.Dense{Float64, Vector{Float64}}
    source
    op(opname::String,sites::Vector{<:Index},n::Int; kwargs...)

    Return an ITensor corresponding to the operator named opname for the n'th Index in the array sites.

    Example

    s = siteinds("S=1/2", 4)
    +Sz2 = op("Sz", s, 2)
    source
    ITensors.SiteTypes.stateFunction
    state(s::Index, name::String; kwargs...)

    Return an ITensor corresponding to the state named name for the Index s. The returned ITensor will have s as its only index.

    The terminology here is based on the idea of a single-site state or wavefunction in physics.

    The state function is implemented for various Index tags by overloading either the state or state! methods which take a SiteType argument corresponding to one of the tags of the Index s and an StateName"name" argument that corresponds to the input state name.

    The state system is used by the MPS type to construct product-state MPS and for other purposes.

    Example

    s = Index(2, "Site,S=1/2")
     sup = state(s,"Up")
     sdn = state(s,"Dn")
     sxp = state(s,"X+")
    -sxm = state(s,"X-")
    source
    ITensors.valFunction
    val(q::QN,name)

    Get the value within the QN q corresponding to the string name

    source
    val(s::Index, name::String)

    Return an integer corresponding to the name of a certain value the Index s can take. In other words, the val function maps strings to specific integer values within the range 1:dim(s).

    The val function is implemented for various Index tags by overloading methods named val which take a SiteType argument corresponding to one of the tags of the Index s and an ValName"name" argument that corresponds to the input name.

    Example

    s = Index(2, "Site,S=1/2")
    +sxm = state(s,"X-")
    source
    ITensors.valFunction
    val(q::QN,name)

    Get the value within the QN q corresponding to the string name

    source
    val(s::Index, name::String)

    Return an integer corresponding to the name of a certain value the Index s can take. In other words, the val function maps strings to specific integer values within the range 1:dim(s).

    The val function is implemented for various Index tags by overloading methods named val which take a SiteType argument corresponding to one of the tags of the Index s and an ValName"name" argument that corresponds to the input name.

    Example

    s = Index(2, "Site,S=1/2")
     val(s,"Up") == 1
     val(s,"Dn") == 2
     
     s = Index(2, "Site,Fermion")
     val(s,"Emp") == 1
    -val(s,"Occ") == 2
    source
    ITensors.spaceFunction
    space(::SiteType"Qubit";
    +val(s,"Occ") == 2
    source
    ITensors.spaceFunction
    space(::SiteType"Qubit";
           conserve_qns = false,
           conserve_parity = conserve_qns,
           conserve_number = false,
           qnname_parity = "Parity",
    -      qnname_number = "Number")

    Create the Hilbert space for a site of type "Qubit".

    Optionally specify the conserved symmetries and their quantum number labels.

    source
    space(::SiteType"S=1/2";
    +      qnname_number = "Number")

    Create the Hilbert space for a site of type "Qubit".

    Optionally specify the conserved symmetries and their quantum number labels.

    source
    space(::SiteType"S=1/2";
           conserve_qns = false,
           conserve_sz = conserve_qns,
           conserve_szparity = false,
           qnname_sz = "Sz",
    -      qnname_szparity = "SzParity")

    Create the Hilbert space for a site of type "S=1/2".

    Optionally specify the conserved symmetries and their quantum number labels.

    source
    space(::SiteType"S=1";
    +      qnname_szparity = "SzParity")

    Create the Hilbert space for a site of type "S=1/2".

    Optionally specify the conserved symmetries and their quantum number labels.

    source
    space(::SiteType"S=1";
           conserve_qns = false,
           conserve_sz = conserve_qns,
    -      qnname_sz = "Sz")

    Create the Hilbert space for a site of type "S=1".

    Optionally specify the conserved symmetries and their quantum number labels.

    source
    space(::SiteType"Fermion";
    +      qnname_sz = "Sz")

    Create the Hilbert space for a site of type "S=1".

    Optionally specify the conserved symmetries and their quantum number labels.

    source
    space(::SiteType"Fermion";
           conserve_qns=false,
           conserve_nf=conserve_qns,
           conserve_nfparity=conserve_qns,
           qnname_nf = "Nf",
           qnname_nfparity = "NfParity",
           qnname_sz = "Sz",
    -      conserve_sz = false)

    Create the Hilbert space for a site of type "Fermion".

    Optionally specify the conserved symmetries and their quantum number labels.

    source
    space(::SiteType"Electron";
    +      conserve_sz = false)

    Create the Hilbert space for a site of type "Fermion".

    Optionally specify the conserved symmetries and their quantum number labels.

    source
    space(::SiteType"Electron";
           conserve_qns = false,
           conserve_sz = conserve_qns,
           conserve_nf = conserve_qns,
           conserve_nfparity = conserve_qns,
           qnname_sz = "Sz",
           qnname_nf = "Nf",
    -      qnname_nfparity = "NfParity")

    Create the Hilbert space for a site of type "Electron".

    Optionally specify the conserved symmetries and their quantum number labels.

    source
    space(::SiteType"tJ";
    +      qnname_nfparity = "NfParity")

    Create the Hilbert space for a site of type "Electron".

    Optionally specify the conserved symmetries and their quantum number labels.

    source
    space(::SiteType"tJ";
           conserve_qns = false,
           conserve_sz = conserve_qns,
           conserve_nf = conserve_qns,
           conserve_nfparity = conserve_qns,
           qnname_sz = "Sz",
           qnname_nf = "Nf",
    -      qnname_nfparity = "NfParity")

    Create the Hilbert space for a site of type "tJ".

    Optionally specify the conserved symmetries and their quantum number labels.

    source
    space(::SiteType"Qudit";
    +      qnname_nfparity = "NfParity")

    Create the Hilbert space for a site of type "tJ".

    Optionally specify the conserved symmetries and their quantum number labels.

    source
    space(::SiteType"Qudit";
           dim = 2,
           conserve_qns = false,
           conserve_number = false,
    -      qnname_number = "Number")

    Create the Hilbert space for a site of type "Qudit".

    Optionally specify the conserved symmetries and their quantum number labels.

    source
    space(::SiteType"Boson";
    +      qnname_number = "Number")

    Create the Hilbert space for a site of type "Qudit".

    Optionally specify the conserved symmetries and their quantum number labels.

    source
    space(::SiteType"Boson";
           dim = 2,
           conserve_qns = false,
           conserve_number = false,
    -      qnname_number = "Number")

    Create the Hilbert space for a site of type "Boson".

    Optionally specify the conserved symmetries and their quantum number labels.

    source
    + qnname_number = "Number")

    Create the Hilbert space for a site of type "Boson".

    Optionally specify the conserved symmetries and their quantum number labels.

    source diff --git a/previews/PR1514/Sweeps.html b/previews/PR1514/Sweeps.html index 4e8fc3bcfe..509161cdde 100644 --- a/previews/PR1514/Sweeps.html +++ b/previews/PR1514/Sweeps.html @@ -1,5 +1,5 @@ -Sweeps · ITensors.jl

    Sweeps

    ITensors.ITensorMPS.SweepsType

    A Sweeps objects holds information about the various parameters controlling a density matrix renormalization group (DMRG) or similar matrix product state (MPS) calculation.

    For a Sweeps object sw the available parameters are:

    • nsweep(sw) – the number of sweeps to do
    • maxdim(sw,n) – maximum MPS bond dimension for sweep n
    • mindim(sw,n) – minimum MPS bond dimension for sweep n
    • cutoff(sw,n) – truncation error cutoff for sweep n
    • noise(sw,n) – noise term coefficient for sweep n
    source
    ITensors.ITensorMPS.SweepsMethod
    Sweeps(d::AbstractMatrix)
    +Sweeps · ITensors.jl

    Sweeps

    ITensors.ITensorMPS.SweepsType

    A Sweeps objects holds information about the various parameters controlling a density matrix renormalization group (DMRG) or similar matrix product state (MPS) calculation.

    For a Sweeps object sw the available parameters are:

    • nsweep(sw) – the number of sweeps to do
    • maxdim(sw,n) – maximum MPS bond dimension for sweep n
    • mindim(sw,n) – minimum MPS bond dimension for sweep n
    • cutoff(sw,n) – truncation error cutoff for sweep n
    • noise(sw,n) – noise term coefficient for sweep n
    source
    ITensors.ITensorMPS.SweepsMethod
    Sweeps(d::AbstractMatrix)
     
     Sweeps(nsweep::Int, d::AbstractMatrix)

    Make a sweeps object from a matrix of input values. The first row should be strings that define which variables are being set ("maxdim", "cutoff", "mindim", and "noise").

    If the number of sweeps are not specified, they are determined from the size of the input matrix.

    Examples

    julia > Sweeps(
       [
    @@ -18,5 +18,5 @@
     3cutoff = 1.0E-12, maxdim = 200, mindim = 20, noise = 1.0E-10
     4cutoff = 1.0E-12, maxdim = 400, mindim = 20, noise = 0.0E+00
     5cutoff = 1.0E-12, maxdim = 800, mindim = 20, noise = 1.0E-11
    -6cutoff = 1.0E-12, maxdim = 800, mindim = 20, noise = 0.0E+00
    source

    Modifying Sweeps Objects

    ITensors.ITensorMPS.setmaxdim!Function
    maxdim!(sw::Sweeps,maxdims::Int...)

    Set the maximum MPS bond dimension for each sweep by providing up to nsweep(sw) values. If fewer values are provided, the last value is repeated for the remaining sweeps.

    source
    ITensors.ITensorMPS.setcutoff!Function
    cutoff!(sw::Sweeps,maxdims::Int...)

    Set the MPS truncation error used for each sweep by providing up to nsweep(sw) values. If fewer values are provided, the last value is repeated for the remaining sweeps.

    source
    ITensors.ITensorMPS.setnoise!Function
    noise!(sw::Sweeps,maxdims::Int...)

    Set the noise-term coefficient used for each sweep by providing up to nsweep(sw) values. If fewer values are provided, the last value is repeated for the remaining sweeps.

    source
    ITensors.ITensorMPS.setmindim!Function
    mindim!(sw::Sweeps,maxdims::Int...)

    Set the minimum MPS bond dimension for each sweep by providing up to nsweep(sw) values. If fewer values are provided, the last value is repeated for the remaining sweeps.

    source

    Getting Sweeps Object Data

    NDTensors.maxdimMethod
    maxdim(sw::Sweeps,n::Int)

    Maximum MPS bond dimension allowed by the Sweeps object sw during sweep n

    source
    NDTensors.mindimMethod
    mindim(sw::Sweeps,n::Int)

    Minimum MPS bond dimension allowed by the Sweeps object sw during sweep n

    source
    +6cutoff = 1.0E-12, maxdim = 800, mindim = 20, noise = 0.0E+00
    source

    Modifying Sweeps Objects

    ITensors.ITensorMPS.setmaxdim!Function
    maxdim!(sw::Sweeps,maxdims::Int...)

    Set the maximum MPS bond dimension for each sweep by providing up to nsweep(sw) values. If fewer values are provided, the last value is repeated for the remaining sweeps.

    source
    ITensors.ITensorMPS.setcutoff!Function
    cutoff!(sw::Sweeps,maxdims::Int...)

    Set the MPS truncation error used for each sweep by providing up to nsweep(sw) values. If fewer values are provided, the last value is repeated for the remaining sweeps.

    source
    ITensors.ITensorMPS.setnoise!Function
    noise!(sw::Sweeps,maxdims::Int...)

    Set the noise-term coefficient used for each sweep by providing up to nsweep(sw) values. If fewer values are provided, the last value is repeated for the remaining sweeps.

    source
    ITensors.ITensorMPS.setmindim!Function
    mindim!(sw::Sweeps,maxdims::Int...)

    Set the minimum MPS bond dimension for each sweep by providing up to nsweep(sw) values. If fewer values are provided, the last value is repeated for the remaining sweeps.

    source

    Getting Sweeps Object Data

    NDTensors.maxdimMethod
    maxdim(sw::Sweeps,n::Int)

    Maximum MPS bond dimension allowed by the Sweeps object sw during sweep n

    source
    NDTensors.mindimMethod
    mindim(sw::Sweeps,n::Int)

    Minimum MPS bond dimension allowed by the Sweeps object sw during sweep n

    source
    diff --git a/previews/PR1514/UpgradeGuide_0.1_to_0.2.html b/previews/PR1514/UpgradeGuide_0.1_to_0.2.html index 097409940d..569a7e323f 100644 --- a/previews/PR1514/UpgradeGuide_0.1_to_0.2.html +++ b/previews/PR1514/UpgradeGuide_0.1_to_0.2.html @@ -185,4 +185,4 @@ 2: QN("Sz",-1) => 1 (dim=2|id=810|"S=1/2,Site,n=4") <Out> 1: QN("Sz",1) => 1 - 2: QN("Sz",-1) => 1

    This shouldn't affect end users in general. The new convention is a bit more intuitive since the quantum number can be thought of as counting the total number of 1 bits in the state, though the conventions can be mapped to each other with a constant.

    maxlinkdim for MPS/MPO with no indices

    maxlinkdim(::MPS/MPO) returns a minimum of 1 (previously it returned 0 for MPS/MPO without and link indices) (PR #663).

    + 2: QN("Sz",-1) => 1

    This shouldn't affect end users in general. The new convention is a bit more intuitive since the quantum number can be thought of as counting the total number of 1 bits in the state, though the conventions can be mapped to each other with a constant.

    maxlinkdim for MPS/MPO with no indices

    maxlinkdim(::MPS/MPO) returns a minimum of 1 (previously it returned 0 for MPS/MPO without and link indices) (PR #663).

    diff --git a/previews/PR1514/examples/DMRG.html b/previews/PR1514/examples/DMRG.html index ed29409bac..0f7063259d 100644 --- a/previews/PR1514/examples/DMRG.html +++ b/previews/PR1514/examples/DMRG.html @@ -315,4 +315,4 @@ After sweep 4, |psi| = 2.863 MiB, |PH| = 7.246 MiB After sweep 4 energy=-44.127710946536645 maxlinkdim=56 maxerr=9.99E-09 time=0.445 After sweep 5, |psi| = 3.108 MiB, |PH| = 7.845 MiB -After sweep 5 energy=-44.127736798226536 maxlinkdim=57 maxerr=9.98E-09 time=0.564 +After sweep 5 energy=-44.127736798226536 maxlinkdim=57 maxerr=9.98E-09 time=0.564 diff --git a/previews/PR1514/examples/ITensor.html b/previews/PR1514/examples/ITensor.html index e4b1ca735d..a034f19a1a 100644 --- a/previews/PR1514/examples/ITensor.html +++ b/previews/PR1514/examples/ITensor.html @@ -23,8 +23,8 @@ T = random_itensor(k,m) @show T
    T = ITensor ord=2
    -Dim 1: (dim=4|id=684|"index_k")
    -Dim 2: (dim=2|id=241|"index_m")
    +Dim 1: (dim=4|id=945|"index_k")
    +Dim 2: (dim=2|id=111|"index_m")
     NDTensors.Dense{Float64, Vector{Float64}}
      4×2
      -0.037025446544394686   0.4967800795298839
    @@ -57,21 +57,21 @@
     T .= myf.(T)

    Making an ITensor with a Single Non-Zero Element

    It is often useful to make ITensors with all elements zero except for a specific element that is equal to 1.0. Use cases can include making product-state quantum wavefunctions or contracting single-element ITensors with other ITensors to set their indices to a fixed value.

    To make such an ITensor, use the onehot function. Borrowing terminology from engineering, a "one hot" vector or tensor has a single element equal to 1.0 and the rest zero. (In previous versions of ITensor this function was called setelt.)

    The ITensor function onehot takes one or more Index-value Pairs such as i=>2 and j=>1 and returns an ITensor with a 1.0 in the location specified by the Index values:

    i = Index(2)
     O1 = onehot(i=>1)
     println(O1)
    ITensor ord=1
    -Dim 1: (dim=2|id=60)
    +Dim 1: (dim=2|id=407)
     NDTensors.Dense{Float64, Vector{Float64}}
      2-element
      1.0
      0.0
    O2 = onehot(i=>2)
     println(O2)
    ITensor ord=1
    -Dim 1: (dim=2|id=371)
    +Dim 1: (dim=2|id=984)
     NDTensors.Dense{Float64, Vector{Float64}}
      2-element
      0.0
      1.0
    j = Index(3)
     T = onehot(i=>2,j=>3)
     println(T)
    ITensor ord=2
    -Dim 1: (dim=2|id=924)
    -Dim 2: (dim=3|id=642)
    +Dim 1: (dim=2|id=236)
    +Dim 2: (dim=3|id=635)
     NDTensors.Dense{Float64, Vector{Float64}}
      2×3
      0.0  0.0  0.0
    @@ -105,11 +105,11 @@
     @show norm(U*S*V-T)
     @show (norm(U*S*V - T)/norm(T))^2

    QR Factorization

    Computing the QR factorization of an ITensor works in a similar way as for the SVD. In addition to passing the ITensor you want to factorize, you must also pass the indices you want to end up on the tensor Q, in other words to be treated as the "row" indices for the purpose of defining the QR factorization.

    Say we want to compute the QR factorization of an ITensor T with indices i,j,k, putting the indices i and k onto Q and the remaining indices onto R. We can do this as follows:

    T = random_itensor(i,j,k)
     Q,R = qr(T,(i,k);positive=true)

    Note the use of the optional positive=true keyword argument, which ensures that the diagonal elements of R are non-negative. With this option, the QR factorization is unique, which can be useful in certain cases.

    Combining Multiple Indices into One Index

    It can be very useful to combine or merge multiple indices of an ITensor into a single Index. Say we have an ITensor with indices i,j,k and we want to combine Index i and Index k into a new Index. This new Index (call it c) will have a dimension whose size is the dimension of i times the dimension of k.

    To carry out this procedure we can make a special kind of ITensor: a combiner. To make a combiner, call the function combiner, passing the indices you want to combine:

    C = combiner(i,k; tags="c")

    Then if we have an ITensor

    T = random_itensor(i,j,k)
    -@show inds(T)
    ((dim=4|id=604|"i"), (dim=3|id=46|"j"), (dim=2|id=616|"k"))

    we can combine indices i and k by contracting with the combiner:

    CT = C * T

    Printing out the indices of the new ITensor CT we can see that it has only two indices:

    @show inds(CT)
    ((dim=8|id=903|"c"), (dim=3|id=46|"j"))

    The first is the newly made combined Index, which was made for us by the combiner function and the second is the j Index of T which was not part of the combining process. To access the combined Index you can call the combinedind function on the combiner:

    ci = combinedind(C)
    (dim=8|id=903|"c")

    We can visualize all of the steps above as follows:

    Combining is not limited to two indices and you can combine any number of indices, in any order, using a combiner.

    To undo the combining process and uncombine the Index c back into i,k, just contract with the conjugate of the combiner ITensor dag(C).

    UT = dag(C) * CT
    -@show inds(UT)
    ((dim=4|id=604|"i"), (dim=2|id=616|"k"), (dim=3|id=46|"j"))

    Write and Read an ITensor to Disk with HDF5

    Info

    Make sure to install the HDF5 package to use this feature. (Run julia> ] add HDF5 in the Julia REPL console.)

    Saving ITensors to disk can be very useful. For example, you might encounter a bug in your own code, and by reading the ITensors involved from disk you can shortcut the process of running a lengthy algorithm over many times to reproduce the bug. Or you can save the output of an expensive calculation, such as a DMRG calculation, and use it as a starting point for multiple follow-up calculations such as computing time-dependent properties.

    ITensors can be written to files using the HDF5 format. HDF5 offers many benefits such as being portable across different machine types, and offers a standard interface across various libraries and languages.

    Writing an ITensor to an HDF5 File

    Let's say you have an ITensor T which you have made or obtained from a calculation. To write it to an HDF5 file named "myfile.h5" you can use the following pattern:

    using HDF5
    +@show inds(T)
    ((dim=4|id=192|"i"), (dim=3|id=838|"j"), (dim=2|id=133|"k"))

    we can combine indices i and k by contracting with the combiner:

    CT = C * T

    Printing out the indices of the new ITensor CT we can see that it has only two indices:

    @show inds(CT)
    ((dim=8|id=496|"c"), (dim=3|id=838|"j"))

    The first is the newly made combined Index, which was made for us by the combiner function and the second is the j Index of T which was not part of the combining process. To access the combined Index you can call the combinedind function on the combiner:

    ci = combinedind(C)
    (dim=8|id=496|"c")

    We can visualize all of the steps above as follows:

    Combining is not limited to two indices and you can combine any number of indices, in any order, using a combiner.

    To undo the combining process and uncombine the Index c back into i,k, just contract with the conjugate of the combiner ITensor dag(C).

    UT = dag(C) * CT
    +@show inds(UT)
    ((dim=4|id=192|"i"), (dim=2|id=133|"k"), (dim=3|id=838|"j"))

    Write and Read an ITensor to Disk with HDF5

    Info

    Make sure to install the HDF5 package to use this feature. (Run julia> ] add HDF5 in the Julia REPL console.)

    Saving ITensors to disk can be very useful. For example, you might encounter a bug in your own code, and by reading the ITensors involved from disk you can shortcut the process of running a lengthy algorithm over many times to reproduce the bug. Or you can save the output of an expensive calculation, such as a DMRG calculation, and use it as a starting point for multiple follow-up calculations such as computing time-dependent properties.

    ITensors can be written to files using the HDF5 format. HDF5 offers many benefits such as being portable across different machine types, and offers a standard interface across various libraries and languages.

    Writing an ITensor to an HDF5 File

    Let's say you have an ITensor T which you have made or obtained from a calculation. To write it to an HDF5 file named "myfile.h5" you can use the following pattern:

    using HDF5
     f = h5open("myfile.h5","w")
     write(f,"T",T)
     close(f)

    Above, the string "T" can actually be any string you want such as "ITensor T" or "Result Tensor" and doesn't have to have the same name as the reference T. Closing the file f is optional and you can also write other objects to the same file before closing it.

    Reading an ITensor from an HDF5 File

    Say you have an HDF5 file "myfile.h5" which contains an ITensor stored as a dataset with the name "T". (Which would be the situation if you wrote it as in the example above.) To read this ITensor back from the HDF5 file, use the following pattern:

    using HDF5
     f = h5open("myfile.h5","r")
     T = read(f,"T",ITensor)
    -close(f)

    Note the ITensor argument to the read function, which tells Julia which read function to call and how to interpret the data stored in the HDF5 dataset named "T". In the future we might lift the requirement of providing the type and have it be detected automatically from the data stored in the file.

    +close(f)

    Note the ITensor argument to the read function, which tells Julia which read function to call and how to interpret the data stored in the HDF5 dataset named "T". In the future we might lift the requirement of providing the type and have it be detected automatically from the data stored in the file.

    diff --git a/previews/PR1514/examples/MPSandMPO.html b/previews/PR1514/examples/MPSandMPO.html index 7bb7ba14d2..56cb154099 100644 --- a/previews/PR1514/examples/MPSandMPO.html +++ b/previews/PR1514/examples/MPSandMPO.html @@ -89,4 +89,4 @@ H = MPO(os,sites) # Compute <psi|H|psi> -energy_psi = inner(psi',H,psi)

    Note the MPS argument to the read function, which tells Julia which read function to call and how to interpret the data stored in the HDF5 dataset named "psi". In the future we might lift the requirement of providing the type and have it be detected automatically from the data stored in the file.

    Writing and Reading MPOs

    To write or read MPOs to or from HDF5 files, just follow the examples above but use the type MPO when reading an MPO from the file instead of the type MPS.

    +energy_psi = inner(psi',H,psi)

    Note the MPS argument to the read function, which tells Julia which read function to call and how to interpret the data stored in the HDF5 dataset named "psi". In the future we might lift the requirement of providing the type and have it be detected automatically from the data stored in the file.

    Writing and Reading MPOs

    To write or read MPOs to or from HDF5 files, just follow the examples above but use the type MPO when reading an MPO from the file instead of the type MPS.

    diff --git a/previews/PR1514/examples/Physics.html b/previews/PR1514/examples/Physics.html index 172fd735ee..25394f40ab 100644 --- a/previews/PR1514/examples/Physics.html +++ b/previews/PR1514/examples/Physics.html @@ -102,8 +102,8 @@ 0 0 0 -3/2]

    As you can see, the function is passed two objects: an OpName and a SiteType. The strings "Sz" and "S=3/2" are also part of the type of these objects, and have the meaning of which operator name we are defining and which site type these operators are defined for.

    The body of this overload of ITensors.op constructs and returns a Julia matrix which gives the matrix elements of the operator we are defining.

    Once this function is defined, and if you have an Index such as

    s = Index(4,"S=3/2")

    then, for example, you can get the "Sz" operator for this Index and print it out by doing:

    using ITensors, ITensorMPS
     Sz = op("Sz",s)
     println(Sz)
    ITensor ord=2
    -Dim 1: (dim=4|id=966|"S=3/2")'
    -Dim 2: (dim=4|id=966|"S=3/2")
    +Dim 1: (dim=4|id=102|"S=3/2")'
    +Dim 2: (dim=4|id=102|"S=3/2")
     NDTensors.Dense{Float64, Vector{Float64}}
      4×4
      1.5  0.0   0.0   0.0
    @@ -145,4 +145,4 @@
     

    Now let's look at each part of the code above.

    The space function

    In the previous code example above, we discussed that the function space tells the ITensor library the basic information about how to construct an Index associated with a special Index tag, in this case the tag "S=3/2". As in that code formula, if the user does not request that quantum numbers be included (the case conserve_qns=false) then all that the space function returns is the number 4, indicating that a "S=3/2" Index should be of dimension 4.

    But if the conserve_qns keyword argument gets set to true, the space function we defined above returns an array of QN=>Int pairs. (The notation a=>b in Julia constructs a Pair object.) Each pair in the array denotes a subspace. The QN part of each pair says what quantum number the subspace has, and the integer following it indicates the dimension of the subspace.

    After defining the space function this way, you can write code like:

    using ITensors, ITensorMPS
     s = siteind("S=3/2"; conserve_qns=true)

    to obtain a single "S=3/2" Index which carries quantum number information. The siteind function built into ITensor relies on your custom space function to ask how to construct a "S=3/2" Index but also includes some other Index tags which are conventional for all site indices.

    You can now also call code like:

    using ITensors, ITensorMPS
     N = 100
    -sites = siteinds("S=3/2",N; conserve_qns=true)

    to obtain an array of N "S=3/2" indices which carry quantum numbers.

    The op Function in the Quantum Number Case

    Note that the op function overloads are exactly the same as for the more basic case of defining an "S=3/2" Index type that does not carry quantum numbers. There is no need to upgrade any of the op functions for the QN-conserving case. The reason is that all QN, block-sparse information about an ITensor is deduced from the indices of the tensor, and setting elements of such tensors does not require any other special code.

    However, only operators which have a well-defined QN flux–-meaning they always change the quantum number of a state they act on by a well-defined amount–-can be used in practice in the case of QN conservation. Attempting to build an operator, or any ITensor, without a well-defined QN flux out of QN-conserving indices will result in a run time error. An example of an operator that would lead to such an error would be the "Sx" spin operator since it alternately increases $S^z$ or decreases $S^z$ depending on the state it acts on, thus it does not have a well-defined QN flux. But it is perfectly fine to define an op overload for the "Sx" operator and to make this operator when working with dense, non-QN-conserving ITensors or when $S^z$ is not conserved.

    +sites = siteinds("S=3/2",N; conserve_qns=true)

    to obtain an array of N "S=3/2" indices which carry quantum numbers.

    The op Function in the Quantum Number Case

    Note that the op function overloads are exactly the same as for the more basic case of defining an "S=3/2" Index type that does not carry quantum numbers. There is no need to upgrade any of the op functions for the QN-conserving case. The reason is that all QN, block-sparse information about an ITensor is deduced from the indices of the tensor, and setting elements of such tensors does not require any other special code.

    However, only operators which have a well-defined QN flux–-meaning they always change the quantum number of a state they act on by a well-defined amount–-can be used in practice in the case of QN conservation. Attempting to build an operator, or any ITensor, without a well-defined QN flux out of QN-conserving indices will result in a run time error. An example of an operator that would lead to such an error would be the "Sx" spin operator since it alternately increases $S^z$ or decreases $S^z$ depending on the state it acts on, thus it does not have a well-defined QN flux. But it is perfectly fine to define an op overload for the "Sx" operator and to make this operator when working with dense, non-QN-conserving ITensors or when $S^z$ is not conserved.

    diff --git a/previews/PR1514/faq/DMRG.html b/previews/PR1514/faq/DMRG.html index 9ab4f9007e..2b8833e65f 100644 --- a/previews/PR1514/faq/DMRG.html +++ b/previews/PR1514/faq/DMRG.html @@ -15,4 +15,4 @@ end hterms += "Sz",1,"Sz",N # term 'wrapping' around the ring -H = MPO(hterms,sites)

    For two-dimensional DMRG calculations, where the most common approach is to use periodic boundary conditions in the y-direction only, and not in the x-direction, you do a similar step in making your OpSum input to ITensor DMRG: you include terms wrapping around the periodic cylinder in the y direction but not in the x direction.

    However, fully periodic boundary conditions are only recommended for small systems when absolutely needed, and in general are not recommended. For a longer discussion of alternatives to using fully periodic boundaries, see the next section below.

    The reason fully periodic boundary conditions (periodic in x in 1D, and periodic in both x and y in 2D) are not recommended in general is that the DMRG algorithm, as we are defining it here, optimizes an open-boundary MPS. So if you input a periodic-boundary Hamiltonian, there is a kind of "mismatch" that happens where you can still get the correct answer, but it requires much more resources (a larger bond dimension and more sweeps) to get good accuracy. There has been some research into "truly" periodic DMRG, [Pippan] that is DMRG that optimizes an MPS with a ring-like topology, but it is not widely used, is still an open area of algorithm development, and is not currently available in ITensor.

    What boundary conditions should I choose: open, periodic, or infinite?

    One of the weaknesses of the density matrix renormalization group (DMRG), and its time-dependent or finite-temperature extensions, is that it works poorly with periodic boundary conditions. This stems from the fact that conventional DMRG optimizes over open-boundary matrix product state (MPS) wavefunctions whether or not the Hamiltonian includes periodic interactions.

    But this begs the question, when are periodic boundary conditions (PBC) really needed? DMRG offers some compelling alternatives to PBC:

    However, there are a handful of cases where PBC remains preferable despite the extra overhead. A few such cases are:

    (Note that in the remaining discussion, by PBC I mean fully periodic boundary conditions in all directions. For the case of DMRG applied to quasi-two-dimensional systems, it remains a good practice to use periodic boundaries in the shorter direction, while still using open (or infinite) boundaries in the longer direction along the DMRG/MPS path.)

    Below I discuss more about the problems with using PBC, as well as some misconceptions about when PBC seems necessary even though there are better alternatives.

    Drawbacks of Periodic Boundary Conditions

    Periodic boundary conditions are straightforward to implement in conventional DMRG. The simplest approach is to include a "long bond" directly connecting site 1 to site N in the Hamiltonian. However this naive approach has a major drawback: if open-boundary DMRG achieves a given accuracy when keeping $m$ states (bond dimension of size $m$), then to reach the same accuracy with PBC one must keep closer to $m^2$ states! The reason is that now every bond of the MPS not only carries local entanglement as with OBC, but also the entanglement between the first and last sites. (There is an alternative DMRG algorithm[Pippan] for periodic systems which may have better scaling than the above approach but has not been widely applied and tested, as far as I am aware, especially for 2D or critical systems .)

    The change in scaling from $m$ to $m^2$ is a severe problem. For example, many gapped one-dimensional systems only require about $m=100$ to reach good accuracy (truncation errors of less than 1E-9 or so). To reach the same accuracy with naive PBC would then require using 10,000 states, which can easily fill the RAM of a typical desktop computer for a large enough system, not to mention the extra time needed to work with larger matrices.

    But poor scaling is not the only drawback of PBC. Systems that exhibit spontaneous symmetry breaking are simple to work with under OBC, where one has the additional freedom of applying edge pinning terms to drive the bulk into a specific symmetry sector. Using edge pinning reduces the bulk entanglement and makes measuring order parameters straightforward. Similarly one can use infinite DMRG to directly observe symmetry breaking effects.

    But under PBC, order parameters remain equal to zero and can only be accessed through correlation functions. Though using correlation functions is often presented as the "standard" or "correct" approach, such reasoning pre-supposes that PBC is the best choice. Recent work in the quantum Monte Carlo community demonstrates that open boundaries with pinning fields can actually be a superior approach.[Assaad]

    Cases Where Periodic BC Seems Necessary, But Open/Infinite BC Can be Better

    Below are some cases where periodic boundary conditions seem to be necessary at a first glance. But in many of these cases, not only can open or infinite boundaries be just as successful, they can even be the better choice.

    In conclusion, consider carefully whether you really need to use periodic boundary conditions, as they impose a steep computational cost within DMRG. Periodic BC can actually be worse for the very types of measurements where they are often presented as the best or "standard" choice. Many of the issues periodic boundaries circumvent can be avoided more elegantly by using infinite DMRG, or when that is not applicable, by using open boundary conditions with sufficient care.

    +H = MPO(hterms,sites)

    For two-dimensional DMRG calculations, where the most common approach is to use periodic boundary conditions in the y-direction only, and not in the x-direction, you do a similar step in making your OpSum input to ITensor DMRG: you include terms wrapping around the periodic cylinder in the y direction but not in the x direction.

    However, fully periodic boundary conditions are only recommended for small systems when absolutely needed, and in general are not recommended. For a longer discussion of alternatives to using fully periodic boundaries, see the next section below.

    The reason fully periodic boundary conditions (periodic in x in 1D, and periodic in both x and y in 2D) are not recommended in general is that the DMRG algorithm, as we are defining it here, optimizes an open-boundary MPS. So if you input a periodic-boundary Hamiltonian, there is a kind of "mismatch" that happens where you can still get the correct answer, but it requires much more resources (a larger bond dimension and more sweeps) to get good accuracy. There has been some research into "truly" periodic DMRG, [Pippan] that is DMRG that optimizes an MPS with a ring-like topology, but it is not widely used, is still an open area of algorithm development, and is not currently available in ITensor.

    What boundary conditions should I choose: open, periodic, or infinite?

    One of the weaknesses of the density matrix renormalization group (DMRG), and its time-dependent or finite-temperature extensions, is that it works poorly with periodic boundary conditions. This stems from the fact that conventional DMRG optimizes over open-boundary matrix product state (MPS) wavefunctions whether or not the Hamiltonian includes periodic interactions.

    But this begs the question, when are periodic boundary conditions (PBC) really needed? DMRG offers some compelling alternatives to PBC:

    However, there are a handful of cases where PBC remains preferable despite the extra overhead. A few such cases are:

    (Note that in the remaining discussion, by PBC I mean fully periodic boundary conditions in all directions. For the case of DMRG applied to quasi-two-dimensional systems, it remains a good practice to use periodic boundaries in the shorter direction, while still using open (or infinite) boundaries in the longer direction along the DMRG/MPS path.)

    Below I discuss more about the problems with using PBC, as well as some misconceptions about when PBC seems necessary even though there are better alternatives.

    Drawbacks of Periodic Boundary Conditions

    Periodic boundary conditions are straightforward to implement in conventional DMRG. The simplest approach is to include a "long bond" directly connecting site 1 to site N in the Hamiltonian. However this naive approach has a major drawback: if open-boundary DMRG achieves a given accuracy when keeping $m$ states (bond dimension of size $m$), then to reach the same accuracy with PBC one must keep closer to $m^2$ states! The reason is that now every bond of the MPS not only carries local entanglement as with OBC, but also the entanglement between the first and last sites. (There is an alternative DMRG algorithm[Pippan] for periodic systems which may have better scaling than the above approach but has not been widely applied and tested, as far as I am aware, especially for 2D or critical systems .)

    The change in scaling from $m$ to $m^2$ is a severe problem. For example, many gapped one-dimensional systems only require about $m=100$ to reach good accuracy (truncation errors of less than 1E-9 or so). To reach the same accuracy with naive PBC would then require using 10,000 states, which can easily fill the RAM of a typical desktop computer for a large enough system, not to mention the extra time needed to work with larger matrices.

    But poor scaling is not the only drawback of PBC. Systems that exhibit spontaneous symmetry breaking are simple to work with under OBC, where one has the additional freedom of applying edge pinning terms to drive the bulk into a specific symmetry sector. Using edge pinning reduces the bulk entanglement and makes measuring order parameters straightforward. Similarly one can use infinite DMRG to directly observe symmetry breaking effects.

    But under PBC, order parameters remain equal to zero and can only be accessed through correlation functions. Though using correlation functions is often presented as the "standard" or "correct" approach, such reasoning pre-supposes that PBC is the best choice. Recent work in the quantum Monte Carlo community demonstrates that open boundaries with pinning fields can actually be a superior approach.[Assaad]

    Cases Where Periodic BC Seems Necessary, But Open/Infinite BC Can be Better

    Below are some cases where periodic boundary conditions seem to be necessary at a first glance. But in many of these cases, not only can open or infinite boundaries be just as successful, they can even be the better choice.

    In conclusion, consider carefully whether you really need to use periodic boundary conditions, as they impose a steep computational cost within DMRG. Periodic BC can actually be worse for the very types of measurements where they are often presented as the best or "standard" choice. Many of the issues periodic boundaries circumvent can be avoided more elegantly by using infinite DMRG, or when that is not applicable, by using open boundary conditions with sufficient care.

    diff --git a/previews/PR1514/faq/Development.html b/previews/PR1514/faq/Development.html index 4f27089def..53ac6cfcda 100644 --- a/previews/PR1514/faq/Development.html +++ b/previews/PR1514/faq/Development.html @@ -1,2 +1,2 @@ -ITensor Development FAQs · ITensors.jl

    ITensor Development Frequently Asked Questions

    What are the steps to contribute code to ITensor?

    1. Please contact us (support at itensor.org) if you are planning to submit a major contribution (more than a few lines of code, say). If so, we would like to discuss your plan and design before you spend significant time on it, to increase the chances we will merge your pull request.

    2. Fork the ITensors.jl Github repo, create a new branch and make changes (commits) on that branch. ITensor imposes code formatting for contributions. Please run using JuliaFormatter; format(".") in the project directory to ensure formatting. As an alternative you may also use pre-commit. Install pre-commit with e.g. pip install pre-commit, then run pre-commit install in the project directory in order for pre-commit to run automatically before any commit.

    3. Run the ITensor unit tests by going into the test/ folder and running julia runtests.jl. To run individual test scripts, start a Julia REPL (interactive terminal) session and include each script, such as include("itensor.jl").

    4. Push your new branch and changes to your forked repo. Github will give you the option to make a pull request (PR) out of your branch that will be submitted to us, and which you can view under the list of ITensors.jl pull requests. If your PR's tests pass and we approve your changes, we will merge it or ask you to merge it. If you merge your PR, please use the Squash and Merge option. We may also ask you to make more changes to bring your PR in line with our design goals or technical requirements.

    +ITensor Development FAQs · ITensors.jl

    ITensor Development Frequently Asked Questions

    What are the steps to contribute code to ITensor?

    1. Please contact us (support at itensor.org) if you are planning to submit a major contribution (more than a few lines of code, say). If so, we would like to discuss your plan and design before you spend significant time on it, to increase the chances we will merge your pull request.

    2. Fork the ITensors.jl Github repo, create a new branch and make changes (commits) on that branch. ITensor imposes code formatting for contributions. Please run using JuliaFormatter; format(".") in the project directory to ensure formatting. As an alternative you may also use pre-commit. Install pre-commit with e.g. pip install pre-commit, then run pre-commit install in the project directory in order for pre-commit to run automatically before any commit.

    3. Run the ITensor unit tests by going into the test/ folder and running julia runtests.jl. To run individual test scripts, start a Julia REPL (interactive terminal) session and include each script, such as include("itensor.jl").

    4. Push your new branch and changes to your forked repo. Github will give you the option to make a pull request (PR) out of your branch that will be submitted to us, and which you can view under the list of ITensors.jl pull requests. If your PR's tests pass and we approve your changes, we will merge it or ask you to merge it. If you merge your PR, please use the Squash and Merge option. We may also ask you to make more changes to bring your PR in line with our design goals or technical requirements.

    diff --git a/previews/PR1514/faq/HPC.html b/previews/PR1514/faq/HPC.html index b2d45bd985..71fa8ebcab 100644 --- a/previews/PR1514/faq/HPC.html +++ b/previews/PR1514/faq/HPC.html @@ -1,2 +1,2 @@ -High-Performance Computing FAQs · ITensors.jl

    High Performance Computing (HPC) Frequently Asked Questions

    My code is using a lot of RAM - what can I do about this?

    Tensor network algorithms can often use a large amount of RAM. On top of this essential fact, the Julia programming languge is "garbage collected" which means that unused memory isn't given back to the operating system right away, but only when the Julia runtime dynamically reclaims it. When your code allocates memory very rapidly, this can lead to high memory usage overall.

    Fortunately there are various steps you can take to keep the memory usage of your code under control.

    1. Avoid Repeatedly Allocating, Especially in Fast or "Hot" Loops

    More memory gets used whenever your code "allocates", which happens most commonly when you use dynamic storage types like Vector and Matrix. If you have a code pattern where you allocate or resize an array or vector inside a 'hot' loop, meaning a loop that iterates quickly very many times, the memory from the previous allocations may pile up very quickly before the next garbage collector run.

    To avoid this, allocate the array once before the loop begins if possible, then overwrite its contents during each iteration. More generally, try as much as possible to estimate the sizes of dynamic resources ahead of time. Or do one allocation that creates a large enough "workspace" that dynamic algorithms can reuse part of without reallocating the whole workspace (i.e. making a large array once then using portions of it when smaller arrays are needed).

    2. Use the --heap-size-hint Flag

    A simple step you can take to help with overall memory usage is to pass the --heap-size-hint flag to the Julia program when you start it. For example, you can call Julia as:

    julia --heap-size-hint=60G

    When you pass this heap size, Julia will try to keep the memory usage at or below this value if possible.

    In cases where this does not work, your code simply may be allocating too much memory. Be sure not to allocate over and over again inside of "hot" loops which execute many times.

    Another possibility is that you are simply working with a tensor network with large bond dimensions, which may fundamentally use a lot of memory. In those cases, you can try to use features such as "write to disk mode" of the ITensor DMRG code or other related techniques. (See the write_when_maxdim_exceeds keyword of the ITensor dmrg function.)

    3. In Rare Case, Force a Garbage Collection Run

    In some rare cases, such as when your code cannot be optimized to avoid any more allocations or when the --heap-size-hint provided above is not affecting the behavior of the Julia garbage collector, you can force the garbage collector (GC) to run at a specific point in your code by calling:

    GC.gc()

    Alternatively, you can call GC.gc(true) to force a "full run" rather than just collecting a more 'young' subset of previous allocations.

    While this approach works well to reduce memory usage, it can have the unfortunate downside of slowing down your code each time the garbage collector runs, which can be especially harmful to multithreaded or parallel algorithms. Therefore, if this approach must be used try calling GC.gc() as infrequently as possible and ideally only in the outermost functions and loops of your code (highest levels of your code).

    Can Julia Be Used to Perform Parallel, Distributed Calculations on Large Clusters?

    Yes. The Julia ecosystem offers multiple approaches to parallel computing across multiple machines including on large HPC clusters and including GPU resources.

    For an overall view of some of these options, the Julia on HPC Clusters website is a good resource.

    Some of the leading approaches to parallelism in Julia are:

    • MPI, through the MPI.jl package. Has the advantage of optionally using an MPI backend that is optimized for a particular cluster and possibly using fast interconnects like Infiniband.
    • Dagger, a framework for parallel computing across all kinds of resources, like CPUs and GPUs, and across multiple threads and multiple servers.
    • Distributed. Part of the base Julia library, giving tools to perform calculations distributed across multiple machines.

    Does My Cluster Admin Have to Install Julia for Me? What are the Best Practices for Installing Julia on Clusters?

    The most common approach to installing and using Julia on clusters is for users to install their own Julia binary and dependencies, which is quite easy to do. However, for certain libraries like MPI.jl, there may be MPI backends that are preferred by the cluster administrator. Fortunately, it is possible for admins to set global defaults for such backends and other library preferences.

    For more information on best practices for installing Julia on clusters, see the Julia on HPC Clusters website.

    +High-Performance Computing FAQs · ITensors.jl

    High Performance Computing (HPC) Frequently Asked Questions

    My code is using a lot of RAM - what can I do about this?

    Tensor network algorithms can often use a large amount of RAM. On top of this essential fact, the Julia programming languge is "garbage collected" which means that unused memory isn't given back to the operating system right away, but only when the Julia runtime dynamically reclaims it. When your code allocates memory very rapidly, this can lead to high memory usage overall.

    Fortunately there are various steps you can take to keep the memory usage of your code under control.

    1. Avoid Repeatedly Allocating, Especially in Fast or "Hot" Loops

    More memory gets used whenever your code "allocates", which happens most commonly when you use dynamic storage types like Vector and Matrix. If you have a code pattern where you allocate or resize an array or vector inside a 'hot' loop, meaning a loop that iterates quickly very many times, the memory from the previous allocations may pile up very quickly before the next garbage collector run.

    To avoid this, allocate the array once before the loop begins if possible, then overwrite its contents during each iteration. More generally, try as much as possible to estimate the sizes of dynamic resources ahead of time. Or do one allocation that creates a large enough "workspace" that dynamic algorithms can reuse part of without reallocating the whole workspace (i.e. making a large array once then using portions of it when smaller arrays are needed).

    2. Use the --heap-size-hint Flag

    A simple step you can take to help with overall memory usage is to pass the --heap-size-hint flag to the Julia program when you start it. For example, you can call Julia as:

    julia --heap-size-hint=60G

    When you pass this heap size, Julia will try to keep the memory usage at or below this value if possible.

    In cases where this does not work, your code simply may be allocating too much memory. Be sure not to allocate over and over again inside of "hot" loops which execute many times.

    Another possibility is that you are simply working with a tensor network with large bond dimensions, which may fundamentally use a lot of memory. In those cases, you can try to use features such as "write to disk mode" of the ITensor DMRG code or other related techniques. (See the write_when_maxdim_exceeds keyword of the ITensor dmrg function.)

    3. In Rare Case, Force a Garbage Collection Run

    In some rare cases, such as when your code cannot be optimized to avoid any more allocations or when the --heap-size-hint provided above is not affecting the behavior of the Julia garbage collector, you can force the garbage collector (GC) to run at a specific point in your code by calling:

    GC.gc()

    Alternatively, you can call GC.gc(true) to force a "full run" rather than just collecting a more 'young' subset of previous allocations.

    While this approach works well to reduce memory usage, it can have the unfortunate downside of slowing down your code each time the garbage collector runs, which can be especially harmful to multithreaded or parallel algorithms. Therefore, if this approach must be used try calling GC.gc() as infrequently as possible and ideally only in the outermost functions and loops of your code (highest levels of your code).

    Can Julia Be Used to Perform Parallel, Distributed Calculations on Large Clusters?

    Yes. The Julia ecosystem offers multiple approaches to parallel computing across multiple machines including on large HPC clusters and including GPU resources.

    For an overall view of some of these options, the Julia on HPC Clusters website is a good resource.

    Some of the leading approaches to parallelism in Julia are:

    • MPI, through the MPI.jl package. Has the advantage of optionally using an MPI backend that is optimized for a particular cluster and possibly using fast interconnects like Infiniband.
    • Dagger, a framework for parallel computing across all kinds of resources, like CPUs and GPUs, and across multiple threads and multiple servers.
    • Distributed. Part of the base Julia library, giving tools to perform calculations distributed across multiple machines.

    Does My Cluster Admin Have to Install Julia for Me? What are the Best Practices for Installing Julia on Clusters?

    The most common approach to installing and using Julia on clusters is for users to install their own Julia binary and dependencies, which is quite easy to do. However, for certain libraries like MPI.jl, there may be MPI backends that are preferred by the cluster administrator. Fortunately, it is possible for admins to set global defaults for such backends and other library preferences.

    For more information on best practices for installing Julia on clusters, see the Julia on HPC Clusters website.

    diff --git a/previews/PR1514/faq/JuliaAndCpp.html b/previews/PR1514/faq/JuliaAndCpp.html index 61c41df215..5edebf007d 100644 --- a/previews/PR1514/faq/JuliaAndCpp.html +++ b/previews/PR1514/faq/JuliaAndCpp.html @@ -1,2 +1,2 @@ -Programming Language (Julia, C++, ...) FAQs · ITensors.jl

    Programming Language (Julia, C++) Frequently Asked Questions

    Should I use the Julia or C++ version of ITensor?

    We recommend the Julia version of ITensor for most people, because:

    • Julia ITensor has more and newer features than C++ ITensor, and we are developing it more rapidly
    • Julia is a more productive language than C++ with more built-in features, such as linear algebra, iteration tools, etc.
    • Julia is a compiled language with performance rivaling C++ (see next question below for a longer discussion)
    • Julia has a rich ecosystem with a package manager, many well-designed libraries, and helpful tutorials

    Even if Julia is not available by default on your computer cluster, it is easy to set up your own local install of Julia on a cluster.

    However, some good reasons to use the C++ version of ITensor are:

    • using ITensor within existing C++ codes
    • you already have expertise in C++ programming
    • multithreading support in C++, such as with OpenMP, offer certain sophisticated features compared to Julia multithreading (though Julia's support for multithreading has other benefits such as composability and is rapidly improving)
    • you need other specific features of C++, such as control over memory management or instant start-up times

    Which is faster: Julia or C++ ?

    Julia and C++ offer about the same performance.

    Each language gets compiled to optimized assembly code and offer arrays and containers which can efficiently stored and iterated. Well-written Julia code can be even faster than comparable C++ codes in many cases.

    The longer answer is of course that it depends:

    • Julia is a more productive language than C++, with many highly-optimized libraries for numerical computing tasks, and excellent tools for profiling and benchmarking. These features help significantly to tune Julia codes for optimal performance.
    • C++ offers much more fine-grained control over memory management, which can enhance performance in certain applications and control memory usage.
    • Julia codes can slow down significantly during refactoring or when introducing new code if certain best practices are not followed. The most important of these is writing type-stable code. For more details see the Performance Tips section of the Julia documentation.
    • C++ applications start instantly, while Julia codes can be slow to start. However, if this start-up time is subtracted, the rest of the time of running a Julia application is similar to C++.

    Why did you choose Julia over Python for ITensor?

    Julia offers much better performance than Python, while still having nearly all of Python's benefits. One consequence is that ITensor can be written purely in Julia, whereas to write high-performance Python libraries it is necessary to implement many parts in C or C++ (the "two-language problem").

    The main reasons Julia codes can easily outperform Python codes are:

    1. Julia is a (just-in-time) compiled language with functions specialized for the types of the arguments passed to them
    2. Julia arrays and containers are specialized to the types they contain, and perform similarly to C or C++ arrays when all elements have the same type
    3. Julia has sophisticated support for multithreading while Python has significant problems with multithreading

    Of course there are some drawbacks of Julia compared to Python, including a less mature ecosystem of libraries (though it is simple to call Python libraries from Julia using PyCall), and less widespread adoption.

    Is Julia ITensor a wrapper around the C++ version?

    No. The Julia version of ITensor is a complete, ground-up port of the ITensor library to the Julia language and is written 100% in Julia.

    +Programming Language (Julia, C++, ...) FAQs · ITensors.jl

    Programming Language (Julia, C++) Frequently Asked Questions

    Should I use the Julia or C++ version of ITensor?

    We recommend the Julia version of ITensor for most people, because:

    • Julia ITensor has more and newer features than C++ ITensor, and we are developing it more rapidly
    • Julia is a more productive language than C++ with more built-in features, such as linear algebra, iteration tools, etc.
    • Julia is a compiled language with performance rivaling C++ (see next question below for a longer discussion)
    • Julia has a rich ecosystem with a package manager, many well-designed libraries, and helpful tutorials

    Even if Julia is not available by default on your computer cluster, it is easy to set up your own local install of Julia on a cluster.

    However, some good reasons to use the C++ version of ITensor are:

    • using ITensor within existing C++ codes
    • you already have expertise in C++ programming
    • multithreading support in C++, such as with OpenMP, offer certain sophisticated features compared to Julia multithreading (though Julia's support for multithreading has other benefits such as composability and is rapidly improving)
    • you need other specific features of C++, such as control over memory management or instant start-up times

    Which is faster: Julia or C++ ?

    Julia and C++ offer about the same performance.

    Each language gets compiled to optimized assembly code and offer arrays and containers which can efficiently stored and iterated. Well-written Julia code can be even faster than comparable C++ codes in many cases.

    The longer answer is of course that it depends:

    • Julia is a more productive language than C++, with many highly-optimized libraries for numerical computing tasks, and excellent tools for profiling and benchmarking. These features help significantly to tune Julia codes for optimal performance.
    • C++ offers much more fine-grained control over memory management, which can enhance performance in certain applications and control memory usage.
    • Julia codes can slow down significantly during refactoring or when introducing new code if certain best practices are not followed. The most important of these is writing type-stable code. For more details see the Performance Tips section of the Julia documentation.
    • C++ applications start instantly, while Julia codes can be slow to start. However, if this start-up time is subtracted, the rest of the time of running a Julia application is similar to C++.

    Why did you choose Julia over Python for ITensor?

    Julia offers much better performance than Python, while still having nearly all of Python's benefits. One consequence is that ITensor can be written purely in Julia, whereas to write high-performance Python libraries it is necessary to implement many parts in C or C++ (the "two-language problem").

    The main reasons Julia codes can easily outperform Python codes are:

    1. Julia is a (just-in-time) compiled language with functions specialized for the types of the arguments passed to them
    2. Julia arrays and containers are specialized to the types they contain, and perform similarly to C or C++ arrays when all elements have the same type
    3. Julia has sophisticated support for multithreading while Python has significant problems with multithreading

    Of course there are some drawbacks of Julia compared to Python, including a less mature ecosystem of libraries (though it is simple to call Python libraries from Julia using PyCall), and less widespread adoption.

    Is Julia ITensor a wrapper around the C++ version?

    No. The Julia version of ITensor is a complete, ground-up port of the ITensor library to the Julia language and is written 100% in Julia.

    diff --git a/previews/PR1514/faq/JuliaPkg.html b/previews/PR1514/faq/JuliaPkg.html index 7f3af10ff1..d3738ac08d 100644 --- a/previews/PR1514/faq/JuliaPkg.html +++ b/previews/PR1514/faq/JuliaPkg.html @@ -1,3 +1,3 @@ Julia Package Manager FAQs · ITensors.jl

    Julia Package Manager Frequently Asked Questions

    What if I can't upgrade ITensors.jl to the latest version?

    Sometimes you may find that doing ] update ITensors or equivalently doing ] up ITensors within Julia package manager mode doesn't result in the ITensors package actually being upgraded. You may see that the current version you have remains stuck to a version that is lower than the latest one which you can check here.

    What is most likely going on is that you have other packages installed which are blocking ITensors from being updated.

    To get more information into which packages may be doing this, and what versions they are requiring, you can do the following. First look up the latest version of ITensors.jl. Let's say for this example that it is v0.3.0.

    Next, input the following command while in package manager mode:

    julia> ]
    -pkg> add ITensors@v0.3.0

    If the package manager cannot update to this version, it will list all of the other packages that are blocking this from happening and give information about why. To go into a little more depth, each package has a compatibility or "compat" entry in its Project.toml file which says which versions of the ITensors package it is compatible with. If these versions do not include the latest one, perhaps because the package has not been updated, then it can block the ITensors package from being updated on your system.

    Generally the solution is to just update each of these packages, then try again to update ITensors. If that does not work, then check the following

    • Are any of the blocking packages in "dev mode" meaning you called dev PackageName on them in the past? Try doing free PackageName if so to bring them out of dev mode.
    • Are any of the blocking packages unregistered packages that were installed through a GitHub repo link? If so, you may need to do something like add https://github.com/Org/PackageName#main to force update that package to the latest code available on its main branch.

    If you still can't get the ITensors package update, feel free to post a question or contact us for help.

    +pkg> add ITensors@v0.3.0

    If the package manager cannot update to this version, it will list all of the other packages that are blocking this from happening and give information about why. To go into a little more depth, each package has a compatibility or "compat" entry in its Project.toml file which says which versions of the ITensors package it is compatible with. If these versions do not include the latest one, perhaps because the package has not been updated, then it can block the ITensors package from being updated on your system.

    Generally the solution is to just update each of these packages, then try again to update ITensors. If that does not work, then check the following

    If you still can't get the ITensors package update, feel free to post a question or contact us for help.

    diff --git a/previews/PR1514/faq/QN.html b/previews/PR1514/faq/QN.html index c8a9bf86b0..d7da541094 100644 --- a/previews/PR1514/faq/QN.html +++ b/previews/PR1514/faq/QN.html @@ -1,4 +1,4 @@ Quantum Number (QN) FAQs · ITensors.jl

    Quantum Number Frequently Asked Questions

    Can I mix different types of quantum numbers within the same system?

    Yes, you can freely mix quantum numbers (QNs) of different types. For example, you can make the sites of your systems alternate between sites carrying spin "Sz" QNs and fermion sites carrying particle number "Nf" QNs. The QNs will not mix with each other and will separately be conserved to the original value you set for your initial wavefunction.

    How can I separately conserve QNs which have the same name?

    If you have two physically distinct types of sites, such as "Qudit" sites, but which carry identically named QNs called "Number", and you want the qudit number to be separately conserved within each type of site, you must make the QN names different for the two types of sites.

    For example, the following line of code will make an array of site indices with the qudit number QN having the name "Number_odd" on odd sites and "Number_even" on even sites:

    sites = [isodd(n) ? siteind("Qudit", n; dim=10, conserve_qns=true, qnname_number="Number_odd")
                       : siteind("Qudit", n; dim=2, conserve_qns=true, qnname_number="Number_even")
    -                  for n in 1:2*L]

    (You may have to collapse the above code into a single line for it to run properly.)

    + for n in 1:2*L]

    (You may have to collapse the above code into a single line for it to run properly.)

    diff --git a/previews/PR1514/faq/RelationshipToOtherLibraries.html b/previews/PR1514/faq/RelationshipToOtherLibraries.html index 872f104979..41e69933ab 100644 --- a/previews/PR1514/faq/RelationshipToOtherLibraries.html +++ b/previews/PR1514/faq/RelationshipToOtherLibraries.html @@ -1,2 +1,2 @@ -Relationship of ITensor to other tensor libraries FAQs · ITensors.jl

    Relationship of ITensor to other tensor libraries

    Here we will describe the relationship of ITensor to more traditional Julia Arrays or deep learning libraries like TensorFlow and PyTorch. There are a few things that distinguish ITensor from those approaches:

    1. ITensors have dimensions with labels that get passed around, which makes it simple to perform certain operations like contraction, addition, and tensor decompositions with a high level interface, independent of memory layout. This is along the same lines as Julia packages like NamedDims.jl and AxisArrays.jl and libraries in Python like xarray, however I would argue that the ITensor approach is a little more sophisticated (the dimensions have more metadata which makes them easier to manipulate for different situations, random ids to help avoid name clashes, etc.). This design was inspired by the needs of tensor network algorithms, where there are many tensor dimensions in the computation (of which many of them are dynamically created during the calculation), but would be helpful for writing other algorithms too.

    2. The ITensor type has a dynamic high level interface, where the type itself is mutable and the data can be swapped out. This allows for conveniently allocating the data of an ITensor on the fly "as needed", which makes for a nicer, more flexible interface (like initializing an empty ITensor before a loop, and filling it with the correct data type when the first value is set), at the expense of a small overhead for accessing data in the ITensor. We have found this tradeoff is worth it, since we expect ITensors to be used for medium to large scale calculations where operations on the tensors like contraction, addition, and tensor decomposition dominate the cost of the calculation, and code can be designed with function barriers to speed up operations when data is being accessed repeatedly.

    3. Another feature that ITensor has that goes beyond what is available in standard Julia, TensorFlow, and PyTorch is tensors which are symmetric under a group action. The physical interpretation of these tensors are ones that have a conserved quantity (like a quantum state with a conserved number of particles), so that feature is more physics-oriented, but could have applications in other areas like machine learning as well. In practice, these tensors are block sparse, and have extra metadata on the dimensions labeling representations of the group.

    4. Based on the features above, the ITensor library provides high level implementations of tensor network algorithms (algebraic operations of very high dimensional tensors, such as addition, multiplication, and finding dominant eigenvectors). In general these algorithms can (and have been) written on top of other libraries like standard Julia Arrays/AD, PyTorch, or TensorFlow, but they might have various downsides (a less convenient interface for dealing with tensor operations, no support for the types of symmetric tensors we often need, limited support for tensors with complex numbers in the case of libraries like PyTorch, though perhaps that has improved since I last checked, etc.).

    Although ITensor has primarily focused on quantum physics and quantum computing applications, there is work using ITensor for machine learning applications (so far focused on applications of tensor networks to machine learning, so no neural network calculations yet as far as I know). In general, these different libraries (ITensor, Flux, PyTorch, TensorFlow) are biased towards their specific methods and application areas that they are used for the most: ITensor is more biased towards tensor network calculations and quantum physics/quantum computing applications, based on the available features and interface, while PyTorch and TensorFlow are more biased towards neural network calculations. However, our goal would be to provide more features to ITensor that would make it useful for neural network applications as well, such as better support for slicing operations.

    +Relationship of ITensor to other tensor libraries FAQs · ITensors.jl

    Relationship of ITensor to other tensor libraries

    Here we will describe the relationship of ITensor to more traditional Julia Arrays or deep learning libraries like TensorFlow and PyTorch. There are a few things that distinguish ITensor from those approaches:

    1. ITensors have dimensions with labels that get passed around, which makes it simple to perform certain operations like contraction, addition, and tensor decompositions with a high level interface, independent of memory layout. This is along the same lines as Julia packages like NamedDims.jl and AxisArrays.jl and libraries in Python like xarray, however I would argue that the ITensor approach is a little more sophisticated (the dimensions have more metadata which makes them easier to manipulate for different situations, random ids to help avoid name clashes, etc.). This design was inspired by the needs of tensor network algorithms, where there are many tensor dimensions in the computation (of which many of them are dynamically created during the calculation), but would be helpful for writing other algorithms too.

    2. The ITensor type has a dynamic high level interface, where the type itself is mutable and the data can be swapped out. This allows for conveniently allocating the data of an ITensor on the fly "as needed", which makes for a nicer, more flexible interface (like initializing an empty ITensor before a loop, and filling it with the correct data type when the first value is set), at the expense of a small overhead for accessing data in the ITensor. We have found this tradeoff is worth it, since we expect ITensors to be used for medium to large scale calculations where operations on the tensors like contraction, addition, and tensor decomposition dominate the cost of the calculation, and code can be designed with function barriers to speed up operations when data is being accessed repeatedly.

    3. Another feature that ITensor has that goes beyond what is available in standard Julia, TensorFlow, and PyTorch is tensors which are symmetric under a group action. The physical interpretation of these tensors are ones that have a conserved quantity (like a quantum state with a conserved number of particles), so that feature is more physics-oriented, but could have applications in other areas like machine learning as well. In practice, these tensors are block sparse, and have extra metadata on the dimensions labeling representations of the group.

    4. Based on the features above, the ITensor library provides high level implementations of tensor network algorithms (algebraic operations of very high dimensional tensors, such as addition, multiplication, and finding dominant eigenvectors). In general these algorithms can (and have been) written on top of other libraries like standard Julia Arrays/AD, PyTorch, or TensorFlow, but they might have various downsides (a less convenient interface for dealing with tensor operations, no support for the types of symmetric tensors we often need, limited support for tensors with complex numbers in the case of libraries like PyTorch, though perhaps that has improved since I last checked, etc.).

    Although ITensor has primarily focused on quantum physics and quantum computing applications, there is work using ITensor for machine learning applications (so far focused on applications of tensor networks to machine learning, so no neural network calculations yet as far as I know). In general, these different libraries (ITensor, Flux, PyTorch, TensorFlow) are biased towards their specific methods and application areas that they are used for the most: ITensor is more biased towards tensor network calculations and quantum physics/quantum computing applications, based on the available features and interface, while PyTorch and TensorFlow are more biased towards neural network calculations. However, our goal would be to provide more features to ITensor that would make it useful for neural network applications as well, such as better support for slicing operations.

    diff --git a/previews/PR1514/getting_started/DebugChecks.html b/previews/PR1514/getting_started/DebugChecks.html index 36d4388a86..6649da27aa 100644 --- a/previews/PR1514/getting_started/DebugChecks.html +++ b/previews/PR1514/getting_started/DebugChecks.html @@ -35,4 +35,4 @@ [8] noprime(::ITensor) @ ITensors ~/.julia/packages/ITensors/cu9Bo/src/itensor.jl:1211 [9] top-level scope - @ REPL[7]:1

    You can track where debug checks are located in the code here, and add your own debug checks to your own code by wrapping your code with the macro ITensors.@debug_check.

    + @ REPL[7]:1

    You can track where debug checks are located in the code here, and add your own debug checks to your own code by wrapping your code with the macro ITensors.@debug_check.

    diff --git a/previews/PR1514/getting_started/Installing.html b/previews/PR1514/getting_started/Installing.html index 5bb11909b9..f8b61ea4b5 100644 --- a/previews/PR1514/getting_started/Installing.html +++ b/previews/PR1514/getting_started/Installing.html @@ -3,4 +3,4 @@ $ mkdir -p bin $ wget https://julialang-s3.julialang.org/bin/linux/x64/1.7/julia-1.7.2-linux-x86_64.tar.gz $ tar xvzf julia-1.7.2-linux-x86_64.tar.gz -$ ln -s julia-1.7.2/bin/julia bin/julia

    If you want to install Julia 1.6.6, you would change 1.7 to 1.6 and 1.7.2 to 1.6.6. In general we recommend using the current stable release of Julia, which you can find out by going to the Julia Downloads page. We also don't recommend using versions of Julia below 1.6, which are no longer compatible with ITensors.jl as of ITensors 0.3.

    After these steps, you should be able to type julia from your terminal to run Julia in interactive mode. If that works, then you have the Julia language and can run it in all the usual ways. If it does not work, you may need to log out and back in, and check that the bin directory is in your program execution path (PATH environment variable).

    Explanation of the sample commands above:

    Installing ITensor (ITensors.jl Package)

    Installing the Julia version of ITensor is easy once you have the Julia language installed. For more information about installing Julia, please see the Julia language downloads page.

    Once you have installed Julia on your machine,

    1. Enter the command julia to launch an interactive Julia session (a.k.a. the Julia "REPL")
    2. Type ] to enter the package manager (pkg> prompt should now show)
    3. Enter the command add ITensors
    4. After installation completes, press backspace to return to the normal julia> prompt
    5. Optional but Recommended: Enter the command julia> using ITensors; ITensors.compile() to compile a large fraction of the ITensor library code and following the instructions afterward to make an alias for loading a pre-built ITensor system image with Julia. This step can take up to 10 minutes to complete but only has to be done once for each version of ITensor. See the section on compiling ITensor for more information.

    Sample screenshot:

    +$ ln -s julia-1.7.2/bin/julia bin/julia

    If you want to install Julia 1.6.6, you would change 1.7 to 1.6 and 1.7.2 to 1.6.6. In general we recommend using the current stable release of Julia, which you can find out by going to the Julia Downloads page. We also don't recommend using versions of Julia below 1.6, which are no longer compatible with ITensors.jl as of ITensors 0.3.

    After these steps, you should be able to type julia from your terminal to run Julia in interactive mode. If that works, then you have the Julia language and can run it in all the usual ways. If it does not work, you may need to log out and back in, and check that the bin directory is in your program execution path (PATH environment variable).

    Explanation of the sample commands above:

    Installing ITensor (ITensors.jl Package)

    Installing the Julia version of ITensor is easy once you have the Julia language installed. For more information about installing Julia, please see the Julia language downloads page.

    Once you have installed Julia on your machine,

    1. Enter the command julia to launch an interactive Julia session (a.k.a. the Julia "REPL")
    2. Type ] to enter the package manager (pkg> prompt should now show)
    3. Enter the command add ITensors
    4. After installation completes, press backspace to return to the normal julia> prompt
    5. Optional but Recommended: Enter the command julia> using ITensors; ITensors.compile() to compile a large fraction of the ITensor library code and following the instructions afterward to make an alias for loading a pre-built ITensor system image with Julia. This step can take up to 10 minutes to complete but only has to be done once for each version of ITensor. See the section on compiling ITensor for more information.

    Sample screenshot:

    diff --git a/previews/PR1514/getting_started/NextSteps.html b/previews/PR1514/getting_started/NextSteps.html index fd2f965642..7f578bc667 100644 --- a/previews/PR1514/getting_started/NextSteps.html +++ b/previews/PR1514/getting_started/NextSteps.html @@ -1,2 +1,2 @@ -Next Steps · ITensors.jl
    +Next Steps · ITensors.jl
    diff --git a/previews/PR1514/getting_started/RunningCodes.html b/previews/PR1514/getting_started/RunningCodes.html index 3a3b5e1311..7e87f83519 100644 --- a/previews/PR1514/getting_started/RunningCodes.html +++ b/previews/PR1514/getting_started/RunningCodes.html @@ -20,4 +20,4 @@ end main(; d1 = 4, d2 = 5)

    which can be useful in interactive mode, particularly if you might want to run your code with a variety of different arguments.

    Running a Script

    Now say you put the above code into a file named code.jl. Then you can run this code on the command line as follows

    $ julia code.jl

    This script-like mode of running Julia is convenient for running longer jobs, such as on a cluster.

    Running Interactively

    However, sometimes you want to do rapid development when first writing and testing a code. For this kind of work, the long startup and compilation times currently incurred by the Julia compiler can be a nuisance. Fortunately a nice solution is to alternate between modifying your code then running it by loading it into an already running Julia session.

    To set up this kind of session, take the following steps:

    1. Enter the interactive mode of Julia, by inputting the command julia on the command line. You will now be in the Julia "REPL" (read-eval-print loop) with the prompt julia> on the left of your screen.

    2. To run a code such as the code.jl file discussed above, input the command

      julia> include("code.jl")

      Note that you must be in the same folder as code.jl for this to work; otherwise input the entire path to the code.jl file. The code will run and you will see its output in the REPL.

    3. Now say you want to modify and re-run the code. To do this, just edit the file in an editor in another window, without closing your Julia session. Now run the command

      julia> include("code.jl")

      again and your updated code will run, but this time skipping any of the precompilation overhead incurred on previous steps.

    The above steps to running a code interactively has a big advantage that you only have to pay the startup time of compiling ITensor and other libraries you are using once. Further changes to your code only incur very small extra compilation times, facilitating rapid development.

    Compiling an ITensor System Image

    The above strategy of running code in the Julia REPL (interactive mode) works well, but still incurs a large start-up penalty for the first run of your code. Fortunately there is a nice way around this issue too: compiling ITensors.jl and making a system image built by the PackageCompiler.jl library.

    To use this approach, we have provided a convenient one-line command:

    julia> using ITensors; ITensors.compile()

    Once ITensors.jl is installed, you can just run this command in an interactive Julia session. It can take a few minutes to run, but you only have to run it once for a given version of ITensors.jl. When it is done, it will create a file sys_itensors.so in the directory ~/.julia/sysimages/.

    To use the compiled system image together with Julia, run the julia command (for interactive mode or scripts) in the following way:

    $ julia --sysimage ~/.julia/sysimages/sys_itensors.so

    A convenient thing to do is to make an alias in your shell for this command. To do this, edit your .bashrc or .zshrc or similar file for the shell you use by adding the following line:

    alias julia_itensors="julia --sysimage ~/.julia/sysimages/sys_itensors.so -e \"using ITensors\" -i "

    where of course you can use the command name you like when defining the alias. Now running commands like julia_itensors code.jl or julia_itensors to start an interactive session will have the ITensor system image pre-loaded and you will notice significantly faster startup times. The arguments -e \"using ITensors\" -i make it so that running julia_itensors also loads the ITensor library as soon as Julia starts up, so that you don't have to type using ITensors every time.

    Using a Compiled Sysimage in Jupyter or VS Code

    If you have compiled a sysimage for ITensor as shown above, you can use it in Jupyter by running the following code:

    using IJulia
    -installkernel("julia_ITensors","--sysimage=~/.julia/sysimages/sys_itensors.so")

    in the Julia REPL (Julia console).

    To load the ITensor sysimage in VS Code, you can add

    "--sysimage ~/.julia/sysimages/sys_itensors.so"

    as an argument under the julia.additionalArgs setting in your Settings.json file.

    For more information on the above, see the following Julia Discourse post.

    +installkernel("julia_ITensors","--sysimage=~/.julia/sysimages/sys_itensors.so")

    in the Julia REPL (Julia console).

    To load the ITensor sysimage in VS Code, you can add

    "--sysimage ~/.julia/sysimages/sys_itensors.so"

    as an argument under the julia.additionalArgs setting in your Settings.json file.

    For more information on the above, see the following Julia Discourse post.

    diff --git a/previews/PR1514/index.html b/previews/PR1514/index.html index 258988cf91..a19cc3e2dc 100644 --- a/previews/PR1514/index.html +++ b/previews/PR1514/index.html @@ -161,4 +161,4 @@ After sweep 3 energy=-138.940080155429 maxlinkdim=92 maxerr=1.00E-10 time=4.522 After sweep 4 energy=-138.940086009318 maxlinkdim=100 maxerr=1.05E-10 time=11.644 After sweep 5 energy=-138.940086058840 maxlinkdim=96 maxerr=1.00E-10 time=12.771 -Final energy = -138.94008605883985

    You can find more examples of running dmrg and related algorithms here.

    +Final energy = -138.94008605883985

    You can find more examples of running dmrg and related algorithms here.

    diff --git a/previews/PR1514/search.html b/previews/PR1514/search.html index 04cc389884..efc3773aaa 100644 --- a/previews/PR1514/search.html +++ b/previews/PR1514/search.html @@ -1,2 +1,2 @@ -Search · ITensors.jl

    Loading search...

      +Search · ITensors.jl

      Loading search...

        diff --git a/previews/PR1514/tutorials/DMRG.html b/previews/PR1514/tutorials/DMRG.html index 1d530d7877..31549fe77c 100644 --- a/previews/PR1514/tutorials/DMRG.html +++ b/previews/PR1514/tutorials/DMRG.html @@ -23,7 +23,7 @@ return end

        Steps of The Code

        The first two lines

        using ITensors, ITensorMPS
         N = 100
        -sites = siteinds("S=1",N)

        tells the function siteinds to make an array of ITensor Index objects which have the properties of $S=1$ spins. This means their dimension will be 3 and they will carry the "S=1" tag, which will enable the next part of the code to know how to make appropriate operators for them.

        Try printing out some of these indices to verify their properties:

        @show sites[1]
        (dim=3|id=416|"S=1,Site,n=1")

        The next part of the code builds the Hamiltonian:

        os = OpSum()
        +sites = siteinds("S=1",N)

        tells the function siteinds to make an array of ITensor Index objects which have the properties of $S=1$ spins. This means their dimension will be 3 and they will carry the "S=1" tag, which will enable the next part of the code to know how to make appropriate operators for them.

        Try printing out some of these indices to verify their properties:

        @show sites[1]
        (dim=3|id=991|"S=1,Site,n=1")

        The next part of the code builds the Hamiltonian:

        os = OpSum()
         for j=1:N-1
           os += "Sz",j,"Sz",j+1
           os += 1/2,"S+",j,"S-",j+1
        @@ -31,4 +31,4 @@
         end
         H = MPO(os,sites)

        An OpSum is an object which accumulates Hamiltonian terms such as "Sz",1,"Sz",2 so that they can be summed afterward into a matrix product operator (MPO) tensor network. The line of code H = MPO(os,sites) constructs the Hamiltonian in the MPO format, with physical indices given by the array sites.

        The line

        psi0 = random_mps(sites;linkdims=10)

        constructs an MPS psi0 which has the physical indices sites and a bond dimension of 10. It is made by a random quantum circuit that is reshaped into an MPS, so that it will have as generic and unbiased properties as an MPS of that size can have. This choice can help prevent the DMRG calculation from getting stuck in a local minimum.

        The lines

        nsweeps = 5
         maxdim = [10,20,100,100,200]
        -cutoff = [1E-10]

        define the number of DMRG sweeps (five) we will instruct the code to do, as well as the parameters that will control the speed and accuracy of the DMRG algorithm within each sweep. The array maxdim limits the maximum MPS bond dimension allowed during each sweep and cutoff defines the truncation error goal of each sweep (if fewer values are specified than sweeps, the last value is used for all remaining sweeps).

        Finally the call

        energy,psi = dmrg(H,psi0;nsweeps,maxdim,cutoff)

        runs the DMRG algorithm included in ITensor, using psi0 as an initial guess for the ground state wavefunction. The optimized MPS psi and its eigenvalue energy are returned.

        After the dmrg function returns, you can take the returned MPS psi and do further calculations with it, such as measuring local operators or computing entanglement entropy.

        +cutoff = [1E-10]

        define the number of DMRG sweeps (five) we will instruct the code to do, as well as the parameters that will control the speed and accuracy of the DMRG algorithm within each sweep. The array maxdim limits the maximum MPS bond dimension allowed during each sweep and cutoff defines the truncation error goal of each sweep (if fewer values are specified than sweeps, the last value is used for all remaining sweeps).

        Finally the call

        energy,psi = dmrg(H,psi0;nsweeps,maxdim,cutoff)

        runs the DMRG algorithm included in ITensor, using psi0 as an initial guess for the ground state wavefunction. The optimized MPS psi and its eigenvalue energy are returned.

        After the dmrg function returns, you can take the returned MPS psi and do further calculations with it, such as measuring local operators or computing entanglement entropy.

        diff --git a/previews/PR1514/tutorials/MPSTimeEvolution.html b/previews/PR1514/tutorials/MPSTimeEvolution.html index 8d64de4fdf..04a651b674 100644 --- a/previews/PR1514/tutorials/MPSTimeEvolution.html +++ b/previews/PR1514/tutorials/MPSTimeEvolution.html @@ -51,4 +51,4 @@ end

        Steps of The Code

        First we setsome parameters, like the system size N and time step $\tau$ to use.

        The line s = siteinds("S=1/2",N;conserve_qns=true) defines an array of spin 1/2 tensor indices (Index objects) which will be the site or physical indices of the MPS.

        Next we make an empty array gates = ITensor[] that will hold ITensors that will be our Trotter gates. Inside the for n=1:N-1 loop that follows the lines

        hj =      op("Sz",s1) * op("Sz",s2) +
             1/2 * op("S+",s1) * op("S-",s2) +
             1/2 * op("S-",s1) * op("S+",s2)

        call the op function which reads the "S=1/2" tag on our site indices (sites j and j+1) and which then knows that we want the spin 1/ 2 version of the "Sz", "S+", and "S-" operators. The op function returns these operators as ITensors and we tensor product and add them together to compute the operator $h_{j,j+1}$ defined as

        \[h_{j,j+1} = S^z_j S^z_{j+1} + \frac{1}{2} S^+_j S^-_{j+1} + \frac{1}{2} S^-_j S^+_{j+1}\]

        which we call hj in the code.

        To make the corresponding Trotter gate Gj we exponentiate hj times a factor $-i \tau/2$ and then append or push this onto the end of the gate array gates.

        Gj = exp(-im * tau/2 * hj)
        -push!(gates,Gj)

        Having made the gates for bonds (1,2),(2,3),(3,4), etc. we still need to append the gates in reverse order to complete the correct Trotter formula. Here we can conveniently do that by just calling the Julia append! function and supply a reversed version of the array of gates we have made so far. This can be done in a single line of code append!(gates,reverse(gates)).

        The line of code psi = MPS(s, n -> isodd(n) ? "Up" : "Dn") initializes our MPS psi as a product state of alternating up and down spins.

        To carry out the time evolution we loop over the range of times from 0.0 to ttotal in steps of tau, using the Julia range notation 0.0:tau:ttotal to easily set up this loop as for t in 0.0:tau:ttotal.

        Inside the loop, we use the expect function to measure the expected value of the "Sz" operator on the center site.

        To evolve the MPS to the next time, we call the function

        psi = apply(gates, psi; cutoff)

        which applies the array of ITensors called gates to our current MPS psi, truncating the MPS at each step using the truncation error threshold supplied as the variable cutoff.

        The apply function is smart enough to determine which site indices each gate has, and then figure out where to apply it to our MPS. It automatically handles truncating the MPS and can even handle non-nearest-neighbor gates, though that feature is not used in this example.

        +push!(gates,Gj)

        Having made the gates for bonds (1,2),(2,3),(3,4), etc. we still need to append the gates in reverse order to complete the correct Trotter formula. Here we can conveniently do that by just calling the Julia append! function and supply a reversed version of the array of gates we have made so far. This can be done in a single line of code append!(gates,reverse(gates)).

        The line of code psi = MPS(s, n -> isodd(n) ? "Up" : "Dn") initializes our MPS psi as a product state of alternating up and down spins.

        To carry out the time evolution we loop over the range of times from 0.0 to ttotal in steps of tau, using the Julia range notation 0.0:tau:ttotal to easily set up this loop as for t in 0.0:tau:ttotal.

        Inside the loop, we use the expect function to measure the expected value of the "Sz" operator on the center site.

        To evolve the MPS to the next time, we call the function

        psi = apply(gates, psi; cutoff)

        which applies the array of ITensors called gates to our current MPS psi, truncating the MPS at each step using the truncation error threshold supplied as the variable cutoff.

        The apply function is smart enough to determine which site indices each gate has, and then figure out where to apply it to our MPS. It automatically handles truncating the MPS and can even handle non-nearest-neighbor gates, though that feature is not used in this example.

        diff --git a/previews/PR1514/tutorials/QN_DMRG.html b/previews/PR1514/tutorials/QN_DMRG.html index df3c32a3c1..c3967c9462 100644 --- a/previews/PR1514/tutorials/QN_DMRG.html +++ b/previews/PR1514/tutorials/QN_DMRG.html @@ -36,4 +36,4 @@ energy, psi = dmrg(H,psi0; nsweeps, maxdim, cutoff) return -end +end