#pragma once #include "model.hpp" #include "mps.hpp" #include #include #include /** * @brief Apply the function f on every element if inRange, but reuse a cached result when two pointers point to the same object */ template std::vector> forEachUnique(std::function f, const std::vector>& inRange); template using Bonds = std::vector>>; /** * @brief Apply time evolution on a bond * @details * \f[U = \exp{ B \, \del t} \f] */ template e::Tensor getTimeEvolutionMatrix(const e::MatrixX& bond, T dt, unsigned localDim); template Bonds getTimeEvolutionMatrices(const std::vector>>& bonds, T dt, unsigned int localDim); /** * @brief Apply U on bond i, perform SVD and truncate, update MPS */ template void updateBond(MPS& psi, size_t i, const e::Tensor& U, unsigned chiMax, T eps); template using postUpdateFunction_t = std::function&, unsigned)>; template void postUpdateNoop(const MPS&, unsigned) {}; /** * @brief Brickwall TEBS scheme * @details * Apply U on every second bond starting from the first, * then apply U on every second bond starting from the second * * First order scheme * * @param postUpdateFunc function that can be used to track expectation values after each update */ template void runTEBD_brickwall(MPS& psi, const Bonds& bonds, unsigned chiMax, T eps, unsigned nSteps, postUpdateFunction_t postUpdateFunc); template void runTEBD_secondOrder(MPS& psi, const Bonds& bonds, const Bonds& bondsHalfDT, unsigned chiMax, T eps, unsigned nSteps, postUpdateFunction_t postUpdateFunc);