遗传算法的原理与实现示例

  遗传算法是一种受生物进化理论启发的随机优化算法,其核心思想是模拟自然界中 “物竞天择、适者生存” 的进化过程,通过对候选解的迭代优化,找到问题的最优解。

一、核心思想

  遗传算法将优化问题的候选解视为生物群体中的“个体”,每个个体的“基因”对应解的参数。通过模拟生物进化中的选择、交叉、变异等过程,让群体中 “适应性强”(即更接近最优解)的个体保留并繁衍,“适应性弱” 的个体被淘汰,最终使群体逐渐逼近最优解。

二、算法步骤

1. 编码:将问题解转化为“基因”形式

  首先需要将实际问题的候选解编码为计算机可处理的字符串(如二进制串、实数编码等),这个字符串称为 “染色体”,其中的每个元素(如二进制位)称为 “基因”。

  例如:若优化问题是求“x 在 [0,31] 范围内使函数 f (x)=x² 最大的解”,可将 x 用 5 位二进制编码(如 x=5 对应染色体“00101”)。

2. 初始化群体

  随机生成一定数量的染色体,组成初始 “群体”(个体数量称为 “种群规模”)。

  例如:随机生成 4 个 5 位二进制串,组成初始群体:00101、11011、01001、10010。

3. 适应度评估:衡量个体的 “优劣”

  定义“适应度函数”,计算每个个体的适应度值,值越高表示该个体(解)越优。

  例如:对上述问题,适应度函数可直接用 f (x)=x²,将染色体转为十进制后计算:
      00101(x=5)→ 适应度 = 25;
      11011(x=27)→ 适应度 = 729(最优)。

4. 遗传操作:模拟进化过程

  通过选择、交叉、变异,生成下一代群体:

  选择(Selection):从当前群体中筛选出适应度高的个体,使其有更高概率繁衍后代(类似 “适者生存”)。
  常用方法:轮盘赌选择(适应度越高的个体,被选中的概率越大)。例如:适应度为 729 的个体被选中的概率远高于25的个体。

  交叉(Crossover):将两个选中的个体(父代染色体)按一定概率(交叉概率)交换部分基因,生成新个体(子代染色体),增加群体多样性。

  例如:对父代11011和10010,随机选择交叉点(如第 3 位后),交换后半部分:
      父代 1:110 | 11 → 子代 1:11010;
      父代 2:100 | 10 → 子代 2:10011。

  变异(Mutation):对子代染色体的基因按一定概率(变异概率)随机改变(如二进制位 0 变 1 或 1 变 0),避免群体陷入局部最优。

  例如:对子代11010的第 4 位进行变异(1→0),得到11000。

5. 终止条件

  重复步骤 3 和 4,直到满足终止条件(如迭代次数达到上限、最优个体的适应度不再提升等),最终输出适应度最高的个体作为最优解。

三、算法特点

  鲁棒性:对问题的数学性质要求低,可处理非线性、多峰等复杂问题。
  并行性:群体中的多个个体可同时优化,适合并行计算。
  随机性:通过随机操作探索解空间,但通过适应度评估引导优化方向,兼顾 “探索” 与 “利用”。

四、应用场景

  遗传算法广泛用于函数优化、机器学习(如神经网络参数优化)、组合优化(如旅行商问题、背包问题)、工程设计(如电路布局)等领域。

五、Python实现示例

  现寻找函数 f ( x ) = − x 2 + 10 f(x)=-x^2+10 f(x)=x2+10在区间 [ − 10 , 10 ] [-10,10] [10,10]内的最大值。理论上,当 x = 0 x=0 x=0时函数取得最大值 f ( 0 ) = 10 f(0)=10 f(0)=10。运行该程序后,可以观察算法如何逐步逼近这个最优解。

import matplotlib
matplotlib.use('TkAgg')import numpy as np
import matplotlib.pyplot as plt# 目标函数:寻找 -x^2 + 10 的最大值
def objective_function(x):return -x ** 2 + 10# 解码染色体为实际数值
def decode_chromosome(chromosome, min_val, max_val):"""将二进制染色体解码为区间内的实数值"""binary_str = ''.join(map(str, chromosome))decimal = int(binary_str, 2)max_decimal = 2 ** len(chromosome) - 1return min_val + (max_val - min_val) * decimal / max_decimal# 计算适应度(函数值越大适应度越高)
def calculate_fitness(population, min_val, max_val):fitness = []for chromosome in population:x = decode_chromosome(chromosome, min_val, max_val)fitness.append(objective_function(x))return fitness# 选择操作(锦标赛选择)
def tournament_selection(population, fitness, tournament_size=3):selected = []for _ in range(len(population)):# 随机选择几个个体进行比较tournament_indices = np.random.choice(len(population), tournament_size)tournament_fitness = [fitness[i] for i in tournament_indices]# 选择适应度最高的个体winner_index = tournament_indices[np.argmax(tournament_fitness)]selected.append(population[winner_index])return selected# 交叉操作(单点交叉)
def crossover(parents, crossover_rate=0.8):children = []for i in range(0, len(parents), 2):parent1 = parents[i]parent2 = parents[i + 1] if i + 1 < len(parents) else parents[0]# 以一定概率进行交叉if np.random.random() < crossover_rate:crossover_point = np.random.randint(1, len(parent1))child1 = parent1[:crossover_point] + parent2[crossover_point:]child2 = parent2[:crossover_point] + parent1[crossover_point:]else:child1, child2 = parent1.copy(), parent2.copy()children.extend([child1, child2])# 确保种群大小不变return children[:len(parents)]# 变异操作
def mutate(population, mutation_rate=0.01):for chromosome in population:for i in range(len(chromosome)):if np.random.random() < mutation_rate:chromosome[i] = 1 - chromosome[i]  # 翻转位return population# 主函数
def genetic_algorithm(pop_size=100, chromosome_length=10, generations=100,min_val=-10, max_val=10):# 初始化种群population = [np.random.randint(0, 2, chromosome_length).tolist() for _ in range(pop_size)]best_fitness_history = []best_solution = Nonebest_x = Nonefor generation in range(generations):# 计算适应度fitness = calculate_fitness(population, min_val, max_val)# 记录最佳解best_idx = np.argmax(fitness)best_fitness_history.append(fitness[best_idx])if best_solution is None or fitness[best_idx] > calculate_fitness([best_solution], min_val, max_val)[0]:best_solution = population[best_idx]best_x = decode_chromosome(best_solution, min_val, max_val)# 选择parents = tournament_selection(population, fitness)# 交叉children = crossover(parents)# 变异population = mutate(children)# 打印进度if generation % 10 == 0:print(f"Generation {generation}: Best fitness = {best_fitness_history[-1]:.4f}, Best x = {best_x:.4f}")# 绘制适应度进化曲线plt.figure(figsize=(10, 6))plt.plot(best_fitness_history)plt.title('Best Fitness Over Generations')plt.xlabel('Generation')plt.ylabel('Fitness')plt.grid(True)plt.savefig('figure/fitness_evolution.png')return best_solution, best_x, objective_function(best_x)# 运行算法
best_chromosome, best_x, best_fitness = genetic_algorithm()print("\n优化结果:")
print(f"最佳染色体: {best_chromosome}")
print(f"最优解 x = {best_x:.6f}")
print(f"最大值 f(x) = {best_fitness:.6f}")

在这里插入图片描述
在这里插入图片描述
  示例通过模拟生物进化过程来寻找函数的最优解。流程包括:
    初始化种群:随机生成一组二进制编码的染色体
    评估适应度:计算每个染色体对应的函数值
    选择操作:使用锦标赛选择法选出较优个体
    交叉操作:对选中的个体进行基因重组
    变异操作:引入随机变异增加多样性
    迭代进化:重复上述过程直到满足终止条件


  通过模拟生物进化的“优胜劣汰”机制,遗传算法能在复杂解空间中高效搜索最优解,是启发式优化算法中的经典方法之一。



End.

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/web/87596.shtml
繁体地址,请注明出处:http://hk.pswp.cn/web/87596.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

centos7 ping127.0.0.1不通

ping 127.0.0.1&#xff0c;localhost和本地ip都不通&#xff0c;所有的配置也是正确的 检查下是否禁止了ping vim /proc/sys/net/ipv4/icmp_echo_ignore_all 内容为 1 禁止ping 内容为0 开启ping sysctl -w net.ipv4.icmp_echo_ignore_all0 变更以上设置即可

【无标题】JavaScript入门

JS 1.JS引入方式 <!DOCTYPE html><html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>JS-引入方式</title><!-- …

(JAVA)自建应用调用企业微信API接口,实现消息推送

建议先简单了解企业微信开发者中心文档&#xff1a;开发前必读 - 文档 - 企业微信开发者中心 了解一下企业微信调用接口的基础参数&#xff1a;基本概念介绍 - 文档 - 企业微信开发者中心 本篇每个步骤都会跟着官网文档走&#xff0c;都会贴上相关链接&#xff0c;看完本篇文…

P/Invoke 在默认封送(marshalling)规则下,常见托管 ⇄ 非托管类型的对应关系

下表整理了 P/Invoke 在默认封送&#xff08;marshalling&#xff09;规则下&#xff0c;常见托管 ⇄ 非托管类型的对应关系。 内容主要依据微软官方 Marshalling Data with Platform Invoke 文档&#xff0c;并补充了常见指针&#xff0f;句柄用法与字符串缓冲区&#xff…

2.isaacsim4.2 教程-初识OmniGraph

1. OmniGraph&#xff08;视觉编程&#xff09; OmniGraph 是 Omniverse 的可视化编程框架。它提供了一个图状结构&#xff0c;将 Omniverse 内多个系统的功能节点串联起来&#xff1b;同时也是一个计算框架&#xff0c;允许你编写高度自定义的节点&#xff0c;将自己的功能无…

MonoGame 游戏开发框架日记 -03

第三章&#xff1a;创建类库 内容介绍 主要内容&#xff1a;创建Core类并编写 创建这个类主要是为了后续开发方便&#xff0c;并介绍游戏开发中的一种非常重要编程模式 单例模式&#xff0c;以及了解MonoGame基本图形渲染知识单例模式&#xff1a; 第一步我们得先了解什么是单例…

AES 256 CBC加密和解密

AES-256-CBC 是一种对称加密算法&#xff0c;使用 256位密钥 和 CBC&#xff08;Cipher Block Chaining&#xff09;模式。它的典型使用场景包括对敏感信息进行加密存储或传输。下面是 AES-256-CBC 的加密与解密的 Python 示例&#xff0c;使用 pycryptodome 库&#xff1a; &a…

Git 版本控制完全指南:从入门到精通

Git 版本控制完全指南&#xff1a;从入门到精通 作为当今最流行的分布式版本控制系统&#xff0c;Git 已经成为开发者必备的技能之一。无论你是独立开发者还是团队协作&#xff0c;Git 都能帮助你高效管理代码版本。本文将带你从零开始&#xff0c;逐步掌握 Git 的核心概念和常…

408第三季part2 - 计算机网络 - 计算机网络分层结构

理解 PCI会放一些控制信息&#xff0c;源地址目的地址都在里面 SDU是放的数据 整个加起来是PDU 每一层的SDU都是上一层的PDU 看一看 也是简单看一看就行 网络层有时候也叫IP数据报 这里断点下载的意思就是&#xff0c;你下载东西的时候网络断了&#xff0c;再连回来的时候会接…

打开摄像头,服务器和客户端传输摄像头图像数据

1&#xff1a;Camera Server 主要功能&#xff0c;打开摄像头&#xff0c;接收客户端请求 接收到客户端请求“R”字符后开始传输摄像头图像。 #include "mainwindow.h" #include "ui_mainwindow.h"#include<QDebug>MainWindow::MainWindow(QWidget…

Android实现获取前台应用信息

Android实现获取前台应用信息 1.前言&#xff1a; 之前需要获取在后台运行的App信息&#xff0c;比如包名、版本这些常规的&#xff0c;今天是讲解获取在前台的App信息&#xff0c;虽然App在前台&#xff0c;但是具体的信息可能不知道&#xff0c;今天就尝试获取一下&#xf…

快讯|美团即时零售日订单已突破1.2亿,餐饮订单占比过亿

据美团内网公布信息显示&#xff0c;截至22时54分&#xff0c;美团即时零售当日订单已经突破了1.2亿单&#xff0c;其中&#xff0c;餐饮订单已超过1亿单。 值得注意的是&#xff0c;就在当晚20时45分&#xff0c;美团内网曾显示即时零售日订单突破了1亿。这也意味着&#xff…

pycharm2018配置gitee操作

一、gitee介绍及下载安装 gitee介绍&#xff1a; gitee别名码云&#xff0c;是中国的一个代码托管平台&#xff0c;类似于GitHub&#xff0c;基于Git技术&#xff0c;提供远程仓库托管、协作功能和开源社区服务&#xff0c;优势包括访问速度快、本地化服务和政策合规git和gite…

数据结构——栈的讲解(超详细)

数据结构——栈的讲解&#xff08;超详细&#xff09;-腾讯云开发者社区-腾讯云 #include"Stack.h" void STInit(ST* ps) {ps->arr NULL;ps->capacity ps->top 0; //总空间个数和有用空间个数都初始化为0 }void STDestroy(ST* ps) {if (ps -> arr) …

MySQL允许root用户远程连接

注意&#xff1a;在实际生产环境中&#xff0c;允许root用户从任意主机&#xff08;‘%’&#xff09;连接存在安全风险&#xff0c;建议使用强密码并限制访问IP&#xff0c;或者创建具有必要权限的单独用户用于远程连接。MySQL 配置远程连接指南 1. 登录 MySQL 服务器 mysql -…

STM32的 syscalls.c 和 sysmem.c

syscalls.c 是 STM32CubeIDE 自动生成的标准系统调用适配文件&#xff0c;用于裸机环境下支持 newlib 标准库&#xff08;如 printf, scanf, malloc&#xff09;的运行。这份文件提供了标准库运行所需的最小系统调用实现。现在我来逐段解析其作用&#xff0c;并补充你可能需要修…

Java零基础笔记01(JKD及开发工具IDEA安装配置)

1.Java简介 Java是一种广泛使用的计算机编程语言&#xff0c;由美国的Sun Microsystems公司&#xff08;Stanford University Network&#xff09;在1995年推出。Java以其跨平台、面向对象、安全性高等特点&#xff0c;广泛应用于企业级应用开发、移动应用开发等领域。2009年&a…

Spark SQL架构及高级用法

Spark SQL 架构概述 架构核心组件 API层&#xff08;用户接口&#xff09; 输入方式&#xff1a;SQL查询&#xff1b;DataFrame/Dataset API。统一性&#xff1a; 所有接口最终转换为逻辑计划树&#xff08;Logical Plan&#xff09;&#xff0c;进入优化流程。 编译器层&…

【机器学习深度学习】什么是下游任务模型?

目录 前言 一、什么是下游任务模型&#xff1f; 二、为什么需要下游任务模型&#xff1f; 三、下游任务模型都在干嘛&#xff1f; 四、下游模型怎么训练出来的&#xff1f; 五、图解理解&#xff1a;上游 vs 下游 六、一个现实案例&#xff1a;BERT做情感分析 原始数据…

补充:问题:CORS ,前后端访问跨域问题

补充&#xff1a;问题&#xff1a;CORS &#xff0c;前后端访问跨域问题 我这边的解决方法是&#xff1a; myAxios.defaults.withCredentials true; // 配置为true&#xff0c;表示前端向后端发送请求的时候&#xff0c;需要携带上凭证cookie整体的&#xff1a; import axio…