https://github.com/ITensor/ITensor
Tip revision: e2c6ea98c5d68dee7ad38c16a7f71574a46589e0 authored by Miles Stoudenmire on 20 August 2015, 19:47:53 UTC
Shortened some type names in cplx_literal.h and names of constants in global.h
Shortened some type names in cplx_literal.h and names of constants in global.h
Tip revision: e2c6ea9
DMRGObserver.h
//
// Distributed under the ITensor Library License, Version 1.1.
// (See accompanying LICENSE file.)
//
#ifndef __ITENSOR_DMRGOBSERVER_H
#define __ITENSOR_DMRGOBSERVER_H
#include "observer.h"
namespace itensor {
//
// Class for monitoring DMRG calculations.
// The measure and checkDone methods are virtual
// so that behavior can be customized in a
// derived class.
//
template<class Tensor>
class DMRGObserver : public Observer
{
public:
DMRGObserver(const MPSt<Tensor>& psi,
const Args& args = Global::args());
virtual ~DMRGObserver() { }
void virtual
measure(const Args& args = Global::args());
bool virtual
checkDone(const Args& args = Global::args());
void virtual
lastSpectrum(const Spectrum& spec) { last_spec_ = spec; }
const MPSt<Tensor>&
psi() const { return psi_; }
const Spectrum&
spectrum() const { return last_spec_; }
private:
/////////////
//
// Data Members
const MPSt<Tensor>& psi_;
Vector center_eigs;
Real energy_errgoal; //Stop DMRG once energy has converged to this precision
bool printeigs; //Print slowest decaying eigenvalues after every sweep
int max_eigs;
Real max_te;
bool done_;
Real last_energy_;
Spectrum last_spec_;
Model::DefaultOpsT default_ops_;
//
/////////////
}; // class DMRGObserver
template<class Tensor>
inline DMRGObserver<Tensor>::
DMRGObserver(const MPSt<Tensor>& psi, const Args& args)
:
psi_(psi),
energy_errgoal(args.getReal("EnergyErrgoal",-1)),
printeigs(args.getBool("PrintEigs",true)),
max_eigs(-1),
max_te(-1),
done_(false),
last_energy_(1000),
default_ops_(psi.sites().defaultOps())
{
}
template<class Tensor>
void inline DMRGObserver<Tensor>::
measure(const Args& args)
{
const int N = psi_.N();
const int b = args.getInt("AtBond",1);
const int sw = args.getInt("Sweep",0);
const int ha = args.getInt("HalfSweep",0);
const Real energy = args.getReal("Energy",0);
if(!args.getBool("Quiet",false) && !args.getBool("NoMeasure",false))
{
if(b < N && b > 0)
{
const Tensor wfb = psi_.A(b)*psi_.A(b+1);
for(const std::string& opname : default_ops_)
{
Complex z =
BraKet(prime(wfb,psi_.sites()(b)),psi_.sites().op(opname,b)*wfb);
if(fabs(z.imag()) < 1E-14)
printfln("<%s>(%d) = %.10E",opname,b,z.real());
else
printfln("<%s>(%d) = (%.10E,%.10E)",opname,b,z.real(),z.imag());
}
}
}
if(printeigs)
{
if(b == N/2 && ha == 2)
{
println();
Vector center_eigs = last_spec_.eigsKept();
Real S = 0;
for(int j = 1; j <= center_eigs.Length(); ++j)
{
S -= center_eigs(j)*log(fabs(center_eigs(j)));
}
printfln(" vN Entropy at center bond b=%d = %.12f",N/2,S);
printf(" Eigs at center bond b=%d: ",N/2);
for(int j = 1; j <= min(center_eigs.Length(),10); ++j)
{
const Real eig = center_eigs(j);
if(eig < 1E-3) break;
printf("%.4f ",eig);
}
println();
}
}
max_eigs = max(max_eigs,last_spec_.numEigsKept());
max_te = max(max_te,last_spec_.truncerr());
if(b == 1 && ha == 2)
{
if(!printeigs) println();
println(" Largest m during sweep ",sw," was ",(max_eigs > 1 ? max_eigs : 1));
max_eigs = -1;
println(" Largest truncation error: ",(max_te > 0 ? max_te : 0.));
max_te = -1;
printfln(" Energy after sweep %d is %.12f",sw,energy);
}
}
template<class Tensor>
bool inline DMRGObserver<Tensor>::
checkDone(const Args& args)
{
const int sw = args.getInt("Sweep",0);
const Real energy = args.getReal("Energy",0);
if(sw == 1) last_energy_ = 1000;
if(energy_errgoal > 0 && sw%2 == 0)
{
Real dE = fabs(energy-last_energy_);
if(dE < energy_errgoal)
{
printfln(" Energy error goal met (dE = %.3E < %.3E); returning after %d sweeps.",
dE, energy_errgoal, sw);
last_energy_ = 1000;
return true;
}
}
last_energy_ = energy;
//If STOP_DMRG found, will return true (i.e. done) once, but
//outer calling using same Observer may continue running e.g. infinite dmrg calling finite dmrg.
if(fileExists("STOP_DMRG"))
{
println("File STOP_DMRG found: stopping this DMRG run after sweep ",sw);
system("rm -f STOP_DMRG");
return true;
}
//Set done_ flag to true so any outer callers using this Observer will also terminate.
if(fileExists("STOP_DMRG_ALL"))
{
println("File STOP_DMRG_ALL found: stopping this run after sweep ",sw);
system("rm -f STOP_DMRG_ALL");
done_ = true;
return done_;
}
return done_;
}
} //namespace itensor
#endif // __ITENSOR_DMRGOBSERVER_H