diff --git a/Documentation/Examples/Solvers/Linear/CMakeLists.txt b/Documentation/Examples/Solvers/Linear/CMakeLists.txt
index a07ba232a43e240f18caf81cc95a1ace6d70b02a..0d2354617cb0a18822e2934f543938df68ae67d7 100644
--- a/Documentation/Examples/Solvers/Linear/CMakeLists.txt
+++ b/Documentation/Examples/Solvers/Linear/CMakeLists.txt
@@ -2,6 +2,7 @@ set( COMMON_EXAMPLES
    IterativeLinearSolverExample
    IterativeLinearSolverWithMonitorExample
    IterativeLinearSolverWithTimerExample
+   IterativeLinearSolverWithPreconditionerExample
 )
 
 if( BUILD_CUDA )
diff --git a/Documentation/Examples/Solvers/Linear/IterativeLinearSolverWithPreconditionerExample.cpp b/Documentation/Examples/Solvers/Linear/IterativeLinearSolverWithPreconditionerExample.cpp
new file mode 100644
index 0000000000000000000000000000000000000000..3a2141074403af91ec96a9f1f019114aa9adb4ea
--- /dev/null
+++ b/Documentation/Examples/Solvers/Linear/IterativeLinearSolverWithPreconditionerExample.cpp
@@ -0,0 +1,87 @@
+#include <iostream>
+#include <memory>
+#include <TNL/Algorithms/ParallelFor.h>
+#include <TNL/Matrices/SparseMatrix.h>
+#include <TNL/Devices/Host.h>
+#include <TNL/Devices/Cuda.h>
+#include <TNL/Solvers/Linear/TFQMR.h>
+#include <TNL/Solvers/Linear/Preconditioners/Diagonal.h>
+
+template< typename Device >
+void iterativeLinearSolverExample()
+{
+   /***
+    * Set the following matrix (dots represent zero matrix elements):
+    *
+    *   /  2.5 -1    .    .    .   \
+    *   | -1    2.5 -1    .    .   |
+    *   |  .   -1    2.5 -1.   .   |
+    *   |  .    .   -1    2.5 -1   |
+    *   \  .    .    .   -1    2.5 /
+    */
+   using MatrixType = TNL::Matrices::SparseMatrix< double, Device >;
+   using Vector = TNL::Containers::Vector< double, Device >;
+   const int size( 5 );
+   auto matrix_ptr = std::make_shared< MatrixType >();
+   matrix_ptr->setDimensions( size, size );
+   matrix_ptr->setRowCapacities( Vector( { 2, 3, 3, 3, 2 } ) );
+
+   auto f = [=] __cuda_callable__ ( typename MatrixType::RowView& row ) mutable {
+      const int rowIdx = row.getRowIndex();
+      if( rowIdx == 0 )
+      {
+         row.setElement( 0, rowIdx,    2.5 );    // diagonal element
+         row.setElement( 1, rowIdx+1, -1 );      // element above the diagonal
+      }
+      else if( rowIdx == size - 1 )
+      {
+         row.setElement( 0, rowIdx-1, -1.0 );    // element below the diagonal
+         row.setElement( 1, rowIdx,    2.5 );    // diagonal element
+      }
+      else
+      {
+         row.setElement( 0, rowIdx-1, -1.0 );    // element below the diagonal
+         row.setElement( 1, rowIdx,    2.5 );    // diagonal element
+         row.setElement( 2, rowIdx+1, -1.0 );    // element above the diagonal
+      }
+   };
+
+   /***
+    * Set the matrix elements.
+    */
+   matrix_ptr->forAllRows( f );
+   std::cout << *matrix_ptr << std::endl;
+
+   /***
+    * Set the right-hand side vector.
+    */
+   Vector x( size, 1.0 );
+   Vector b( size );
+   matrix_ptr->vectorProduct( x, b );
+   x = 0.0;
+   std::cout << "Vector b = " << b << std::endl;
+
+   /***
+    * Solve the linear system using diagonal (Jacobi) preconditioner.
+    */
+   using LinearSolver = TNL::Solvers::Linear::TFQMR< MatrixType >;
+   using Preconditioner = TNL::Solvers::Linear::Preconditioners::Diagonal< MatrixType >;
+   auto preconditioner_ptr = std::make_shared< Preconditioner >;
+   preconditioner_ptr->update( matrix_ptr );
+   LinearSolver solver;
+   solver.setMatrix( matrix_ptr );
+   solver.setPreconditioner( preconditioner_ptr );
+   solver.solve( b, x );
+   std::cout << "Vector x = " << x << std::endl;
+}
+
+int main( int argc, char* argv[] )
+{
+   std::cout << "Solving linear system on host: " << std::endl;
+   iterativeLinearSolverExample< TNL::Devices::Sequential >();
+
+#ifdef HAVE_CUDA
+   std::cout << "Solving linear system on CUDA device: " << std::endl;
+   iterativeLinearSolverExample< TNL::Devices::Cuda >();
+#endif
+}
diff --git a/Documentation/Examples/Solvers/Linear/IterativeLinearSolverWithPreconditionerExample.cu b/Documentation/Examples/Solvers/Linear/IterativeLinearSolverWithPreconditionerExample.cu
new file mode 120000
index 0000000000000000000000000000000000000000..dc6a12a588680a497eed939cf34de678a3f1f87b
--- /dev/null
+++ b/Documentation/Examples/Solvers/Linear/IterativeLinearSolverWithPreconditionerExample.cu
@@ -0,0 +1 @@
+IterativeLinearSolverWithPreconditionerExample.cpp
\ No newline at end of file
diff --git a/Documentation/Tutorials/Solvers/Linear/tutorial_Linear_solvers.md b/Documentation/Tutorials/Solvers/Linear/tutorial_Linear_solvers.md
index c58b28273fdfd851aca4b55ff92e9255db6372d4..25e709ae663608d381f75575d48f7211cccbeec9 100644
--- a/Documentation/Tutorials/Solvers/Linear/tutorial_Linear_solvers.md
+++ b/Documentation/Tutorials/Solvers/Linear/tutorial_Linear_solvers.md
@@ -79,4 +79,14 @@ The result looks as follows:
 
 \include IterativeLinearSolverWithTimerExample.out
 
-## Setup with preconditioner
\ No newline at end of file
+## Setup with preconditioner
+
+Preconditioners of iterative solvers can significantly improve the performance of the solver. In the case of the linear systems, they are used mainly with the Krylov subspace methods. Preconditioners cannot be used with the starionary methods (\ref TNL::Solvers::Linear::Jacobi and \ref TNL::Solvers::Linear::SOR). The following example shows how to setup an iterative solver of linear systems with preconditioning.
+
+\includelineno Solvers/Linear/IterativeLinearSolverWithPreconditionerExample.cpp
+
+In this example, we solve the same problem as in all other examples in this section. The only differences concerning the preconditioner happen on the lines (68-72). Similar to the matrix of the linear system, the preconditioner is passed to the solver by the means of  smart shared pointer (\ref std::shared_ptr). The instance is created on the lines 68 and 69. Next we just need to connect the solver with the preconditioner (line 72, \ref TNL::Solvers::Linear::LinearSolver).
+
+The result looks as follows:
+
+\include IterativeLinearSolverWithPreconditionerExample.out
diff --git a/src/TNL/Solvers/IterativeSolverMonitor.h b/src/TNL/Solvers/IterativeSolverMonitor.h
index 651f38cdedc7b061077e96dc22750e3f867413ae..8f42a6b28d7f8ac2a4bdfcf0e925e94ad09c60ac 100644
--- a/src/TNL/Solvers/IterativeSolverMonitor.h
+++ b/src/TNL/Solvers/IterativeSolverMonitor.h
@@ -20,6 +20,24 @@ namespace TNL {
  *
  * \tparam Real is a type of the floating-point arithmetics.
  * \tparam Index is an indexing type.
+ *
+ * The following example shows how to use the iterative solver monitor for monitoring
+ * convergence of linear iterative solver:
+ *
+ * \includelineno Solvers/Linear/IterativeLinearSolverWithMonitorExample.cpp
+ *
+ * The result looks as follows:
+ *
+ * \include IterativeLinearSolverWithMonitorExample.out
+ *
+ * The following example shows how to employ timer (\ref TNL::Timer) to the monitor
+ * of iterative solvers:
+ *
+ * \includelineno Solvers/Linear/IterativeLinearSolverWithTimerExample.cpp
+ *
+ * The result looks as follows:
+ *
+ * \include IterativeLinearSolverWithTimerExample.out
  */
 template< typename Real = double,
           typename Index = int >
diff --git a/src/TNL/Solvers/Linear/LinearSolver.h b/src/TNL/Solvers/Linear/LinearSolver.h
index 858f08189040aea4851b29a5e8865c4c51708375..f2de9a8f4a8c34cdc4755f365ed2f84f3e39ace3 100644
--- a/src/TNL/Solvers/Linear/LinearSolver.h
+++ b/src/TNL/Solvers/Linear/LinearSolver.h
@@ -34,6 +34,16 @@ namespace TNL {
  * \ref LinearSolver::setPreconditioner.
  *
  * \tparam Matrix is type of matrix representing the linear system.
+ *
+ * The following example demonstrates the use iterative linear solvers:
+ *
+ * \includelineno Solvers/Linear/IterativeLinearSolverExample.cpp
+ *
+ * The result looks as follows:
+ *
+ * \include IterativeLinearSolverExample.out
+ *
+ * See also \ref TNL::Solvers::IterativeSolverMonitor for monitoring of iterative solvers.
  */
 template< typename Matrix >
 class LinearSolver
diff --git a/src/TNL/Solvers/Linear/Preconditioners/ILU0.h b/src/TNL/Solvers/Linear/Preconditioners/ILU0.h
index 4b35864484f0ebdb6aba3a0fab094778c41bcfe3..6ce3fef5155edd23f510814ad28f0a03c8b5f6d8 100644
--- a/src/TNL/Solvers/Linear/Preconditioners/ILU0.h
+++ b/src/TNL/Solvers/Linear/Preconditioners/ILU0.h
@@ -76,6 +76,7 @@ class ILU0_impl< Matrix, Real, Devices::Host, Index >
 : public Preconditioner< Matrix >
 {
    public:
+
       /**
        * \brief Floating point type used for computations.
        */
diff --git a/src/TNL/Solvers/Linear/Preconditioners/Preconditioner.h b/src/TNL/Solvers/Linear/Preconditioners/Preconditioner.h
index 43b429da7484bab6019470861f0a153ef18e2cbe..e0dc721e362e361d983e4bdaa68e56104d4b7ebd 100644
--- a/src/TNL/Solvers/Linear/Preconditioners/Preconditioner.h
+++ b/src/TNL/Solvers/Linear/Preconditioners/Preconditioner.h
@@ -27,6 +27,15 @@ namespace TNL {
  * \brief Base class for preconditioners of of iterative solvers of linear systems.
  *
  * \tparam Matrix is type of matrix describing the linear system.
+ *
+ * The following example shows how to setup an iterative solver of linear systems with
+ * preconditioning:
+ *
+ * \includelineno Solvers/Linear/IterativeLinearSolverWithPreconditionerExample.cpp
+ *
+ * The result looks as follows:
+ *
+ * \include IterativeLinearSolverWithPreconditionerExample.out
  */
 template< typename Matrix >
 class Preconditioner