Lab 2: Exploring Basis Functions in deal.II
Contents
Lab 2: Exploring Basis Functions in deal.II#
This laboratory session is designed to help you explore the shapes of basis functions using the deal.II library. You will learn how to create and refine meshes, distribute degrees of freedom, define finite elements, and visualize the basis functions.
Degrees of Freedom per Object#
In the context of finite element methods (FEM), the term «degrees of freedom» (DoF) refers to the number of independent scalar values that are used to represent a function on a given finite element mesh. Each degree of freedom corresponds to the result of evaluating a linear functional on the mesh where the solution is approximated. The most common degrees of freedom used in FEM are function values at interpolation points (vertices, midpoint of vertices, midpoint of cells, etc.).
Degrees of Freedom per Object#
The concept of «degrees of freedom per object» refers to the distribution of degrees of freedom across various geometrical entities within the finite element mesh. These objects can include:
Vertices (Nodes): Points where elements meet. In lower-order elements, degrees of freedom are often associated with the vertices.
Edges: Line segments connecting vertices. Higher-order elements may have additional degrees of freedom on edges.
Quads/Triangles: Surfaces bounded by edges. Higher-order elements may have degrees of freedom associated with Quads or Triangles.
Hexes/Tets: Volumes bounded by Quads or Triangles. Higher-order elements can have degrees of freedom inside Hexes or Tets.
The degrees of freedom per object vector (dpo
) is used by DoFHandler
to decide how to distribute degrees of freedom on a Triangulation
. It is a dim+1
array, where the i
-th entry indicates the number of degrees of freedom associated to objects of dimension i
.
Examples#
Linear Elements (P1 Elements):
For a 2D triangular element (triangle), linear elements have degrees of freedom only at the vertices. Each vertex of the triangle represents one degree of freedom (
dpo={1,0,0}
).
Quadratic Elements (P2 Elements):
For a 2D triangular element with quadratic shape functions, degrees of freedom exist at the vertices and along the edges. This means each edge of the triangle has additional degrees of freedom, allowing the solution to be more accurately approximated (
dpo={1,1,0}
).
Discontinuous Galerkin Elements (DG Elements):
In DG methods, degrees of freedom are typically associated with the interior of the element and do not need to be continuous across element boundaries. This allows for greater flexibility in handling complex boundary conditions and discontinuities (i.e., DGP(1) on triangles: (
dpo={0,0,3}
)).
Importance in FEM#
Understanding the distribution of degrees of freedom per object is essential for:
Mesh Generation and Refinement: Knowing where degrees of freedom are located helps in creating and refining meshes to better capture the solution’s behavior.
Solution Accuracy: More degrees of freedom generally lead to a more accurate solution, but also increase computational cost.
Element Selection: Different types of elements (linear, quadratic, etc.) distribute degrees of freedom differently, affecting both the accuracy and efficiency of the solution.
Objectives#
By the end of this laboratory, you should be able to:
Create and refine meshes using the
Triangulation
class.Generate different types of grids using the
GridGenerator
namespace.Set up and distribute degrees of freedom using the
DoFHandler
class.Define finite elements with the
FiniteElement
class, specificallyFE_Q
andFE_DGQ
.Output and visualize the shapes of basis functions using the
DataOut
class.
Prerequisites#
Make sure you have the following before you start:
A working installation of the deal.II library.
Basic knowledge of C++.
Example Code#
Here is a simple example to get you started. This code creates a subdivided hyper-rectangle mesh, distributes degrees of freedom, and outputs the shapes of basis functions to a VTU file.
#include <deal.II/base/point.h>
#include <deal.II/dofs/dof_handler.h>
#include <deal.II/fe/fe_dgq.h>
#include <deal.II/fe/fe_q.h>
#include <deal.II/grid/grid_generator.h>
#include <deal.II/grid/grid_out.h>
#include <deal.II/grid/tria.h>
#include <deal.II/lac/vector.h>
#include <deal.II/numerics/data_out.h>
#include <iostream>
using namespace dealii;
int main()
{
// Create a 2D triangulation object
Triangulation<2> triangulation;
// Define a finite element space with polynomial degree 2
FE_Q<2> fe(2); // dpo = {1, 1, 1}
// FE_DGQ<2> fe(2); // dpo = {0, 0, 9}
// Set up the DoF handler with the triangulation
DoFHandler<2> dof_handler(triangulation);
// Generate a subdivided hyper-rectangle mesh
GridGenerator::subdivided_hyper_rectangle(triangulation,
{2, 1},
Point<2>(0, 0),
Point<2>(2, 1));
// Distribute degrees of freedom among the cells
dof_handler.distribute_dofs(fe);
// Output the number of active cells and total cells in the triangulation
std::cout << "Number of active cells: " << triangulation.n_active_cells()
<< std::endl
<< "Total number of cells: " << triangulation.n_cells()
<< std::endl;
// Output the number of degrees of freedom
std::cout << "Number of degrees of freedom: " << dof_handler.n_dofs()
<< std::endl;
// Output the number of dofs per cell
std::cout << "Number of dofs per cell: " << fe.dofs_per_cell << std::endl;
// Output the global dof indices for each cell
std::vector<types::global_dof_index> dofs_per_cell(fe.dofs_per_cell);
for (const auto &cell : dof_handler.active_cell_iterators())
{
std::cout << cell << std::endl;
cell->get_dof_indices(dofs_per_cell);
for (const auto &dof : dofs_per_cell)
{
std::cout << dof << " ";
}
std::cout << std::endl;
}
// Create a DataOut object to visualize the basis functions
DataOut<2> data_out;
// Set flags to write higher order cells
DataOutBase::VtkFlags flags;
flags.write_higher_order_cells = true;
data_out.set_flags(flags);
// Attach the DoF handler to the DataOut object
data_out.attach_dof_handler(dof_handler);
// Create and add basis functions to the DataOut object
std::vector<Vector<double>> basis_functions(
10, Vector<double>(dof_handler.n_dofs()));
for (unsigned int i = 0; i < 10; ++i)
{
Vector<double> &basis_function = basis_functions[i];
basis_function[i] = 1;
data_out.add_data_vector(basis_function,
"basis_function_" + std::to_string(i));
}
// Build patches and write the output to a VTU file
data_out.build_patches(fe.degree);
std::ofstream output("solution.vtu");
data_out.write_vtu(output);
}
Exercises#
Exercise 1: Visualizing Basis Functions for Different Degrees#
Basis Functions for Degree 1:
Modify the example code to use
FE_Q<2>(1)
and visualize the first few basis functions.Observe and describe the shapes of the basis functions in the generated VTU files.
Basis Functions for Higher Degrees:
Change the finite element to
FE_Q<2>(3)
and visualize the first few basis functions.Compare the shapes of the basis functions for polynomial degrees 1, 2, and 3.
Exercise 2: Exploring Discontinuous Galerkin Basis Functions#
Visualizing DG Basis Functions:
Replace
FE_Q
withFE_DGQ
in the example code and visualize the basis functions.Observe how the shapes of the DG basis functions differ from the continuous basis functions.
Comparing Continuous and Discontinuous Basis Functions:
Visualize and compare the basis functions for
FE_Q<2>(2)
andFE_DGQ<2>(2)
.Describe the differences in their shapes and explain why they differ.
Exercise 3: Basis Functions on Different Meshes#
Hyper Shell Mesh:
Modify the example code to create a hyper shell mesh using
GridGenerator::hyper_shell
.Visualize the basis functions on the hyper shell mesh and describe any differences compared to the hyper-rectangle mesh.
Adaptive Mesh Refinement:
Implement a strategy to refine the mesh adaptively around a specific region (e.g., near the origin).
Visualize the basis functions on the adaptively refined mesh and compare them to those on a uniformly refined mesh.
Exercise 4: Higher-Order Elements and Basis Functions#
Using
FE_Nedelec
Elements:Replace the finite element in the example code with
FE_Nedelec
and visualize the basis functions.Describe the shapes of the Nedelec basis functions and their applications.
Using
FE_RaviartThomas
Elements:Replace the finite element with
FE_RaviartThomas
and visualize the basis functions.Compare the shapes of the Raviart-Thomas basis functions with those of
FE_Q
andFE_Nedelec
.
Exercise 5: Output and Visualization#
Output Grids in Different Formats:
Output the grid and basis functions to different formats such as
SVG
,EPS
, andVTK
usingGridOut
andDataOut
.View the generated files using appropriate viewers and compare the visualizations.
Visualizing Basis Functions in Paraview:
Open the VTU files in Paraview and explore the different visualization options.
Create contour plots, vector field plots, and other visualizations of the basis functions.