【python】人工蜂群算法求解函数值


人工蜂群算法

人工蜂群算法(artificial bee colony algorithm, ABC)是一种仿生优化算法,模拟了蜜蜂在寻找食物和选择蜜蜂舞蹈位置的行为。ABC算法的搜索过程包括三个阶段:觅食阶段、跟随阶段和贪婪搜索阶段。

  • 在觅食阶段,蜜蜂根据当前位置在搜索空间中随机选择新的候选解,并通过计算目标函数值来评估候选解的质量。

  • 在跟随阶段,蜜蜂会根据其他蜜蜂的舞蹈信息选择邻域进行搜索。舞蹈信息是一种表示候选解质量的指标,蜜蜂会更倾向于选择质量较好的候选解所在的邻域。

  • 在贪婪搜索阶段,蜜蜂会根据当前位置和搜索历史选择质量最好的候选解作为新的位置,并更新舞蹈信息。

优点

  1. 算法简单易理解
    ABC算法基于简单的蜜蜂行为模型,易于理解和实现。
  2. 全局搜索能力强
    ABC算法通过不断更新和交换信息,能够保持全局搜索的能力,有助于找到更优的解。
  3. 高度并行化
    ABC算法的计算过程可以高度并行化,提高了算法的计算效率。
  4. 适用范围广
    ABC算法适用于多种问题的优化求解,包括函数优化、组合优化、机器学习等。

缺点

  1. 参数设置困难
    ABC算法中涉及到一些参数的设定,如蜜蜂数量、迭代次数等,不同问题需求下的参数选择较为困难。
  2. 收敛速度较慢
    ABC算法在收敛速度上可能较慢,特别是对于复杂的高维问题,可能需要较长的时间才能达到较好的解。
  3. 对问题特性的适应性不强
    ABC算法的搜索过程是基于信息交流的,对于某些问题特性不明显的情况,可能会导致搜索的效果不佳。
  4. 缺乏理论分析
    ABC算法发展较晚,目前对于其理论性质的研究较少,缺乏对算法性能和收敛性的理论分析。

模型搭建流程

在这里对搭建过程进行简单的结束,完整的代码可以直接跳到完整代码模型小节

外接第三方库

pip install ypstruct

测试函数

# 测试函数
def sphere(x):
    return sum(x**2)

问题定义

problem = structure()
problem.costfunc = sphere
problem.nvar = 10
problem.varmin = -100 * np.ones(10)
problem.varmax = 100 * np.ones(10)

ABC参数设置

params = structure()
params.maxit = 500
params.npop = 100
params.nonlooker = 100
params.a = 1

运行ABC

out = artificial_bee_colony.run(problem, params)

运行结果输出

plt.rcParams['font.sans-serif'] = ['KaiTi']  #设置字体为楷体
plt.plot(out.bestcost)
print("最优解:{}".format(out.bestsol))
end = time.time()              # 运行结束时刻
print('运行时间:{}s'.format(end-start))

迭代图表输出

plt.xlim(0, params.maxit)
plt.xlabel('迭代次数')
plt.ylabel('全局最优目标函数值')
plt.title('人工蜂群算法')
plt.grid(True)
plt.show()

完整代码模型

代码模型已经上传CSDN资源共享
在这里插入图片描述

app.py

import matplotlib.pyplot as plt
import numpy as np
from ypstruct import structure
import time
import artificial_bee_colony

start = time.time()         #运行开始时刻
# 测试函数
def sphere(x):
    return sum(x**2)

# 问题定义
problem = structure()
problem.costfunc = sphere
problem.nvar = 10
problem.varmin = -100 * np.ones(10)
problem.varmax = 100 * np.ones(10)

# ABC参数
params = structure()
params.maxit = 500
params.npop = 100
params.nonlooker = 100
params.a = 1

# 运行ABC
out = artificial_bee_colony.run(problem, params)
# 运行结果
plt.rcParams['font.sans-serif'] = ['KaiTi']  #设置字体为楷体
plt.plot(out.bestcost)
print("最优解:{}".format(out.bestsol))
end = time.time()              # 运行结束时刻
print('运行时间:{}s'.format(end-start))

plt.xlim(0, params.maxit)
plt.xlabel('迭代次数')
plt.ylabel('全局最优目标函数值')
plt.title('人工蜂群算法')
plt.grid(True)
plt.show()

artificial_bee_colony.py

import numpy as np
from ypstruct import structure

def run(problem, params):
    # 函数信息
    costfunc = problem.costfunc
    nvar = problem.nvar
    varmin = problem.varmin
    varmax = problem.varmax

    # 参数信息
    maxit = params.maxit
    npop = params.npop
    nonlooker = params.nonlooker
    limit = int(np.round(0.6*nvar*npop))
    a = params.a

    # 空的蜂群结构
    empty_bee = structure()
    empty_bee.position = None
    empty_bee.cost = None

    # 临时蜂群结构
    newbee = structure()
    newbee.position = None
    newbee.cost = None

    # 初始化全局最优解
    bestsol = empty_bee.deepcopy()
    bestsol.cost = np.inf

    # 种群初始化
    pop = empty_bee.repeat(npop)

    for i in range(npop):
        pop[i].position = np.random.uniform(varmin, varmax, nvar)
        pop[i].cost = costfunc(pop[i].position)
        if pop[i].cost < bestsol.cost:
            bestsol = pop[i].deepcopy()

    # 初始化每个个体的抛弃次数
    count = np.empty(npop)

    # 记录每一代中全局最优个体目标函数值
    bestcost = np.empty(maxit)

    # 人工蜂群算法主循环
    for it in range(maxit):

        # 引领蜂
        for i in range(npop):

            # 随机选择k,不等于i
            K = np.append(np.arange(0,i),np.arange(i+1,npop))
            k = K[np.random.randint(K.size)]

            # 定义加速系数
            phi = a * np.random.uniform(-1, 1, nvar)

            # 新的蜜蜂位置
            newbee.position = pop[i].position + phi * (pop[i].position - pop[k].position)

            # 计算新蜜蜂目标函数值
            newbee.cost = costfunc(newbee.position)

            # 通过比较目标函数值,更新第i个蜜蜂的位置
            if newbee.cost < pop[i].cost:
                pop[i] = newbee.deepcopy()
            else:
                count[i] += 1

        # 计算适应度值和选择概率
        fit = np.empty(npop)
        meancost = np.mean([pop[i].cost for i in range(npop)])
        for i in range(npop):
            fit[i] = np.exp(-pop[i].cost/meancost)     #将目标函数值转换为适应度值

        probs = fit / np.sum(fit)

        # 跟随蜂
        for m in range(nonlooker):

            # 通过轮盘赌的方式选择蜜源
            i = roulette_wheel_selection(probs)

            # 随机选择k,不等于i
            K = np.append(np.arange(0, i), np.arange(i + 1, npop))
            k = K[np.random.randint(K.size)]

            # 定义加速系数
            phi = a * np.random.uniform(-1, 1, nvar)

            # 新的蜜蜂位置
            newbee.position = pop[i].position + phi * (pop[i].position - pop[k].position)

            # 计算新蜜蜂目标函数值
            newbee.cost = costfunc(newbee.position)

            # 通过比较目标函数值,更新第i个蜜蜂的位置
            if newbee.cost < pop[i].cost:
                pop[i] = newbee.deepcopy()
            else:
                count[i] += 1

        # 侦察蜂
        for i in range(npop):
            if count[i] > limit:
                pop[i].position = np.random.uniform(varmin, varmax, nvar)
                pop[i].cost = costfunc(pop[i].position)
                count[i] = 0

        # 更新全局最优解
        for i in range(npop):
            if pop[i].cost < bestsol.cost:
                bestsol = pop[i].deepcopy()

        # 存储每一代全局最优解的目标函数值
        bestcost[it] = bestsol.cost

        # 展示迭代信息
        print("Iteration {}: Best Cost = {}".format(it, bestcost[it]))

    # 返回值
    out = structure()
    out.pop = pop
    out.bestsol = bestsol
    out.bestcost = bestcost
    return out

def roulette_wheel_selection(p):
    c = np.cumsum(p)
    r = sum(p) * np.random.rand()
    ind = np.argwhere(r <= c)
    return ind[0][0]

  • 0
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

傻傻虎虎

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值