张麟 2010年12月11日 星期六 21:38 | 3497次浏览 | 3条评论
PyGEP的一个说用说明。及一些例子。
PyGEP说用说明
简介
PYGEP包是一个基因表达式编程(gene express programming)的一个开源python代码包
里面最核心的程序就是population.py和chromosome.py,只要读懂这两个程序,整个GEP的思想也在其中了
还是先看一下该方法的框架吧!
从流程图可以看出,基本上的步骤就是
1、生成初始种群(由染色体组成chromosomes),每个染色体表达了问题的一个解。这本质上就是一个测试空间。我们所有的工作就是在该测试空间下通过一系列的操作以寻找出符合我们要求的解。
2、对每个染色体进行解码(Execute),得到我们的一个评估结果。每个染色体对应于一个适应性函数(fitness),和遗传算法(GA)类似。
3、对当前种群的改造,生成新的种群,重复2-3直到得到我们需要的解。
PyGEP结构介绍
一、生成种群
初始化种群时需要如下参数的设置,这些参数将决定如何对现在种群进行改造或者叫遗传。
__init__()函数:def __init__(self, cls, size, head, genes=1, linker=default_linker)
cls表示种群对应的染色体类型,
size表示种群大小,
head种群头部长度,
genes 基因组数,
linker基因链接的方式
这些就是我们调用population的接口。整个改造过程是有cycle()函数完成的。
类中还定义了一个best属性用以标示当前的最好染色体。
二、染色体
问题的描述主要是在染色体中进行的。所以实质上染色体的构造是我们对模型的构造。而种群是对算法的构造。
之所以说GEP算法在并行上有很大的优势是因为她天生就是并行的。因为对于每个染色体的解码是与其他染色体无关的。
分析chromosome类的外部作用,我们需要知道的是如何生成chromosome类,并且需要定义如何评估(fitness及solved)
chromosome类提供了一个generate函数:
def generate(cls, head, genes=1, linker=default_linker)
head是头部长度,genes是基因组数,linker是指链接方式。
同样还有:
def __init__(self, genes, head, linker=default_linker)同上。
关于fitness和solved,PyGEP提供了两个属性来连接。
fitness = property(lambda self: self._fitness(), doc='Fitness value')
solved = property(lambda self: self._solved(), doc='Problem solved')
而所有具体问题的描述都将在fitness和solved中进行。
当然在进行如上操作之前,我们必须要设置符号空间 functions和 terminals ,用以表示表达式。
三、一个例子
这个例子是包中的例子(majority),理解一下就清楚如何使用这个程序包了。
首先定义一个类:
该方程实质上是给了Majority这个bool运算函数的一个运算法则。我们想让gep通过符号运算来得到这法则。
在这个地方我所知道的其实就是不同的参数对应的Majority的值。
这部分继承chromosome类,初始化了functions和terminals的算符表,并重写了_fitness()和_solved()函数。
这部分所做的,调用了整个过程。由这个例子可以看出,如果我们要使用该包去计算一个问题时,我们所要做的是
1、建立一个变量列表所对应的数据表(如Majority中给出的是a,b,c,及对应的majority)
2、建立一个chromosome的类的继承,给定算符空间(functions and terminals)在继承中重写_fitness()和_solved()函数,
以满足我们具体问题的需要。
3、生成初始种群P0,并调用_solved()函数检查是否满足要求,调用cycle()函数进行种群改造。
这些就是我们所要做的事,如果你对比如选择算子,变异算子,等的概率设置不满意,同样可以到population类代码中进行修改。
同样对其他不满意也可以自己去修改,因为代码就在你手中。
又一个例子
:
寻找一个多项式y=x**4+x**3+x**2+x
数据存在data.txt文件里。data.txt文件内容:
9.500366 9103.54918799079
-6.130432 1213.47815867463
3.252685 160.18146840485
7.88797 4432.23529247904
9.090484 7671.79392962917
1.485199 11.8327154232663
-3.950531 193.570365560718
10.003326 11124.3786278048
-.607453 -.326443121119579
5.469299 1093.78835980998
源代码:
result:
+*x++x*x/*xxxxxxxxxxx [2814]: 4
+x*x++x***xxxxxxxxxxx [2951]: 5
+*x++x*x/*xxxxxxxxxxx [2945]: 4
+*+++x*x/*xxxxxxxxxxx [2952]: 0
+*/++x*x/*xxxxxxxxxxx [2953]: 0
+*x++x*x/*xxxxxxxxxxx [2908]: 4
+*x++x*x/*xxxxxxxxxxx [2908]: 4
+*x++x*-+*xxxxxxxxxxx [2954]: 0
+*x++x*x/*xxxxxxxxxxx [2814]: 4
+*x++x*x/*xxxxxxxxxxx [2814]: 4
SOLVED: +x*x++x***xxxxxxxxxxx
so we get the result:x**4+x**3+x**2+x.
代码分析与应用
继续我们的讨论哈!
#--encoding=UTF-8--
#分析PyGEP
一. Chromosome类分析
1.MetaChromosome
这个metachromosome类只重写了type的new方法,并在new方法中为meta实例
添加了symbols字段,用以表示符号集
arity字段,用以识别符号集中的最大操作数目.比如'+'法函数又两个参数'a+b',则arity=2
_fitness字段用以得到适应性函数
2.Chromosome
__metaclass__ = MetaChromosome #得到类型名
__next_id = 1
gene_type = KarvaGene #表示gene类型是KarvaGene的
#下面重点分析一下代码
@classmethod
def generate(cls, head, genes=1, linker=default_linker):
tail = head * (cls.arity - 1) + 1 #得到尾部长度
while True:
new_genes = [None] * genes #新建一个列表
for i in xrange(genes):
#调用KarvaGene,新建染色体序列
new_genes
= cls.gene_type(
[random.choice(cls.symbols) for _ in xrange(head)] + \
[random.choice(cls.terminals) for _ in xrange(tail)], head
)
yield cls(new_genes, head, linker) #返回该染色体,也就是Chromosome的一个实例
#注意这里使用yield生成的是一个代码序列.
#以下__init__代码:
def __init__(self, genes, head, linker=default_linker):
''
if head < 0:
raise ValueError('Head length must be at least 0')
if not genes:
raise ValueError('Must have at least 1 gene')
self.genes = genes
self.head = head
self.linker = linker
# Unique number of the organism
self.__id = type(self).__next_id #每一个实例对应一个id
type(self).__next_id += 1 #这里设置了一个迭代,这个迭代是在上面的while True:..中实现.
#__call__函数
def __call__(self, obj):
return self.linker(*[g(obj) for g in self.genes])
#该函数只有一句,在return中执行,却是得到的我们染色体序列的值.具体需要分析karvaGene.这里不做解释.
#其他是一些操作.暂时不做解释.这些不影响我们的使用.
#现在有一个问题,如果我希望将一些染色体本身作为染色体序列中的一个gene来进行处理,那么该如何去做呢?
#这个问题的背景可以设想为,比如我知道当前的符号表达式是一些三角多项式,我要固定三角函数的形式,比如
#sin(x),那么我希望在生成的序列中要么是一些变量比如x,要么是sin(x)的形式.或者更复杂一些,我希望是
#sin(f(a)*x)的形式,其中f(a)是一些常数的组合,这些常数是需要我们在基因生成过程中来选择的.
#那么就需要考虑刚才的问题,这让我想到了<盗梦空间>中的多层梦境.
#和rainboy约定sin(x**2+x)*x来演示sin(f(x))*h(x),其中f,h需要在GEP中模拟.
#发现分别用种群来模拟,很难加入到种群遗传序列里.
#最终不得不用了一个最为简单的办法.重写__call__()函数.
#这里是两个例子.可见该方法还是有效的,目前看是最简单的,显然也是有很大限制的,这个一方面是由于我用的这个
#勉强使用的办法的缺陷,另一方面,由于需要在整体上进行进化,这也是不得已的办法.希望rainboy和各位朋友能想
#出更好的办法.
例子:sin(f(x))*h(x)
1.f(x)=x**2+x,h(x)=x
代码:(有很多与以上例子重复的地方,但是为了清晰起见还是把他放在这里.
运行结果:
*++xxxxx+xxxxx [876]: 1
*++xxxxx/xxxxx [902]: 1
*++xxxxxx*xxxx [853]: 1
x+xxxxxx//xxxx [904]: 0
*++xxxxx/xxxxx [878]: 1
*+*xxxxxxxxxxx [901]: 0
-++xxxxxx-xxxx [903]: 0
*++xxxxx++xxxx [897]: 1
*++xxxxx+xxxxx [876]: 1
+x*xxxxx+xxxxx [898]: 5
SOLVED: +x*xxxxx+xxxxx
这个代码就是表示x**2+x 和x
另外也算了一个h(x)=x**3+x**2的例子.
只需要在以上代码里将mp类中的初始数据改一下就可以了.
运行结果:
[Generation: 442 | Best: #4664 (5) | Mean: 0.8]
012345678901234567890123456789012345678901
------------------------------------------
+-*--/+*//xxxxxxxxxxx*-*/+/*-+/xxxxxxxxxxx [4624]: 1
+-*--/+*//xxxxxxxxxxx*-*/+/*-+/xxxxxxxxxxx [4670]: 1
+-*+-/+*//xxxxxxxxxxx*-*/+-*-+/xxxxxxxxxxx [4661]: 0
+-**-/+*//xxxxxxxxxxx*-*/+//-+/xxxxxxxxxxx [4669]: 0
+-*--/+*//xxxxxxxxxxx*-*/+/+-+/xxxxxxxxxxx [4663]: 0
+-*--/--+*xxxxxxxxxxx*-*/+/*-+/xxxxxxxxxxx [4664]: 5
+-*-//+*/-xxxxxxxxxxx--*-*/+**-xxxxxxxxxxx [4666]: 0
+-*--/+*//xxxxxxxxxxx*--/+/*-+/xxxxxxxxxxx [4667]: 0
+-*--/+*//xxxxxxxxxxx*-*/+/*-+/xxxxxxxxxxx [4624]: 1
*-*/+/*-+/xxxxxxxxxxx+-*--/+*//xxxxxxxxxxx [4668]: 0
SOLVED: +-*--/--+*xxxxxxxxxxx*-*/+/*-+/xxxxxxxxxxx
这个符号序列解码出来就是x**2+x和x**3+x**2.
希望有更好的办法,开始的时候一直希望能通过设置算符,来满足要求,结果却没有实现哈.d
Zeuux © 2025
京ICP备05028076号
回复 王俊亚 2011年04月20日 星期三 14:42