NSGA-II算法理解及matlab代码详解(二)

2019.7.26更新

时间隔了这么久,没想到当年写的这篇文章竟然有这么多人问问题。我只帮各位找到当年我用的代码,代码在https://download.csdn.net/download/xiaoaoqing/11431980

------------------------------------------------------------------------------------------------------------------

------------------------------------------------------------------------------------------------------------------

------------------------------------------------------------------------------------------------------------------

过了这么久才回来写这篇算法的理解,有点抱歉,实在有很多事。
现在就进入正题吧,这个是一个多目标函数优化算法。
多目标函数优化有一种方法是,假如现在有n个目标函数fi,首先将每个目标函数乘以一个适当的参数alfai,再将所有的目标函数加起来,得到一个目标函数。这就将多目标函数转化为单目标函数了。
还有一种方法,是真正的多目标函数优化算法。
首先介绍下非支配解(Non-dominated solution)与支配解(dominated solution)
All nondominated solutions in the combined population are assigned a fitness based on the number of solutions they dominate and dominated solutions are assigned fitnessworse than the worst fitness of any nondominated solution.
这句话的意思是:在合并的种群中,所有非支配解将根据它们支配解的数目分配一个适应度值,支配解分配的适应度值比最差的非支配解的适应度值还要差。
补充一下:在NSGA中,介绍了Pareto最优集:对于多目标优化问题,通常存在一个解集,这些解之间就全体目标函数而言是无法比较优劣的,其特点是:无法在改进任何目标函数的同时不削弱至少一个其他目标函数。这种解称作非支配解或Pareto最优解。对于组成Pareto最优解集的所有Pareto最优解,其对应目标空间中的目标矢量所构成的曲面称作Pareto最优前沿。

一、快速非支配排序法

N 种群大小,M为目标函数数目。
这是一个复杂度为O(MN2)的算法
下面这个matlab是求解的最小值,再来补充下支配的概念,当不相等的两个解集p与解集q时,他们俩相比,当p的每个目标函数的值不存在大于q的,那么p支配q

 

    for i = 1 : N
    % Number of individuals that dominate this individual
    individual(i).n = 0;
    % Individuals which this individual dominate
    individual(i).p = [];
    for j = 1 : N
        dom_less = 0;
        dom_equal = 0;
        dom_more = 0;
        for k = 1 : M
            if (x(i,V + k) < x(j,V + k))
                dom_less = dom_less + 1;
            elseif (x(i,V + k) == x(j,V + k))
                dom_equal = dom_equal + 1;
            else
                dom_more = dom_more + 1;
            end
        end
        if dom_less == 0 && dom_equal ~= M             %%%j 支配了i
            individual(i).n = individual(i).n + 1;
        elseif dom_more == 0 && dom_equal ~= M     %%%i支配了j
            individual(i).p = [individual(i).p j];
        end
    end   
    if individual(i).n == 0  %%%当i不存在被支配的,那么它划分的等级为1
        x(i,M + V + 1) = 1;
        F(front).f = [F(front).f i];
    end
end
%%%%%可以从上面看出查找一级front的复杂度为O(MN2),在上面individual已经保存了各自的支配解,这个很巧妙
%%%%%%%%%%%%
% Find the subsequent fronts     发现后续的front集合
while ~isempty(F(front).f)     %%%这里是精妙开始之处,检测当然front的支配集的非支配解,当没检测到一次,减少一,最先减为0,也就是除了小于等于front的可以支配它,剩下的就是它支配了,也就是当前下一个front集合里的</span>
   Q = [];                              %%%Q存放下一个front集合的
   for i = 1 : length(F(front).f)
       if ~isempty(individual(F(front).f(i)).p)
        	for j = 1 : length(individual(F(front).f(i)).p)
            	individual(individual(F(front).f(i)).p(j)).n = ...
                	individual(individual(F(front).f(i)).p(j)).n - 1;
        	   	if individual(individual(F(front).f(i)).p(j)).n == 0
               		x(individual(F(front).f(i)).p(j),M + V + 1) = ...
                        front + 1;
                    Q = [Q individual(F(front).f(i)).p(j)];
                end
            end
       end
   end
   front =  front + 1;
   F(front).f = Q;
end


[temp,index_of_fronts] = sort(x(:,M + V + 1));
for i = 1 : length(index_of_fronts)
    sorted_based_on_front(i,:) = x(index_of_fronts(i),:);
end
current_index = 0;

支配集数目最多N-1,每个解最多访问N-1,那么第二部分最大复杂度为O(N2),所以算法复杂度为O(N),空间复杂度为O(N2)。

先到这吧,有什么问题的可以留言讨论

 

 

 

  • 11
    点赞
  • 76
    收藏
    觉得还不错? 一键收藏
  • 30
    评论
### 回答1: 抱歉,我不熟悉matlab代码,但我可以提供关于NSGA-II算法的一些基本信息。NSGA-II是一种多目标优化算法,用于求解复杂的多目标优化问题。它的基本思想是使用解决多目标优化问题的种群搜索算法,通过不断迭代来改善种群中的解。它的优点在于可以有效地搜索出多个最优解,而不是仅仅找到一个最优解。 ### 回答2: NSGA-II是一种常用的多目标优化算法,它结合了多种搜索策略和遗传算法的思想,用于求解具有多个目标函数的优化问题。下面是NSGA-II算法Matlab中的简化代码示例。 首先,定义目标函数的个数`nObj`和种群大小`popSize`。假设有两个目标函数和种群大小为100: ```matlab nObj = 2; popSize = 100; ``` 接下来,定义问题的搜索空间,包括每个决策变量的上界和下界。假设有两个决策变量,它们的取值范围分别是[0, 1]和[0, 5]: ```matlab nVar = 2; varMin = [0, 0]; varMax = [1, 5]; ``` 然后,初始化种群,生成一组随机的解。将解表示为一个矩阵,其中每一行代表一个个体的决策变量: ```matlab popDec = unifrnd(varMin, varMax, popSize, nVar); ``` 接着,对于每个个体,计算它的目标函数值。这里假设目标函数是`objFun`,且可以同时计算多个个体的目标函数值: ```matlab popObj = objFun(popDec); ``` 然后,使用快速非支配排序对种群中的个体进行排序。这里使用外部函数`fastNonDominatedSort`实现: ```matlab popRank = fastNonDominatedSort(popObj); ``` 接下来,根据排序的结果,计算每个个体的拥挤距离,用于进行下一步的选择。这里使用外部函数`crowdingDistance`实现: ```matlab popDist = crowdingDistance(popObj, popRank); ``` 然后,根据个体的排序和拥挤距离,按照NSGA-II的选择策略,生成新一代种群。这里使用外部函数`tournamentSelection`实现: ```matlab pop = tournamentSelection(popDec, popObj, popRank, popDist); ``` 最后,对新一代种群进行交叉和变异操作,生成下一代种群。这里使用外部函数`crossover`和`mutation`实现: ```matlab popDec = crossover(pop, varMin, varMax); popDec = mutation(popDec, varMin, varMax); ``` 以上是NSGA-II算法Matlab中的简化代码示例。注意,实际使用中需要根据具体问题进行适当的修改和调整,并且可能需要自定义的目标函数和操作函数。这里的代码只是提供一个基本的框架和示例。 ### 回答3: NSGA-II(Non-dominated Sorting Genetic Algorithm II)是一种基于遗传算法的多目标优化算法。它通过不断迭代生成的种群,利用非支配排序和拥挤度距离来筛选出帕累托最优解集。以下是基于MatlabNSGA-II算法代码: 首先,定义问题的目标函数和约束条件,在此代码中我们以目标函数最小化为例: ```matlab % 定义目标函数 function f = objFcn(x) f(1) = sin(x(1)*pi/180) + cos(x(2)*pi/180); f(2) = 2*x(1)*x(2) - 1; end % 定义约束条件 function [c, ceq] = constFcn(x) c = []; ceq = x(1)^2 + x(2)^2 - 1; end ``` 然后,定义NSGA-II算法函数: ```matlab function nsga2() % 设置算法参数 popSize = 100; % 种群大小 maxGen = 100; % 最大迭代次数 nVar = 2; % 变量维度 nObj = 2; % 目标函数个数 % 初始化种群 pop = rand(popSize, nVar); objVal = zeros(popSize, nObj); for i = 1:popSize objVal(i,:) = objFcn(pop(i,:)); end % NSGA-II主循环 for gen = 1:maxGen % 计算非支配排序和拥挤度距离 fronts = nonDominatedSorting(objVal); crowdingDist = crowdingDistance(objVal, fronts); % 选择新一代种群 newPop = []; newObjVal = []; for i = 1:length(fronts)-1 sortedFront = sortrows([objVal(fronts{i},:), crowdingDist(fronts{i})], 3); [~, rank] = sortrows(sortedFront(:,2), 'descend'); n = length(fronts{i}) - sum(sortedFront(rank(1:end-1),2) == sortedFront(rank(2:end),2)); newPop = [newPop; pop(fronts{i}(rank(1:n)),:)]; newObjVal = [newObjVal; objVal(fronts{i}(rank(1:n)),:)]; end % 交叉和变异产生子代种群 offspringPop = []; for i = 1:popSize/2 parentIdx = randperm(length(newPop), 2); parent1 = newPop(parentIdx(1),:); parent2 = newPop(parentIdx(2),:); offspring = crossover(parent1, parent2); offspring = mutation(offspring); offspringPop = [offspringPop; offspring]; end % 合并新一代种群和子代种群 pop = [newPop; offspringPop]; objVal = zeros(size(pop,1), nObj); for i = 1:size(pop,1) objVal(i,:) = objFcn(pop(i,:)); end end % 输出最终帕累托最优解集 paretoIdx = nonDominatedSorting(objVal); paretoSet = pop(paretoIdx{1},:); paretoFront = objVal(paretoIdx{1},:); disp('Pareto optimal set:'); disp(paretoSet); disp('Pareto optimal front:'); disp(paretoFront); end % 非支配排序算法 function fronts = nonDominatedSorting(objVal) nPop = size(objVal, 1); S = cell(nPop,1); n = zeros(nPop, 1); % 初始化支配等级和被支配个体集合 fronts = {}; for p = 1:nPop S{p} = []; n(p) = 0; for q = 1:nPop if any(objVal(p,:) < objVal(q,:)) && all(objVal(p,:) <= objVal(q,:)) S{p} = [S{p}, q]; elseif any(objVal(p,:) > objVal(q,:)) && all(objVal(p,:) >= objVal(q,:)) n(p) = n(p) + 1; end end if n(p) == 0 fronts{1} = [fronts{1}, p]; end end % 迭代计算支配等级并更新被支配个体集合 iFront = 1; while ~isempty(fronts{iFront}) Q = []; for p = fronts{iFront} for q = S{p} n(q) = n(q) - 1; if n(q) == 0 Q = [Q, q]; end end end iFront = iFront + 1; fronts{iFront} = Q; end end % 计算拥挤度距离 function D = crowdingDistance(objVal, fronts) nFront = length(fronts); nObj = size(objVal, 2); D = zeros(size(objVal,1),1); for iFront = 1:nFront fSize = length(fronts{iFront}); if fSize > 2 fValues = objVal(fronts{iFront},:); [~, sortedIdx] = sortrows(fValues); D(fronts{iFront}(sortedIdx(1))) = inf; D(fronts{iFront}(sortedIdx(end))) = inf; for j = 2:fSize-1 D(fronts{iFront}(sortedIdx(j))) = ... D(fronts{iFront}(sortedIdx(j))) + ... norm(fValues(sortedIdx(j+1),:) - fValues(sortedIdx(j-1),:)); end end end end % 交叉操作 function offspring = crossover(parent1, parent2) nVar = length(parent1); rc = randi(nVar-1); offspring = [parent1(1:rc), parent2(rc+1:end)]; end % 变异操作 function offspring = mutation(parent) nVar = length(parent); rm = randi(nVar); offspring = parent; offspring(rm) = rand(1); end ``` 以上是一个简单的NSGA-II算法Matlab代码实现,可以根据需要进行修改和扩展。请注意代码中的注释,以便更好地理解算法的实现细节。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 30
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值