LORENE
evolution.C
1 /*
2  * Methods of template class Evolution
3  *
4  * (see file evolution.h for documentation).
5  *
6  */
7 
8 /*
9  * Copyright (c) 2004 Eric Gourgoulhon & Jerome Novak
10  *
11  * This file is part of LORENE.
12  *
13  * LORENE is free software; you can redistribute it and/or modify
14  * it under the terms of the GNU General Public License version 2
15  * as published by the Free Software Foundation.
16  *
17  * LORENE is distributed in the hope that it will be useful,
18  * but WITHOUT ANY WARRANTY; without even the implied warranty of
19  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20  * GNU General Public License for more details.
21  *
22  * You should have received a copy of the GNU General Public License
23  * along with LORENE; if not, write to the Free Software
24  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
25  *
26  */
27 
28 /*
29  * $Id: evolution.C,v 1.20 2025/03/18 14:16:32 j_novak Exp $
30  * $Log: evolution.C,v $
31  * Revision 1.20 2025/03/18 14:16:32 j_novak
32  * Corrected a missing include
33  *
34  * Revision 1.19 2025/03/18 14:06:02 j_novak
35  * Added extrapolation method (constant,linear or parabolic) in the Evolution template class.
36  *
37  * Revision 1.18 2014/10/13 08:52:38 j_novak
38  * Lorene classes and functions now belong to the namespace Lorene.
39  *
40  * Revision 1.17 2014/03/27 16:59:41 j_novak
41  * Added methods next_position(int) and previous_position(int). Changed (corrected + simplified) the interpolation method.
42  *
43  * Revision 1.16 2013/07/19 15:50:24 j_novak
44  * Implementation of the interpolation function for Evolution, with order=0, 1 or 2.
45  *
46  * Revision 1.15 2008/09/19 13:24:29 j_novak
47  * Added the third-order scheme for the time derivativre computation.
48  *
49  * Revision 1.14 2004/05/14 08:51:01 p_grandclement
50  * *** empty log message ***
51  *
52  * Revision 1.13 2004/05/14 08:41:05 e_gourgoulhon
53  * Added declaration "class Tbl ;" before the declarations of
54  * write_formatted.
55  *
56  * Revision 1.12 2004/05/13 21:30:32 e_gourgoulhon
57  * Use of function write_formatted in method save( ).
58  *
59  * Revision 1.11 2004/05/11 20:12:49 e_gourgoulhon
60  * Added methods j_min, j_max and save.
61  *
62  * Revision 1.10 2004/05/03 15:23:22 e_gourgoulhon
63  * Method downdate: changed the order of conditions (pos_jtop>=0)
64  * and (val[pos_jtop] == 0x0) in the while(...) test.
65  *
66  * Revision 1.9 2004/03/31 20:24:04 e_gourgoulhon
67  * Method time_derive: result object created by the copy constructor of
68  * class TyT, since the arithmetics may not return an object of exactly
69  * class TyT.
70  *
71  * Revision 1.8 2004/03/26 13:31:09 j_novak
72  * Definition of the macro UNDEF_STEP for non-defined time-steps.
73  * Changes in the way the time derivative is calculated.
74  *
75  * Revision 1.7 2004/03/26 08:22:13 e_gourgoulhon
76  * *** Full reorganization of class Evolution ***
77  * Introduction of the notion of absoluteuniversal time steps,
78  * stored in the new array 'step'.
79  * The new function position(int j) makes a correspondence
80  * between a universal time step j and the position in the
81  * arrays step, the_time and val.
82  * Only method update is now virtual.
83  * Methods operator[], position, is_known, downdate belong to
84  * the base class.
85  *
86  * Revision 1.6 2004/03/24 14:55:47 e_gourgoulhon
87  * Added method last_value().
88  *
89  * Revision 1.5 2004/03/23 14:50:41 e_gourgoulhon
90  * Added methods is_updated, downdate, get_jlast, get_size,
91  * as well as constructors without any initial value.
92  * Formatted documentation for Doxygen.
93  *
94  * Revision 1.4 2004/03/06 21:13:15 e_gourgoulhon
95  * Added time derivation (method time_derive).
96  *
97  * Revision 1.3 2004/02/17 22:13:34 e_gourgoulhon
98  * Suppressed declaration of global char[] evolution_C = ...
99  *
100  * Revision 1.2 2004/02/15 21:55:33 e_gourgoulhon
101  * Introduced derived classes Evolution_full and Evolution_std.
102  * Evolution is now an abstract base class.
103  *
104  * Revision 1.1 2004/02/13 15:53:20 e_gourgoulhon
105  * New (template) class for time evolution.
106  *
107  *
108  * $Header: /cvsroot/Lorene/C++/Include/Template/evolution.C,v 1.20 2025/03/18 14:16:32 j_novak Exp $
109  *
110  */
111 
112 // C headers
113 #include <cassert>
114 
115 // C++ headers
116 #include "headcpp.h"
117 
118 namespace Lorene {
119 class Tbl ;
120 
121 void write_formatted(const double&, ostream& ) ;
122 void write_formatted(const Tbl&, ostream& ) ;
123 
124 
125  //-------------------------//
126  // Constructors //
127  //-------------------------//
128 
129 
130 template<typename TyT>
131 Evolution<TyT>::Evolution(const TyT& initial_value, int initial_step,
132  double initial_time, int size_i)
133  : size(size_i),
134  pos_jtop(0) {
135 
136  step = new int[size] ;
137  step[0] = initial_step ;
138  for (int j=1; j<size; j++) {
139  step[j] = UNDEF_STEP ;
140  }
141 
142  the_time = new double[size] ;
143  the_time[0] = initial_time ;
144  for (int j=1; j<size; j++) {
145  the_time[j] = -1e20 ;
146  }
147 
148  val = new TyT*[size] ;
149  val[0] = new TyT(initial_value) ;
150  for (int j=1; j<size; j++) {
151  val[j] = 0x0 ;
152  }
153 
154 }
155 
156 
157 template<typename TyT>
159  : size(size_i),
160  pos_jtop(-1) {
161 
162  step = new int[size] ;
163  for (int j=0; j<size; j++) {
164  step[j] = UNDEF_STEP ;
165  }
166 
167  the_time = new double[size] ;
168  for (int j=0; j<size; j++) {
169  the_time[j] = -1e20 ;
170  }
171 
172  val = new TyT*[size] ;
173  for (int j=0; j<size; j++) {
174  val[j] = 0x0 ;
175  }
176 
177 }
178 
179 
180 
181 template<typename TyT>
183  : size(evo.size),
184  pos_jtop(evo.pos_jtop) {
185 
186  step = new int[size] ;
187  for (int j=0; j<size; j++) {
188  step[j] = evo.step[j] ;
189  }
190 
191  the_time = new double[size] ;
192  for (int j=0; j<size; j++) {
193  the_time[j] = evo.the_time[j] ;
194  }
195 
196  val = new TyT*[size] ;
197  for (int j=0; j<size; j++) {
198  if (evo.val[j] != 0x0) {
199  val[j] = new TyT( *(evo.val[j]) ) ;
200  }
201  else {
202  val[j] = 0x0 ;
203  }
204  }
205 
206 
207 }
208 
209 
210 
211 
212  //-----------------------//
213  // Destructor //
214  //-----------------------//
215 
216 
217 template<typename TyT>
219 
220  delete [] step ;
221  delete [] the_time ;
222 
223  for (int j=0; j<size; j++) {
224  if (val[j] != 0x0) delete val[j] ;
225  }
226 
227  delete [] val ;
228 
229 }
230 
231 
232 
233  //-----------------------//
234  // Mutators //
235  //-----------------------//
236 
237 
238 template<typename TyT>
240 
241  cerr << "void Evolution<TyT>::operator= : not implemented yet ! \n" ;
242  abort() ;
243 
244 }
245 
246 
247 template<typename TyT>
249 
250  if ( !(is_known(j)) ) return ; // a never updated step cannot
251  // be downdated
252 
253  int pos = position(j) ;
254 
255  assert( val[pos] != 0x0) ;
256 
257  delete val[pos] ;
258  val[pos] = 0x0 ;
259  step[pos] = UNDEF_STEP ;
260  the_time[pos] = -1e20 ;
261 
262  if (pos == pos_jtop) { // pos_jtop must be decreased
263  pos_jtop-- ;
264  while ( (pos_jtop>=0) && (val[pos_jtop] == 0x0) ) pos_jtop-- ;
265  }
266 
267 }
268 
269 
270 
271 
272  //------------//
273  // Accessors //
274  //------------//
275 
276 template<typename TyT>
277 int Evolution<TyT>::position(int j) const {
278 
279  assert(pos_jtop >= 0) ;
280  int jmax = step[pos_jtop] ;
281 
282  if (j == jmax) return pos_jtop ; // for efficiency purpose
283 
284  int pos = - 1 ;
285 
286  if ( (j>=step[0]) && (j<jmax) ) {
287 
288  for (int i=pos_jtop-1; i>=0; i--) { // cas i=pos_jtop treated above
289  if (step[i] == j) {
290  pos = i ;
291  break ;
292  }
293  }
294  }
295 
296  if (pos == -1) {
297  cerr << "Evolution<TyT>::position: time step j = " <<
298  j << " not found !" << endl ;
299  abort() ;
300  }
301 
302  return pos ;
303 }
304 
305 template<typename TyT>
307 
308  assert( (j>=0) && (j<=pos_jtop) ) ;
309  assert( step[j] != UNDEF_STEP ) ;
310 
311  int n_pos = -1 ;
312 
313  while ( (n_pos == -1) && ( j < pos_jtop ) ) {
314 
315  j++ ;
316  if (step[j] != UNDEF_STEP) n_pos = j ;
317  }
318  return n_pos ;
319 }
320 
321 template<typename TyT>
323 
324  assert( (j>=0) && (j<=pos_jtop) ) ;
325  assert( step[j] != UNDEF_STEP ) ;
326 
327  int n_pos = -1 ;
328 
329  while ( (n_pos == -1) && ( j > 0 ) ) {
330 
331  j-- ;
332  if (step[j] != UNDEF_STEP) n_pos = j ;
333  }
334  return n_pos ;
335 }
336 
337 
338 template<typename TyT>
339 bool Evolution<TyT>::is_known(int j) const {
340 
341  if (pos_jtop == -1) return false ;
342 
343  assert(pos_jtop >= 0) ;
344 
345  int jmax = step[pos_jtop] ;
346 
347  if (j == jmax) {
348  return ( val[pos_jtop] != 0x0 ) ;
349  }
350 
351  if ( (j>=step[0]) && (j<jmax) ) {
352 
353  for (int i=pos_jtop-1; i>=0; i--) { // cas i=pos_jtop treated above
354 
355  if (step[i] == j) return ( val[i] != 0x0 ) ;
356  }
357  }
358 
359  return false ;
360 }
361 
362 
363 template<typename TyT>
364 const TyT& Evolution<TyT>::operator[](int j) const {
365 
366  TyT* pval = val[position(j)] ;
367  assert(pval != 0x0) ;
368  return *pval ;
369 
370 }
371 
372 
373 template<typename TyT>
374 TyT Evolution<TyT>::operator()(double t, int order) const {
375 
376  int imin = position( j_min() ) ;
377 
378  if ( ( t < the_time[ imin ] ) || ( t > the_time[pos_jtop] ) ) {
379  cerr << "Evolution<TyT>::operator()(double t, int order) : \n"
380  << "Requested time outside stored range!" << endl ;
381  abort() ;
382  }
383  assert ( order <= 2 ) ;
384  assert ( pos_jtop > order ) ;
385 
386  int j0 = imin ;
387 
388  while ((the_time[j0] < t) && ( j0 < pos_jtop )) {
389  j0 = next_position(j0) ;
390  assert( j0 != -1 ) ;
391  }
392 
393  switch (order) {
394 
395  case 0: {
396 
397  return (*val[j0]) ;
398 
399  break ;
400  }
401 
402  case 1: {
403 
404  int j1 = ( (j0 > imin) ? previous_position(j0) : next_position(j0) ) ;
405  assert( j1 != -1) ;
406 
407  double x0 = the_time[j1] - t ;
408  double x1 = the_time[j0] - t ;
409  double dx = the_time[j0] - the_time[j1] ;
410 
411  double a0 = x1 / dx ;
412  double a1 = x0 / dx ;
413 
414  return (a0*(*val[j0]) - a1*(*val[j1])) ;
415 
416  break ;
417  }
418 
419  case 2: {
420 
421  int j1 = ( (j0 == imin) ? next_position(j0) : previous_position(j0) ) ;
422  assert( j1 != -1) ;
423 
424  int j2 = -1 ;
425  if (j0 == imin)
426  j2 = next_position(j1) ;
427  else {
428  if (j1 == imin)
429  j2 = next_position(j0) ;
430  else
431  j2 = previous_position(j1) ;
432  }
433  assert (j2 != -1 ) ;
434 
435  double x0 = t - the_time[j0] ;
436  double x1 = t - the_time[j1] ;
437  double dx = the_time[j0] - the_time[j1] ;
438 
439  double x2 = t - the_time[j2] ;
440 
441  double dx0 = the_time[j1] - the_time[j2] ;
442  double dx1 = the_time[j0] - the_time[j2] ;
443  double dx2 = dx ;
444 
445  double a0 = ( x2*x1 ) / ( dx2*dx1 ) ;
446  double a1 = ( x0*x2 ) / ( dx0*dx2 ) ;
447  double a2 = ( x0*x1 ) / ( dx0*dx1 ) ;
448 
449  return ( a0*(*val[j0]) - a1*(*val[j1]) + a2*(*val[j2]) ) ;
450 
451  break ;
452  }
453 
454  default: {
455  cerr << " Evolution<TyT>::operator()(double t, int order) : \n" << endl ;
456  cerr << " The case order = " << order << " is not implemented!" << endl ;
457  abort() ;
458  break ;
459  }
460  }
461 
462  return *val[j0] ;
463 
464 }
465 
466 template<typename TyT>
467 TyT Evolution<TyT>::extrapolate(double t, int order) const {
468 
469  if ( t < the_time[pos_jtop] ) {
470  cerr << "Evolution<TyT>::extrapolate(double t, int order) : \n"
471  << "Requested time is inside stored range..." << endl ;
472  abort() ;
473  }
474  assert ( pos_jtop >= order ) ;
475 
476  switch (order) {
477 
478  case 0: {
479  return *val[pos_jtop];
480  break ;
481  }
482 
483  case 1: {
484  int j0 = UNDEF_STEP ;
485  for (int i=pos_jtop-1; i>=0; i--) {
486  if (step[i] != UNDEF_STEP) {
487  j0 = i ;
488  break ;
489  }
490  }
491  assert(j0 != UNDEF_STEP) ;
492  int j1 = pos_jtop ;
493 
494  double fac = (t - the_time[j0]) / (the_time[j1] - the_time[j0]) ;
495  return fac*(*val[j1]) + (1. - fac)*(*val[j0]) ;
496  break ;
497  }
498 
499  case 2: {
500  int j2 = pos_jtop ;
501  int j1 = UNDEF_STEP ;
502  for (int i=j2-1; i>=0; i--) {
503  if (step[i] != UNDEF_STEP) {
504  j1 = i ;
505  break ;
506  }
507  }
508  assert(j1 != UNDEF_STEP) ;
509  int j0 = UNDEF_STEP ;
510  for (int i=j1-1; i>=0; i--) {
511  if (step[i] != UNDEF_STEP) {
512  j0 = i ;
513  break ;
514  }
515  }
516  assert(j0 != UNDEF_STEP) ;
517 
518  double t0 = the_time[j0] ;
519  double t1 = the_time[j1] ;
520  double t2 = the_time[j2] ;
521  double t01 = t0 - t1 ;
522  double t02 = t0 - t2 ;
523  double t12 = t1 - t2 ;
524 
525  return ((t - t1)*(t - t2)/(t01*t02))*(*val[j0])
526  - ((t - t0)*(t - t2)/(t01*t12))*(*val[j1])
527  + ((t - t0)*(t - t1)/(t02*t12))*(*val[j2]) ;
528  break ;
529  }
530 
531  default: {
532  cerr << " Evolution<TyT>::extrapolate(double t, int order) : \n" << endl ;
533  cerr << " The case order = " << order << " is not implemented!" << endl ;
534  abort() ;
535  break ;
536  }
537  }
538 
539  return *val[pos_jtop] ;
540 }
541 
542 
543 
544 template<typename TyT>
546 
547  int resu = UNDEF_STEP ;
548  for (int i=0; i<=pos_jtop; i++) {
549  if (step[i] != UNDEF_STEP) {
550  resu = step[i] ;
551  break ;
552  }
553  }
554 
555  if (resu == UNDEF_STEP) {
556  cerr << "Evolution<TyT>::j_min() : no valid time step found !" << endl ;
557  abort() ;
558  }
559 
560  return resu ;
561 }
562 
563 template<typename TyT>
565 
566  if (pos_jtop == -1) {
567  cerr << "Evolution<TyT>::j_max() : no valid time step found !" << endl ;
568  abort() ;
569  }
570 
571  assert(pos_jtop >=0) ;
572  int jmax = step[pos_jtop] ;
573  assert(jmax != UNDEF_STEP) ;
574  return jmax ;
575 }
576 
577 
578 
579 
580 
581  //-----------------------//
582  // Time derivative //
583  //-----------------------//
584 
585 template<typename TyT>
586 TyT Evolution<TyT>::time_derive(int j, int n) const {
587 
588  if (n == 0) {
589  TyT resu ( operator[](j) ) ;
590  resu = 0 * resu ;
591 
592  return resu ;
593 
594  }
595 
596  else {
597 
598  int pos = position(j) ;
599  assert ( pos > 0 ) ;
600  assert ( step[pos-1] != UNDEF_STEP ) ;
601 
602  switch (n) {
603 
604  case 1 : {
605 
606  double dt = the_time[pos] - the_time[pos-1] ;
607 
608  return ( (*val[pos]) - (*val[pos-1]) ) / dt ;
609  break ;
610  }
611 
612  case 2 : {
613 
614  assert ( pos > 1 ) ;
615  assert ( step[pos-2] != UNDEF_STEP ) ;
616  double dt = the_time[pos] - the_time[pos-1] ;
617 #ifndef NDEBUG
618  double dt2 = the_time[pos-1] - the_time[pos-2] ;
619  if (fabs(dt2 -dt) > 1.e-13) {
620  cerr <<
621  "Evolution<TyT>::time_derive: the current version is"
622  << " valid only for \n"
623  << " a constant time step !" << endl ;
624  abort() ;
625  }
626 #endif
627  return ( 1.5* (*val[pos]) - 2.* (*val[pos-1]) + 0.5* (*val[pos-2]) ) / dt ;
628  break ;
629  }
630 
631  case 3 : {
632 
633  assert ( pos > 2 ) ;
634  assert ( step[pos-2] != UNDEF_STEP ) ;
635  assert ( step[pos-3] != UNDEF_STEP ) ;
636  double dt = the_time[pos] - the_time[pos-1] ;
637 #ifndef NDEBUG
638  double dt2 = the_time[pos-1] - the_time[pos-2] ;
639  double dt3 = the_time[pos-2] - the_time[pos-3] ;
640  if ((fabs(dt2 -dt) > 1.e-13)||(fabs(dt3 -dt2) > 1.e-13)) {
641  cerr <<
642  "Evolution<TyT>::time_derive: the current version is valid only for \n"
643  << " a constant time step !" << endl ;
644  abort() ;
645  }
646 #endif
647  return ( 11.*(*val[pos]) - 18.*(*val[pos-1]) + 9.*(*val[pos-2])
648  - 2.*(*val[pos-3])) / (6.*dt) ;
649  break ;
650  }
651  default : {
652  cerr << "Evolution<TyT>::time_derive: the case n = " << n
653  << " is not implemented !" << endl ;
654  abort() ;
655  break ;
656  }
657  }
658 
659  }
660  return operator[](j) ;
661 }
662 
663 
664 
665  //-----------------------//
666  // Outputs //
667  //-----------------------//
668 
669 
670 template<typename TyT>
671 void Evolution<TyT>::save(const char* filename) const {
672 
673  ofstream fich(filename) ;
674 
675  time_t temps = time(0x0) ;
676 
677  fich << "# " << filename << " " << ctime(&temps) ;
678  fich << "# " << size << " size" << '\n' ;
679  fich << "# " << pos_jtop << " pos_jtop" << '\n' ;
680  fich << "# t value... \n" ;
681 
682  fich.precision(14) ;
683  fich.setf(ios::scientific) ;
684  fich.width(20) ;
685  for (int i=0; i<=pos_jtop; i++) {
686  if (step[i] != UNDEF_STEP) {
687  fich << the_time[i] ; fich.width(23) ;
688  assert(val[i] != 0x0) ;
689  write_formatted(*(val[i]), fich) ;
690  fich << '\n' ;
691  }
692  }
693 
694  fich.close() ;
695 }
696 
697 
698 
699 
700 
701 
702 
703 
704 
705 
706 
707 
708 
709 
710 
711 
712 
713 
714 }
Lorene prototypes.
Definition: app_hor.h:67
TyT ** val
Array of pointers onto the values (size = size).
Definition: evolution.h:140
Evolution(const TyT &initial_value, int initial_j, double initial_time, int initial_size)
Constructor from some initial value.
Definition: evolution.C:131
int * step
Array of time step indices (size = size).
Definition: evolution.h:134
double * the_time
Array of values of t at the various time steps (size = size).
Definition: evolution.h:137
Time evolution.
Definition: evolution.h:123