模拟退火的基本思想:

(1) 初始化:初始温度T(充分大),初始解状态S(是算法迭代的起点),每个T值的迭代次数L

(2) 对k=1,……,L做第(3)至第6步:

(3) 产生新解$S\prime $

(4) 计算增量$\Delta t\prime  = C\left( {S\prime } \right) - C\left( S \right)$,其中$C\left( S \right)$为评价函数

(5) 若$\Delta t\prime  < 0$则接受$S\prime $作为新的当前解,否则以概率$exp\left( { - \Delta t\prime /T} \right)$接受$S\prime $作为新的当前解.

(6) 如果满足终止条件则输出当前解作为最优解,结束程序。 终止条件通常取为连续若干个新解都没有被接受时终止算法。

(7) T逐渐减少,且T->0,然后转第2步。

从算法的流程上看,模拟退火算法包括三函数、两准则,即状态产生函数、状态接受函数、温度更新函数、内循环终止准则和外循环终止准则,这些环节的设计将决定模拟退火算法的优化性能。此外,初温的选择对模拟退火算法性能也有很大的影响。

状态产生函数:

   原则:设计状态产生函数(领域函数)的出发点应该是尽可能保证产生的候选解遍布全部的解空间。通常,状态产生函数由两部分组成,即产生候选解的方式和候选解吃剩的概率分布。
方法:在当前状态下的领域结构内以一定的概率方式(均匀分布、正态分布、指数分布等)产生

状态接受函数:

   原则:函数一般以概率的方式给出,不同接受函数的差别主要在于接受概率的形式不同。设计状态接受概率,应该遵循以下原则:

1)在固定温度下,接受使目标函数下降的候选解的概率要大于使目标函数上升的候选解的概率; 
2)随温度的下降,接受目标函数上升的解的概率要逐渐减少; 
3)当温度趋于零时,只能接受目标函数下降的解。方法:状态接受函数的引入是模拟退火算法实现全局搜索的最关键的因素,模拟退火算法中通常用作为状态接受函数。

初始温度、温度更新函数、内循环终止准则和外循环终止准则通常被称为退火历程。

初始温度:

    原则:通过理论分析可以得到初温的解析式,但解决实际问题难以得到精确的参数;实际应用往往要让初温充分大。实验表明:初温越大,获得高质量解的机率越大,但花费较多的计算时间。

方法: 
(1)均匀抽样一组状态,以各状态目标值的方差为初温; 
(2)随机产生一组状态,确定两两状态间的最大目标差值,根据差值,利用一定的函数确定初温,譬如 ,其中为初始接受函数; 
(3)利用经验公式。

温度更新函数:

 温度更新函数,即温度下降方式,用于在啊外循环中修改温度值。

内循环终止准则:

常用的Metropolis抽样准则: 
1)检验目标函数的均值是否稳定; 
2)连续若干步的目标值变化较小; 
3)按一定步数抽样。

外循环终止准则:

 1)设置终止温度的阈值;
2)设置外循环迭代次数;
3)算法搜索到的最优值连续若干步保持不变;
4)概率分析方法。

\[p(dE) = exp(dE/kT)\]其中k是一个常数,exp表示自然指数,且dE<0。这条公式说白了就是:温度越高,出现一次能量差为dE的降温的概率就越大;温度越低,则出现降温的概率就越小。又由于dE总是小于0(否则就不叫退火了),因此dE/kT < 0 ,所以P(dE)的函数取值范围是(0,1) 。

若ΔT<0则接受S′作为新的当前解S,否则以概率exp(-ΔT/T)接受S′作为新的当前解S。

http://acm.hdu.edu.cn/showproblem.php?pid=1109

题意:给定X,Y,M,求区域内点到已知点最短距离的最大值。

解题关键:

1、注意一定不要忘记设立初始解。

2、T_min、T_max转化为步长,以保证精确度。

3、内循环、外循环一定要理解。

4、此程序其实还有一个问题未解决,即以一定的概率接受非最优解,而由于未找到良好的评估函数,故可以直接用时间之比即可,若有时候WA时,尝试一下下面第二个程序。

5、在这里其实就是将外循环的降温当做误差范围。

6、第一个程序相当于多点爬山?不算随机算法。

 #include<cstdio>
#include<cstring>
#include<algorithm>
#include<cstdlib>
#include<cmath>
#include<iostream>
using namespace std;
typedef long long ll;
#define INF 100000000
#define N 30
#define PI acos(-1.0)
double X,Y,M;
double step,eps=1e-,k=0.55;//step代表最长的跨度 ,注意这种用法 ,用来控制精度
struct point{
double x,y;
}people[],num[];
double d[]; double dist(point a,point b){
return sqrt((a.x-b.x)*(a.x-b.x)+(a.y-b.y)*(a.y-b.y));
} bool judge(point a){
return a.x>=&&a.x<=X&&a.y>=&&a.y<=Y;
} double rand1(double l,double r){//随机数生成函数
return rand()%/10000.0*(r-l)+l;
} double fun(point a){
double td=INF;
for(int i=;i<M;i++){
td=min(td,dist(num[i],a));
}
return td;
}
int main(){
int t;
scanf("%d",&t);
while(t--){
double max1=-1.0;
int maxn=;
cin>>X>>Y>>M;
for(int i=;i<M;i++){
scanf("%lf%lf",&num[i].x,&num[i].y);
} for(int i=;i<N;i++){
people[i].x=rand1(,(double)X);
people[i].y=rand1(,(double)Y);
d[i]=fun(people[i]);
} step=max(X,Y);
while(step>eps){
for(int i=;i<N;i++){//初始状态一般为30个
for(int j=;j<;j++){//一般循环50次左右,看题目要求的时间可以变化
point tem;
double angle=rand1(,*PI);//枚举任何角度,使得到的新点向四周扩散
tem.x=people[i].x+cos(angle)*step;
tem.y=people[i].y+sin(angle)*step;
if(!judge(tem)) continue;
double dd=fun(tem);
if(dd>d[i]){ //这里一定要注意,必须要更新,不要忘记
d[i]=dd;
people[i]=tem;
}
}
}
step*=k;
}
for(int i=;i<N;i++){//找到退火后的状态中,最优的解
if((d[i]-max1)>eps){
maxn=i;
max1=d[i];
}
}
printf("The safest point is (%.1f, %.1f).\n",people[maxn].x,people[maxn].y);
}
}
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
#define INF 100000000
#define EPS 1e-6
#define N 30
#define PI acos(-1.0)
double X,Y,M;
double step,eps=1e-,k=0.55;//step代表最长的跨度 ,注意这种用法 ,用来控制精度
struct point{
double x,y;
}people[],num[];
double d[]; double dist(point a,point b){
return sqrt((a.x-b.x)*(a.x-b.x)+(a.y-b.y)*(a.y-b.y));
} bool judge(point a){
return a.x>=&&a.x<=X&&a.y>=&&a.y<=Y;
} double rand1(double l,double r){//随机数生成函数
return rand()%/10000.0*(r-l)+l;
} double fun(point a){
double td=INF;
for(int i=;i<M;i++){
td=min(td,dist(num[i],a));
}
return td;
}
int main(){
int t;
scanf("%d",&t);
while(t--){
double max1=-1.0;
int maxn=;
cin>>X>>Y>>M;
for(int i=;i<M;i++){
scanf("%lf%lf",&num[i].x,&num[i].y);
} for(int i=;i<N;i++){
people[i].x=rand1(,(double)X);
people[i].y=rand1(,(double)Y);
d[i]=fun(people[i]);
} step=max(X,Y);
while(step>eps){
for(int i=;i<N;i++){//初始状态一般为30个
for(int j=;j<;j++){//一般循环50次左右,看题目要求的时间可以变化
point tem;
double angle=rand1(,*PI);//枚举任何角度,使得到的新点向四周扩散
tem.x=people[i].x+cos(angle)*step;
tem.y=people[i].y+sin(angle)*step;
if(!judge(tem)) continue;
double dd=fun(tem);
if(dd>d[i]){ //这里一定要注意,必须要更新,不要忘记
d[i]=dd;
people[i]=tem;
}
else{
if(rand()%/10000.0>exp(k-)){
d[i]=dd;
people[i]=tem;
}
}
}
}
step*=k;
}
for(int i=;i<N;i++){//找到退火后的状态中,最优的解
if((d[i]-max1)>EPS){
maxn=i;
max1=d[i];
}
}
printf("The safest point is (%.1f, %.1f).\n",people[maxn].x,people[maxn].y);
}
}

[HDU1109]模拟退火算法的更多相关文章

  1. 初探 模拟退火算法 POJ2420 HDU1109

    模拟退火算法来源于固体退火原理,更多的化学物理公式等等这里不再废话,我们直接这么来看 模拟退火算法简而言之就是一种暴力搜索算法,用来在一定概率下查找全局最优解 找的过程和固体退火原理有所联系,一般来讲 ...

  2. 模拟退火算法-[HDU1109]

    模拟退火算法的原理模拟退火算法来源于固体退火原理,将固体加温至充分高,再让其徐徐冷却,加温时,固体内部粒子随温升变为无序状,内能增大,而徐徐冷却时粒子渐趋有序,在每个温度都达到平衡态,最后在常温时达到 ...

  3. 【高级算法】模拟退火算法解决3SAT问题(C++实现)

    转载请注明出处:http://blog.csdn.net/zhoubin1992/article/details/46453761 ---------------------------------- ...

  4. 模拟退火算法(SA)求解TSP 问题(C语言实现)

    这篇文章是之前写的智能算法(遗传算法(GA).粒子群算法(PSO))的补充.其实代码我老早之前就写完了,今天恰好重新翻到了,就拿出来给大家分享一下,也当是回顾与总结了. 首先介绍一下模拟退火算法(SA ...

  5. 原创:工作指派问题解决方案---模拟退火算法C实现

    本文忽略了对于模拟退火的算法的理论讲解,读者可参考相关的博文或者其他相关资料,本文着重于算法的实现: /************************************************ ...

  6. BZOJ 3680: 吊打XXX【模拟退火算法裸题学习,爬山算法学习】

    3680: 吊打XXX Time Limit: 10 Sec  Memory Limit: 128 MBSec  Special JudgeSubmit: 3192  Solved: 1198[Sub ...

  7. OI骗分神器——模拟退火算法

    前言&&为什么要学模拟退火 最近一下子学了一大堆省选算法,所以搞一个愉快一点的东西来让娱乐一下 其实是为了骗到更多的分,然后证明自己的RP. 说实话模拟退火是一个集物理与IT多方面知识 ...

  8. 模拟退火算法 R语言

    0 引言 模拟退火算法是用来解决TSP问题被提出的,用于组合优化. 1 原理 一种通用的概率算法,用来在一个打的搜索空间内寻找命题的最优解.它的原理就是通过迭代更新当前值来得到最优解.模拟退火通常使用 ...

  9. 模拟退火算法(西安网选赛hdu5017)

    Ellipsoid Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) Total ...

随机推荐

  1. c# 文件IO操作 StreamReader StreamWriter Split 使用

    StreamWriter(String,Boolean) 若要追加数据到该文件中,则为 true:若要覆盖该文件,则为 false. 如果指定的文件不存在,该参数无效,且构造函数将创建一个新文件. 例 ...

  2. Python 3 并发编程多进程之进程与线程

    Python 3 进程与线程 进程即正在执行的一个过程.进程是对正在运行程序的一个抽象. 进程的概念起源于操作系统,是操作系统最核心的概念,也是操作系统提供的最古老也是最重要的抽象概念之一.操作系统的 ...

  3. Elasticsearch核心知识大纲脑图

  4. java.sql.SQLException: Column count doesn't match value count at row 1 Query: insert into category values(null,?,?,?) Parameters: [1111111, 1111, 软件]

    java.sql.SQLException 问题: java.sql.SQLException: Column count doesn't match value count at row 1 Que ...

  5. 修改myEclipse2014web项目名称

    重命名项目名称后 右键点击你的项目,然后选择属性---->然后点击myeclipse—>Project Facets—> web 选项,修改web context-root名称为你要 ...

  6. Java---变量与常量

    Java中的关键字 Java 语言中有一些具有特殊用途的词被称为关键字.关键字对 Java 的编译器有着特殊的意义,在程序中应用时一定要慎重 Java标识符 标识符就是用于给 Java 程序中变量.类 ...

  7. spring MVC--配置注解

    <context-param> 作用:该元素用来声明应用范围(整个WEB项目)内的上下文初始化参数 param-name 设定上下文的参数名称.必须是唯一名称 param-value 设定 ...

  8. flex 和bison的安装和使用

    1.在ubutu上安装 yacc的命令: sudo apt-get install flex bison flex:词法分析器 flex是一个词法分析器.用来将一个.l文件生成一个.c程序文件.即生成 ...

  9. 8个Javascript小技巧,让你写的代码有腔调

    如果你想确保你的JavaScript在大多数浏览器和移动设备中都可以工作,那么我从大漠等大神指导,原来可以使用f2etest,也可以使用Endtest,browserstack等 1. 使用 + 字符 ...

  10. CodeForces - 1019D(BZOJ3707圈地):Large Triangle (几何,找面积为S的三角形)

    题意:给定平面上N个点,问是否存在三角形,其面积为S. 思路:选择Y轴,枚举这个Y轴,面积大小只与|y-Y|有关,然后二分,具体的可以先去做BZOJ3707. 具体的: 1,先对点排序,X坐标为第一关 ...