Simulate Anneal模拟退火算法,是一种用于得到最优解的随机化算法。

如果可以打一手漂亮的随机化搜索,也许当你面对一筹莫展的神仙题时就有一把趁手的兵器了。

这篇题解将教你什么?SA的基本思路,什么时候能用SA。

标题是浅谈,所以本篇博客参杂了些许个人简介,若有疑问或异议,欢迎提出指正。

我也很感谢你们给出的建议,它们真的能让我变好、变强。

那么我们进入本篇正题。

1. 什么是模拟退火:

模拟退火是一种在广大的搜索空间寻找最优解的随机化算法。我们看名字就明白,这个算法实在模拟物理中退火的过程。知识很多时候都是相通的,我们学习的大部分知识都是有用的。

为什么通过模拟退火的过程可以得出最优解呢?在物理中,固体物质的退火过程和一般的组合优化问题有着很高的相似度。

2. 模拟退火基本要素:

满足这些你就可以将SA算法愉快地嵌入你的程序里了。

第一个要素就是状态空间和状态的产生函数,我们需要有一个搜索空间,而且范围较大。

什么是搜索空间呢?我们学习搜索的时候应该接触过,我们的搜索算法本质就是在问题的求解空间中进行遍历,寻找最优解。模拟退火的状态空间也类似,它就是我们自定义出来的最优解的有限集合。

我们光有状态空间还不够啊,我们真正需要的是状态。学函数的时候,老师就说过了,一个函数要有“域”。我们的状态空间就是状态产生函数的“域”。而我们如果想写一个好的退火,我们的搜索空间要足够大,足够让函数生成不同的新解。

第二个就是候选解,我们在状态空间内通过生成的随机数在一定密度内随机选择我们的候选解。

其实还有一条是概率分布,大部分采用均匀分布,少数情况我们会用到指数分布。

至于状态转移概率,我目前接触到的SA算法统统采用了Metropolis准则,我接下来会介绍它。

3. 模拟退火基本流程:

一. 由一个状态产生函数从当前解产生一个新解,一般采用增量构造(即由原解加上/减去产生函数产生的值)来得到。

二. 计算新解的目标函数差Δt'。

三. 判断新解是否被接受,

这里介绍Metropolis准则:若Δt'<0,我们接受它,否则以exp(-Δt'/T)的多项式概率接受它。

**T是我们模拟退火过程中的温度**

四. 当新解被接受时,用新解代替当前解,否则继续下一轮试验。

4. 模拟退火中的参数控制:

调参可以说是SA算法最难的部分,也是决定你的算法得分率的部分。

这里分享一下神仙FlashHu(LCT导师Orz)写SA时调参的经验,我总结了一下就是这样:

对于eps,可以根据数据范围和精度要求粗略得到eps的大概大小,手动微调可以得到最终的eps。

对于T和ΔT(温度的变化率,一般在0.95-0.99间),先开大一点跑出最优解,然后一边退火一边输出当前的T、ans等信息,大致感受解的下降速率,越均匀表示参数越好(神仙称之为观察法)

5. 注意事项:

SA可能会陷入当前解卡在局部最小的情况,也就是这样:

图中,红色是我们的当前解,蓝色是我们的最优解,红色的线代表SA算法得到的新解,我们会发现,如果参数不佳,我们就有可能出现卡在局部最优解的情况。本身算法的设计就有避免这一种情况,还记得吗?Metropolis准则,就算这不是最优解,我们也以一定概率接受它(答案不更新),就是为了防止这种情况,然而在参数不佳的情况下,它仍可能发生。所以我们要改进对温度的控制方式。

这里还是以那道经典到不能再经典的模拟退火模板做例题:平衡点/吊打xxx

这里直接贴上代码,关于算法中需要注意的地方我会打上注释。

#include<bits/stdc++.h>
#define down 0.997//ΔT,模拟徐徐降温
using namespace std;
inline int read(){
int data=,w=;char ch=;
while(ch!='-' && (ch<''||ch>''))ch=getchar();
if(ch=='-')w=-,ch=getchar();
while(ch>='' && ch<='')data=data*+ch-'',ch=getchar();
return data*w;
}
int n;
struct point{
int x,y,w;
}object[];//物体信息
double ansx,ansy,answ;//答案
double energy(double x,double y){//物理学知识:能量总和越小越稳定
double r=,dx,dy;
for(int i=;i<=n;i++){
dx=x-object[i].x;dy=y-object[i].y;
r+=sqrt(dx*dx+dy*dy)*object[i].w;//力臂乘重力
}
return r;
}
void SA(){
double t=3e3+;//初始温度要高
while(t>1e-){//exp略大于0
double ex=ansx+(rand()*-RAND_MAX)*t;
double ey=ansy+(rand()*-RAND_MAX)*t;
double ew=energy(ex,ey);
double de=ew-answ;
if(de<){//此答案更优
ansx=ex;ansy=ey;answ=ew;
}else if(exp(-de/t)*RAND_MAX>rand()){//Metropolis准则,以多项式概率接受
ansx=ex;ansy=ey;
}t*=down;//逐步降温
}
}
void solve(){
SA();SA();SA();SA();SA();
}
int main(){
n=read();
for(int i=;i<=n;i++){
object[i].x=read();object[i].y=read();object[i].w=read();
ansx+=object[i].x;ansy+=object[i].y;
}
ansx/=n;ansy/=n;//设平均值为初值
answ=energy(ansx,ansy);
solve();
printf("%.3lf %.3lf\n",ansx,ansy);
return ;
}

游戏结束。

[随机化算法] 听天由命?浅谈Simulate Anneal模拟退火算法的更多相关文章

  1. Manacher算法(马拉车算法)浅谈

    什么是Manacher算法? 转载自百度百科 Manachar算法主要是处理字符串中关于回文串的问题的,它可以在 O(n) 的时间处理出以字符串中每一个字符为中心的回文串半径,由于将原字符串处理成两倍 ...

  2. 大数质因解:浅谈Miller-Rabin和Pollard-Rho算法

    2017-07-19 08:54 Amphetamine:能发一下代码吗? 应我那位谜一样好友的邀约,我打算好好看一看Miller-Rabin和Pollard-Rho算法.很奇怪,各种地方有很多代码描 ...

  3. 浅谈Java数据结构和算法

    今天的突然看集合底层的时候发现了好多算法和数据结构.再次就比较一下和汇总一下. 数据结构分类:线性结构和非线性结构 问题一: 什么是线性和非线性: 我个人的理解是:数据结构中线性结构指的是数据元素之间 ...

  4. [HDU1109]模拟退火算法

    模拟退火的基本思想: (1) 初始化:初始温度T(充分大),初始解状态S(是算法迭代的起点),每个T值的迭代次数L (2) 对k=1,……,L做第(3)至第6步: (3) 产生新解$S\prime $ ...

  5. 模拟退火算法SA原理及python、java、php、c++语言代码实现TSP旅行商问题,智能优化算法,随机寻优算法,全局最短路径

    模拟退火算法SA原理及python.java.php.c++语言代码实现TSP旅行商问题,智能优化算法,随机寻优算法,全局最短路径 模拟退火算法(Simulated Annealing,SA)最早的思 ...

  6. 浅谈URLEncoder编码算法

    一.为什么要用URLEncoder 客户端在进行网页请求的时候,网址中可能会包含非ASCII码形式的内容,比如中文. 而直接把中文放到网址中请求是不允许的,所以需要用URLEncoder编码地址, 将 ...

  7. 浅谈Hex编码算法

    一.什么是Hex 将每一个字节表示的十六进制表示的内容,用字符串来显示. 二.作用 将不可见的,复杂的字节数组数据,转换为可显示的字符串数据 类似于Base64编码算法 区别:Base64将三个字节转 ...

  8. 浅谈Base64编码算法

    一.什么是编码解码 编码:利用特定的算法,对原始内容进行处理,生成运算后的内容,形成另一种数据的表现形式,可以根据算法,再还原回来,这种操作称之为编码. 解码:利用编码使用的算法的逆运算,对经过编码的 ...

  9. 浅谈算法和数据结构: 七 二叉查找树 八 平衡查找树之2-3树 九 平衡查找树之红黑树 十 平衡查找树之B树

    http://www.cnblogs.com/yangecnu/p/Introduce-Binary-Search-Tree.html 前文介绍了符号表的两种实现,无序链表和有序数组,无序链表在插入的 ...

随机推荐

  1. ubuntu linux 修改ip 超扎心。

    老大说“终于搞定了,快记下来,不然以后又忘了”(露出慈母般的微笑) 参考地址:https://jingyan.baidu.com/article/adc815139ddcc4f723bf7339.ht ...

  2. Spring 梳理-运行时动态注入bean

    动态注入的方法 使用占位符 使用Spring表达式

  3. 对vue nextTick深入理解-vue性能优化、DOM更新时机、事件循环机制

    一.定义[nextTick.事件循环] nextTick的由来: 由于VUE的数据驱动视图更新,是异步的,即修改数据的当下,视图不会立刻更新,而是等同一事件循环中的所有数据变化完成之后,再统一进行视图 ...

  4. Python之——爱心代码参与情人节

    一行代码实现输出爱心图,参考https://zhuanlan.zhihu.com/p/23321351 原理: 1.借助数学函数——((x * 0.05) ** 2 + (y * 0.1) ** 2 ...

  5. navicat工具 pymysql模块

    目录 一 IDE工具介绍(Navicat) 二 pymysql模块 一 IDE工具介绍(Navicat) 生产环境还是推荐使用mysql命令行,但为了方便我们测试,可以使用IDE工具,我们使用Navi ...

  6. 快学Scala 第二十课 (trait的构造顺序)

    trait的构造顺序: 首先调用超类构造器 特质构造器在超类构造器之后,类构造器之前执行 特质从左向右被构造 每个特质当中,父特质先被构造 如果多个特质共有一个父特质,而那个父特质已经被构造,则不会被 ...

  7. Redis开发与运维:linux安装

    Linux 安装 我的系统是inux 系统,官网下载 https://redis.io/download redis-5.0.5.tar.gz 解压: 编译安装: 官网和文档说得已经很清楚了,现在就执 ...

  8. 去掉文件 BOM 头

    什么是 BOM? BOM 全称是 Byte Order Mark,意思是字节顺序标记.常用来当作标示文件是以 UTF-8.UTF-16 或者 UTF-32 编码的标记. 去除 BOM 头方法 vim ...

  9. Tomcat源码分析一:编译Tomcat源码

    Tomcat源码分析一:编译Tomcat源码 1 内容介绍 在之前的<Servlet与Tomcat运行示例>一文中,给大家带来如何在Tomcat中部署Servlet应用的相关步骤,本文将就 ...

  10. 确认自己所用的python版本

    总结: 目前有两个版本的python处于活跃状态:python2,python3 有多种流行的python运行环境:cpython(应用最广泛的python解释器,如无对解释器有要求,一般用这个,默认 ...