?? sga1.cpp
字號:
//基本遺傳算法,求最大最小值問題,根據遺傳算法原理及其應用(周明)書中程序,修改了隨機數產生語句,使得該程序能在vc中使用 由于隨機數的原因,結果有時對,有時不對(結果應為二十個零),具體原因還沒有弄明白,請各位試試,大家交流一下。
由于隨機數的原因,結果有時對,有時不對(結果應為二十個零)
具體原因還沒有弄明白,請各位試試,大家交流一下。
#include "stdio.h"
#include "stdlib.h"
#include "time.h"
#include "math.h"
//定義常量
#define POPSIZE 500
#define MAXIMIZATION 1
#define MINIMIZATION 2
//定義使用數據,對于不同問題而不同
#define Cmax 100
#define Cmin 0
#define LENGTH1 10
#define LENGTH2 10
#define CHROMLENGTH LENGTH1+LENGTH2
int FunctionMode=MAXIMIZATION;
int PopSize=80;
int MaxGeneration=200;
double Pc=0.6;
double Pm=0.001;
//定義數據結構
struct individual
{
char chrom[CHROMLENGTH];
double value;
double fitness;
};
//定義全局變量
int generation;
int best_index;
int worst_index;
struct individual bestindividual;
struct individual worstindividual;
struct individual currentbest;
struct individual population[POPSIZE];
//函數聲明
void GenerateInitialPopulation(void);
void GenerateNextPopulation(void);
void EvaluatePopulation(void);
long DecodeChromosome(char*,int,int);
void CalculateObjectValue(void);
void CalculateFitnessValue(void);
void FindBestAndWorstIndividual(void);
void PerformEvolution(void);
void SelectionOperator(void);
void CrossOverOperator(void);
void MutationOperator(void);
void OutputTextReport(void);
//主函數
void main(void)
{
//srand((unsigned)time(NULL));
generation=0;
GenerateInitialPopulation();
EvaluatePopulation();
while(generation<MaxGeneration){
generation++;
GenerateNextPopulation();
EvaluatePopulation();
PerformEvolution();
OutputTextReport();
}
}
//初始化,第一代
void GenerateInitialPopulation(void)
{
int i,j;
srand((unsigned)time(NULL));
for(i=0;i<PopSize;i++){
for(j=0;j<CHROMLENGTH;j++){
population[i].chrom[j]=(((rand()%10)<5)?'0':'1');
}
population[i].chrom[CHROMLENGTH]='\0';
}
}
//產生下一代
void GenerateNextPopulation(void)
{
SelectionOperator();
CrossOverOperator();
MutationOperator();
}
//評價
void EvaluatePopulation(void)
{
CalculateObjectValue();
CalculateFitnessValue();
FindBestAndWorstIndividual();
}
//把二進制解碼為十進制
long DecodeChromosome(char *string,int point,int length)
{
int i;
long decimal=0;//
char *pointer;
for(i=0,pointer=string+point;i<length;i++,pointer++)
{
decimal+=(*pointer-'0')<<(length-1-i);
}
return(decimal);
}
//計算目標值
void CalculateObjectValue(void)
{
int i;
long temp1,temp2;
double x1,x2;
for(i=0;i<PopSize;i++)
{
temp1=DecodeChromosome(population[i].chrom,0,LENGTH1);
temp2=DecodeChromosome(population[i].chrom,LENGTH1,LENGTH2);
x1=4.096*temp1/1023.0-2.048;
x2=4.096*temp2/1023.0-2.048;
population[i].value=100*(x1*x1-x2)*(x1*x1-x2)+(1-x1)*(1-x1);
}
}
//計算適應度
void CalculateFitnessValue(void)
{
int i;
double temp;
for(i=0;i<PopSize;i++)
{
if(FunctionMode==MAXIMIZATION){
if((population[i].value+Cmin)>0.0){
temp=Cmin+population[i].value;
}
else{
temp=0.0;
}
}
else if(FunctionMode==MINIMIZATION){
if(population[i].value<Cmax){
temp=Cmax-population[i].value;
}else{
temp=0.0;
}
}
population[i].fitness=temp;
}
}
//在當前代尋找最好個體
void FindBestAndWorstIndividual(void)
{
int i;
double sum=0.0;
bestindividual=population[0];
worstindividual=population[0];
for(i=1;i<PopSize;i++){
if(population[i].fitness<worstindividual.fitness){
worstindividual=population[i];
worst_index=i;
}
sum+=population[i].fitness;
}
if(generation==0){
currentbest=bestindividual;
}else{
if(bestindividual.fitness>currentbest.fitness){
currentbest=bestindividual;
}
}
}
//用最好個體替代最差個體
void PerformEvolution(void)
{
if(bestindividual.fitness>currentbest.fitness){
currentbest=population[best_index];
}else{
population[worst_index]=currentbest;
}
}
//比例選擇產生染色體
void SelectionOperator(void)
{
int i,index;
double p,sum=0.0;
double cfitness[POPSIZE];
struct individual newpopulation[POPSIZE];
//計算相對適應度
for(i=0;i<PopSize;i++){
sum+=population[i].fitness;
}
for(i=0;i<PopSize;i++){
cfitness[i]=population[i].fitness/sum;
}
//計算累積適應度
for(i=1;i<PopSize;i++){
cfitness[i]=cfitness[i-1]+cfitness[i];
}
//選擇
for(i=0;i<PopSize;i++){
p=rand()%1000/1000.0;
index=0;
while(p>cfitness[index]){
index++;
}
newpopulation[i]=population[index];
}
for(i=0;i<PopSize;i++){
population[i]=newpopulation[i];
}
}
//交叉
void CrossOverOperator(void)
{
int i,j;
int index[POPSIZE];
int point,temp;
double p;
char ch;
//任意配對
for(i=0;i<PopSize;i++)
{
index[i]=i;
}
//for(i=0;i<PopSize;i++)
// {
// point=rand()%PopSize-1;
// temp=index[i];
// index[i]=index[point+1];
// index[point+1]=temp;
// }
//一點交叉
for(i=0;i<PopSize-1;i+=2)
{
p=rand()%1000/1000.0;
if(p<Pc){
point=rand()%CHROMLENGTH+1;
for(j=point;j<CHROMLENGTH;j++)
{
ch=population[index[i]].chrom[j];
population[index[i]].chrom[j]=population[index[i+1]].chrom[j];
population[index[i+1]].chrom[j]=ch;
}
}
}
}
//變異
void MutationOperator(void)
{
int i,j;
double p;
for(i=0;i<PopSize;i++){
for(j=0;j<CHROMLENGTH;j++){
p=rand()%1000/1000.0;
if(p<Pm){
population[i].chrom[j]=(population[i].chrom[j]=='0')?'1':'0';
}
}
}
}
//輸出結果
void OutputTextReport(void)
{
int i;
double sum;
double average;
sum=0.0;
for(i=0;i<PopSize;i++){
sum+=population[i].value;
}
average=sum/PopSize;
printf("gen=%d,avg=%f,best=%f,",generation,average,currentbest.value);
printf("chromsome=");
for(i=0;i<CHROMLENGTH;i++){
printf("%c",currentbest.chrom[i]);
}
printf("\n");
}
?? 快捷鍵說明
復制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -