遗传算法解flowshop问题

2023-12-16 14:38

本文主要是介绍遗传算法解flowshop问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

遗传算法解决flowshop问题

1.遗传算法简介

参考:https://blog.csdn.net/u010451580/article/details/51178225

百度百科:遗传算法

总结遗传算法过程:选择 -> 交叉 -> 变异 -> 更新 -> 终止

2.flowshop问题简介

已知:有n 个工件需要在m台机器上流水加工。

工件上的约束:所有工件均在0时刻释放且在各机器上的加工顺序相同,每个工件在每台机器上只加工一次。

机器上的约束:每个机器某一个时刻最多只能加工一个工件,而且执行过程是非抢占的。

目标:给出调度方案,使调度总完工时间最小。

要求:算法复杂度在多项式时间内。

总结来说就是:n个工件在m台机器上流水加工,所有工件0时刻释放,且在机器上加工顺序相同,只加工一次。机器每时刻最多只能加工一个工件,同时,执行过程是非抢占的。那么转化为组合优化问题就是,需要给出m个1-n的排列,使得每台机器按相应的一个排列对工件进行加工,从而使得总加工时间最短。那么就需要找到这样的m个1-n的排列作为一个解,使总加工时间最短。当给出一个解时,通过动态规划可以很容易计算得到加工时间。

输入如下:

 11 50 375 1  12 2 142 3 245 4 4120 632 1 452 2 758 3 278 4 3980  12 1 876 2 124 3 534 4 7650 460 1 542 2 523 3 120 4 4990 528 1 101 2 789 3 124 4 9990 796 1 245 2 632 3 375 4 1230 532 1 230 2 543 3 896 4 4520  14 1 124 2 214 3 543 4 7850 257 1 527 2 753 3 210 4 4630 896 1 896 2 214 3 258 4 2590 532 1 302 2 501 3 765 4 988

optimum result:7038

说明:第一行是工件数和机器数,2-12行是每个工件在每台标号的机器上的加工时长如第二行就是第一个工件在0机器加工375,1机器加工12…

3.遗传算法代码

对于遗传算法,其关键点在初始化,交叉算子,变异算子、选择算子以及适应度的选择与确定。对于适应度的选择,本文以每个解的总时间花费作为一个个体的适应度;针对初始化,生成一个具有一定规模大小的种群,每个个体有m个染色体,每条染色体有n个基因,即一个染色体上有一个随机的排列;选择算子目前包括精英选择和轮盘赌方式,本文选择轮盘赌方式;对于交叉算子和变异算子,现有的交叉算子有AP,CX,ER,OX1,OX2,PMX,POS,VRR共八种,变异算子有DM,EM,ISM,IVM,SIM,SM共六种,本文选择PMX交叉算子和EM变异算子,二者操作方式如图;最后就是迭代种群,本文采取的方式是,将父代种群与子代种群合并,然后选取适应度小的一些个体作为下一代种群个体,同时保证种群大小不变。

具体遗传算子总结参考:https://blog.csdn.net/u012750702/article/details/54563515

PMX和EM操作如下:

/*** date: 2018-06-14* author: silvester* (utf-8) 编译环境:gcc version 5.3.0 (GCC) -std=c++11*/
#include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<math.h>
#include<time.h>
#define INF 0x3f3f3f3f
#define MACHINE 10
#define JOB 50int m,n;//m台机器,n个工件
int infor[JOB][MACHINE];//记录n个工件在每台机器上加工时长void inputInformation()//读取输入信息
{int k,t;freopen("ins3.txt","r",stdin);scanf("%d %d",&n,&m);for(int i=0;i<n;i++){for(int j=0;j<m;j++){scanf("%d %d",&k,&t);infor[i][k]=t;}}
}void swap(int *x,int *y)//交换两个元素
{int temp=*x;*x=*y;*y=temp;
}
void randArray(int *a)//获取一个随机序列
{for(int i=0;i<n;i++){//先初始化该序列a[i]=i;}for(int i=0;i<n;i++){//然后随机交换两个位置n次int rd=rand()%(n-i)+i;swap(&a[i],&a[rd]);}
}int max(int a,int b){//取大return a>=b?a:b;
}/******GA*******/
#define GROUPSCALE 50//种群大小,即规模
#define SONNUM 50//孩子种群最大规模
int inheritance=50000;//迭代代数
double mutation=0.1;//变异概率
double crossover=0.8;//交叉概率,即生产儿子概率struct Unit{//一个个体定义int chromosome[MACHINE][JOB];//个体染色体MACHINE条,每条染色体JOB个基因int fitness;//个体适应度double proba;//根据适应度计算得到个体在种群中被选择做父本的概率
};struct Unit group[GROUPSCALE];//定义种群
struct Unit sonGroup[SONNUM];//定义儿子种群
int newbornNum;//新生儿子个数
int indexcross_i;//即将交叉染色体片段起始位置
int indexcross_j;//即将交叉染色体片段终止位置void printfUnit(Unit u){//输出某一个个体信息for(int i=0;i<m;i++){//输出基因型for(int j=0;j<n;j++){printf("%d%c",u.chromosome[i][j],j==n-1?'\n':' ');}}//输出适应度printf("%d\n",u.fitness);printf("----------------\n");
}int calculateTime_GA(Unit unit)//适应度函数,flowshop问题中,即为加工这些工件的时间花费
{int time=0,fin_time[JOB];memset(fin_time,0,sizeof(fin_time));//对于m个排列,每台机器按照一个相应的排列顺序来加工工件,若工件还未到达,则等待//动态规划计算一个解决方案的花费时间for(int i=0;i<m;i++){for(int j=0;j<n;j++){time=max(time,fin_time[unit.chromosome[i][j]])+infor[unit.chromosome[i][j]][i];fin_time[unit.chromosome[i][j]]=time;}if(i==m-1) return time;else time=0;}return 0;
}void calculateProbality(int total)//计算种群各个个体被选择作为父本概率,传入的数为种群fitness之和
{double tempTotalP=0.0;// 由于是fitness越小,被选择概率越大,因此,需要做相应函数变化,而不能直接按比例得到for(int i=0;i<GROUPSCALE;i++){group[i].proba=(1.0/(double)group[i].fitness)*(double)total;tempTotalP+=group[i].proba;}for(int i=0;i<GROUPSCALE;i++){group[i].proba=group[i].proba/tempTotalP;}
}void init_GA()//初始化种群
{int total=0;for(int i=0;i<GROUPSCALE;i++){for(int j=0;j<m;j++){randArray(group[i].chromosome[j]);//每条染色体随机初始化一个排列}group[i].fitness=calculateTime_GA(group[i]);//计算这个个体fitnesstotal+=group[i].fitness;//累计种群fitness之和}calculateProbality(total);//计算种群个体被选择做父本概率
}
//挑选父本方式1:精英选择,直接选择种群中适应度最小的两个个体
void selectTwo(int &father_pos,int &mother_pos)
{int one=0,minCost=INF;for(int i=0;i<GROUPSCALE;i++){//挑选第一个if(minCost>group[i].fitness){minCost=group[i].fitness;one=i;}}father_pos=one;minCost=INF;one=0;for(int i=0;i<GROUPSCALE;i++){//挑选第二个if(i==father_pos) continue;if(minCost>group[i].fitness){minCost=group[i].fitness;one=i;}}mother_pos=one;
}//挑选父本方式2:轮盘赌选择,模拟自然选择,依据被选择概率随机选择两个个体做父本
int selectOne(int conf)
{double selectP=((double)(rand()%999)/998.0);double sumP=0.0;for(int i=0;i<GROUPSCALE;i++){sumP+=group[i].proba;if(selectP<sumP){if(i==conf) return i+1;//解决选取的两个父本是同一个的问题else return i;}}return 0;
}void getConflict(int Detection[],int Model[],int len_cross,int &len_conflict,int *conflict)
{len_conflict=0;for(int i=0;i<len_cross;i++)//如果对Detection中的某个基因在Model中也存在,则不存在冲突否则存在冲突{int flag=1;for(int j=0;j<len_cross;j++){if(Detection[i]==Model[j]){j=len_cross;flag=0;}}if(flag){conflict[len_conflict]=Detection[i];//将冲突基因存在conflict中len_conflict++;}}
}void handleConflict(Unit conflictUnit,int *Detection_Conflict,int *Model_Conflict,int len_conflict,int k,int *p)
{// 解决冲突,使得一个排列中不出现重复的编号for(int i=0;i<len_conflict;i++){//共len_conflict个冲突编号int flag=0;int index=0;//重复的编号在Model_Conflict中//以下对非交换区域找出编号重复的位置for(index=0;index<indexcross_i;index++){if(Model_Conflict[i]==conflictUnit.chromosome[k][index]){flag=1;break;}}if(!flag){for(index=indexcross_j+1;index<n;index++){if(Model_Conflict[i]==conflictUnit.chromosome[k][index])break;}}//将重复的编号替换,替换编号在Detection_Conflict中conflictUnit.chromosome[k][index]=Detection_Conflict[i];}for(int i=0;i<n;i++){//终止将这条新染色体赋值,传给孩子p[i]=conflictUnit.chromosome[k][i];}
}void Crossover_PMX(Unit fa,Unit mo)//采用PMX方式进行交叉变异
{Unit son_one;//孩子1Unit son_two;//孩子2son_one.fitness=0;son_two.fitness=0;for(int k=0;k<m;k++){//对m条染色体交叉indexcross_i=rand()%n;indexcross_j=rand()%n;//确定交叉片段起始和终止位置if(indexcross_i>indexcross_j){int temp=indexcross_i;indexcross_i=indexcross_j;indexcross_j=temp;}int father_cross[JOB];//记录父亲交叉片段int mother_cross[JOB];//记录母亲交叉片段int len_cross=0;for(int i=indexcross_i;i<=indexcross_j;i++){father_cross[len_cross]=fa.chromosome[k][i];mother_cross[len_cross]=mo.chromosome[k][i];len_cross++;}int conflict_fa[JOB];//记录父亲产生的冲突片段int conflict_ma[JOB];//记录母亲产生的冲突片段int len_conflict=0;//冲突基因个数getConflict(father_cross,mother_cross,len_cross,len_conflict,conflict_fa);//获取父亲冲突基因片段getConflict(mother_cross,father_cross,len_cross,len_conflict,conflict_ma);//获取母亲冲突基因片段for(int i=indexcross_i;i<=indexcross_j;i++){//将父亲和母亲交叉片段进行交换int temp_node=fa.chromosome[k][i];fa.chromosome[k][i]=mo.chromosome[k][i];mo.chromosome[k][i]=temp_node;}//分别对父亲和母亲处理冲突的基因片段,这样可以得到两个孩子handleConflict(fa,conflict_fa,conflict_ma,len_conflict,k,son_one.chromosome[k]);handleConflict(mo,conflict_ma,conflict_fa,len_conflict,k,son_two.chromosome[k]);    }sonGroup[newbornNum++]=son_one;//将两个孩子加入子代种群sonGroup[newbornNum++]=son_two;
}void Mutation(int index,int k)//对第index个个体的第k条染色体进行变异
{int gen_i=rand()%n;int gen_j=rand()%n;//变异采取的方式是随机交换两个基因int temp=sonGroup[index].chromosome[k][gen_i];sonGroup[index].chromosome[k][gen_i]=sonGroup[index].chromosome[k][gen_j];sonGroup[index].chromosome[k][gen_j]=temp;
}int UpdateGroup()//种群更新
{Unit tempP;//对新生种群按适应度从小到大排序for(int i=0;i<newbornNum;i++){for(int j=newbornNum-1;j>i;j--){if(sonGroup[i].fitness>sonGroup[j].fitness){tempP=sonGroup[i];sonGroup[i]=sonGroup[j];sonGroup[j]=tempP;}}}//对原始种群按适应度从小到大排序for(int i=0;i<GROUPSCALE;i++){for(int j=GROUPSCALE-1;j>i;j--){if(group[i].fitness>group[j].fitness){tempP=group[i];group[i]=group[j];group[j]=tempP;}}}//将新生种群中fitness较小的个体替换至原始种群,从而得到新种群int j=GROUPSCALE-1;for(int i=0;i<newbornNum;i++){if(sonGroup[i].fitness<group[j].fitness){group[j]=sonGroup[i];j--;}else break;}int total=0;for(int i=0;i<GROUPSCALE;i++){//计算新种群所以个体适应度之和total+=group[i].fitness;}return total;
}void GA()
{init_GA();int iter=0;int minFitness=INF,minGeneration;Unit minUnit;while(iter<inheritance){int M=GROUPSCALE-GROUPSCALE/2;//每次交叉产生两个儿子,故只需M次交叉newbornNum=0;//记录新生儿个数while(M){//1.选择,两种选择方式//(1)轮盘赌选择int pos1=selectOne(-1);int pos2=selectOne(pos1);//(2)精英选择//int pos1,pos2;//selectTwo(pos1,pos2);//确定两个父本个体Unit father=group[pos1];Unit mother=group[pos2];//2.交叉// int M=GROUPSCALE-GROUPSCALE/2;//每次交叉产生两个儿子,故只需M次交叉// newbornNum=0;//记录新生儿个数// while(M){double Is_crossover=((double)(rand()%999)/998.0);if(Is_crossover<=crossover){//依概率判断是否交叉//采用PMX遗传算子交叉Crossover_PMX(father,mother);}M--;}//3.变异for(int i=0;i<newbornNum;i++){//对每个新生儿,每条染色体依概率判断是否发生变异for(int k=0;k<m;k++){double rateVaration=double(rand()%999/998.0);if(rateVaration<mutation){Mutation(i,k);}}sonGroup[i].fitness=calculateTime_GA(sonGroup[i]);//计算新生儿适应度}//4.更新int totaltime=UpdateGroup();//更新种群并得到新种群个体适应度之和calculateProbality(totaltime);//计算新种群个体被选择概率iter++;//代数自加int minCost=INF,pos;//求出此代最优for(int i=0;i<GROUPSCALE;i++){if(minCost>group[i].fitness){minCost=group[i].fitness;pos=i;}}//记录历代最优if(minFitness>minCost){//更新整体历代最优minFitness=minCost;minUnit=group[pos];minGeneration=iter;}}//5.输出最优printf("Best, No.%d generation:\n",minGeneration);//最优出现代数printfUnit(minUnit);//最优解
}int main()
{inputInformation();//输入long start_time=clock();srand((unsigned)(time(NULL)));//随机种子/*****GA*****/printf("---------GA----------\n");GA();long end_time=clock();//输出算法运行时间printf("time = %ld ms\n",end_time-start_time);return 0;
}

输出结果:

这里写图片描述

五个测试用例如下:

 11 50 375 1  12 2 142 3 245 4 4120 632 1 452 2 758 3 278 4 3980  12 1 876 2 124 3 534 4 7650 460 1 542 2 523 3 120 4 4990 528 1 101 2 789 3 124 4 9990 796 1 245 2 632 3 375 4 1230 532 1 230 2 543 3 896 4 4520  14 1 124 2 214 3 543 4 7850 257 1 527 2 753 3 210 4 4630 896 1 896 2 214 3 258 4 2590 532 1 302 2 501 3 765 4 98813 40 654 1 147 2 345 3 4470 321 1 520 2 789 3 7020  12 1 147 2 630 3 2550 345 1 586 2 214 3 8660 678 1 532 2 275 3 3320 963 1 145 2 302 3 2250  25 1  24 2 142 3 5890 874 1 517 2  24 3 9960 114 1 896 2 520 3 5410 785 1 543 2 336 3 2340 203 1 210 2 699 3 7840 696 1 784 2 855 3 5120 302 1 512 2 221 3 34512 50 456 1 537 2 123 3 214 4 2340 789 1 854 2 225 3 528 4 1230 876 1 632 2 588 3 896 4 4560 543 1 145 2 669 3 325 4 7890 210 1 785 2 966 3 147 4 8760 123 1 214 2 332 3 856 4 5430 456 1 752 2 144 3 321 4 2100 789 1 143 2 755 3 427 4 1230 876 1 698 2 322 3 546 4 4560 543 1 532 2 100 3 321 4 7890 210 1 145 2 114 3 401 4 8760 124 1 247 2 753 3 214 4 54314 40 456 1 856 2 963 3 6960 789 1 930 2  21 3 3200 630 1 214 2 475 3 1420 214 1 257 2 320 3 7530 573 1 896 2 124 3 2140 218 1 532 2 752 3 5280 653 1 142 2 147 3 6530 214 1 547 2 532 3 2140 204 1 865 2 145 3 5270 785 1 321 2 763 3 5360 696 1 124 2 214 3 2140 532 1  12 2 257 3 5280  12 1 345 2 854 3 8880 457 1 678 2 123 3 99910 60 333 1 991 2 996 3 123 4 145 5 2340 333 1 111 2 663 3 456 4 785 5 5320 252 1 222 2 222 3 789 4 214 5 5860 222 1 204 2 114 3 876 4 752 5 5320 255 1 477 2 123 3 543 4 143 5 1420 555 1 566 2 456 3 210 4 698 5 5730 558 1 899 2 789 3 124 4 532 5  120 888 1 965 2 876 3 537 4 145 5  140 889 1 588 2 543 3 854 4 247 5 5270 999 1 889 2 210 3 632 4 451 5 856

注:以上所有操作均在作者在网上搜集资料后,在个人电脑上实验成功,若读者实验时失败,可能由一些未知因素导致,可与作者联系。编写的教程可能由于疏忽出错,请与作者联系。

这篇关于遗传算法解flowshop问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/500828

相关文章

关于@MapperScan和@ComponentScan的使用问题

《关于@MapperScan和@ComponentScan的使用问题》文章介绍了在使用`@MapperScan`和`@ComponentScan`时可能会遇到的包扫描冲突问题,并提供了解决方法,同时,... 目录@MapperScan和@ComponentScan的使用问题报错如下原因解决办法课外拓展总结@

MybatisGenerator文件生成不出对应文件的问题

《MybatisGenerator文件生成不出对应文件的问题》本文介绍了使用MybatisGenerator生成文件时遇到的问题及解决方法,主要步骤包括检查目标表是否存在、是否能连接到数据库、配置生成... 目录MyBATisGenerator 文件生成不出对应文件先在项目结构里引入“targetProje

C#使用HttpClient进行Post请求出现超时问题的解决及优化

《C#使用HttpClient进行Post请求出现超时问题的解决及优化》最近我的控制台程序发现有时候总是出现请求超时等问题,通常好几分钟最多只有3-4个请求,在使用apipost发现并发10个5分钟也... 目录优化结论单例HttpClient连接池耗尽和并发并发异步最终优化后优化结论我直接上优化结论吧,

Java内存泄漏问题的排查、优化与最佳实践

《Java内存泄漏问题的排查、优化与最佳实践》在Java开发中,内存泄漏是一个常见且令人头疼的问题,内存泄漏指的是程序在运行过程中,已经不再使用的对象没有被及时释放,从而导致内存占用不断增加,最终... 目录引言1. 什么是内存泄漏?常见的内存泄漏情况2. 如何排查 Java 中的内存泄漏?2.1 使用 J

numpy求解线性代数相关问题

《numpy求解线性代数相关问题》本文主要介绍了numpy求解线性代数相关问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 在numpy中有numpy.array类型和numpy.mat类型,前者是数组类型,后者是矩阵类型。数组

解决systemctl reload nginx重启Nginx服务报错:Job for nginx.service invalid问题

《解决systemctlreloadnginx重启Nginx服务报错:Jobfornginx.serviceinvalid问题》文章描述了通过`systemctlstatusnginx.se... 目录systemctl reload nginx重启Nginx服务报错:Job for nginx.javas

Redis缓存问题与缓存更新机制详解

《Redis缓存问题与缓存更新机制详解》本文主要介绍了缓存问题及其解决方案,包括缓存穿透、缓存击穿、缓存雪崩等问题的成因以及相应的预防和解决方法,同时,还详细探讨了缓存更新机制,包括不同情况下的缓存更... 目录一、缓存问题1.1 缓存穿透1.1.1 问题来源1.1.2 解决方案1.2 缓存击穿1.2.1

vue解决子组件样式覆盖问题scoped deep

《vue解决子组件样式覆盖问题scopeddeep》文章主要介绍了在Vue项目中处理全局样式和局部样式的方法,包括使用scoped属性和深度选择器(/deep/)来覆盖子组件的样式,作者建议所有组件... 目录前言scoped分析deep分析使用总结所有组件必须加scoped父组件覆盖子组件使用deep前言

解决Cron定时任务中Pytest脚本无法发送邮件的问题

《解决Cron定时任务中Pytest脚本无法发送邮件的问题》文章探讨解决在Cron定时任务中运行Pytest脚本时邮件发送失败的问题,先优化环境变量,再检查Pytest邮件配置,接着配置文件确保SMT... 目录引言1. 环境变量优化:确保Cron任务可以正确执行解决方案:1.1. 创建一个脚本1.2. 修

Python 标准库time时间的访问和转换问题小结

《Python标准库time时间的访问和转换问题小结》time模块为Python提供了处理时间和日期的多种功能,适用于多种与时间相关的场景,包括获取当前时间、格式化时间、暂停程序执行、计算程序运行时... 目录模块介绍使用场景主要类主要函数 - time()- sleep()- localtime()- g