欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Python實(shí)現(xiàn)雞群算法的示例代碼

 更新時(shí)間:2022年11月22日 14:58:36   作者:微小冷  
雞群算法,縮寫(xiě)為CSO(Chicken?Swarm?Optimization),盡管具備所謂仿生學(xué)的背景,但實(shí)質(zhì)上是粒子群算法的一個(gè)變體。本文將利用Python語(yǔ)言實(shí)現(xiàn)這一算法,感興趣的可以了解一下

算法簡(jiǎn)介

雞群算法,縮寫(xiě)為CSO(Chicken Swarm Optimization),盡管具備所謂仿生學(xué)的背景,但實(shí)質(zhì)上是粒子群算法的一個(gè)變體。

簡(jiǎn)單來(lái)說(shuō),粒子群就是一群粒子,每個(gè)粒子都有自己的位置和速度,而且每個(gè)粒子都要受到最佳粒子的吸引,除了這兩條規(guī)則之外,粒子之間完全平等,彼此之間除了位置和速度之外,完全相等。

當(dāng)然,粒子群算法本身也是有仿生學(xué)背景的,據(jù)說(shuō)靈感來(lái)自于鳥(niǎo)群覓食,這個(gè)當(dāng)然不重要,無(wú)非是一群平等的粒子變成了一群平等的鳥(niǎo)罷了。

而雞群算法,則是為這些粒子,或者這些鳥(niǎo),添加了不同的身份特征,使得彼此之間不再等同。

雞群中至少有三個(gè)階層,分別是公雞、母雞和小雞,每只雞都有其位置和速度。但區(qū)別之處在于,

  • 公雞最神氣,原則上可以隨便踱步,只是有的時(shí)候注意到其他公雞的時(shí)候,會(huì)有搶食的想法,相當(dāng)于隨機(jī)抽選一只其他公雞,對(duì)其位置產(chǎn)生影響。
  • 母雞最憋屈,一方面要接受公雞的領(lǐng)導(dǎo),另一方面還要和其他母雞搶食
  • 小雞最無(wú)憂無(wú)慮,跟著母雞走就是了。

隨著位置關(guān)系的變化,母雞和小雞可能會(huì)逐漸遺忘最初的首領(lǐng),也就是說(shuō)種群關(guān)系可能會(huì)發(fā)生變化。

Python實(shí)現(xiàn)雞和雞群

首先,要實(shí)現(xiàn)一個(gè)雞類,一只雞,有兩種基本屬性,即位置和類別。

import numpy as np
from random import gauss, random
randint = np.random.randint
uniRand = np.random.uniform

class Chicken:
    def __init__(self, N, xRange, order=0, kind=0):
        # 生成(N)維參數(shù)
        self.x = uniRand(*xRange, (N,))
        self.best = np.inf
        self.xBest = np.zeros((N,))
        self.kind = kind            # 雞的類別
        self.order = order          # 雞的編號(hào)
    
    # 設(shè)置自己的首領(lǐng)公雞
    def setCock(self, i):
        self.cock = i

    # 設(shè)置自己的監(jiān)護(hù)母雞
    def setHen(self, i):
        self.hen = i

其中kind分為三類,分別是公雞、母雞和小雞。其中,每只母雞都有自己的首領(lǐng)公雞,每只小雞都有自己的監(jiān)護(hù)母雞。

order為這只雞在雞群中的編號(hào),主要在雞群中得以體現(xiàn)。

雞群和粒子群有一個(gè)很大的區(qū)別,后者說(shuō)到底只有一個(gè)群,而雞群中,每個(gè)公雞都有自己的母雞和小雞,相當(dāng)于一個(gè)小群體。但雞和雞之間的關(guān)系,并不取決于雞自己,故而需要在雞群中實(shí)現(xiàn)

randint = np.random.randint
class Swarm:
    # cNum 雞數(shù),是三個(gè)元素的列表,分別是公雞、母雞和小雞數(shù)
    # N參數(shù)維度
    def __init__(self, cNum, N, xRange):
        self.initCs(cNum, N, xRange)
        self.bestCS = deepcopy(self.cs)     #最佳雞群
        self.best = np.inf  #全局最優(yōu)值
        self.xBest = np.zeros((N,)) #全局最優(yōu)參數(shù)
        self.N = N

    def initCs(self, cNum, N, xRange, vRange):
        self.cs = []
        self.cNum = cNum
        self.cocks = np.arange(cNum[0])     # 公雞編號(hào)
        self.hens = np.arange(cNum[0], cNum[0]+cNum[1]) #母雞編號(hào)
        self.chicks = np.arange(cNum[0]+cNum[1], np.sum(cNum))  #小雞編號(hào)
        kinds = np.repeat([0,1,2], cNum)
        for i in range(sum(cNum)):
            self.cs.append(Chicken(N,xRange, vRange, i, kinds[i]))
            if kinds[i] > 0:
                cock = randint(0, cNum[0])
                self.cs[i].setCock(cock)
            if kinds[i] > 1:
                hen = randint(cNum[0], cNum[0]+cNum[1])
                self.cs[i].setHen(hen)

其中,initCs是初始化雞群的函數(shù),其中母雞、小雞的首領(lǐng)公雞,小雞的監(jiān)護(hù)母雞,都是隨機(jī)生成的。

雞群更新

接下來(lái)就是算法的核心環(huán)節(jié),不同的雞要遵循不同的更新規(guī)則,其中,公雞最瀟灑,其下一步位置只取決于自己,以及另一只隨便挑選的公雞。

公雞

記當(dāng)前這只公雞的編號(hào)是i,隨機(jī)挑選的公雞編號(hào)是j , j=?i,則第i只公雞位置的更新方法為

xi?(t+1)=xi?(t)⋅(1+r)

其中,r是通過(guò)正態(tài)分布生成的隨機(jī)數(shù),可表示為1∼N(0,σ2),其中σ2

其中f一般叫做適應(yīng)因子,相當(dāng)于將某只雞塞到待搜解的函數(shù)中得到的值。例如要搜索y=2的最小值,如果當(dāng)前這只雞的位置1.5,那么f=1.52=2.25。ε是一個(gè)防止除零錯(cuò)誤的小量。

但需要注意,上文中所有的x,表示的并非一個(gè)標(biāo)量,而是一個(gè)數(shù)組。

其Python實(shí)現(xiàn)為

# 寫(xiě)在Swarm類中
def cockStep(self):
    for i in self.cocks:
        # 第j只公雞
        j = np.random.randint(self.cNum[0])
        if j==i:
            j = (j+1) % self.cNum[0]
        # 第i只公雞
        ci = self.cs[i]
        # 第j只公雞
        cj = self.cs[self.cocks[j]]
        sigma = 1 if cj.best > ci.best else np.exp(
            (cj.best-ci.best)/(np.abs(ci.best)+1e-15))
        ci.x *= 1 + gauss(0, sigma)

母雞

設(shè)當(dāng)前母雞編號(hào)為i,這只母雞既要追隨首領(lǐng)公雞,又要和其他母雞搶食。

xi?(t+1)=xi?(t)+k1?r1?(xc?−xi?)+k2?r2?(xj?−xi?)

其中,xc?為其首領(lǐng)公雞,xj?為另一只母雞或者公雞。k1,k2為系數(shù),其更新邏輯與公雞的k是一樣的,當(dāng)fi較大時(shí),表示為

代碼實(shí)現(xiàn)為

def henStep(self):
    nGuarder = self.cNum[0] + self.cNum[1] - 2
    for i in self.hens:
        guarders = list(self.cocks) + list(self.hens)
        c = self.cs[i].cock     #首領(lǐng)公雞
        guarders.remove(i)
        guarders.remove(c)
        # 隨機(jī)生成另一只監(jiān)護(hù)雞
        j = guarders[np.random.randint(nGuarder)]
        ci = self.cs[i]
        cj = self.cs[j]
        cc = self.cs[c]
        k1, k2 = random(), random()
        if cc.best > ci.best:
            k1 *= np.exp((ci.best-cc.best)/(np.abs(ci.best)+1e-15))
        if cj.best < ci.best:
            k2 *=  np.exp(cj.best-ci.best)
        ci.x += k1*(cc.x-ci.x)+k2*(cj.x-ci.x)

小雞

最后是小雞的更新邏輯,小雞在母雞的周圍找食物,其更新邏輯為

xi?(t+1)=xi?(t)+r(xh?(t)−xi?(t))

其中,xh為其監(jiān)護(hù)母雞,r為隨機(jī)數(shù),算法實(shí)現(xiàn)為

def chickStep(self):
    for i in self.chicks:
        ci = self.cs[i]
        ci.x += 2*random()*(self.cs[ci.hen].x-ci.x)

整個(gè)雞群

正所謂,算法源于生活而高于生活,自然界里講求輩分,但在雞群算法里,講究的確是實(shí)力。如果小雞運(yùn)氣爆棚,得到了比公雞還厲害的優(yōu)化結(jié)果,那么這只小雞就會(huì)進(jìn)化成公雞。

也就是說(shuō),每隔一段時(shí)間,雞群里的雞會(huì)被重新安排身份,優(yōu)化效果最好的就是頭領(lǐng)公雞,差一點(diǎn)的是監(jiān)護(hù)母雞,最差的就只能是小雞了。

def update(self):
    cn = np.sum(self.cNum)
    c1, c2 = self.cNum[0], self.cNum[0]+self.cNum[1]
    fitness = [self.cs[i].best for i in range(cn)]
    index = np.argsort(fitness)
    self.cocks = index[np.arange(c1)]
    self.hens = index[np.arange(c1,c2)]
    self.chicks = index[np.arange(c2,cn)]
    for i in self.cocks:
        self.cs[i].kind = 0
    for i in self.hens:
        self.cs[i].kind = 1
    for i in self.chicks:
        self.cs[i].kind = 2
    for i in range(cn):
        if self.cs[i].kind > 0:
            cock = self.cocks[randint(0, c1)]
            self.cs[i].setCock(cock)
        if self.cs[i].kind > 1:
            hen = self.hens[randint(c1,c2)]
            self.cs[i].setHen(hen)

優(yōu)化迭代

至此,集群算法的框架算是搭建成功了,接下來(lái)就實(shí)現(xiàn)最關(guān)鍵的部分,優(yōu)化。

其基本邏輯是,輸入一個(gè)待優(yōu)化func,通過(guò)將每只雞的位置x帶入到這個(gè)函數(shù)中,得到一個(gè)判定值,最后通過(guò)這個(gè)判定值,來(lái)不斷更新雞群。

除了這個(gè)函數(shù)之外,還需要輸入一些其他參數(shù),比如整個(gè)雞群算法的迭代次數(shù),以及雞群更新的頻次等等

# func為待優(yōu)化函數(shù)
# N為迭代次數(shù)
# T為雞群更新周期
def optimize(self, func, N, T, msgT):
    for n in range(N):
        # 計(jì)算優(yōu)化參數(shù)
        for c in self.cs:
            c.best = func(c.x)
        # 分別更新公雞、母雞和小雞
        self.cockStep()
        self.henStep()
        self.chickStep()
        if (n+1)%T == 0:
            self.update()   #每T次更新一次種群
            self.printBest(n)
    self.printBest(n)

其中,printBest可以將當(dāng)前最佳結(jié)果打印出來(lái),其形式為

def printBest(self,n):
    fitness = [c.best for c in self.cs]
    best = np.min(fitness)
    ind = np.where(fitness==best)[0]
    msg = f"已經(jīng)迭代{n}次,最佳優(yōu)化結(jié)果為{np.min(fitness)},參數(shù)為:\n"
    msg += ", ".join([f"{x:.6f}" for x in self.cs[ind].x])
    print(msg)

測(cè)試

算法完成之后,當(dāng)然要找個(gè)函數(shù)測(cè)試一下,測(cè)試函數(shù)為

def test(xs):
    _sum = 0.0
    for i in range(len(xs)):
        _sum = _sum + np.cos((xs[i]*i)/5)*(i+1)
    return _sum

???????if __name__ == "__main__":
    cNum = [15,20,100]
    s = Swarm(cNum, 5, (-5,5))
    s.optimize(test, 20, 5)

測(cè)試結(jié)果如下

已經(jīng)迭代4次,最佳優(yōu)化結(jié)果為-5.793762423022024,參數(shù)為:
-6.599526, 3.117137, 5.959538, 7.225785, 5.204990
已經(jīng)迭代9次,最佳優(yōu)化結(jié)果為-10.61594651972434,參數(shù)為:
-7.003724, -5.589730, 0.981409, 12.920325, -19.006112
已經(jīng)迭代14次,最佳優(yōu)化結(jié)果為-9.143596747975293,參數(shù)為:
5.388234, -3.714421, -5.254391, -5.216215, -6.079223
已經(jīng)迭代19次,最佳優(yōu)化結(jié)果為-11.097888385616995,參數(shù)為:
-9.156244, -5.914600, -5.960154, 4.550833, 4.127889
已經(jīng)迭代19次,最佳優(yōu)化結(jié)果為-11.097888385616995,參數(shù)為:
-9.156244, -5.914600, -5.960154, 4.550833, 4.127889

以上就是Python實(shí)現(xiàn)雞群算法的示例代碼的詳細(xì)內(nèi)容,更多關(guān)于Python雞群算法的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • python requests包的request()函數(shù)中的參數(shù)-params和data的區(qū)別介紹

    python requests包的request()函數(shù)中的參數(shù)-params和data的區(qū)別介紹

    這篇文章主要介紹了python requests包的request()函數(shù)中的參數(shù)-params和data的區(qū)別介紹,具有很好參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-05-05
  • 探索Python中雙下劃線的特殊方法屬性魔法世界

    探索Python中雙下劃線的特殊方法屬性魔法世界

    這篇文章主要為大家介紹了Python中雙下劃線的特殊方法屬性魔法世界探索,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-11-11
  • python中的bool數(shù)組取反案例

    python中的bool數(shù)組取反案例

    這篇文章主要介紹了python中的bool數(shù)組取反案例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2021-03-03
  • python單向鏈表實(shí)例詳解

    python單向鏈表實(shí)例詳解

    這篇文章主要為大家詳細(xì)介紹了python單向鏈表實(shí)例,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-05-05
  • PyTorch零基礎(chǔ)入門(mén)之邏輯斯蒂回歸

    PyTorch零基礎(chǔ)入門(mén)之邏輯斯蒂回歸

    PyTorch是一個(gè)開(kāi)源的Python機(jī)器學(xué)習(xí)庫(kù),基于Torch,用于自然語(yǔ)言處理等應(yīng)用程序,它是一個(gè)可續(xù)計(jì)算包,提供兩個(gè)高級(jí)功能:1、具有強(qiáng)大的GPU加速的張量計(jì)算(如NumPy)。2、包含自動(dòng)求導(dǎo)系統(tǒng)的深度神經(jīng)網(wǎng)絡(luò)
    2021-10-10
  • Python 用__new__方法實(shí)現(xiàn)單例的操作

    Python 用__new__方法實(shí)現(xiàn)單例的操作

    這篇文章主要介紹了Python 用__new__方法實(shí)現(xiàn)單例的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-12-12
  • Python中bisect的用法及示例詳解

    Python中bisect的用法及示例詳解

    這篇文章主要介紹了Python中bisect的用法及示例詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-07-07
  • python3中關(guān)于excel追加寫(xiě)入格式被覆蓋問(wèn)題(實(shí)例代碼)

    python3中關(guān)于excel追加寫(xiě)入格式被覆蓋問(wèn)題(實(shí)例代碼)

    這篇文章主要介紹了python3中關(guān)于excel追加寫(xiě)入格式被覆蓋問(wèn)題,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-01-01
  • 關(guān)于Python下載大文件時(shí)哪種方式速度更快

    關(guān)于Python下載大文件時(shí)哪種方式速度更快

    這篇文章主要介紹了關(guān)于Python下載大文件時(shí)哪種方式速度更快,通常,我們都會(huì)用 requests 庫(kù)去下載,這個(gè)庫(kù)用起來(lái)太方便了,需要的朋友可以參考下
    2023-04-04
  • 解決Django生產(chǎn)環(huán)境無(wú)法加載靜態(tài)文件問(wèn)題的解決

    解決Django生產(chǎn)環(huán)境無(wú)法加載靜態(tài)文件問(wèn)題的解決

    這篇文章主要介紹了解決Django生產(chǎn)環(huán)境無(wú)法加載靜態(tài)文件問(wèn)題的解決,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2019-04-04

最新評(píng)論