粒子群算法(PSO) Python實現(xiàn)

1 原理

粒子群算法是群智能一種承二,是基于對鳥群覓食行為的研究和模擬而來的淋淀。假設(shè)在鳥群覓食范圍瘟芝,只在一個地方有食物易桃,所有鳥兒看不到食物(不知道食物的具體位置),但是能聞到食物的味道(能知道食物距離自己位置)锌俱。最好的策略就是結(jié)合自己的經(jīng)驗在距離鳥群中距離食物最近的區(qū)域搜索晤郑。

利用粒子群算法解決實際問題本質(zhì)上就是利用粒子群算法求解函數(shù)的最值。因此需要事先把實際問題抽象為一個數(shù)學(xué)函數(shù)贸宏,稱之為適應(yīng)度函數(shù)造寝。在粒子群算法中,每只鳥都可以看成是問題的一個解吭练,這里我們通常把鳥稱之為粒子诫龙,每個粒子都擁有:

  • 位置,可以理解函數(shù)的自變量的值鲫咽;
  • 經(jīng)驗签赃,也即是自身經(jīng)歷過的距離食物最近的位置谷异;
  • 速度,可以理解為自變量的變化值锦聊;
  • 適應(yīng)度歹嘹,距離食物的位置,也就是函數(shù)值括丁。

粒子群算法的過程

PSO流程圖
  1. 初始化荞下。包括根據(jù)給定的粒子個數(shù),初始化粒子史飞,包括初始化一下的值:
  • 位置:解空間內(nèi)的隨機值尖昏;
  • 經(jīng)驗:與初始位置相等;
  • 速度:0构资;
  • 適應(yīng)度:根據(jù)位置抽诉,帶入適應(yīng)度函數(shù),得到適應(yīng)度值吐绵。
  1. 更新迹淌。包括兩部分:
  • 粒子自身信息:包括根據(jù)下面的公式更新粒子的速度、位置己单,根據(jù)適應(yīng)度函數(shù)更新適應(yīng)度唉窃,然后和用更新后的適應(yīng)度和自身經(jīng)驗進(jìn)行比較,如果新的適應(yīng)度由于經(jīng)驗的適應(yīng)度纹笼,就利用當(dāng)前位置更新經(jīng)驗纹份;


    速度更新公式

    位置更新公式

上面公式中:i表示粒子編號;t表示時刻廷痘,反映在迭代次數(shù)上蔓涧;w是慣性權(quán)重,一般設(shè)置在0.4左右笋额;c表示學(xué)習(xí)因子元暴,一般都取值為2;Xpbest表示的是粒子i的經(jīng)驗兄猩,也即是粒子i所到過最佳位置茉盏;Xgbest代表的是全局最優(yōu)粒子的位置;r是0到1之間的隨機值厦滤。

  • 種群信息:把當(dāng)前適應(yīng)度和全局最優(yōu)位置的適應(yīng)度進(jìn)行比較援岩,如果當(dāng)前適應(yīng)度優(yōu)于全局最優(yōu)的適應(yīng)度,那么久用當(dāng)前粒子替換群居最優(yōu)掏导。
  1. 判斷結(jié)束條件。結(jié)束條件包括最大迭代次數(shù)和適應(yīng)度的閾值羽峰。

2 代碼

  • 實驗環(huán)境為python 2.7.11趟咆。
  • 這個代碼最初是用于求解一維最大熵分割圖像問題的添瓷,因此是求解函數(shù)最大值,如果需要求解最小值值纱,把代碼中的大于號全部改成小于號就可以了鳞贷。

首先需要解決的是粒子的存儲,我第一反應(yīng)是利用結(jié)構(gòu)體來存儲虐唠,但是python并沒有相應(yīng)的數(shù)據(jù)結(jié)構(gòu)搀愧,所以我選擇用一個類來表示粒子結(jié)構(gòu),該類的一個對象就是一個粒子疆偿,上代碼:

class bird:
    """
    speed:速度
    position:位置
    fit:適應(yīng)度
    lbestposition:經(jīng)歷的最佳位置
    lbestfit:經(jīng)歷的最佳的適應(yīng)度值
    """
    def __init__(self, speed, position, fit, lBestPosition, lBestFit):
        self.speed = speed
        self.position = position
        self.fit = fit
        self.lBestFit = lBestPosition
        self.lBestPosition = lPestFit

接下來就是粒子群算法的主干部分咱筛,用一個類來封裝,代碼:

import random

class PSO:
    """
    fitFunc:適應(yīng)度函數(shù)
    birdNum:種群規(guī)模
    w:慣性權(quán)重
    c1,c2:個體學(xué)習(xí)因子杆故,社會學(xué)習(xí)因子
    solutionSpace:解空間迅箩,列表類型:[最小值,最大值]
    """
    def __init__(self, fitFunc, birdNum, w, c1, c2, solutionSpace):
        self.fitFunc = fitFunc
        self.w = w
        self.c1 = c1
        self.c2 = c2
        self.birds, self.best = self.initbirds(birdNum, solutionSpace)

    def initbirds(self, size, solutionSpace):
        birds = []
        for i in range(size):
            position = random.uniform(solutionSpace[0], solutionSpace[1])
            speed = 0
            fit = self.fitFunc(position)
            birds.append(bird(speed, position, fit, position, fit))
        best = birds[0]
        for bird in birds:
            if bird.fit > best.fit:
                best = bird
        return birds,best

    def updateBirds(self):
        for bird in self.birds:
            # 更新速度
            bird.speed = self.w * bird.speed + self.c1 * random.random() * (bird.lBestPosition - bird.position) + self.c2 * random.random() * (self.best.position - bird.position)
            # 更新位置
            bird.position = bird.position + bird.speed
            # 跟新適應(yīng)度
            bird.fit = self.fitFunc(bird.position)
            # 查看是否需要更新經(jīng)驗最優(yōu)
            if bird.fit > bird.lBestFit:
                bird.lBestFit = bird.fit
                bird.lBestPosition = bird.position

    def solve(self, maxIter):
        # 只考慮了最大迭代次數(shù)处铛,如需考慮閾值饲趋,添加判斷語句就好
        for i in range(maxIter):
            # 更新粒子
            self.updateBirds()
            for bird in self.birds:
                # 查看是否需要更新全局最優(yōu)
                if bird.fit > self.best.fit:
                    self.best = bird

有了以上代碼,只需要自定義適應(yīng)度函數(shù)fitFunc就可以進(jìn)行求解撤蟆,但是需要注意的是只適用于求解** 一維問題 **奕塑。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市家肯,隨后出現(xiàn)的幾起案子龄砰,更是在濱河造成了極大的恐慌,老刑警劉巖息楔,帶你破解...
    沈念sama閱讀 206,839評論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件寝贡,死亡現(xiàn)場離奇詭異,居然都是意外死亡值依,警方通過查閱死者的電腦和手機圃泡,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,543評論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來愿险,“玉大人颇蜡,你說我怎么就攤上這事×究鳎” “怎么了风秤?”我有些...
    開封第一講書人閱讀 153,116評論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長扮叨。 經(jīng)常有香客問我缤弦,道長,這世上最難降的妖魔是什么彻磁? 我笑而不...
    開封第一講書人閱讀 55,371評論 1 279
  • 正文 為了忘掉前任碍沐,我火速辦了婚禮狸捅,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘累提。我一直安慰自己尘喝,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 64,384評論 5 374
  • 文/花漫 我一把揭開白布斋陪。 她就那樣靜靜地躺著朽褪,像睡著了一般。 火紅的嫁衣襯著肌膚如雪无虚。 梳的紋絲不亂的頭發(fā)上缔赠,一...
    開封第一講書人閱讀 49,111評論 1 285
  • 那天,我揣著相機與錄音骑科,去河邊找鬼橡淑。 笑死,一個胖子當(dāng)著我的面吹牛咆爽,可吹牛的內(nèi)容都是我干的梁棠。 我是一名探鬼主播,決...
    沈念sama閱讀 38,416評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼斗埂,長吁一口氣:“原來是場噩夢啊……” “哼符糊!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起呛凶,我...
    開封第一講書人閱讀 37,053評論 0 259
  • 序言:老撾萬榮一對情侶失蹤男娄,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后漾稀,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體模闲,經(jīng)...
    沈念sama閱讀 43,558評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,007評論 2 325
  • 正文 我和宋清朗相戀三年崭捍,在試婚紗的時候發(fā)現(xiàn)自己被綠了尸折。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片夯尽。...
    茶點故事閱讀 38,117評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡旷太,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出访递,到底是詐尸還是另有隱情粒梦,我是刑警寧澤亮航,帶...
    沈念sama閱讀 33,756評論 4 324
  • 正文 年R本政府宣布,位于F島的核電站匀们,受9級特大地震影響缴淋,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,324評論 3 307
  • 文/蒙蒙 一宴猾、第九天 我趴在偏房一處隱蔽的房頂上張望圆存。 院中可真熱鬧叼旋,春花似錦仇哆、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,315評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至详民,卻和暖如春延欠,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背沈跨。 一陣腳步聲響...
    開封第一講書人閱讀 31,539評論 1 262
  • 我被黑心中介騙來泰國打工由捎, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人饿凛。 一個月前我還...
    沈念sama閱讀 45,578評論 2 355
  • 正文 我出身青樓狞玛,卻偏偏與公主長得像,于是被迫代替她去往敵國和親涧窒。 傳聞我的和親對象是個殘疾皇子心肪,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,877評論 2 345

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

  • 姓名:彭帥 學(xué)號:17021210850 【嵌牛導(dǎo)讀】:蟻群算法(ACO)是受自然界中螞蟻搜索食物行為的啟發(fā),是一...
    重露成涓滴閱讀 38,559評論 0 8
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 171,517評論 25 707
  • 王者榮耀辣雞隊友太坑纠吴,哎哎
    別和我比閱讀 244評論 0 0
  • 前幾日去打針硬鞍,排隊等候時,身邊一位30歲模樣的壯漢說戴已,這個針痛得很固该。 閑來無事,我接話說糖儡,我之前打過的伐坏,的確痛,但...
    純銀V閱讀 3,717評論 0 54
  • 【石豐畫語】不畫閑情逸致休玩,不畫風(fēng)花雪月著淆,不畫鶯歌燕舞,不畫表象魅惑拴疤;只畫生命滴血永部,只畫人性局限,只畫我心飛翔呐矾,只畫...
    石豐當(dāng)代藝術(shù)閱讀 1,118評論 5 4