PSCF v1.2
rpc/fts/analyzer/AverageListAnalyzer.tpp
1#ifndef RPC_AVERAGE_LIST_ANALYZER_TPP
2#define RPC_AVERAGE_LIST_ANALYZER_TPP
3
4/*
5* PSCF - Polymer Self-Consistent Field Theory
6*
7* Copyright 2016 - 2022, The Regents of the University of Minnesota
8* Distributed under the terms of the GNU General Public License.
9*/
10
11#include "AverageListAnalyzer.h"
12
13#include <rpc/System.h>
14#include <util/format/Int.h>
15#include <util/format/Dbl.h>
16#include <util/misc/FileMaster.h>
17#include <util/misc/ioUtil.h>
18
19namespace Pscf {
20namespace Rpc
21{
22
23 using namespace Util;
24
25 /*
26 * Constructor.
27 */
28 template <int D>
30 : Analyzer<D>(),
31 systemPtr_(&system),
32 nSamplePerOutput_(1),
33 nValue_(0),
34 hasAccumulators_(false)
35 {}
36
37 /*
38 * Destructor.
39 */
40 template <int D>
43
44 /*
45 * Read interval and outputFileName.
46 */
47 template <int D>
49 {
51 nSamplePerOutput_ = 1;
52 readOptional(in,"nSamplePerOutput", nSamplePerOutput_);
53 if (nSamplePerOutput() > 0) {
54 std::string fileName = outputFileName(".dat");
55 system().fileMaster().openOutputFile(fileName, outputFile_);
56 }
57 // Note: ReadParameters method of derived classes should call this,
58 // determine nValue and then call initializeAccumulators(nValue).
59 }
60
61 /*
62 * Clear accumulators (do nothing on slave processors).
63 */
64 template <int D>
66 {
67 UTIL_CHECK(hasAccumulators_);
68 clearAccumulators();
69 }
70
71 /*
72 * Setup before system.
73 */
74 template <int D>
76 {
77 UTIL_CHECK(hasAccumulators_);
78 clear();
79 }
80
81 /*
82 * Compute and sample current values.
83 */
84 template <int D>
86 {
87 UTIL_CHECK(hasAccumulators_);
88 if (!isAtInterval(iStep)) return;
89 compute();
90 updateAccumulators(iStep);
91 }
92
93 /*
94 * Output results after a system is completed.
95 */
96 template <int D>
98 {
99 UTIL_CHECK(hasAccumulators_);
100
101 // Close data file, if any
102 if (outputFile_.is_open()) {
103 outputFile_.close();
104 }
105
106 #if 0
107 // Write parameter (*.prm) file
108 system().fileMaster().openOutputFile(outputFileName(".prm"), outputFile_);
109 ParamComposite::writeParam(outputFile_);
110 outputFile_.close();
111 #endif
112
113 // Write average (*.ave) and error analysis (*.aer) files
114 outputAccumulators();
115
116 }
117
121 template <int D>
123 {
124 UTIL_CHECK(nValue > 0);
125 UTIL_CHECK(nValue_ == 0);
126 UTIL_CHECK(nSamplePerOutput_ >= 0);
127
128 // Allocate arrays
129 accumulators_.allocate(nValue);
130 names_.allocate(nValue);
131 values_.allocate(nValue);
132 nValue_ = nValue;
133 hasAccumulators_ = true;
134
135 // Set the accumulators to compute block averages with
136 // nSamplePerOutput_ sampled values per block
137 for (int i = 0; i < nValue_; ++i) {
138 accumulators_[i].setNSamplePerBlock(nSamplePerOutput_);
139 }
140
141 clearAccumulators();
142 }
143
144 /*
145 * Clear accumulators.
146 */
147 template <int D>
149 {
150 UTIL_CHECK(hasAccumulators_);
151 UTIL_CHECK(nValue_ > 0);
152 for (int i = 0; i < nValue_; ++i) {
153 accumulators_[i].clear();
154 }
155 }
156
157 template <int D>
158 void AverageListAnalyzer<D>::setName(int i, std::string name)
159 {
160 UTIL_CHECK(hasAccumulators_);
161 UTIL_CHECK(i >= 0 && i < nValue_);
162 names_[i] = name;
163 }
164
165 /*
166 * Update accumulators for all current values.
167 */
168 template <int D>
170 {
171 UTIL_CHECK(hasAccumulators_);
172 UTIL_CHECK(accumulators_.capacity() == nValue_);
173
174 // Update accumulators.
175 for (int i = 0; i < nValue(); ++i) {
176 double data = value(i);
177 accumulators_[i].sample(data);
178 }
179
180 // Output block averages
181 if (nSamplePerOutput_ > 0) {
182 if (accumulators_[0].isBlockComplete()) {
183 UTIL_CHECK(outputFile_.is_open());
184 int beginStep = iStep - (nSamplePerOutput_ - 1)*interval();
185 outputFile_ << Int(beginStep);
186 for (int i = 0; i < nValue(); ++i) {
187 UTIL_CHECK(accumulators_[i].isBlockComplete());
188 double block = accumulators_[i].blockAverage();
189 outputFile_ << Dbl(block);
190 }
191 outputFile_ << "\n";
192 }
193 }
194
195 }
196
197 /*
198 * Output results to file after simulation is completed.
199 */
200 template <int D>
202 {
203 UTIL_CHECK(hasAccumulators_);
204
205 // Close data (*.dat) file, if any
206 if (outputFile_.is_open()) {
207 outputFile_.close();
208 }
209
210 // Compute maximum length of name field
211 int nameWidth = 0;
212 int length;
213 for (int i = 0; i < nValue_; ++i) {
214 length = names_[i].length();
215 if (length > nameWidth) {
216 nameWidth = length;
217 }
218 }
219 nameWidth += 2;
220
221 // Write average (*.ave) file
222 std::string fileName = outputFileName(".ave");
223 system().fileMaster().openOutputFile(fileName, outputFile_);
224 double ave, err;
225 for (int i = 0; i < nValue_; ++i) {
226 ave = accumulators_[i].average();
227 err = accumulators_[i].blockingError();
228 outputFile_ << " " << std::left << std::setw(nameWidth)
229 << names_[i] << " ";
230 outputFile_ << Dbl(ave) << " +- " << Dbl(err, 9, 2) << "\n";
231 }
232 outputFile_.close();
233
234 // Write error analysis (*.aer) file
235 fileName = outputFileName(".aer");
236 system().fileMaster().openOutputFile(fileName, outputFile_);
237 std::string line;
238 line =
239 "---------------------------------------------------------------------";
240 for (int i = 0; i < nValue_; ++i) {
241 outputFile_ << line << std::endl;
242 outputFile_ << names_[i] << " :" << std::endl;
243 accumulators_[i].output(outputFile_);
244 outputFile_ << std::endl;
245 }
246 outputFile_.close();
247
248 #if 0
249 // Write data format file (*.dfm) file
250 fileName = outputFileName();
251 fileName += ".dfm";
252 system().fileMaster().openOutputFile(fileName, outputFile_);
253 outputFile_ << "Value = " << nValue() << std::endl;
254 outputFile_ << "iStep ";
255 for (int i = 0; i < nValue_; ++i) {
256 outputFile_ << names_[i] << " ";
257 }
258 outputFile_ << std::endl;
259 outputFile_.close();
260 #endif
261
262 }
263
264}
265}
266#endif
Abstract base for periodic output and/or analysis actions.
virtual void readParameters(std::istream &in)
Read parameters from archive.
void updateAccumulators(long iStep)
Add current value to accumulator, output block average if needed.
void clearAccumulators()
Clear internal state of the accumulator.
void initializeAccumulators(int nValue)
Instantiate Average accumulators and set nSamplePerOutput set nValue.
virtual void output()
Write final results to file after a simulation.
void setName(int i, std::string name)
Set name of variable.
virtual void sample(long iStep)
Compute a sampled value and update the accumulator.
AverageListAnalyzer(System< D > &system)
Constructor.
void outputAccumulators()
Write results of statistical analysis to files.
virtual void readParameters(std::istream &in)
Read interval, outputFileName and (optionally) nSamplePerOutput.
Main class for SCFT or PS-FTS simulation of one system.
Definition rpc/System.h:100
Wrapper for a double precision number, for formatted ostream output.
Definition Dbl.h:40
Wrapper for an int, for formatted ostream output.
Definition Int.h:37
virtual void writeParam(std::ostream &out) const
Write all parameters to an output stream.
#define UTIL_CHECK(condition)
Assertion macro suitable for serial or parallel production code.
Definition global.h:68
PSCF package top-level namespace.
Definition param_pc.dox:1
Utility classes for scientific computation.