At least you will need a basic python interpreter as well as the popular python scientific libraries:
- numpy: matrix/vector types with basic linear algebra routines.
- scipy: advanced algorithms (e.g. sparse linear solvers).
- matplotlib: display of 2D and 3D plots.
If you already have a python distribution on your computer, use it.
There are many ways to install python. Some examples:
- Available in all Linux distributions ( `apt install python3` on Ubuntu).
-[Python.org](python.org): the reference binaries available from the official python website. <br/> You should use "PyPI" (`pip` command) for the 3 missing modules: <br/>
`pip install numpy scipy matplotlib`<br/>This is the cheapest option in terms of disk space (<0.5Gb).
-[Anaconda](https://www.anaconda.com/): full IDE around ``conda''. Features: package management, multi-python environments and optimised proprietary libraries (Intel MKL, NVIDIA CUDA). <br/> (comes with >10Gb of libraries/programs already installed, as well as a text editor - no command line needed).
- [Miniconda](https://conda.io/miniconda.html): minimal conda environment - requires the installation of packages with the command line ($\approx$1.6Gb on disk).<br/> `conda install numpy scipy matplotlib`
$\Rightarrow$ You may choose **Anaconda** if you are not familiar with programming and/or the command line.
## Anaconda installation
- Go to https://www.anaconda.com/download/
- Download the installer for your OS:

- Download the installer for your system (it should be auto-detected).
- Start the installer and follow the instructions.
- Uncheck python registration if you plan to use other python versions for other projects:

- Run ``Anaconda Navigator'' (from your Start Menu if Windows):

- Lots of (too many?) programs have been installed. You just need the "Spyder IDE", which can be run from outside this application.
- Anaconda may ask you to perform an update of "Anaconda Navigator":

- It is a **bad idea** to update the base environment. The update procedure might fail leading the program to an unstable state!
- The same remark holds for Spyder or any other python package.
- If you have previously installed an old version of Anaconda for another project, you can create a new and clean ``python 3'' environment:

- Anaconda is able to handle several versions of python in separate environments.
- Launch ``Spyder'' (IDE similar to the one of MATLAB):

## How to learn Python?
- Use the links from the "help" menu:

- Use `CTRL-I` when your cursor is above any unknown command.
- Free Python books from Springer: https://lib.uliege.be/
- If you are familiar with MATLAB:
-[translation of MATLAB commands](http://mathesaurus.sourceforge.net/matlab-numpy.html)
-[numpy for MATLAB users](https://numpy.org/doc/stable/user/numpy-for-matlab-users.html)
Python is said to be **object-oriented**, as many modern programming languages, which means that you can create **your own types** of data beyond the basic types provided by the language (integers, floating-point numbers, strings, etc.).
Learning to write good object-oriented code is not easy but it can be learnt **incrementally**. You may take advantage of this homework to get familiar with OOP and you may try to write your first objects (with our help).
This opportunity to learn OOP was requested by the students from last years, although we never asked them to write object-oriented programs. Using OOP in your homework is **not mandatory**.
Leaning OOP with python is much easier than learning it with compiled languages such as C++, Java, C#, etc.
## Classes
New types are defined in **classes**.
A class gathers several variables which are used together. This is similar to a structure (`struct`) in C.
There are many ways to define "objects" in the frame of the implementation of the finite element method. **There is no unique or best choice.**
Ideas of objects could come from mathematical concepts or words that are used in the statement of the exercise.
Possible objects:
- BoundaryValueProblem
- WeakForm
- ShapeFunction
- Node
- FiniteElement
- DirichletBC, NeumannBC
- LinearSolver
The choice depends on the size of the simulation code, the number of features that should be implemented, the amount of time you want to invest in learning OOP.
OOP is also closely related to **refactoring techniques**. Refactoring means "improving the structure of the code without adding new features".
It means that you can start writing the first lines of your code with a very limited number of objects (or even no object at all).
Then, as the number of lines increases, if you see that many functions act on the same set of variables, it is usually a good idea to "refactor" your program and create a class gathering these variables together.
By doing this, you do not create classes that are unnecessary. Indeed, using too many classes can lead to unreadable code. This is a common problem while learning OOP ("**do not make each byte an object**").
## Example
Implementation of a basic finite element solver for the Helmholtz equation in 1D:
$$
\frac{d^2u}{dx^2}+k^2u = 0\qquad x\in ]0,L[ \text{ and } k\in\mathbb{R}
$$
with appropriate boundary conditions.
The problem is solved by an object named `Solver` which contains the main parameters and a `solve()` function
Note that the same level of clarity could be easily obtained with well-defined functions in a traditional functional-programming approach!
## Final remarks on Object-Oriented Programming
- Once again: using objects is **not mandatory** for the project. Use classical functional programming (as you already did in C or any other language) if you are new to Python and/or finite elements.
- Keep in mind that using objects is **usually not recommended for such small programs** unless you want to "play" with objects and learn how OOP works in the frame of python.
- This introduction lacks one of the most interesting feature of OOP: **polymorphism**. It allows you to define derived types of data (by **inheritance**) and to manipulate their instances without knowing their exact types.