import math
from typing import Callable, Optional
import torch
from evox.core import Algorithm, Mutable
from evox.operators.crossover import simulated_binary_half
from evox.operators.mutation import polynomial_mutation
from evox.operators.sampling import uniform_sampling
from evox.utils import clamp, minimum
[文档]
def pbi(f: torch.Tensor, w: torch.Tensor, z: torch.Tensor):
norm_w = torch.linalg.norm(w, dim=1)
f = f - z
d1 = torch.sum(f * w, dim=1) / norm_w
d2 = torch.linalg.norm(f - (d1[:, None] * w / norm_w[:, None]), dim=1)
return d1 + 5 * d2
[文档]
class MOEAD(Algorithm):
"""
Implementation of the Original MOEA/D algorithm.
:references:
[1] Q. Zhang and H. Li, "MOEA/D: A Multiobjective Evolutionary Algorithm Based on Decomposition,"
IEEE Transactions on Evolutionary Computation, vol. 11, no. 6, pp. 712-731, 2007. Available:
https://ieeexplore.ieee.org/document/4358754
```{note}
This implementation closely follows the original paper and reference code.
It is not optimized for tensorized computation and may therefore be less efficient on GPU.
```
"""
def __init__(
self,
pop_size: int,
n_objs: int,
lb: torch.Tensor,
ub: torch.Tensor,
selection_op: Optional[Callable] = None,
mutation_op: Optional[Callable] = None,
crossover_op: Optional[Callable] = None,
device: torch.device | None = None,
):
"""Initializes the MOEA/D algorithm.
:param pop_size: The size of the population.
:param n_objs: The number of objective functions in the optimization problem.
:param lb: The lower bounds for the decision variables (1D tensor).
:param ub: The upper bounds for the decision variables (1D tensor).
:param selection_op: The selection operation for evolutionary strategy (optional).
:param mutation_op: The mutation operation, defaults to `polynomial_mutation` if not provided (optional).
:param crossover_op: The crossover operation, defaults to `simulated_binary_half` if not provided (optional).
:param device: The device on which computations should run (optional). Defaults to PyTorch's default device.
"""
super().__init__()
self.pop_size = pop_size
self.n_objs = n_objs
if device is None:
device = torch.get_default_device()
# check
assert lb.shape == ub.shape and lb.ndim == 1 and ub.ndim == 1
assert lb.dtype == ub.dtype and lb.device == ub.device
self.dim = lb.shape[0]
# write to self
self.lb = lb.to(device=device)
self.ub = ub.to(device=device)
self.selection = selection_op
self.mutation = mutation_op
self.crossover = crossover_op
self.device = device
if self.mutation is None:
self.mutation = polynomial_mutation
if self.crossover is None:
self.crossover = simulated_binary_half
w, _ = uniform_sampling(self.pop_size, self.n_objs)
self.pop_size = w.size(0)
self.n_neighbor = int(math.ceil(self.pop_size / 10))
length = ub - lb
population = torch.rand(self.pop_size, self.dim, device=device)
population = length * population + lb
neighbors = torch.cdist(w, w)
self.neighbors = torch.argsort(neighbors, dim=1, stable=True)[:, : self.n_neighbor]
self.w = w
self.pop = Mutable(population)
self.fit = Mutable(torch.empty((self.pop_size, self.n_objs), device=device).fill_(torch.inf))
self.z = Mutable(torch.zeros((self.n_objs,), device=device))
[文档]
def init_step(self):
"""
Perform the initialization step of the workflow.
Calls the `init_step` of the algorithm if overwritten; otherwise, its `step` method will be invoked.
"""
self.fit = self.evaluate(self.pop)
self.z = torch.min(self.fit, dim=0)[0]
[文档]
def step(self):
"""Perform a single optimization step of the workflow."""
for i in range(self.pop_size):
parents = self.neighbors[i][torch.randperm(self.n_neighbor, device=self.device)]
crossovered = self.crossover(self.pop[parents[:2]])
offspring = self.mutation(crossovered, self.lb, self.ub)
offspring = clamp(offspring, self.lb, self.ub)
off_fit = self.evaluate(offspring)
self.z = minimum(self.z, off_fit)
g_old = pbi(self.fit[parents], self.w[parents], self.z)
g_new = pbi(off_fit, self.w[parents], self.z)
self.fit[parents[g_old >= g_new]] = off_fit
self.pop[parents[g_old >= g_new]] = offspring