昨天我开始探索遗传算法,当我结束了一些基本理论时,我尝试在 Python 上编写简单的 GA,求解丢番图方程。我是 Python 和 GA 的新手,所以请不要严格判断我的代码。
由于过早收敛,我无法得到任何结果(有一些不返回点(n-population),population[n] == population[n+i],其中 i 是任何整数。即使是随机变异元素无法改变这一点,这一代正在迅速退化)
GA 正在使用交叉育种,以及 parent 的加权选择。
代码:
# -*- coding: utf-8 -*-
from random import randint
from copy import deepcopy
from math import floor
import random
class Organism:
#initiate
def __init__(self, alleles, fitness, likelihood):
self.alleles = alleles
self.fitness = fitness
self.likelihood = likelihood
self.result = 0
def __unicode__(self):
return '%s [%s - %s]' % (self.alleles, self.fitness, self.likelihood)
class CDiophantine:
def __init__(self, coefficients, result):
self.coefficients = coefficients
self.result = result
maxPopulation = 40
organisms = []
def GetGene (self,i):
return self.organisms[i]
def OrganismFitness (self,gene):
gene.result = 0
for i in range (0, len(self.coefficients)):
gene.result += self.coefficients[i]*gene.alleles[i]
gene.fitness = abs(gene.result - self.result)
return gene.fitness
def Fitness (self):
for organism in self.organisms:
organism.fitness = self.OrganismFitness(organism)
if organism.fitness == 0:
return organism
return None
def MultiplyFitness (self):
coefficientSum = 0
for organism in self.organisms:
coefficientSum += 1/float(organism.fitness)
return coefficientSum
def GenerateLikelihoods (self):
last = 0
multiplyFitness = self.MultiplyFitness()
for organism in self.organisms:
last = ((1/float(organism.fitness)/multiplyFitness)*100)
#print '1/%s/%s*100 - %s' % (organism.fitness, multiplyFitness, last)
organism.likelihood = last
def Breed (self, parentOne, parentTwo):
crossover = randint (1,len(self.coefficients)-1)
child = deepcopy(parentOne)
initial = 0
final = len(parentOne.alleles) - 1
if randint (1,100) < 50:
father = parentOne
mother = parentTwo
else:
father = parentTwo
mother = parentOne
child.alleles = mother.alleles[:crossover] + father.alleles[crossover:]
if randint (1,100) < 5:
for i in range(initial,final):
child.alleles[i] = randint (0,self.result)
return child
def CreateNewOrganisms (self):
#generating new population
tempPopulation = []
for _ in self.organisms:
iterations = 0
father = deepcopy(self.organisms[0])
mother = deepcopy(self.organisms[1])
while father.alleles == mother.alleles:
father = self.WeightedChoice()
mother = self.WeightedChoice()
iterations+=1
if iterations > 35:
break
kid = self.Breed(father,mother)
tempPopulation.append(kid)
self.organisms = tempPopulation
def WeightedChoice (self):
list = []
for organism in self.organisms:
list.append((organism.likelihood,organism))
list = sorted((random.random() * x[0], x[1]) for x in list)
return list[-1][1]
def AverageFitness (self):
sum = 0
for organism in self.organisms:
sum += organism.fitness
return float(sum)/len(self.organisms)
def AverageLikelihoods (self):
sum = 0
for organism in self.organisms:
sum += organism.likelihood
return sum/len(self.organisms)
def Solve (self):
solution = None
for i in range(0,self.maxPopulation):
alleles = []
#
for j in range(0, len(self.coefficients)):
alleles.append(randint(0, self.result))
self.organisms.append(Organism(alleles,0,0))
solution = self.Fitness()
if solution:
return solution.alleles
iterations = 0
while not solution and iterations <3000:
self.GenerateLikelihoods()
self.CreateNewOrganisms()
solution = self.Fitness()
if solution:
print 'SOLUTION FOUND IN %s ITERATIONS' % iterations
return solution.alleles
iterations += 1
return -1
if __name__ == "__main__":
diophantine = CDiophantine ([1,2,3,4],30)
#cProfile.run('diophantine.Solve()')
print diophantine.Solve()
我尝试改变品种和加权随机选择逻辑但没有结果。这个 GA 应该可以工作,但我不知道出了什么问题。 我知道 Python 上有一些 GA 库,我现在正试图理解它们——它们对我来说似乎很复杂。抱歉犯错,英语不是我的母语。感谢您的理解。
死灵更新: 以格雷码而非整数形式存储染色体。
最佳答案
轻微的逻辑错误:parentTwo 更可能是父亲而不是母亲。偶数赔率是 randint (1,100) <= 50,而不是="" randint="" (1,100)="">=><>
在如此小的种群规模下,200-300 代解决问题并不奇怪。如果你增加人口,它应该减少所需的世代。
注意:我发现了一些几年前为解决类似问题而编写的旧代码。它在 C 中,并使用锦标赛选择,但也许它可以给你一些想法:
/*Diophantine equation solving genetic algorithm
Copyright (C) 2009- by Joel Rein
Licensed under the terms of the MIT License*/
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#define POP 100
//number of variables to solve for
#define VAR 4
//maximum value for a) result and b) variables
#define MAX 100
#define MAX_GENS 500
//probability of crossover (otherwise just one parent will be used)
#define CROSSOVER 0.7
//probability of mutation (per gene)
#define MUTATION 0.4
//print out debug information each generation (recommended: if used, keep RUNS low)
#define DEBUG
//print result of each run individually
#define PRINT_RESULT
//how many times to run the GA
#define RUNS 1
int pop[POP][VAR], scores[POP], new[POP][VAR];
int coefficients[VAR];
int result=0;
int score(int index){
int sum=0;
for(int i=0;i<VAR;i++)
sum+=coefficients[i]*pop[index][i];
return abs(sum-result);
}
int tournament(int size){
int best=rand()%POP;
for(int i=1;i<size;i++){
int comp=rand()%POP;
if(scores[comp]<scores[best])
best=comp;
}
return best;
}
void breed(int target){
int a=tournament(3), b=tournament(3);
//copy a
for(int i=0;i<VAR;i++)
new[target][i]=pop[a][i];
//crossover
if((float)rand()/RAND_MAX<CROSSOVER){
int x=rand()%VAR;
for(int i=x;i<VAR;i++)
new[target][i]=pop[b][i];
}
//mutation
for(int i=0;i<VAR;i++)
if((float)rand()/RAND_MAX<MUTATION)
new[target][i]=rand()%(result*2)-result;
}
void debug(int gen, int best){
#ifdef DEBUG
printf("Gen: %3i Score: %3i --- ", gen, scores[best]);
int sum=0;
for(int i=0;i<VAR;i++){
sum+=coefficients[i]*pop[best][i];
printf("%3i*%3i+", coefficients[i], pop[best][i]);
}
printf("0= %3i (target: %i)\n", sum, result);
#endif
}
int ga(int run){
srand(time(NULL)+run);
//calculate a result for the equation.
//this mustn't be 0, else we get division-by-zero errors while initialising & mutating.
while(!result)
result=rand()%MAX;
for(int i=0;i<VAR;i++)
coefficients[i]=rand()%result;
//initialise population
for(int i=0;i<POP;i++)
for(int j=0;j<VAR;j++)
pop[i][j]=rand()%(result*2)-result;
//main loop
int gen, best;
for(gen=0;gen<MAX_GENS;gen++){
best=0;
//evaluate population
for(int i=0;i<POP;i++){
scores[i]=score(i);
if(scores[i]<scores[best])
best=i;
}
debug(gen, best);
if(scores[best]==0)
break;
//breed and replace
for(int i=0;i<POP;i++)
breed(i);
for(int i=0;i<POP;i++)
for(int j=0;j<VAR;j++)
pop[i][j]=new[i][j];
}
#ifdef PRINT_RESULT
printf("Terminated after %4i generations with a score of %3i\n", gen, scores[best]);
#else
printf(".");
#endif
return gen;
}
int main(){
int total=0;
for(int i=0;i<RUNS;i++)
total+=ga(i);
printf("\nAverage runtime: %i generations\n", total/RUNS);
}
关于python - 如何修复简单 GA(Python)中的过早收敛?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6512275/
我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div
总的来说,我对ruby还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用
我试图在一个项目中使用rake,如果我把所有东西都放到Rakefile中,它会很大并且很难读取/找到东西,所以我试着将每个命名空间放在lib/rake中它自己的文件中,我添加了这个到我的rake文件的顶部:Dir['#{File.dirname(__FILE__)}/lib/rake/*.rake'].map{|f|requiref}它加载文件没问题,但没有任务。我现在只有一个.rake文件作为测试,名为“servers.rake”,它看起来像这样:namespace:serverdotask:testdoputs"test"endend所以当我运行rakeserver:testid时
作为我的Rails应用程序的一部分,我编写了一个小导入程序,它从我们的LDAP系统中吸取数据并将其塞入一个用户表中。不幸的是,与LDAP相关的代码在遍历我们的32K用户时泄漏了大量内存,我一直无法弄清楚如何解决这个问题。这个问题似乎在某种程度上与LDAP库有关,因为当我删除对LDAP内容的调用时,内存使用情况会很好地稳定下来。此外,不断增加的对象是Net::BER::BerIdentifiedString和Net::BER::BerIdentifiedArray,它们都是LDAP库的一部分。当我运行导入时,内存使用量最终达到超过1GB的峰值。如果问题存在,我需要找到一些方法来更正我的代
关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。
Rails2.3可以选择随时使用RouteSet#add_configuration_file添加更多路由。是否可以在Rails3项目中做同样的事情? 最佳答案 在config/application.rb中:config.paths.config.routes在Rails3.2(也可能是Rails3.1)中,使用:config.paths["config/routes"] 关于ruby-on-rails-Rails3中的多个路由文件,我们在StackOverflow上找到一个类似的问题
给定这段代码defcreate@upgrades=User.update_all(["role=?","upgraded"],:id=>params[:upgrade])redirect_toadmin_upgrades_path,:notice=>"Successfullyupgradeduser."end我如何在该操作中实际验证它们是否已保存或未重定向到适当的页面和消息? 最佳答案 在Rails3中,update_all不返回任何有意义的信息,除了已更新的记录数(这可能取决于您的DBMS是否返回该信息)。http://ar.ru
我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t
我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚
Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack