evox.algorithms.mo.moead 源代码

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