-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathstep-3.cc
680 lines (618 loc) · 34.4 KB
/
step-3.cc
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
/* ---------------------------------------------------------------------
*
* Copyright (C) 1999 - 2022 by the deal.II authors
*
* This file is part of the deal.II library.
*
* The deal.II library is free software; you can use it, redistribute
* it, and/or modify it under the terms of the GNU Lesser General
* Public License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
* The full text of the license can be found in the file LICENSE.md at
* the top level directory of deal.II.
*
* ---------------------------------------------------------------------
*
* Authors: Wolfgang Bangerth, 1999,
* Guido Kanschat, 2011
*/
// @sect3{Many new include files}
// These include files are already known to you. They declare the classes
// which handle triangulations and enumeration of degrees of freedom:
#include <deal.II/grid/tria.h>
#include <deal.II/dofs/dof_handler.h>
// And this is the file in which the functions are declared that create grids:
#include <deal.II/grid/grid_generator.h>
// This file contains the description of the Lagrange interpolation finite
// element:
#include <deal.II/fe/fe_q.h>
// And this file is needed for the creation of sparsity patterns of sparse
// matrices, as shown in previous examples:
#include <deal.II/dofs/dof_tools.h>
// The next two files are needed for assembling the matrix using quadrature on
// each cell. The classes declared in them will be explained below:
#include <deal.II/fe/fe_values.h>
#include <deal.II/base/quadrature_lib.h>
// The following three include files we need for the treatment of boundary
// values:
#include <deal.II/base/function.h>
#include <deal.II/numerics/vector_tools.h>
#include <deal.II/numerics/matrix_tools.h>
// We're now almost to the end. The second to last group of include files is
// for the linear algebra which we employ to solve the system of equations
// arising from the finite element discretization of the Laplace equation. We
// will use vectors and full matrices for assembling the system of equations
// locally on each cell, and transfer the results into a sparse matrix. We
// will then use a Conjugate Gradient solver to solve the problem, for which
// we need a preconditioner (in this program, we use the identity
// preconditioner which does nothing, but we need to include the file anyway):
#include <deal.II/lac/vector.h>
#include <deal.II/lac/full_matrix.h>
#include <deal.II/lac/sparse_matrix.h>
#include <deal.II/lac/dynamic_sparsity_pattern.h>
#include <deal.II/lac/solver_cg.h>
#include <deal.II/lac/precondition.h>
// Finally, this is for output to a file and to the console:
#include <deal.II/numerics/data_out.h>
#include <fstream>
#include <iostream>
// ...and this is to import the deal.II namespace into the global scope:
using namespace dealii;
// @sect3{The <code>Step3</code> class}
// Instead of the procedural programming of previous examples, we encapsulate
// everything into a class for this program. The class consists of functions
// which each perform certain aspects of a finite element program, a `main`
// function which controls what is done first and what is done next, and a
// list of member variables.
// The public part of the class is rather short: it has a constructor and a
// function `run` that is called from the outside and acts as something like
// the `main` function: it coordinates which operations of this class shall be
// run in which order. Everything else in the class, i.e. all the functions
// that actually do anything, are in the private section of the class:
class Step3
{
public:
Step3();
void run();
// Then there are the member functions that mostly do what their names
// suggest and whose have been discussed in the introduction already. Since
// they do not need to be called from outside, they are made private to this
// class.
private:
void make_grid();
void setup_system();
void assemble_system();
void solve();
void output_results() const;
// And finally we have some member variables. There are variables describing
// the triangulation and the global numbering of the degrees of freedom (we
// will specify the exact polynomial degree of the finite element in the
// constructor of this class)...
Triangulation<2> triangulation;
FE_Q<2> fe;
DoFHandler<2> dof_handler;
// ...variables for the sparsity pattern and values of the system matrix
// resulting from the discretization of the Laplace equation...
SparsityPattern sparsity_pattern;
SparseMatrix<double> system_matrix;
// ...and variables which will hold the right hand side and solution
// vectors.
Vector<double> solution;
Vector<double> system_rhs;
};
// @sect4{Step3::Step3}
// Here comes the constructor. It does not much more than first to specify
// that we want bi-linear elements (denoted by the parameter to the finite
// element object, which indicates the polynomial degree), and to associate
// the dof_handler variable to the triangulation we use. (Note that the
// triangulation isn't set up with a mesh at all at the present time, but the
// DoFHandler doesn't care: it only wants to know which triangulation it will
// be associated with, and it only starts to care about an actual mesh once
// you try to distribute degree of freedom on the mesh using the
// distribute_dofs() function.) All the other member variables of the Step3
// class have a default constructor which does all we want.
Step3::Step3()
: fe(1)
, dof_handler(triangulation)
{}
// @sect4{Step3::make_grid}
// Now, the first thing we've got to do is to generate the triangulation on
// which we would like to do our computation and number each vertex with a
// degree of freedom. We have seen these two steps in step-1 and step-2
// before, respectively.
//
// This function does the first part, creating the mesh. We create the grid
// and refine all cells five times. Since the initial grid (which is the
// square $[-1,1] \times [-1,1]$) consists of only one cell, the final grid
// has 32 times 32 cells, for a total of 1024.
//
// Unsure that 1024 is the correct number? We can check that by outputting the
// number of cells using the <code>n_active_cells()</code> function on the
// triangulation.
void Step3::make_grid()
{
GridGenerator::hyper_cube(triangulation, -1, 1);
triangulation.refine_global(5);
std::cout << "Number of active cells: " << triangulation.n_active_cells()
<< std::endl;
}
// @note We call the Triangulation::n_active_cells() function, rather than
// Triangulation::n_cells(). Here, <i>active</i> means the cells that aren't
// refined any further. We stress the adjective "active" since there are more
// cells, namely the parent cells of the finest cells, their parents, etc, up
// to the one cell which made up the initial grid. Of course, on the next
// coarser level, the number of cells is one quarter that of the cells on the
// finest level, i.e. 256, then 64, 16, 4, and 1. If you called
// <code>triangulation.n_cells()</code> instead in the code above, you would
// consequently get a value of 1365 instead. On the other hand, the number of
// cells (as opposed to the number of active cells) is not typically of much
// interest, so there is no good reason to print it.
// @sect4{Step3::setup_system}
// Next we enumerate all the degrees of freedom and set up matrix and vector
// objects to hold the system data. Enumerating is done by using
// DoFHandler::distribute_dofs(), as we have seen in the step-2 example. Since
// we use the FE_Q class and have set the polynomial degree to 1 in the
// constructor, i.e. bilinear elements, this associates one degree of freedom
// with each vertex. While we're at generating output, let us also take a look
// at how many degrees of freedom are generated:
void Step3::setup_system()
{
dof_handler.distribute_dofs(fe);
std::cout << "Number of degrees of freedom: " << dof_handler.n_dofs()
<< std::endl;
// There should be one DoF for each vertex. Since we have a 32 times 32
// grid, the number of DoFs should be 33 times 33, or 1089.
// As we have seen in the previous example, we set up a sparsity pattern by
// first creating a temporary structure, tagging those entries that might be
// nonzero, and then copying the data over to the SparsityPattern object
// that can then be used by the system matrix.
DynamicSparsityPattern dsp(dof_handler.n_dofs());
DoFTools::make_sparsity_pattern(dof_handler, dsp);
sparsity_pattern.copy_from(dsp);
// Note that the SparsityPattern object does not hold the values of the
// matrix, it only stores the places where entries are. The entries
// themselves are stored in objects of type SparseMatrix, of which our
// variable system_matrix is one.
//
// The distinction between sparsity pattern and matrix was made to allow
// several matrices to use the same sparsity pattern. This may not seem
// relevant here, but when you consider the size which matrices can have,
// and that it may take some time to build the sparsity pattern, this
// becomes important in large-scale problems if you have to store several
// matrices in your program.
system_matrix.reinit(sparsity_pattern);
// The last thing to do in this function is to set the sizes of the right
// hand side vector and the solution vector to the right values:
solution.reinit(dof_handler.n_dofs());
system_rhs.reinit(dof_handler.n_dofs());
}
// @sect4{Step3::assemble_system}
// The next step is to compute the entries of the matrix and right hand side
// that form the linear system from which we compute the solution. This is the
// central function of each finite element program and we have discussed the
// primary steps in the introduction already.
//
// The general approach to assemble matrices and vectors is to loop over all
// cells, and on each cell compute the contribution of that cell to the global
// matrix and right hand side by quadrature. The point to realize now is that
// we need the values of the shape functions at the locations of quadrature
// points on the real cell. However, both the finite element shape functions
// as well as the quadrature points are only defined on the reference
// cell. They are therefore of little help to us, and we will in fact hardly
// ever query information about finite element shape functions or quadrature
// points from these objects directly.
//
// Rather, what is required is a way to map this data from the reference cell
// to the real cell. Classes that can do that are derived from the Mapping
// class, though one again often does not have to deal with them directly:
// many functions in the library can take a mapping object as argument, but
// when it is omitted they simply resort to the standard bilinear Q1
// mapping. We will go this route, and not bother with it for the moment (we
// come back to this in step-10, step-11, and step-12).
//
// So what we now have is a collection of three classes to deal with: finite
// element, quadrature, and mapping objects. That's too much, so there is one
// type of class that orchestrates information exchange between these three:
// the FEValues class. If given one instance of each three of these objects
// (or two, and an implicit linear mapping), it will be able to provide you
// with information about values and gradients of shape functions at
// quadrature points on a real cell.
//
// Using all this, we will assemble the linear system for this problem in the
// following function:
void Step3::assemble_system()
{
// Ok, let's start: we need a quadrature formula for the evaluation of the
// integrals on each cell. Let's take a Gauss formula with two quadrature
// points in each direction, i.e. a total of four points since we are in
// 2D. This quadrature formula integrates polynomials of degrees up to three
// exactly (in 1D). It is easy to check that this is sufficient for the
// present problem:
QGauss<2> quadrature_formula(fe.degree + 1);
// And we initialize the object which we have briefly talked about above. It
// needs to be told which finite element we want to use, and the quadrature
// points and their weights (jointly described by a Quadrature object). As
// mentioned, we use the implied Q1 mapping, rather than specifying one
// ourselves explicitly. Finally, we have to tell it what we want it to
// compute on each cell: we need the values of the shape functions at the
// quadrature points (for the right hand side $(\varphi_i,f)$), their
// gradients (for the matrix entries $(\nabla \varphi_i, \nabla
// \varphi_j)$), and also the weights of the quadrature points and the
// determinants of the Jacobian transformations from the reference cell to
// the real cells.
//
// This list of what kind of information we actually need is given as a
// collection of flags as the third argument to the constructor of
// FEValues. Since these values have to be recomputed, or updated, every
// time we go to a new cell, all of these flags start with the prefix
// <code>update_</code> and then indicate what it actually is that we want
// updated. The flag to give if we want the values of the shape functions
// computed is #update_values; for the gradients it is
// #update_gradients. The determinants of the Jacobians and the quadrature
// weights are always used together, so only the products (Jacobians times
// weights, or short <code>JxW</code>) are computed; since we need them, we
// have to list #update_JxW_values as well:
FEValues<2> fe_values(fe,
quadrature_formula,
update_values | update_gradients | update_JxW_values);
// The advantage of this approach is that we can specify what kind of
// information we actually need on each cell. It is easily understandable
// that this approach can significantly speed up finite element computations,
// compared to approaches where everything, including second derivatives,
// normal vectors to cells, etc are computed on each cell, regardless of
// whether they are needed or not.
//
// @note The syntax <code>update_values | update_gradients |
// update_JxW_values</code> is not immediately obvious to anyone not
// used to programming bit operations in C for years already. First,
// <code>operator|</code> is the <i>bitwise or operator</i>, i.e.,
// it takes two integer arguments that are interpreted as bit
// patterns and returns an integer in which every bit is set for
// which the corresponding bit is set in at least one of the two
// arguments. For example, consider the operation
// <code>9|10</code>. In binary, <code>9=0b1001</code> (where the
// prefix <code>0b</code> indicates that the number is to be
// interpreted as a binary number) and <code>10=0b1010</code>. Going
// through each bit and seeing whether it is set in one of the
// argument, we arrive at <code>0b1001|0b1010=0b1011</code> or, in
// decimal notation, <code>9|10=11</code>. The second piece of
// information you need to know is that the various
// <code>update_*</code> flags are all integers that have <i>exactly
// one bit set</i>. For example, assume that
// <code>update_values=0b00001=1</code>,
// <code>update_gradients=0b00010=2</code>,
// <code>update_JxW_values=0b10000=16</code>. Then
// <code>update_values | update_gradients | update_JxW_values =
// 0b10011 = 19</code>. In other words, we obtain a number that
// <i>encodes a binary mask representing all of the operations you
// want to happen</i>, where each operation corresponds to exactly
// one bit in the integer that, if equal to one, means that a
// particular piece should be updated on each cell and, if it is
// zero, means that we need not compute it. In other words, even
// though <code>operator|</code> is the <i>bitwise OR operation</i>,
// what it really represents is <i>I want this AND that AND the
// other</i>. Such binary masks are quite common in C programming,
// but maybe not so in higher level languages like C++, but serve
// the current purpose quite well.
// For use further down below, we define a shortcut for a value that will
// be used very frequently. Namely, an abbreviation for the number of degrees
// of freedom on each cell (since we are in 2D and degrees of freedom are
// associated with vertices only, this number is four, but we rather want to
// write the definition of this variable in a way that does not preclude us
// from later choosing a different finite element that has a different
// number of degrees of freedom per cell, or work in a different space
// dimension).
//
// In general, it is a good idea to use a symbolic name instead of
// hard-coding these numbers even if you know them, since for example,
// you may want to change the finite element at some time. Changing the
// element would have to be done in a different function and it is easy
// to forget to make a corresponding change in another part of the program.
// It is better to not rely on your own calculations, but instead ask
// the right object for the information: Here, we ask the finite element
// to tell us about the number of degrees of freedom per cell and we
// will get the correct number regardless of the space dimension or
// polynomial degree we may have chosen elsewhere in the program.
//
// The shortcut here, defined primarily to discuss the basic concept
// and not because it saves a lot of typing, will then make the following
// loops a bit more readable. You will see such shortcuts in many places in
// larger programs, and `dofs_per_cell` is one that is more or less the
// conventional name for this kind of object.
const unsigned int dofs_per_cell = fe.n_dofs_per_cell();
// Now, we said that we wanted to assemble the global matrix and vector
// cell-by-cell. We could write the results directly into the global matrix,
// but this is not very efficient since access to the elements of a sparse
// matrix is slow. Rather, we first compute the contribution of each cell in
// a small matrix with the degrees of freedom on the present cell, and only
// transfer them to the global matrix when the computations are finished for
// this cell. We do the same for the right hand side vector. So let's first
// allocate these objects (these being local objects, all degrees of freedom
// are coupling with all others, and we should use a full matrix object
// rather than a sparse one for the local operations; everything will be
// transferred to a global sparse matrix later on):
FullMatrix<double> cell_matrix(dofs_per_cell, dofs_per_cell);
Vector<double> cell_rhs(dofs_per_cell);
// When assembling the contributions of each cell, we do this with the local
// numbering of the degrees of freedom (i.e. the number running from zero
// through dofs_per_cell-1). However, when we transfer the result into the
// global matrix, we have to know the global numbers of the degrees of
// freedom. When we query them, we need a scratch (temporary) array for
// these numbers (see the discussion at the end of the introduction for
// the type, types::global_dof_index, used here):
std::vector<types::global_dof_index> local_dof_indices(dofs_per_cell);
// Now for the loop over all cells. We have seen before how this works for a
// triangulation. A DoFHandler has cell iterators that are exactly analogous
// to those of a Triangulation, but with extra information about the degrees
// of freedom for the finite element you're using. Looping over the active
// cells of a degree-of-freedom handler works the same as for a triangulation.
//
// Note that we declare the type of the cell as `const auto &` instead of
// `auto` this time around. In step 1, we were modifying the cells of the
// triangulation by flagging them with refinement indicators. Here we're only
// examining the cells without modifying them, so it's good practice to
// declare `cell` as `const` in order to enforce this invariant.
for (const auto &cell : dof_handler.active_cell_iterators())
{
// We are now sitting on one cell, and we would like the values and
// gradients of the shape functions be computed, as well as the
// determinants of the Jacobian matrices of the mapping between
// reference cell and true cell, at the quadrature points. Since all
// these values depend on the geometry of the cell, we have to have the
// FEValues object re-compute them on each cell:
fe_values.reinit(cell);
// Next, reset the local cell's contributions to global matrix and
// global right hand side to zero, before we fill them:
cell_matrix = 0;
cell_rhs = 0;
// Now it is time to start integration over the cell, which we
// do by looping over all quadrature points, which we will
// number by q_index.
for (const unsigned int q_index : fe_values.quadrature_point_indices())
{
// First assemble the matrix: For the Laplace problem, the
// matrix on each cell is the integral over the gradients of
// shape function i and j. Since we do not integrate, but
// rather use quadrature, this is the sum over all
// quadrature points of the integrands times the determinant
// of the Jacobian matrix at the quadrature point times the
// weight of this quadrature point. You can get the gradient
// of shape function $i$ at quadrature point with number q_index by
// using <code>fe_values.shape_grad(i,q_index)</code>; this
// gradient is a 2-dimensional vector (in fact it is of type
// Tensor@<1,dim@>, with here dim=2) and the product of two
// such vectors is the scalar product, i.e. the product of
// the two shape_grad function calls is the dot
// product. This is in turn multiplied by the Jacobian
// determinant and the quadrature point weight (that one
// gets together by the call to FEValues::JxW() ). Finally,
// this is repeated for all shape functions $i$ and $j$:
for (const unsigned int i : fe_values.dof_indices())
for (const unsigned int j : fe_values.dof_indices())
cell_matrix(i, j) +=
(fe_values.shape_grad(i, q_index) * // grad phi_i(x_q)
fe_values.shape_grad(j, q_index) * // grad phi_j(x_q)
fe_values.JxW(q_index)); // dx
// We then do the same thing for the right hand side. Here,
// the integral is over the shape function i times the right
// hand side function, which we choose to be the function
// with constant value one (more interesting examples will
// be considered in the following programs).
for (const unsigned int i : fe_values.dof_indices())
cell_rhs(i) += (fe_values.shape_value(i, q_index) * // phi_i(x_q)
1. * // f(x_q)
fe_values.JxW(q_index)); // dx
}
// Now that we have the contribution of this cell, we have to transfer
// it to the global matrix and right hand side. To this end, we first
// have to find out which global numbers the degrees of freedom on this
// cell have. Let's simply ask the cell for that information:
cell->get_dof_indices(local_dof_indices);
// Then again loop over all shape functions i and j and transfer the
// local elements to the global matrix. The global numbers can be
// obtained using local_dof_indices[i]:
for (const unsigned int i : fe_values.dof_indices())
for (const unsigned int j : fe_values.dof_indices())
system_matrix.add(local_dof_indices[i],
local_dof_indices[j],
cell_matrix(i, j));
// And again, we do the same thing for the right hand side vector.
for (const unsigned int i : fe_values.dof_indices())
system_rhs(local_dof_indices[i]) += cell_rhs(i);
}
// Now almost everything is set up for the solution of the discrete
// system. However, we have not yet taken care of boundary values (in fact,
// Laplace's equation without Dirichlet boundary values is not even uniquely
// solvable, since you can add an arbitrary constant to the discrete
// solution). We therefore have to do something about the situation.
//
// For this, we first obtain a list of the degrees of freedom on the
// boundary and the value the shape function shall have there. For
// simplicity, we only interpolate the boundary value function, rather than
// projecting it onto the boundary. There is a function in the library which
// does exactly this: VectorTools::interpolate_boundary_values(). Its
// parameters are (omitting parameters for which default values exist and
// that we don't care about): the DoFHandler object to get the global
// numbers of the degrees of freedom on the boundary; the component of the
// boundary where the boundary values shall be interpolated; the boundary
// value function itself; and the output object.
//
// The component of the boundary is meant as follows: in many cases, you may
// want to impose certain boundary values only on parts of the boundary. For
// example, you may have inflow and outflow boundaries in fluid dynamics, or
// clamped and free parts of bodies in deformation computations of
// bodies. Then you will want to denote these different parts of the
// boundary by indicators, and tell the interpolate_boundary_values
// function to only compute the boundary values on a certain part of the
// boundary (e.g. the clamped part, or the inflow boundary). By default,
// all boundaries have a 0 boundary indicator, unless otherwise specified. If
// sections of the boundary have different boundary conditions, you have to
// number those parts with different boundary indicators. The function call
// below will then only determine boundary values for those parts of the
// boundary for which the boundary indicator is in fact the zero specified as
// the second argument.
//
// The function describing the boundary values is an object of type Function
// or of a derived class. One of the derived classes is
// Functions::ZeroFunction, which describes (not unexpectedly) a function
// which is zero everywhere. We create such an object in-place and pass it to
// the VectorTools::interpolate_boundary_values() function.
//
// Finally, the output object is a list of pairs of global degree of freedom
// numbers (i.e. the number of the degrees of freedom on the boundary) and
// their boundary values (which are zero here for all entries). This mapping
// of DoF numbers to boundary values is done by the <code>std::map</code>
// class.
std::map<types::global_dof_index, double> boundary_values;
VectorTools::interpolate_boundary_values(dof_handler,
0,
Functions::ZeroFunction<2>(),
boundary_values);
// Now that we got the list of boundary DoFs and their respective boundary
// values, let's use them to modify the system of equations
// accordingly. This is done by the following function call:
MatrixTools::apply_boundary_values(boundary_values,
system_matrix,
solution,
system_rhs);
}
// @sect4{Step3::solve}
// The following function solves the discretized equation. As discussed in
// the introduction, we want to use an iterative solver to do this,
// specifically the Conjugate Gradient (CG) method.
//
// The way to do this in deal.II is a three-step process:
// - First, we need to have an object that knows how to tell the CG algorithm
// when to stop. This is done by using a SolverControl object, and as
// stopping criterion we say: stop after a maximum of 1000 iterations (which
// is far more than is needed for 1089 variables; see the results section to
// find out how many were really used), and stop if the norm of the residual
// is below $\tau=10^{-6}\|\mathbf b\|$ where $\mathbf b$ is the right hand
// side vector. In practice, this latter criterion will be the one
// which stops the iteration.
// - Then we need the solver itself. The template parameter to the SolverCG
// class is the type of the vectors we are using.
// - The last step is to actually solve the system of equations. The CG solver
// takes as arguments the components of the linear system $Ax=b$ (in the
// order in which they appear in this equation), and a preconditioner
// as the fourth argument. We don't feel ready to delve into preconditioners
// yet, so we tell it to use the identity operation as preconditioner. Later
// tutorial programs will spend significant amount of time and space on
// constructing better preconditioners.
//
// At the end of this process, the `solution` variable contains the
// nodal values of the solution function.
void Step3::solve()
{
SolverControl solver_control(1000, 1e-6 * system_rhs.l2_norm());
SolverCG<Vector<double>> solver(solver_control);
solver.solve(system_matrix, solution, system_rhs, PreconditionIdentity());
}
// @sect4{Step3::output_results}
// The last part of a typical finite element program is to output the results
// and maybe do some postprocessing (for example compute the maximal stress
// values at the boundary, or the average flux across the outflow, etc). We
// have no such postprocessing here, but we would like to write the solution
// to a file.
void Step3::output_results() const
{
// To write the output to a file, we need an object which knows about output
// formats and the like. This is the DataOut class, and we need an object of
// that type:
DataOut<2> data_out;
// Now we have to tell it where to take the values from which it shall
// write. We tell it which DoFHandler object to use, and the solution vector
// (and the name by which the solution variable shall appear in the output
// file). If we had more than one vector which we would like to look at in
// the output (for example right hand sides, errors per cell, etc) we would
// add them as well:
data_out.attach_dof_handler(dof_handler);
data_out.add_data_vector(solution, "solution");
// After the DataOut object knows which data it is to work on, we have to
// tell it to process them into something the back ends can handle. The
// reason is that we have separated the frontend (which knows about how to
// treat DoFHandler objects and data vectors) from the back end (which knows
// many different output formats) and use an intermediate data format to
// transfer data from the front- to the backend. The data is transformed
// into this intermediate format by the following function:
data_out.build_patches();
// Now we have everything in place for the actual output. Just open a file
// and write the data into it, using VTK format (there are many other
// functions in the DataOut class we are using here that can write the
// data in postscript, AVS, GMV, Gnuplot, or some other file
// formats):
std::ofstream output("solution.vtk");
data_out.write_vtk(output);
}
// @sect4{Step3::run}
// Finally, the last function of this class is the main function which calls
// all the other functions of the <code>Step3</code> class. The order in which
// this is done resembles the order in which most finite element programs
// work. Since the names are mostly self-explanatory, there is not much to
// comment about:
void Step3::run()
{
make_grid();
setup_system();
assemble_system();
solve();
output_results();
}
// @sect3{The <code>main</code> function}
// This is the main function of the program. Since the concept of a
// main function is mostly a remnant from the pre-object oriented era
// before C++ programming, it often does not do much more than
// creating an object of the top-level class and calling its principle
// function.
//
// Finally, the first line of the function is used to enable output of
// some diagnostics that deal.II can generate. The @p deallog
// variable (which stands for deal-log, not de-allog) represents a
// stream to which some parts of the library write output. For
// example, iterative solvers will generate diagnostics (starting
// residual, number of solver steps, final residual) as can be seen
// when running this tutorial program.
//
// The output of @p deallog can be written to the console, to a file,
// or both. Both are disabled by default since over the years we have
// learned that a program should only generate output when a user
// explicitly asks for it. But this can be changed, and to explain how
// this can be done, we need to explain how @p deallog works: When
// individual parts of the library want to log output, they open a
// "context" or "section" into which this output will be placed. At
// the end of the part that wants to write output, one exits this
// section again. Since a function may call another one from within
// the scope where this output section is open, output may in fact be
// nested hierarchically into these sections. The LogStream class of
// which @p deallog is a variable calls each of these sections a
// "prefix" because all output is printed with this prefix at the left
// end of the line, with prefixes separated by colons. There is always
// a default prefix called "DEAL" (a hint at deal.II's history as the
// successor of a previous library called "DEAL" and from which the
// LogStream class is one of the few pieces of code that were taken
// into deal.II).
//
// By default, @p logstream only outputs lines with zero prefixes --
// i.e., all output is disabled because the default "DEAL" prefix is
// always there. But one can set a different maximal number of
// prefixes for lines that should be output to something larger, and
// indeed here we set it to two by calling
// LogStream::depth_console(). This means that for all screen output,
// a context that has pushed one additional prefix beyond the default
// "DEAL" is allowed to print its output to the screen ("console"),
// whereas all further nested sections that would have three or more
// prefixes active would write to @p deallog, but @p deallog does not
// forward this output to the screen. Thus, running this example (or
// looking at the "Results" section), you will see the solver
// statistics prefixed with "DEAL:CG", which is two prefixes. This is
// sufficient for the context of the current program, but you will see
// examples later on (e.g., in step-22) where solvers are nested more
// deeply and where you may get useful information by setting the
// depth even higher.
int main()
{
deallog.depth_console(2);
Step3 laplace_problem;
laplace_problem.run();
return 0;
}