题解 P1337 【[JSOI2004]平衡点 / 吊打XXX】
这道题调了好久,果然非洲人是得不到眷顾的吗。。。
本题采用模拟退火解决。
模拟退火是一种简洁明了而又高效的近似算法,基本上可以套到任何求最优解的题目上去。
它的原理是模拟物理中金属退火的现象,凭借选手逆天的RP跳出局部最优解,来到全局最优解。
比较常用的近似算法还有爬山和遗传,但是我个人觉得没太大必要掌握(像我这种脸黑的选手有一次遗传算法WA52。。)。
模拟退火总体就是一个持续降温的流程。
在降温的过程中,坐标随机跳动的范围会越变越小。
当然,为了防止搞错,我们每一次都会以一定概率接受一个比较奇怪的解,这一点有些类似遗传算法中的突变。
inline void simulateAnneal(){
double x=ansx,y=ansy;t=5000;
while(t>1e-14){
double X=x+((rand()<<1)-RAND_MAX)*t,Y=y+((rand()<<1)-RAND_MAX)*t;
double now=calcEnergy(X,Y),dist=now-ans;
if(dist<0){
ansx=x=X,ansy=y=Y,ans=now;
}else if(exp(-dist/t)*RAND_MAX>rand())x=X,y=Y;
t*=delta;
}
}
这里t代表温度,它会一直递减。
然后每一次我们会根据t的值(控制范围)随机跳一个新的解。
求一下它的值,看一下值能不能接受。
假如这个值不能接受,那就以一定概率接受一个奇奇怪怪的解,选择解的方法遵循Metropolis准则(我也不知道这是什么玩意别问我)。
求值的流程因题而异,总之就是一个衡量解的标准。
template<typename T>inline T calcEnergy(T x,T y){
T tmp=0;
for(register int i=1;i<=n;++i){
T disx=x-a[i].x,disy=y-a[i].y;
tmp+=sqrt(disx*disx+disy*disy)*a[i].val;
}
return tmp;
}
主要的部分大概就这么多了,其他要注意的无非就是常数什么的了。
M_sea大佬用的常数是
delta=0.993
t=2000
loop=5
(loop指的是运行次数)
然而这个常数对于脸黑的我来说比较窒息,所以做了一点小调整,最后是
delta=0.9932
t=5000
loop=5
最后,AC代码如下:
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cmath>
#include<cstdlib>
using namespace std;
namespace StandardIO{
template<typename T>inline void read(T &x){
x=0;T f=1;char c=getchar();
for(;c<'0'||c>'9';c=getchar())if(c=='-')f=-1;
for(;c>='0'&&c<='9';c=getchar())x=x*10+c-'0';
x*=f;
}
template<typename T>inline void write(T x){
if(x<0)putchar('-'),x*=-1;
if(x>=10)write(x/10);
putchar(x%10+'0');
}
}
using namespace StandardIO;
namespace Solve{
const int N=1010;
const double delta=0.9932;
struct SA{
private:
int n,sumx,sumy;
double ans=1e18,t;
struct node{
int x,y,val;
}a[N];
template<typename T>inline T calcEnergy(T x,T y){
T tmp=0;
for(register int i=1;i<=n;++i){
T disx=x-a[i].x,disy=y-a[i].y;
tmp+=sqrt(disx*disx+disy*disy)*a[i].val;
}
return tmp;
}
inline void simulateAnneal(){
double x=ansx,y=ansy;t=5000;
while(t>1e-14){
double X=x+((rand()<<1)-RAND_MAX)*t,Y=y+((rand()<<1)-RAND_MAX)*t;
double now=calcEnergy(X,Y),dist=now-ans;
if(dist<0){
ansx=x=X,ansy=y=Y,ans=now;
}else if(exp(-dist/t)*RAND_MAX>rand())x=X,y=Y;
t*=delta;
}
}
public:
SA(){}
~SA(){}
double ansx,ansy;
inline void init(){
srand(19260817),srand(rand()),srand(rand());
read(n);
for(register int i=1;i<=n;++i){
read(a[i].x),read(a[i].y),read(a[i].val);
sumx+=a[i].x,sumy+=a[i].y;
}
}
template<typename T>inline void SimulateAnneal(T times){
ansx=(double)sumx/n,ansy=(double)sumy/n;
for(register int i=1;i<=times;++i)simulateAnneal();
}
}ljz;
inline void solve(){
ljz.init();
ljz.SimulateAnneal(5);
printf("%.3lf %.3lf",ljz.ansx,ljz.ansy);
}
}
using namespace Solve;
int main(){
solve();
}
题解 P1337 【[JSOI2004]平衡点 / 吊打XXX】的更多相关文章
- 洛谷 P1337 [JSOI2004]平衡点 / 吊打XXX 解题报告
P1337 [JSOI2004]平衡点 / 吊打XXX 题目描述 有 \(n\) 个重物,每个重物系在一条足够长的绳子上.每条绳子自上而下穿过桌面上的洞,然后系在一起.\(X\)处就是公共的绳结.假设 ...
- 洛谷 P1337 [JSOI2004]平衡点 / 吊打XXX
洛谷 P1337 [JSOI2004]平衡点 / 吊打XXX 点击进入FakeHu的模拟退火博客 神仙模拟退火...去看fakehu的博客吧...懒得写了... 因为精度问题要在求得的最优解附近(大约 ...
- 洛谷P1337 [JSOI2004]平衡点 / 吊打XXX(模拟退火)
题目描述 如图:有n个重物,每个重物系在一条足够长的绳子上.每条绳子自上而下穿过桌面上的洞,然后系在一起.图中X处就是公共的绳结.假设绳子是完全弹性的(不会造成能量损失),桌子足够高(因而重物不会垂到 ...
- P1337 [JSOI2004]平衡点 / 吊打XXX 模拟退火
链接 https://www.luogu.org/problemnew/show/P1337 思路 交了好多发,都是wrong 初始值取平均数就1A了 真的是玄学的算法 代码 // luogu-jud ...
- [洛谷P1337][JSOI2004]平衡点 / 吊打XXX
题目大意:有$n$个重物,每个重物系在一条绳子上.所有绳子系在一起,问绳结最终平衡于何处. 题解:$NOIP$前学学模拟退火,但发现我脸好黑啊... 卡点:脸黑 C++ Code: #include ...
- 题解:[JSOI2004]平衡点 / 吊打XXX
这个题目算是一个模拟退火的板子题 物重一定,绳子越短,重物越低,势能越小,势能又与物重成正比 使得$\sum_{i=1}^nd[i]*w[i]$也就是总的重力势能最小,可以使得系统平衡 交了两面半.. ...
- P1337 [JSOI2004]平衡点 / 吊打XXX
题目描述 如图:有n个重物,每个重物系在一条足够长的绳子上.每条绳子自上而下穿过桌面上的洞,然后系在一起.图中X处就是公共的绳结.假设绳子是完全弹性的(不会造成能量损失),桌子足够高(因而重物不会垂到 ...
- Luogu P1337 [JSOI2004]平衡点 / 吊打XXX
一道入门模拟退火的经典题,还是很考验RP的 首先我们发现神TM这道题又和物理扯上了关系,其实是一道求广义费马点的题目 首先我们可以根据物理知识得到,当系统处于平衡状态时,系统的总能量最小 又此时系统的 ...
- 洛谷P1337 [JSOI2004]平衡点 / 吊打XXX(模拟退火)
传送门 先坑着,联赛活着回来的话我就写(意思就是我绝对不会写了) //minamoto #include<cstdio> #include<cmath> #include< ...
- LUOGU P1337 [JSOI2004]平衡点 / 吊打XXX(模拟退火)
传送门 解题思路 学习了一下玄学算法--模拟退火,首先要求平衡处,也就是求势能最小的地方,就是求这个点到所有点的距离*重量最小.剩下的几乎是模拟退火的板子了. #include<iostream ...
随机推荐
- 实战:一、使用mongo做一个注册的小demo
思路:1.使用mongoose 进行 数据库的链接 2.使用Schema来进行传输字段的定义 3.安装koa-router进行数据处理4.安装koa-bodyparser 进行post数据交互5.解决 ...
- Java基础学习总结(62)——Java中的流和Socket
按行读入方式: BufferedReader(); 1.以行为读取单位,读取比较方便. 按行读一般都是字符读. BufferedReader和PrintWriter的内存分析图: 数据流: 输入 输出 ...
- LinkedList 注意事项
public E getFirst() 返回此列表的第一个元素. public E getLast() 返回此列表的最后一个元素. public E removeFirst() 移除并返回此列表的 ...
- gem update --system
gem update --system 修改完gem sources之后,进行gem update: gem update --system 之后的输出: C:\Sites\test01>gem ...
- [Tailwind] Extending Tailwind with Responsive Custom Utility Classes
You are able to extend the custom css with hover, focus, group-hover, responsive variants class in t ...
- CountDownLatch使用方法
CountDownLatch是一个同步辅助类,在完毕一组正在其它线程中运行的操作之前.它同意一个或多个线程一直等待. 如果我们周末要去旅游.出游前须要提前订好机票.巴士和酒店,都订好后就能够出发了.这 ...
- 深刻理解Nginx之Nginx完整安装
1. Nginx安装 1.1预先准备 CentOS系统下,安装Nginx的库包依赖. 安装命令例如以下: sudo yum groupinstall "DevelopmentTools& ...
- tensorflow利用预训练模型进行目标检测(一):安装tensorflow detection api
一.tensorflow安装 首先系统中已经安装了两个版本的tensorflow,一个是通过keras安装的, 一个是按照官网教程https://www.tensorflow.org/install/ ...
- tp中使用事务
是什么 事务是为了防止,多个操作,其中有失败,数据有部分被执行成功的时候使用的. 比如,银行,用户转账.张三钱扣了,结果李四钱还增加! 这个时候需要使用事务,确保张三钱扣了,李四的钱也增加,才真正的成 ...
- netty可靠性
Netty的可靠性 首先,我们要从Netty的主要用途来分析它的可靠性,Netty目前的主流用法有三种: 1) 构建RPC调用的基础通信组件,提供跨节点的远程服务调用能力: 2) NIO通信框架,用于 ...