source: src/Formula.cpp@ 95e6b1

Action_Thermostats Add_AtomRandomPerturbation Add_FitFragmentPartialChargesAction Add_RotateAroundBondAction Add_SelectAtomByNameAction Added_ParseSaveFragmentResults AddingActions_SaveParseParticleParameters Adding_Graph_to_ChangeBondActions Adding_MD_integration_tests Adding_ParticleName_to_Atom Adding_StructOpt_integration_tests AtomFragments Automaking_mpqc_open AutomationFragmentation_failures Candidate_v1.5.4 Candidate_v1.6.0 Candidate_v1.6.1 Candidate_v1.7.0 ChangeBugEmailaddress ChangingTestPorts ChemicalSpaceEvaluator CombiningParticlePotentialParsing Combining_Subpackages Debian_Package_split Debian_package_split_molecuildergui_only Disabling_MemDebug Docu_Python_wait EmpiricalPotential_contain_HomologyGraph EmpiricalPotential_contain_HomologyGraph_documentation Enable_parallel_make_install Enhance_userguide Enhanced_StructuralOptimization Enhanced_StructuralOptimization_continued Example_ManyWaysToTranslateAtom Exclude_Hydrogens_annealWithBondGraph FitPartialCharges_GlobalError Fix_BoundInBox_CenterInBox_MoleculeActions Fix_ChargeSampling_PBC Fix_ChronosMutex Fix_FitPartialCharges Fix_FitPotential_needs_atomicnumbers Fix_ForceAnnealing Fix_IndependentFragmentGrids Fix_ParseParticles Fix_ParseParticles_split_forward_backward_Actions Fix_PopActions Fix_QtFragmentList_sorted_selection Fix_Restrictedkeyset_FragmentMolecule Fix_StatusMsg Fix_StepWorldTime_single_argument Fix_Verbose_Codepatterns Fix_fitting_potentials Fixes ForceAnnealing_goodresults ForceAnnealing_oldresults ForceAnnealing_tocheck ForceAnnealing_with_BondGraph ForceAnnealing_with_BondGraph_continued ForceAnnealing_with_BondGraph_continued_betteresults ForceAnnealing_with_BondGraph_contraction-expansion FragmentAction_writes_AtomFragments FragmentMolecule_checks_bonddegrees GeometryObjects Gui_Fixes Gui_displays_atomic_force_velocity ImplicitCharges IndependentFragmentGrids IndependentFragmentGrids_IndividualZeroInstances IndependentFragmentGrids_IntegrationTest IndependentFragmentGrids_Sole_NN_Calculation JobMarket_RobustOnKillsSegFaults JobMarket_StableWorkerPool JobMarket_unresolvable_hostname_fix MoreRobust_FragmentAutomation ODR_violation_mpqc_open PartialCharges_OrthogonalSummation PdbParser_setsAtomName PythonUI_with_named_parameters QtGui_reactivate_TimeChanged_changes Recreated_GuiChecks Rewrite_FitPartialCharges RotateToPrincipalAxisSystem_UndoRedo SaturateAtoms_findBestMatching SaturateAtoms_singleDegree StoppableMakroAction Subpackage_CodePatterns Subpackage_JobMarket Subpackage_LinearAlgebra Subpackage_levmar Subpackage_mpqc_open Subpackage_vmg Switchable_LogView ThirdParty_MPQC_rebuilt_buildsystem TrajectoryDependenant_MaxOrder TremoloParser_IncreasedPrecision TremoloParser_MultipleTimesteps TremoloParser_setsAtomName Ubuntu_1604_changes stable
Last change on this file since 95e6b1 was e5c0a1, checked in by Tillmann Crueger <crueger@…>, 15 years ago

Made the periodentafel return only "const element *" instead of "element * const", to avoid unexpected changes to stored elements

  • Note: returning "type * const" from a function does not make sense, since the pointer has to be copied anyway (calling convention), so the calling function can just do another copy to get a non const version of the pointer
  • Property mode set to 100644
File size: 14.4 KB
Line 
1/*
2 * Formula.cpp
3 *
4 * Created on: Jul 21, 2010
5 * Author: crueger
6 */
7
8#include "Formula.hpp"
9
10#include <sstream>
11
12#include "World.hpp"
13#include "periodentafel.hpp"
14#include "element.hpp"
15#include "Helpers/Assert.hpp"
16#include "Helpers/Range.hpp"
17
18using namespace std;
19
20Formula::Formula() :
21 numElements(0)
22{}
23
24Formula::Formula(const Formula &src) :
25 elementCounts(src.elementCounts),
26 numElements(src.numElements)
27{}
28
29Formula::Formula(const string &formula) :
30 numElements(0)
31{
32 fromString(formula);
33}
34
35Formula::~Formula()
36{}
37
38Formula &Formula::operator=(const Formula &rhs){
39 // No self-assignment check needed
40 elementCounts=rhs.elementCounts;
41 numElements=rhs.numElements;
42 return *this;
43}
44
45std::string Formula::toString() const{
46 stringstream sstr;
47 for(const_iterator iter=end();iter!=begin();){
48 --iter;
49 sstr << (*iter).first->getSymbol();
50 if((*iter).second>1)
51 sstr << (*iter).second;
52 }
53 return sstr.str();
54}
55
56void Formula::fromString(const std::string &formula) throw(ParseError){
57 // make this transactional, in case an error is thrown
58 Formula res;
59 string::const_iterator begin = formula.begin();
60 string::const_iterator end = formula.end();
61 res.parseFromString(begin,end,static_cast<char>(0));
62 (*this)=res;
63}
64
65int Formula::parseMaybeNumber(string::const_iterator &it,string::const_iterator &end) throw(ParseError){
66 static const range<char> Numbers = makeRange('0',static_cast<char>('9'+1));
67 int count = 0;
68 while(it!=end && Numbers.isInRange(*it))
69 count = (count*10) + ((*it++)-Numbers.first);
70 // one is implicit
71 count = (count!=0)?count:1;
72 return count;
73}
74
75void Formula::parseFromString(string::const_iterator &it,string::const_iterator &end,char delimiter) throw(ParseError){
76 // some constants needed for parsing... Assumes ASCII, change if other encodings are used
77 static const range<char> CapitalLetters = makeRange('A',static_cast<char>('Z'+1));
78 static const range<char> SmallLetters = makeRange('a',static_cast<char>('z'+1));
79 map<char,char> delimiters;
80 delimiters['('] = ')';
81 delimiters['['] = ']';
82 // clean the formula
83 clear();
84 for(/*send from above*/;it!=end && *it!=delimiter;/*updated in loop*/){
85 // we might have a sub formula
86 if(delimiters.count(*it)){
87 Formula sub;
88 char nextdelim=delimiters[*it];
89 sub.parseFromString(++it,end,nextdelim);
90 if(!sub.getElementCount()){
91 throw(ParseError(__FILE__,__LINE__));
92 }
93 int count = parseMaybeNumber(++it,end);
94 addFormula(sub,count);
95 continue;
96 }
97 string shorthand;
98 // Atom names start with a capital letter
99 if(!CapitalLetters.isInRange(*it))
100 throw(ParseError(__FILE__,__LINE__));
101 shorthand+=(*it++);
102 // the rest of the name follows
103 while(it!=end && SmallLetters.isInRange(*it))
104 shorthand+=(*it++);
105 int count = parseMaybeNumber(it,end);
106 // test if the shorthand exists
107 if(!World::getInstance().getPeriode()->FindElement(shorthand))
108 throw(ParseError(__FILE__,__LINE__));
109 // done, we can get the next one
110 addElements(shorthand,count);
111 }
112 if(it==end && delimiter!=0){
113 throw(ParseError(__FILE__,__LINE__));
114 }
115}
116
117bool Formula::checkOut(ostream *output) const{
118 bool result = true;
119 int No = 1;
120
121 if (output != NULL) {
122 *output << "# Ion type data (PP = PseudoPotential, Z = atomic number)" << endl;
123 *output << "#Ion_TypeNr.\tAmount\tZ\tRGauss\tL_Max(PP)L_Loc(PP)IonMass\t# chemical name, symbol" << endl;
124 for(const_iterator iter=begin(); iter!=end();++iter){
125 result = result && (*iter).first->Checkout(output, No++, (*iter).second);
126 }
127 return result;
128 } else
129 return false;
130}
131
132unsigned int Formula::getElementCount() const{
133 return numElements;
134}
135
136bool Formula::hasElement(const element *element) const{
137 ASSERT(element,"Invalid pointer in Formula::hasElement(element*)");
138 return hasElement(element->getNumber());
139}
140
141bool Formula::hasElement(atomicNumber_t Z) const{
142 ASSERT(Z>0,"Invalid atomic Number");
143 ASSERT(World::getInstance().getPeriode()->FindElement(Z),"No Element with this number in Periodentafel");
144 return elementCounts.size()>=Z && elementCounts[Z-1];
145}
146
147bool Formula::hasElement(const string &shorthand) const{
148 const element * element = World::getInstance().getPeriode()->FindElement(shorthand);
149 return hasElement(element);
150}
151
152void Formula::operator+=(const element *element){
153 ASSERT(element,"Invalid pointer in increment of Formula");
154 operator+=(element->getNumber());
155}
156
157void Formula::operator+=(atomicNumber_t Z){
158 ASSERT(Z>0,"Invalid atomic Number");
159 ASSERT(World::getInstance().getPeriode()->FindElement(Z),"No Element with this number in Periodentafel");
160 elementCounts.resize(max<atomicNumber_t>(Z,elementCounts.size()),0); // No-op when we already have the right size
161 // might need to update number of elements
162 if(!elementCounts[Z-1]){
163 numElements++;
164 }
165 elementCounts[Z-1]++; // atomic numbers start at 1
166}
167
168void Formula::operator+=(const string &shorthand){
169 const element * element = World::getInstance().getPeriode()->FindElement(shorthand);
170 operator+=(element);
171}
172
173void Formula::operator-=(const element *element){
174 ASSERT(element,"Invalid pointer in decrement of Formula");
175 operator-=(element->getNumber());
176}
177
178void Formula::operator-=(atomicNumber_t Z){
179 ASSERT(Z>0,"Invalid atomic Number");
180 ASSERT(World::getInstance().getPeriode()->FindElement(Z),"No Element with this number in Periodentafel");
181 ASSERT(elementCounts.size()>=Z && elementCounts[Z-1], "Element not in Formula upon decrement");
182 elementCounts[Z-1]--; // atomic numbers start at 1
183 // might need to update number of elements
184 if(!elementCounts[Z-1]){
185 numElements--;
186 // resize the Array if this was at the last position
187 if(Z==elementCounts.size()){
188 // find the first element from the back that is not equal to zero
189 set_t::reverse_iterator riter = find_if(elementCounts.rbegin(),
190 elementCounts.rend(),
191 bind1st(not_equal_to<mapped_type>(),0));
192 // see how many elements are in this range
193 set_t::reverse_iterator::difference_type diff = riter - elementCounts.rbegin();
194 elementCounts.resize(elementCounts.size()-diff);
195 }
196 }
197}
198
199void Formula::operator-=(const string &shorthand){
200 const element * element = World::getInstance().getPeriode()->FindElement(shorthand);
201 operator-=(element);
202}
203
204void Formula::addElements(const element *element,unsigned int count){
205 ASSERT(element,"Invalid pointer in Formula::addElements(element*)");
206 addElements(element->getNumber(),count);
207}
208
209void Formula::addElements(atomicNumber_t Z,unsigned int count){
210 if(count==0) return;
211 ASSERT(Z>0,"Invalid atomic Number");
212 ASSERT(World::getInstance().getPeriode()->FindElement(Z),"No Element with this number in Periodentafel");
213 elementCounts.resize(max<atomicNumber_t>(Z,elementCounts.size()),0); // No-op when we already have the right size
214 // might need to update number of elements
215 if(!elementCounts[Z-1]){
216 numElements++;
217 }
218 elementCounts[Z-1]+=count;
219}
220
221void Formula::addElements(const string &shorthand,unsigned int count){
222 const element * element = World::getInstance().getPeriode()->FindElement(shorthand);
223 addElements(element,count);
224}
225
226void Formula::addFormula(const Formula &formula,unsigned int n){
227 for(Formula::const_iterator iter=formula.begin();iter!=formula.end();++iter){
228 this->addElements(iter->first,iter->second*n);
229 }
230}
231
232enumeration<Formula::key_type> Formula::enumerateElements() const{
233 enumeration<key_type> res(1);
234 for(Formula::const_iterator iter=begin();iter!=end();++iter){
235 res.add(iter->first);
236 }
237 return res;
238}
239
240const unsigned int Formula::operator[](const element *element) const{
241 ASSERT(element,"Invalid pointer in access of Formula");
242 return operator[](element->getNumber());
243}
244
245const unsigned int Formula::operator[](atomicNumber_t Z) const{
246 ASSERT(Z>0,"Invalid atomic Number");
247 ASSERT(World::getInstance().getPeriode()->FindElement(Z),"No Element with this number in Periodentafel");
248 if(elementCounts.size()<Z)
249 return 0;
250 return elementCounts[Z-1]; // atomic numbers start at 1
251}
252
253const unsigned int Formula::operator[](string shorthand) const{
254 const element * element = World::getInstance().getPeriode()->FindElement(shorthand);
255 return operator[](element);
256}
257
258bool Formula::operator==(const Formula &rhs) const{
259 // quick check... number of elements used
260 if(numElements != rhs.numElements){
261 return false;
262 }
263 // second quick check, size of vectors (== last element in formula)
264 if(elementCounts.size()!=rhs.elementCounts.size()){
265 return false;
266 }
267 // slow check: all elements
268 // direct access to internal structure means all element-counts have to be compared
269 // this avoids access to periodentafel to find elements though and is probably faster
270 // in total
271 return equal(elementCounts.begin(),
272 elementCounts.end(),
273 rhs.elementCounts.begin());
274}
275
276bool Formula::operator!=(const Formula &rhs) const{
277 return !operator==(rhs);
278}
279
280Formula::iterator Formula::begin(){
281 return iterator(elementCounts,0);
282}
283Formula::const_iterator Formula::begin() const{
284 // this is the only place where this is needed, so this is better than making it mutable
285 return const_iterator(const_cast<set_t&>(elementCounts),0);
286}
287Formula::iterator Formula::end(){
288 return iterator(elementCounts);
289}
290Formula::const_iterator Formula::end() const{
291 // this is the only place where this is needed, so this is better than making it mutable
292 return const_iterator(const_cast<set_t&>(elementCounts));
293}
294
295void Formula::clear(){
296 elementCounts.clear();
297 numElements = 0;
298}
299
300/**************** Iterator structure ********************/
301
302template <class result_type>
303Formula::_iterator<result_type>::_iterator(set_t &_set) :
304 set(&_set)
305{
306 pos=set->size();
307}
308
309template <class result_type>
310Formula::_iterator<result_type>::_iterator(set_t &_set,size_t _pos) :
311 set(&_set),pos(_pos)
312{
313 ASSERT(pos<=set->size(),"invalid position in iterator construction");
314 while(pos<set->size() && (*set)[pos]==0) ++pos;
315}
316
317template <class result_type>
318Formula::_iterator<result_type>::_iterator(const _iterator &rhs) :
319 set(rhs.set),pos(rhs.pos)
320{}
321
322template <class result_type>
323Formula::_iterator<result_type>::~_iterator(){}
324
325template <class result_type>
326Formula::_iterator<result_type>&
327Formula::_iterator<result_type>::operator=(const _iterator<result_type> &rhs){
328 set=rhs.set;
329 pos=rhs.pos;
330 return *this;
331}
332
333template <class result_type>
334bool
335Formula::_iterator<result_type>::operator==(const _iterator<result_type> &rhs){
336 return set==rhs.set && pos==rhs.pos;
337}
338
339template <class result_type>
340bool
341Formula::_iterator<result_type>::operator!=(const _iterator<result_type> &rhs){
342 return !operator==(rhs);
343}
344
345template <class result_type>
346Formula::_iterator<result_type>
347Formula::_iterator<result_type>::operator++(){
348 ASSERT(pos!=set->size(),"Incrementing Formula::iterator beyond end");
349 pos++;
350 while(pos<set->size() && (*set)[pos]==0) ++pos;
351 return *this;
352}
353
354template <class result_type>
355Formula::_iterator<result_type>
356Formula::_iterator<result_type>::operator++(int){
357 Formula::_iterator<result_type> retval = *this;
358 ++(*this);
359 return retval;
360}
361
362template <class result_type>
363Formula::_iterator<result_type>
364Formula::_iterator<result_type>::operator--(){
365 ASSERT(pos!=0,"Decrementing Formula::iterator beyond begin");
366 pos--;
367 while(pos>0 && (*set)[pos]==0) --pos;
368 return *this;
369}
370
371template <class result_type>
372Formula::_iterator<result_type>
373Formula::_iterator<result_type>::operator--(int){
374 Formula::_iterator<result_type> retval = *this;
375 --(*this);
376 return retval;
377}
378
379template <class result_type>
380result_type
381Formula::_iterator<result_type>::operator*(){
382 const element *element = World::getInstance().getPeriode()->FindElement(pos+1);
383 ASSERT(element,"Element with position of iterator not found");
384 return make_pair(element,(*set)[pos]);
385}
386
387template <class result_type>
388result_type*
389Formula::_iterator<result_type>::operator->(){
390 // no one can keep this value around, so a static is ok to avoid temporaries
391 static value_type value=make_pair(reinterpret_cast<element*>(0),0); // no default constructor for std::pair
392 const element *element = World::getInstance().getPeriode()->FindElement(pos+1);
393 ASSERT(element,"Element with position of iterator not found");
394 value = make_pair(element,(*set)[pos]);
395 return &value;
396}
397
398// explicit instantiation of all iterator template methods
399// this is quite ugly, but there is no better way unless we expose iterator implementation
400
401// instantiate Formula::iterator
402template Formula::iterator::_iterator(set_t&);
403template Formula::iterator::_iterator(set_t&,size_t);
404template Formula::iterator::_iterator(const Formula::iterator&);
405template Formula::iterator::~_iterator();
406template Formula::iterator &Formula::iterator::operator=(const Formula::iterator&);
407template bool Formula::iterator::operator==(const Formula::iterator&);
408template bool Formula::iterator::operator!=(const Formula::iterator&);
409template Formula::iterator Formula::iterator::operator++();
410template Formula::iterator Formula::iterator::operator++(int);
411template Formula::iterator Formula::iterator::operator--();
412template Formula::iterator Formula::iterator::operator--(int);
413template Formula::value_type Formula::iterator::operator*();
414template Formula::value_type *Formula::iterator::operator->();
415
416// instantiate Formula::const_iterator
417template Formula::const_iterator::_iterator(set_t&);
418template Formula::const_iterator::_iterator(set_t&,size_t);
419template Formula::const_iterator::_iterator(const Formula::const_iterator&);
420template Formula::const_iterator::~_iterator();
421template Formula::const_iterator &Formula::const_iterator::operator=(const Formula::const_iterator&);
422template bool Formula::const_iterator::operator==(const Formula::const_iterator&);
423template bool Formula::const_iterator::operator!=(const Formula::const_iterator&);
424template Formula::const_iterator Formula::const_iterator::operator++();
425template Formula::Formula::const_iterator Formula::const_iterator::operator++(int);
426template Formula::Formula::const_iterator Formula::const_iterator::operator--();
427template Formula::Formula::const_iterator Formula::const_iterator::operator--(int);
428template const Formula::value_type Formula::const_iterator::operator*();
429template const Formula::value_type *Formula::const_iterator::operator->();
430
431/********************** I/O of Formulas ************************************************/
432
433std::ostream &operator<<(std::ostream &ost,const Formula &formula){
434 ost << formula.toString();
435 return ost;
436}
Note: See TracBrowser for help on using the repository browser.