技术好文共享:遗传算法解决函数优化

简介: 技术好文共享:遗传算法解决函数优化

术语说明


由于遗传算法是由进化论和遗传学机理而产生的搜索算法,所以在这个算法中会用到很多生物遗传学知识,下面是我们将会用来的一些术语说明:


一、染色体(Chronmosome)


染色体又可以叫做基因型个体(individuals),一定数量的个体组成了群体(population),群体中个体的数量叫做群体大小。


二、基因(Gene)


基因是串中的元素,基因用于表示个体的特征。例如有一个串S=1011,则其中的1,0,1,1这4个元素分别称为基因。它们的值称为等位基因(Alletes)。


三、基因地点(Locus)


基因地点在算法中表示一个基因在串中的位置称为基因位置(Gene Position),有时也简称基因位。基因位置由串的左向右计算,例如在串 S=1101 中,0的基因位置是3。


四、基因特征值(Gene Feature)


在用串表示整数时,基因的特征值与二进制数的权一致;例如在串 S=1011 中,基因位置3中的1,它的基因特征值为8;基因位置1中的1,它的基因特征值为2。


五、适应度(Fitness)


各个个体对环境的适应程度叫做适应度(fitness)。为了体现染色体的适应能力,引入了对问题中的每一个染色体都能进行度量的函数,叫适应度函数. 这个函数是计算个体在群体中被使用的概率。


操作算法


霍兰德(Holland)教授最初提出的算法也叫简单遗传算法,简单遗传算法的遗传操作主要有三种:选择(selection)、交叉(crossover)、变异(mutation)这也是遗传算法中最常用的三种算法:


1.选择(selection)


选择操作也叫复制操作,从群体中按个体的适应度函数值选择出较适应环境的个体。一般地说,选择将使适应度高的个体繁殖下一代的数目较多,而适应度较小的个体,繁殖下一代的数目较少,甚至被淘汰。最通常的实现方法是轮盘赌(roulette wheel)模型。令Σfi表示群体的适应度值之总和,fi表示种群中第i个染色体的适应度值,它被选择的概率正好为其适应度值所占份额fi/Σfi。


如果适应度值存在负数,我的方法是采用以下公式:


再计算概率和累计概率。


设想群体全部个体的适当性分数由一张饼图来代表 (见图)。


群体中每一染色体指定饼图中一个小块。块的大小与染色体的适应性分数成比例,适应性分数愈高,它在饼图中对应的小块所占面积也愈大。为了选取一个染色体,要做的就是旋转这个轮子,直到轮盘停止时,看指针停止在哪一块上,就选中与它对应的那个染色体。


若产生随机数为0.81,则6号个体被选中。


2.交叉(Crossover)


交叉算子将被选中的两个个体的基因链按一定概率pc进行交叉,从而生成两个新的个体,交叉位置pc是随机的。其中Pc是一个系统参数。根据问题的不同,交叉又为了单点交叉算子(Single Point Crossover)、双点交叉算子(Two Point Crossover)、均匀交叉算子 (Uniform Crossover),在此我们只讨论单点交叉的情况。


单点交叉操作的简单方式是将被选择出的两个个体S1和S2作为父母个体,将两者的部分基因码值进行交换。假设如下两个8位的个体:


S1:1000 1111


S2:1110 1100


产生一个在1到7之间的随机数c,假如现在产生的是2,将S1和S2的低二位交换,后代P1为1100 1111,P2为10101100。在这里高二位与低二位是按从左到右的顺序还是从右到左的顺序,自我感觉效果是一样的,不过也没具体证明,网上是按从右到左的顺序,在这篇文章中的定义都是从左到右的。


3.变异(Mutation)


这是在选中的个体中,将新个体的基因链的各位按概率pm进行异向转化,最简单方式是改变串上某个位置数值。对二进制编码来说将0与1互换:0变异为1,1变异为0。


如下8位二进制编码:


1 1 1 0 1 1 0 0


随机产生一个1至8之间的数i,假如现在k=6,对从左往右的第6位进行变异操作,将原来的1变为0,得到如下串:


1 1 1 0 1 0 0 0


4.精英主义 (Elitism)


仅仅从产生的子代中选择基因去构造新的种群可能会丢失掉上一代种群中的很多信息。也就是说当利用交叉和变异产生新的一代时,我们有很大的可能把在某个中间步骤中得到的最优解丢失。在此我们使用精英主义(Elitism)方法,在每一次产生新的一代时,我们首先把当前最优解原封不动的复制到新的一代中,其他步骤不变。这样任何时刻产生的一个最优解都可以存活到遗传算法结束。不过这篇文章的代码中并没实现这点,以后有机会有时间再贴上代码。


遗传算法的所需参数


说简单点遗传算法就是遍历搜索空间或连接池,从中找出最优的解。搜索空间中全部都是个体,而群体为搜索空间的一个子集。并不是所有被选择了的染色体都要进行交叉操作和变异操作,而是以一定的概率进行,一般在程序设计中交叉发生的概率要比变异发生的概率选取得大若干个数量级。大部分遗传算法的步骤都很类似,常使用如下参数:


Fitness函数:在这篇文章中为:


Fitnessthreshold(适应度阈值):适合度中的设定的阀值,当最优个体的适应度达到给定的阀值,或者最优个体的适应度和群体适应度不再上升时(变化率为零),则算法的迭代过程收敛、算法结束。否则,用经过选择、交叉、变异所得到的新一代群体取代上一代群体,并返回到选择操作处继续循环执行。这篇文章中没使用这个阈值。


P:种群的染色体总数叫种群规模,它对算法的效率有明显的影响,其长度等于它包含的个体数量。太小时难以求出最优解,太大则增长收敛时间导致程序运行时间长。对不同的问题可能有各自适合的种群规模,通常种群规模为 30 至 160。这篇文章中使用的群体规模为20。


pc:在循环中进行交叉操作所用到的概率。交叉概率(Pc)一般取0.6至0.95之间的值,Pc太小时难以向前搜索,太大则容易破坏高适应值的结构。这篇文章中的交叉概率为0.65。


Pm:变异概率,从个体群中产生变异的概率,变异概率一般取0.01至0.03之间的值变异概率Pm太小时难以产生新的基因结构,太大使遗传算法成了单纯的随机搜索。这篇文章中的编译概率为0.01。


另一个系统参数是个体的长度,有定长和变长两种。它对算法的性能也有影响。由于GA是一个概率过程,所以每次迭代的情况是不一样的,系统参数不同,迭代情况也不同。这篇文章中并没用到这个。


遗传步骤


了解了上面的基本参数,下面我们来看看遗传算法的基本步骤。


基本过程为:


对待解决问题进行编码,我们将问题结构变换为位串形式编码表示的过程叫编码;而相反将位串形式编码表示变换为原问题结构的过程叫译码。


随机初始化群体P(0):=(p1, p2, … pn);


计算群体上每个个体的适应度值(Fitness)


评估适应度,对当前群体P(t)中每个个体Pi计算其适应度F(Pi),适应度表示了该个体的性能好坏


按由个体适应度值所决定的某个规则应用选择算子产生中间代Pr(t)


依照Pc选择个体进行交叉操作


仿照Pm对繁殖个体进行变异操作


没有满足某种停止条件,则转第3步,否则进入9


输出种群中适应度值最优的个体


程序的停止条件最简单的有如下二种:完成了预先给定的进化代数则停止;种群中的最优个体在连续若干代没有改进或平均适应度在连续若干代基本没有改进时停止。


代码实现


#include


#include


#include


#include


using namespace std;


const int M = 8; //染色体的数目,群体的大小


const int T = 7; //终止代数


const double pc = 0.6; //交叉概率


const double pm = 0.01; //变异概率


struct Population


{


int g【20】; //将x,y的取值区间分为(2^10-1)份,用20位二进制位表示份数


double x,y;


double fit; //适应函数值


double nfit;


double sumfit; //适应值总和


};


class gaModel{


public:


int rand01() const; //随机生成01


double randab(double,double) const; //随即生成【a,b)的一个数


void initial(Population ); // 初始化函数


void evalueFitness(Population ); // 计算适应度


void select(Population ); // 选择复制函数


void crossover(Population ); // 交叉函数


void mutation(Population ); // 变异函数


void decoding(Population ); // 解码函数


void print(Population ); // 显示函数


Population p【M】;


};


#include "GaModel.h"


int gaModel::rand01() const{


int r = 0;


double q = 0.0;


q = rand()/(RAND_MAX + 0.0);


if(q < 0.5)


r = 0;


else


r = 1;


return r;


}


double gaModel::randab(double a,double b) const{


double c,r;


c = b - a;


r = a + c rand()/(RAND_MAX+1.0);


return r;


}


void gaModel::initial(Population t){


Population po;


srand(time(0));


for(po = t; po < t + M; po++) //初始化群体


for(int j = 0; j < 20; j++)


(po).g【j】 = rand01();


}


void gaModel::evalueFitness(Population t){


double f,xx,yy,temp = 0.0;


Population po1,po2;


double fmax = 0,fmin = 9999999999;


for(po1 = t; po1 < t + M; po1++){ //计算适应值


xx = (po1).x;


yy = (po1).y;


f = xx sin(6 yy) + yy cos(8 xx);//优化函数


(po1).fit = f;


//cout [ (po1).fit [ "测试" [ endl;


}


for(po1 = t; po1 < t + M; po1++){


if((po1).fit > fmax)


fmax = (po1).fit;


if((po1).fit [span style="color: rgba(0, 0, 0, 1)"> fmin)


fmin = (po1).fit;


}


for(po1 = t; po1 < t + M; po1++){


(po1).nfit = ((po1).fit - fmin) / (fmax - fmin + 1);


}


for(po1 = t; po1 < t + M; po1++){ //计算累计适应值,为的是之后的轮盘赌模型


for(po2 = t; po2 <= po1; po2++)


temp = temp + (po2).nfit;


(po1).sumfit = temp;


temp = 0.0;


}


}


void gaModel::select(Population t){ //轮盘赌选择


int i = 0;


Population pt【M】,po;


double s;


srand(time(0));


while(i [span style="color: rgba(0, 0, 0, 1)"> M){


s = randab((t).sumfit,((t + M - 1)).sumfit);


//代码效果参考:http://hnjlyzjd.com/xl/wz_25190.html

for(po = t; po < t + M; po++){

if((po).sumfit >= s){


pt【i】 = (po);


break;


}


else continue;


}


i++;


}


for(i = 0; i < M; i++)


for(int j = 0; j < 20; j++)


t【i】.g【j】 = pt【i】.g【j】;


}


void gaModel::crossover(Population t){ //交叉函数


Population po;


double q; //存放一个0到1的随机数,判定是否交叉


int d,e,tem【20】; //d存放一个从1到19的一个随机整数,用来确定交叉的位置


int k = 0,rpo【5】 = {99,99,99,99,99};//e存放从0到M的一个随机且与当前t【i】中i不同的整数,用来确定交叉的对象


bool tf = true; //rpo存放每次的e,确保所有的染色体都会交叉


srand(time(0));


for(po = t; po < t + M/2; po++){


//代码效果参考:http://hnjlyzjd.com/hw/wz_25188.html

q = rand() / (RAND_MAX + 0.0);

while(true){ //避免自己与自己交叉和交叉过的染色体继续交叉


tf = true;


e = rand() % M;


for(int j = 0; j < M; j++){


if(e == rpo【j】)


tf = false;


}


if(t + e != po && tf)


break;


}


if(q [span style="color: rgba(0, 0, 0, 1)"> pc){


d = 1 + rand() % 19; //确定交叉位置


for(int i = d; i < 20; i++){


tem【i】 = (po).g【i】;


(po).g【i】 = ((t+e)).g【i】;


((t+e)).g【i】 = tem【i】;


}


}


rpo【k】 = e;


k++;


}


}


void gaModel::mutation(Population t){ //变异函数


double q;


Population po;


srand(time(0));


for(po = t; po < t + M; po++){


int i = 0;


while(i < 20){


q = rand() / (RAND_MAX + 0.0);


if(q [span style="color: rgba(0, 0, 0, 1)"> pm)


(po).g【i】 = 1 - (po).g【i】;


i++;


}


}


}


void gaModel::decoding(Population t){ //解码函数


Population po;


int temp,s1 = 0, s2 = 0;


float m,n,dit;


n = ldexp(1.0,10);


dit = (2 - 1) / (n - 1);


for(po = t; po < t + M; po++){


for(int i = 0; i < 10; i++){


temp = ldexp((po).g【i】1.0,i);


s1 = s1 + temp;


}


m = 1 + s1 dit;


(po).x = m;


s1 = 0;


for(int j = 10; j < 20; j++){


temp = ldexp((po).g【j】1.0,j-10);


s2 = s2 + temp;


}


m = 1 + s2 dit;


(po).y = m;


s2 = 0;


}


}


void gaModel::print(Population t){ //显示函数


Population po;


for(po = t; po < t + M; po++){


for(int j = 0; j < 20; j++){


cout [ (po).g【j】;


if((j+1)%5 == 0)


cout [ ' ';


}


cout [ endl;


cout [ "x=" [ (po).x [ ' ' [ "y=" [ (po).y [ ' ' [ "fit=" [ (po).fit [ ' ' [ "nfit=" [ (po).nfit [ ' ' [ "sumfit=" [ (po).sumfit [ endl;


}


}


#include "GaModel.h"


int main(){


int kkk = 0;


gaModel g1;


int gen = 0;


g1.initial(&(g1.p【0】));


g1.print(&(g1.p【0】));


cout [ endl;


g1.decoding(&(g1.p【0】));


g1.print(&(g1.p【0】));


cout [ endl;


g1.evalueFitness(&(g1.p【0】));


g1.print(&(g1.p【0】));


while(gen [span style="color: rgba(0, 0, 0, 1)"> T){


cout [ "gen=" [ gen + 1 [ endl;


g1.select(&(g1.p【0】));


g1.decoding(&(g1.p【0】));


g1.evalueFitness(&(g1.p【0】));


cout [ "sel

相关文章
|
4天前
|
机器学习/深度学习 前端开发 算法
婚恋交友系统平台 相亲交友平台系统 婚恋交友系统APP 婚恋系统源码 婚恋交友平台开发流程 婚恋交友系统架构设计 婚恋交友系统前端/后端开发 婚恋交友系统匹配推荐算法优化
婚恋交友系统平台通过线上互动帮助单身男女找到合适伴侣,提供用户注册、个人资料填写、匹配推荐、实时聊天、社区互动等功能。开发流程包括需求分析、技术选型、系统架构设计、功能实现、测试优化和上线运维。匹配推荐算法优化是核心,通过用户行为数据分析和机器学习提高匹配准确性。
26 3
|
4天前
|
算法
PAI下面的gbdt、xgboost、ps-smart 算法如何优化?
设置gbdt 、xgboost等算法的样本和特征的采样率
21 2
|
19天前
|
算法
基于GA遗传算法的PID控制器参数优化matlab建模与仿真
本项目基于遗传算法(GA)优化PID控制器参数,通过空间状态方程构建控制对象,自定义GA的选择、交叉、变异过程,以提高PID控制性能。与使用通用GA工具箱相比,此方法更灵活、针对性强。MATLAB2022A环境下测试,展示了GA优化前后PID控制效果的显著差异。核心代码实现了遗传算法的迭代优化过程,最终通过适应度函数评估并选择了最优PID参数,显著提升了系统响应速度和稳定性。
|
16天前
|
算法
基于WOA鲸鱼优化的购售电收益与风险评估算法matlab仿真
本研究提出了一种基于鲸鱼优化算法(WOA)的购售电收益与风险评估算法。通过将售电公司购售电收益风险计算公式作为WOA的目标函数,经过迭代优化计算出最优购电策略。实验结果表明,在迭代次数超过10次后,风险价值收益优化值达到1715.1万元的最大值。WOA还确定了中长期市场、现货市场及可再生能源等不同市场的最优购电量,验证了算法的有效性。核心程序使用MATLAB2022a实现,通过多次迭代优化,实现了售电公司收益最大化和风险最小化的目标。
|
20天前
|
算法
通过matlab分别对比PSO,反向学习PSO,多策略改进反向学习PSO三种优化算法
本项目使用MATLAB2022A版本,对比分析了PSO、反向学习PSO及多策略改进反向学习PSO三种优化算法的性能,主要通过优化收敛曲线进行直观展示。核心代码实现了标准PSO算法流程,加入反向学习机制及多种改进策略,以提升算法跳出局部最优的能力,增强全局搜索效率。
|
16天前
|
算法
通过matlab对比遗传算法优化前后染色体的变化情况
该程序使用MATLAB2022A实现遗传算法优化染色体的过程,通过迭代选择、交叉和变异操作,提高染色体适应度,优化解的质量,同时保持种群多样性,避免局部最优。代码展示了算法的核心流程,包括适应度计算、选择、交叉、变异等步骤,并通过图表直观展示了优化前后染色体的变化情况。
|
20天前
|
算法
基于大爆炸优化算法的PID控制器参数寻优matlab仿真
本研究基于大爆炸优化算法对PID控制器参数进行寻优,并通过Matlab仿真对比优化前后PID控制效果。使用MATLAB2022a实现核心程序,展示了算法迭代过程及最优PID参数的求解。大爆炸优化算法通过模拟宇宙大爆炸和大收缩过程,在搜索空间中迭代寻找全局最优解,特别适用于PID参数优化,提升控制系统性能。
|
20天前
|
机器学习/深度学习 算法 数据安全/隐私保护
基于贝叶斯优化CNN-GRU网络的数据分类识别算法matlab仿真
本项目展示了使用MATLAB2022a实现的贝叶斯优化、CNN和GRU算法优化效果。优化前后对比显著,完整代码附带中文注释及操作视频。贝叶斯优化适用于黑盒函数,CNN用于时间序列特征提取,GRU改进了RNN的长序列处理能力。
|
22天前
|
并行计算 算法 测试技术
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面,旨在通过综合策略提升程序性能,满足实际需求。
52 1
|
18天前
|
算法 决策智能
基于遗传优化算法的TSP问题求解matlab仿真
本项目使用遗传算法解决旅行商问题(TSP),目标是在四个城市间找到最短路径。算法通过编码、选择、交叉、变异等步骤,在MATLAB2022A上实现路径优化,最终输出最优路径及距离。
下一篇
DataWorks