DEAP1.2.2(六) 一個(gè)GP實(shí)例——用于函數(shù)發(fā)現(xiàn)

(六) GP實(shí)例——函數(shù)發(fā)現(xiàn)
話不多說直接上代碼

import operator, math, random, numpy
from deap import gp, tools, base, creator, algorithms

toolbox = base.Toolbox()
pset = gp.PrimitiveSet("MAIN", 1)
#define protectedDive function which return their division
def protectedDiv(left, right):
    try:
        return left / right
    except ZeroDivisionError:
        return 1
#adding other functions
pset.addPrimitive(operator.add, 2)
pset.addPrimitive(operator.sub, 2)
pset.addPrimitive(operator.mul, 2)
pset.addPrimitive(protectedDiv, 2)
pset.addPrimitive(operator.neg, 1)
pset.addPrimitive(math.cos, 1)
pset.addPrimitive(math.sin, 1)
#add random constants which is an int type
pset.addEphemeralConstant("rand1", lambda: random.randint(-1,1))
#rename augument x
pset.renameArguments(ARG0='x')

#creating MinFit 
creator.create("FitnessMin", base.Fitness, weights=(-1.0,))
#creating individual
creator.create("Individual", gp.PrimitiveTree, fitness=creator.FitnessMin)

#import toolbox
toolbox = base.Toolbox()
#resigter expr individual population and compile
toolbox.register("expr", gp.genHalfAndHalf, pset=pset, min_=1, max_=2)     #0~3
toolbox.register("individual", tools.initIterate, creator.Individual, toolbox.expr)
toolbox.register("population", tools.initRepeat, list, toolbox.individual)
toolbox.register("compile", gp.compile, pset=pset)
#define evaluating function
def evalSymbReg(individual, points):
    # Transform the tree expression in a callable function
    func = toolbox.compile(expr=individual)
    # Evaluate the mean squared error between the expression
    # and the real function : x**4 + x**3 + x**2 + x
    sqerrors = ((func(x) - x**4 - x**3 - x**2 - x)**2 for x in points) #define their differential
    #return the average fitness
    return math.fsum(sqerrors) / len(points),

#register genetic operations(evaluate/selection/mutate/crossover/)
toolbox.register("evaluate", evalSymbReg, points=[x/10. for x in range(-10,10)]) #the range of points in evaluate function
toolbox.register("select", tools.selTournament, tournsize=3) 
toolbox.register("mate", gp.cxOnePoint)\
#this is expr_mut, if we want to use a GEP, we can mutute the expr at first, then do the expression
toolbox.register("expr_mut", gp.genFull, min_=0, max_=2)
toolbox.register("mutate", gp.mutUniform, expr=toolbox.expr_mut, pset=pset)
#decorating the operator including crossover and mutate, restricting the tree's height and length
toolbox.decorate("mate", gp.staticLimit(key=operator.attrgetter("height"), max_value=17))
toolbox.decorate("mutate", gp.staticLimit(key=operator.attrgetter("height"), max_value=17))

def main():
    #Parameter setting
    CXPB = 0.1
    MUPB = 0.1
    GEN = 400
    POP_SIZE = 300 
    #initializing population
    pop = toolbox.population(n = POP_SIZE)
    
    '''start evolution'''
    print('Start evolution')
    #evaluating the fitness
    fitnesses = list(map(toolbox.evaluate, pop))
    #assign fitness values
    for ind, fit in zip(pop, fitnesses):
        ind.fitness.values = fit
    print('Evaluated %i individuals' %len(pop))
    '''The genetic operations'''
    for g in range(GEN):
        #select
        offspring = toolbox.select(pop, len(pop))
        offspring = list(map(toolbox.clone, offspring))
        #crossover
        for child1, child2 in zip(offspring[::2], offspring[1::2]):
            if random.random() < CXPB:
                toolbox.mate(child1, child2)
                
                del child1.fitness.values
                del child2.fitness.values
        
        #mutation
        for mutant in offspring:
            if random.random() < MUPB:
                toolbox.mutate(mutant)
                del mutant.fitness.values
        
        #evaluate the invalid individuals
        invalid_ind = [ind for ind in offspring if not ind.fitness.valid]
        fitnesses = list(map(toolbox.evaluate, invalid_ind))
        for ind, fit in zip(invalid_ind, fitnesses):
            ind.fitness.values = fit
        print('Evaluated %i individuals' %len(invalid_ind))
        
        #update the pop
        pop[:] = offspring
        
        #statistics
        stats = tools.Statistics(key=lambda ind: ind.fitness.values)
        record = stats.compile(pop)
        stats.register("avg", numpy.mean, axis=0)
        stats.register("min", numpy.min, axis=0)
        stats.register("max", numpy.max, axis=0)
        record = stats.compile(pop)
        logbook = tools.Logbook()
        logbook.record(gen=g, evals=30, **record)
        logbook.header = "gen", "avg", "min", "max"
        print(logbook)
        


    print("-- End of (successful) evolution --")
    
    best_ind = tools.selBest(pop, 1)[0]
    print("Best individual is %s, Best results are %s" % (best_ind, best_ind.fitness.values))
    
        
    

if __name__ == "__main__":
    main()

大部分注釋都有寫三妈,總結(jié)一下這個(gè)例子:
首先,要發(fā)現(xiàn)的函數(shù)是: f(x) = x^4 - x^3 - x^2 - x氢妈, fitness func為 min{sum(func(x) - f(x))^2}/xnum,x in (-1, 1] 間隔0.1蜕着。
接下來首先搭建GP框架
STEP1. 適應(yīng)度函數(shù)的構(gòu)建酬凳,個(gè)體及種群初始化惠况,所用工具包:creator/toolbox
STEP2. 演化操作定義以及樹的size限制
STEP3. 設(shè)定參數(shù)(交叉率、變異率宁仔、迭代次數(shù)以及種群大小)稠屠,初始化種群并計(jì)算適應(yīng)度
STEP4. 按照選擇、交叉翎苫、變異的順序進(jìn)行演化操作权埠,在一次演化操作完成后選出無效個(gè)體(即適應(yīng)性不如先前個(gè)體的個(gè)體,對(duì)這些個(gè)體的適應(yīng)度不予修改)
STEP5. 更新種群煎谍,統(tǒng)計(jì)當(dāng)前種群的適應(yīng)度信息并輸出
STEP6. 當(dāng)循環(huán)結(jié)束攘蔽,輸出全局最優(yōu)解

注意:在GP這種樹結(jié)構(gòu)的算法中,在適應(yīng)度計(jì)算當(dāng)中呐粘,生成的樹結(jié)構(gòu)首先要通過toolbox.compile轉(zhuǎn)化為函數(shù)满俗,再通過函數(shù)進(jìn)行計(jì)算,同時(shí)所有的遺傳操作均是基于樹結(jié)構(gòu)進(jìn)行的作岖,因此在register這些步驟時(shí)候需要使用特定的方法漫雷,如gp.cxOnePoint等。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末鳍咱,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子与柑,更是在濱河造成了極大的恐慌谤辜,老刑警劉巖,帶你破解...
    沈念sama閱讀 210,978評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件价捧,死亡現(xiàn)場(chǎng)離奇詭異丑念,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)结蟋,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,954評(píng)論 2 384
  • 文/潘曉璐 我一進(jìn)店門脯倚,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人嵌屎,你說我怎么就攤上這事推正。” “怎么了宝惰?”我有些...
    開封第一講書人閱讀 156,623評(píng)論 0 345
  • 文/不壞的土叔 我叫張陵植榕,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我尼夺,道長(zhǎng)尊残,這世上最難降的妖魔是什么炒瘸? 我笑而不...
    開封第一講書人閱讀 56,324評(píng)論 1 282
  • 正文 為了忘掉前任,我火速辦了婚禮寝衫,結(jié)果婚禮上顷扩,老公的妹妹穿的比我還像新娘。我一直安慰自己慰毅,他們只是感情好隘截,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,390評(píng)論 5 384
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著事富,像睡著了一般技俐。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上统台,一...
    開封第一講書人閱讀 49,741評(píng)論 1 289
  • 那天雕擂,我揣著相機(jī)與錄音,去河邊找鬼贱勃。 笑死井赌,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的贵扰。 我是一名探鬼主播仇穗,決...
    沈念sama閱讀 38,892評(píng)論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼戚绕!你這毒婦竟也來了纹坐?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,655評(píng)論 0 266
  • 序言:老撾萬榮一對(duì)情侶失蹤舞丛,失蹤者是張志新(化名)和其女友劉穎耘子,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體球切,經(jīng)...
    沈念sama閱讀 44,104評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡谷誓,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,451評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了吨凑。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片捍歪。...
    茶點(diǎn)故事閱讀 38,569評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖鸵钝,靈堂內(nèi)的尸體忽然破棺而出糙臼,到底是詐尸還是另有隱情,我是刑警寧澤蒋伦,帶...
    沈念sama閱讀 34,254評(píng)論 4 328
  • 正文 年R本政府宣布弓摘,位于F島的核電站,受9級(jí)特大地震影響痕届,放射性物質(zhì)發(fā)生泄漏韧献。R本人自食惡果不足惜末患,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,834評(píng)論 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望锤窑。 院中可真熱鬧璧针,春花似錦、人聲如沸渊啰。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,725評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)绘证。三九已至隧膏,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間嚷那,已是汗流浹背胞枕。 一陣腳步聲響...
    開封第一講書人閱讀 31,950評(píng)論 1 264
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留魏宽,地道東北人腐泻。 一個(gè)月前我還...
    沈念sama閱讀 46,260評(píng)論 2 360
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像队询,于是被迫代替她去往敵國(guó)和親派桩。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,446評(píng)論 2 348

推薦閱讀更多精彩內(nèi)容