=============== Getting Started =============== Installing Platypus ------------------- To install the latest version of Platypus, run the following commands. :: pip install -U build setuptools git clone https://github.com/Project-Platypus/Platypus.git cd Platypus python -m build A Simple Example ---------------- As an initial example, we will solve the well-known two objective DTLZ2 problem using the NSGA-II algorithm: .. literalinclude:: ../examples/simple.py :language: python The output shows on each line the objectives for a Pareto optimal solution: .. code:: [1.00289403128, 6.63772921439e-05] [0.000320076737668, 1.00499316652] [1.00289403128, 6.63772921439e-05] [0.705383878891, 0.712701387377] [0.961083112366, 0.285860932437] [0.729124908607, 0.688608373855] ... If *matplotlib* is available, we can also plot the results. Note that *matplotlib* must be installed separately. Running the following code .. literalinclude:: ../examples/simple_plot.py :language: python produce a plot similar to: .. image:: images/figure_1.png :scale: 60 % :alt: Pareto front for the DTLZ2 problem :align: center Note that we did not need to specify many settings when constructing NSGA-II. For any options not specified by the user, Platypus supplies the appropriate settings using best practices. In this example, Platypus inspected the problem definition to determine that the DTLZ2 problem consists of real-valued decision variables and selected the Simulated Binary Crossover (SBX) and Polynomial Mutation (PM) operators. One can easily switch to using different operators, such as Parent-Centric Crossover (PCX): .. code:: python from platypus.algorithms import NSGAII from platypus.problems import DTLZ2 from platypus.operators import PCX problem = DTLZ2() algorithm = NSGAII(problem, variator = PCX()) algorithm.run(10000) Defining Unconstrained Problems ------------------------------- There are several ways to define problems in Platypus, but all revolve around the ``Problem`` class. For unconstrained problems, the problem is defined by a function that accepts a single argument, a list of decision variables, and returns a list of objective values. For example, the bi-objective, Schaffer problem, defined by .. math:: \text{minimize } (x^2, (x-2)^2) \text{ for } x \in [-10, 10] can be programmed as follows: .. literalinclude:: ../examples/custom_problem_1.py :language: python When creating the ``Problem`` class, we provide two arguments: the number if decision variables, ``1``, and the number of objectives, ``2``. Next, we specify the types of the decision variables. In this case, we use a real-valued variable bounded between -10 and 10. Finally, we define the function for evaluating the problem. **Tip:** The notation ``problem.types[:]`` is a shorthand way to assign all decision variables to the same type. This is using Python's slice notation. You can also assign the type of a single decision variable, such as ``problem.types[0]``, or any subset, such as ``problem.types[1:]``. An equivalent but more reusable way to define this problem is extending the ``Problem`` class. The types are defined in the ``__init__`` method, and the actual evaluation is performed in the ``evaluate`` method. .. literalinclude:: ../examples/custom_problem_2.py :language: python Defining Constrained Problems ----------------------------- Constrained problems are defined similarly, but must provide two additional pieces of information. First, they must compute the constraint value (or values if the problem defines more than one constraint). Second, they must specify when constraint is feasible and infeasible. To demonstrate this, we will use the Belegundu problem, defined by: .. math:: \text{minimize } (-2x+y, 2x+y) \text{ subject to } y-x \leq 1 \text{ and } x+y \leq 7 This problem has two inequality constraints. We first simplify the constraints by moving the constant to the left of the inequality. The resulting formulation is: .. math:: \text{minimize } (-2x+y, 2x+y) \text{ subject to } y-x-1 \leq 0 \text{ and } x+y-7 \leq 0 Then, we program this problem within Platypus as follows: .. literalinclude:: ../examples/constrained_problem_1.py :language: python First, we call ``Problem(2, 2, 2)`` to create a problem with two decision variables, two objectives, and two constraints, respectively. Next, we set the decision variable types and the constraint feasibility criteria. The constraint feasibility criteria is specified as the string ``"<=0"``, meaning a solution is feasible if the constraint values are less than or equal to zero. Platypus is flexible in how constraints are defined, and can include inequality and equality constraints such as ``">=0"``, ``"==0"``, or ``"!=5"``. Finally, we set the evaluation function. Note how the ``belegundu`` function returns a tuple (two lists) for the objectives and constraints. The final population could contain infeasible and dominated solutions if the number of function evaluations was insufficient (e.g. ``algorithm.Run(100)``). In this case we would need to filter out the infeasible solutions: .. code:: python feasible_solutions = [s for s in algorithm.result if s.feasible] We could also get only the non-dominated solutions: .. code:: python nondominated_solutions = nondominated(algorithm.result) Alternatively, we can develop a reusable class for this problem by extending the ``Problem`` class. Like before, we move the type and constraint declarations to the ``__init__`` method and assign the solution's ``constraints`` attribute in the ``evaluate`` method. .. literalinclude:: ../examples/constrained_problem_2.py :language: python In these examples, we have assumed that the objectives are being minimized. Platypus is flexible and allows the optimization direction to be changed per objective by setting the ``directions`` attribute. For example: .. code:: python problem.directions[:] = Problem.MAXIMIZE