使用Python实现轮盘赌选择法Roulette Wheel Selection Method in Python

2023-12-08 04:04

本文主要是介绍使用Python实现轮盘赌选择法Roulette Wheel Selection Method in Python,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、引言

        最近在手写遗传算法,想尝试解决一些优化问题。然而,在编码的过程中,自己发现了很多都不懂的问题。比如,交叉的操作,有单点交叉、两点交叉和多点交叉,具体选哪一种会更好呢?未知。还有交叉的概率,以及变异的概率,取多少才算合理呢?未知。最后就是轮盘赌选择法,在实现的时候,也有一点小疑惑,就是:通过Python的random没办法直接使用choice、choices或sample进行选择。所以,只好通过手动的方式写一个了。

二、轮盘赌选择法/Roulette Wheel Selection Method

        轮盘赌选择法,即Roulette Wheel Selection Method,又称为Fitness Proportionate Selection Method。常用于遗传算法染色体的选择。从选择的个数来看,如果只是一个,那么很好实现。但如果是从多个选择之中选择一个,那么就稍微复杂一些了。

        想象有一个轮盘,现在我们将它分割成m个部分,而这里的m代表我们总体中染色体的个数。每条染色体在轮盘上占有的区域面积将根据适应度分数成比例表达出来。例如m=4,而各条染色体的适应度分数分别为28、23、12、34。那么,轮盘的形状如下:

        假设有一个固定的指针固定在圆盘周长上的某一点,轮盘开始旋转,那么当旋转停止时,就能够确定指针停留在哪个区域,并根据此区域能够选择出一个染色体。例如,

        这样,第一个染色体便被选了出来,而且它被选中的概率是35%(34/(28+23+12+34))。此时如果要选出第二个染色体,就拥有两种实现方式了:

        ①轮盘中不删除chromosome4,如果通过旋转随机停下选出来的还是chromosome4,就重新选择,直到选出一个染色体不是chromosome4为止。

        ②轮盘中删除chromosome4,然后按照下面的轮盘进行选择:

        显然,在第二种实现方法中,各染色体被选中的概率已经发生了变化。而且,第一种实现方法存在一个比较明显的缺点,就是:有可能一直都选择同一个染色体,尽管选择的尝试越久、概率越低,这导致选择的效率下降。第一种实现方式还有一个缺点是,选择到两个不同的染色体的概率是不可计算的,因为轮盘选择的结果拥有无数个(例如,每次旋转都被选中相同的chromosome4,导致一个死循环并不断做轮盘选择);而第二种的概率是可以被计算,因为每种选择的结果确定(例如,chromosome4-chromosome1、chromosome4-chromosome2或chromosome4-chromosome3)。

        因此,这里推荐使用第二种实现方式实现轮盘赌选择法。

三、基于Python的实现方式

        编程思想很简单,大概如下。生成一个[0,total_fitness)的随机数R

        如果R≤28,则选择chromosome1;

        如果28<R≤28+23,则选择chromosome2;

        如果28+23<R≤28+23+12,则选择chromosome3;

        如果28+23+12<R≤28+23+12+34,则选择chromosome4。

        当需要选择第二个染色体时,就从list中进行删除,当然fitness所对应的list也需要进行删除。

        实现方式为:

import copy
import random# function: achieve the roulette wheel selection method
# population_list: the list of optional chromosomes
# fitness_list: the corresponding fitness of chromosomes
# num_selection: the number of chromosome selection
def rw_selection(chromosome_list, fitness_list, num_selection):# copy a duplicate of the populationcp_chromosome_list = copy.deepcopy(chromosome_list)cp_fitness_list = copy.deepcopy(fitness_list)# define a list for save the selected chromosomesselected_chromosome_list = []# select m chromosomes from cp_chromosome_listfor _ in range(num_selection):# calculate the current sum of chromosome fitnesstotal_fitness = sum(cp_fitness_list)# judge which chromosome is selectedrandom_value = random.uniform(0, total_fitness)cumulative_fitness = 0for i, chromosome in enumerate(cp_chromosome_list):cumulative_fitness += cp_fitness_list[i]if cumulative_fitness >= random_value:# select a chromosomeselected_chromosome_list.append(chromosome)# remove the corresponding chromosome and fitnesscp_chromosome_list.pop(i)cp_fitness_list.pop(i)break# return the selection resultreturn selected_chromosome_list# enter of the program
if __name__ == '__main__':p = ['chromosome1', 'chromosome2', 'chromosome3', 'chromosome4']f = [28, 23, 12, 34]m = 2for _ in range(10):selected_p = rw_selection(p, f, m)print(selected_p)

四、运行结果和讨论

['chromosome3', 'chromosome4']
['chromosome1', 'chromosome2']
['chromosome2', 'chromosome1']
['chromosome3', 'chromosome1']
['chromosome1', 'chromosome2']
['chromosome1', 'chromosome3']
['chromosome4', 'chromosome1']
['chromosome4', 'chromosome2']
['chromosome3', 'chromosome4']
['chromosome4', 'chromosome1']

Process finished with exit code 0

        从运行的结果来看,chromosome4和chromosome1被优先选择的概率较高,因为chromosome4被选择的次数为5/20,而chromosome1被选择的次数为7/20。另外,chromosome2被选择的次数为4/20,chromosome3被选择的次数为4/20。基本上符合fitness的分布,但由于试验次数太少,chromosome4被选择的次数要少于chromosome1。

五、总结

        其实感觉这篇博客的意义不是很大,但是怕自己以后还遇到这个小问题,所以,还是记录一下比较好。如果博客中出现一些问题,还请广大网友批评指正。

六、参考资料

        1. 遗传算法

        2. Fitness Proportionate Selection

        3. what is roulette wheel selection

这篇关于使用Python实现轮盘赌选择法Roulette Wheel Selection Method in Python的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/468476

相关文章

Flutter实现文字镂空效果的详细步骤

《Flutter实现文字镂空效果的详细步骤》:本文主要介绍如何使用Flutter实现文字镂空效果,包括创建基础应用结构、实现自定义绘制器、构建UI界面以及实现颜色选择按钮等步骤,并详细解析了混合模... 目录引言实现原理开始实现步骤1:创建基础应用结构步骤2:创建主屏幕步骤3:实现自定义绘制器步骤4:构建U

使用Python创建一个功能完整的Windows风格计算器程序

《使用Python创建一个功能完整的Windows风格计算器程序》:本文主要介绍如何使用Python和Tkinter创建一个功能完整的Windows风格计算器程序,包括基本运算、高级科学计算(如三... 目录python实现Windows系统计算器程序(含高级功能)1. 使用Tkinter实现基础计算器2.

SpringBoot中四种AOP实战应用场景及代码实现

《SpringBoot中四种AOP实战应用场景及代码实现》面向切面编程(AOP)是Spring框架的核心功能之一,它通过预编译和运行期动态代理实现程序功能的统一维护,在SpringBoot应用中,AO... 目录引言场景一:日志记录与性能监控业务需求实现方案使用示例扩展:MDC实现请求跟踪场景二:权限控制与

Android实现定时任务的几种方式汇总(附源码)

《Android实现定时任务的几种方式汇总(附源码)》在Android应用中,定时任务(ScheduledTask)的需求几乎无处不在:从定时刷新数据、定时备份、定时推送通知,到夜间静默下载、循环执行... 目录一、项目介绍1. 背景与意义二、相关基础知识与系统约束三、方案一:Handler.postDel

在.NET平台使用C#为PDF添加各种类型的表单域的方法

《在.NET平台使用C#为PDF添加各种类型的表单域的方法》在日常办公系统开发中,涉及PDF处理相关的开发时,生成可填写的PDF表单是一种常见需求,与静态PDF不同,带有**表单域的文档支持用户直接在... 目录引言使用 PdfTextBoxField 添加文本输入域使用 PdfComboBoxField

Git可视化管理工具(SourceTree)使用操作大全经典

《Git可视化管理工具(SourceTree)使用操作大全经典》本文详细介绍了SourceTree作为Git可视化管理工具的常用操作,包括连接远程仓库、添加SSH密钥、克隆仓库、设置默认项目目录、代码... 目录前言:连接Gitee or github,获取代码:在SourceTree中添加SSH密钥:Cl

Python开发文字版随机事件游戏的项目实例

《Python开发文字版随机事件游戏的项目实例》随机事件游戏是一种通过生成不可预测的事件来增强游戏体验的类型,在这篇博文中,我们将使用Python开发一款文字版随机事件游戏,通过这个项目,读者不仅能够... 目录项目概述2.1 游戏概念2.2 游戏特色2.3 目标玩家群体技术选择与环境准备3.1 开发环境3

Python中模块graphviz使用入门

《Python中模块graphviz使用入门》graphviz是一个用于创建和操作图形的Python库,本文主要介绍了Python中模块graphviz使用入门,具有一定的参考价值,感兴趣的可以了解一... 目录1.安装2. 基本用法2.1 输出图像格式2.2 图像style设置2.3 属性2.4 子图和聚

windows和Linux使用命令行计算文件的MD5值

《windows和Linux使用命令行计算文件的MD5值》在Windows和Linux系统中,您可以使用命令行(终端或命令提示符)来计算文件的MD5值,文章介绍了在Windows和Linux/macO... 目录在Windows上:在linux或MACOS上:总结在Windows上:可以使用certuti

CentOS和Ubuntu系统使用shell脚本创建用户和设置密码

《CentOS和Ubuntu系统使用shell脚本创建用户和设置密码》在Linux系统中,你可以使用useradd命令来创建新用户,使用echo和chpasswd命令来设置密码,本文写了一个shell... 在linux系统中,你可以使用useradd命令来创建新用户,使用echo和chpasswd命令来设