Chinaunix首页 | 论坛 | 博客
  • 博客访问: 3589752
  • 博文数量: 365
  • 博客积分: 0
  • 博客等级: 民兵
  • 技术积分: 2522
  • 用 户 组: 普通用户
  • 注册时间: 2019-10-28 13:40
文章分类

全部博文(365)

文章存档

2023年(8)

2022年(130)

2021年(155)

2020年(50)

2019年(22)

我的朋友

分类: Python/Ruby

2021-03-31 16:40:41

步骤1:初始化参数

import numpy as np

import matplotlib.pyplot as plt

import matplotlib as mpl

mpl.rcParams['font.sans-serif'] = ['SimHei']  # 指定默认字体

mpl.rcParams['axes.unicode_minus'] = False  # 解决保存图像是负号'-'显示为方块的问题

# PSO的参数

w = 1  # 惯性因子,一般取1

c1 = 2  # 学习因子,一般取2

c2 = 2  #

r1 = None  # 为两个(0,1)之间的随机数

r2 = None

dim = 2  # 维度的维度#对应2个参数x,y

size = 100  # 种群大小,即种群中小鸟的个数

iter_num = 1000  # 算法最大迭代次数

max_vel = 0.5  # 限制粒子的最大速度为0.5

fitneess_value_list = []  # 记录每次迭代过程中的种群适应度值变化

步骤2:这里定义一些参数,分别是计算适应度函数和计算约束惩罚项函数

def calc_f(X):

    """计算粒子的的适应度值,也就是目标函数值,X 的维度是 size * 2 """

    A = 10

    pi = np.pi

    x = X[0]

    y = X[1]

    return 2 * A + x ** 2 - A * np.cos(2 * pi * x) + y ** 2 - A * np.cos(2 * pi * y)

def calc_e1(X):

    """计算第一个约束的惩罚项"""

    e = X[0] + X[1] - 6

    return max(0, e)

def calc_e2(X):

    """计算第二个约束的惩罚项"""

    e = 3 * X[0] - 2 * X[1] - 5

    return max(0, e)

def calc_Lj(e1, e2):

    """根据每个粒子的约束惩罚项计算Lj权重值,e1, e2列向量,表示每个粒子的第1个第2个约束的惩罚项值"""

    # 注意防止分母为零的情况

    if (e1.sum() + e2.sum()) <= 0:

        return 0, 0

    else:

        L1 = e1.sum() / (e1.sum() + e2.sum())

        L2 = e2.sum() / (e1.sum() + e2.sum())

    return L1, L2

步骤3:定义粒子群算法的速度更新函数,位置更新函数

def velocity_update(V, X, pbest, gbest):

    """

    根据速度更新公式更新每个粒子的速度

     种群size=20

    :param V: 粒子当前的速度矩阵,20*2 的矩阵

    :param X: 粒子当前的位置矩阵,20*2 的矩阵

    :param pbest: 每个粒子历史最优位置,20*2 的矩阵

    :param gbest: 种群历史最优位置,1*2 的矩阵

    """

    r1 = np.random.random((size, 1))

    r2 = np.random.random((size, 1))

    V = w * V + c1 * r1 * (pbest - X) + c2 * r2 * (gbest - X)  # 直接对照公式写就好了

    # 防止越界处理

    V[V < -max_vel] = -max_vel

    V[V > max_vel] = max_vel

    return V

def position_update(X, V):

    """

    根据公式更新粒子的位置

    :param X: 粒子当前的位置矩阵,维度是 20*2

    :param V: 粒子当前的速度举着,维度是 20*2

    """

    X=X+V#更新位置

    size=np.shape(X)[0]#种群大小

    for i in range(size):#遍历每一个例子

        if X[i][0]<=1 or X[i][0]>=2:#x的上下限约束

            X[i][0]=np.random.uniform(1,2,1)[0]#则在12随机生成一个数

        if X[i][1] <= -1 or X[i][0] >= 0:#y的上下限约束

            X[i][1] = np.random.uniform(-1, 0, 1)[0]  # 则在-10随机生成一个数

    return X

步骤4:每个粒子历史最优位置更优函数,货币代码以及整个群体历史最优位置更新函数,和无约束约束优化代码类似,所不同的是添加了违反约束的处理过程

def update_pbest(pbest, pbest_fitness, pbest_e, xi, xi_fitness, xi_e):

    """

    判断是否需要更新粒子的历史最优位置

    :param pbest: 历史最优位置

    :param pbest_fitness: 历史最优位置对应的适应度值

    :param pbest_e: 历史最优位置对应的约束惩罚项

    :param xi: 当前位置

    :param xi_fitness: 当前位置的适应度函数值

    :param xi_e: 当前位置的约束惩罚项

    :return:

    """

    # 下面的 0.0000001 是考虑到计算机的数值精度位置,值等同于0

    # 规则1,如果 pbest xi 都没有违反约束,则取适应度小的

    if pbest_e <= 0.0000001 and xi_e <= 0.0000001:

        if pbest_fitness <= xi_fitness:

            return pbest, pbest_fitness, pbest_e

        else:

            return xi, xi_fitness, xi_e

    # 规则2,如果当前位置违反约束而历史最优没有违反约束,则取历史最优

    if pbest_e < 0.0000001 and xi_e >= 0.0000001:

        return pbest, pbest_fitness, pbest_e

    # 规则3,如果历史位置违反约束而当前位置没有违反约束,则取当前位置

    if pbest_e >= 0.0000001 and xi_e < 0.0000001:

        return xi, xi_fitness, xi_e

    # 规则4,如果两个都违反约束,则取适应度值小的

    if pbest_fitness <= xi_fitness:

        return pbest, pbest_fitness, pbest_e

    else:

        return xi, xi_fitness, xi_e

def update_gbest(gbest, gbest_fitness, gbest_e, pbest, pbest_fitness, pbest_e):

    """

    更新全局最优位置

    :param gbest: 上一次迭代的全局最优位置

    :param gbest_fitness: 上一次迭代的全局最优位置的适应度值

    :param gbest_e:上一次迭代的全局最优位置的约束惩罚项

    :param pbest:当前迭代种群的最优位置

    :param pbest_fitness:当前迭代的种群的最优位置的适应度值

    :param pbest_e:当前迭代的种群的最优位置的约束惩罚项

    :return:

    """

    # 先对种群,寻找约束惩罚项=0的最优个体,如果每个个体的约束惩罚项都大于0,就找适应度最小的个体

    pbest2 = np.concatenate([pbest, pbest_fitness.reshape(-1, 1), pbest_e.reshape(-1, 1)], axis=1)  # 将几个矩阵拼接成矩阵 ,4维矩阵(x,y,fitness,e

    pbest2_1 = pbest2[pbest2[:, -1] <= 0.0000001]  # 找出没有违反约束的个体

    if len(pbest2_1) > 0:

        pbest2_1 = pbest2_1[pbest2_1[:, 2].argsort()]  # 根据适应度值排序

    else:

        pbest2_1 = pbest2[pbest2[:, 2].argsort()]  # 如果所有个体都违反约束,直接找出适应度值最小的

    # 当前迭代的最优个体

    pbesti, pbesti_fitness, pbesti_e = pbest2_1[0, :2], pbest2_1[0, 2], pbest2_1[0, 3]

    # 当前最优和全局最优比较

    # 如果两者都没有约束

    if gbest_e <= 0.0000001 and pbesti_e <= 0.0000001:

        if gbest_fitness < pbesti_fitness:

            return gbest, gbest_fitness, gbest_e

        else:

            return pbesti, pbesti_fitness, pbesti_e

    # 有一个违反约束而另一个没有违反约束

    if gbest_e <= 0.0000001 and pbesti_e > 0.0000001:

        return gbest, gbest_fitness, gbest_e

    if gbest_e > 0.0000001 and pbesti_e <= 0.0000001:

        return pbesti, pbesti_fitness, pbesti_e

    # 如果都违反约束,直接取适应度小的

    if gbest_fitness < pbesti_fitness:

        return gbest, gbest_fitness, gbest_e

    else:

        return pbesti, pbesti_fitness, pbesti_e

阅读(1747) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~