模拟退火相关题目

1.计算函数最值(非单峰函数)

1.Strange fuction

题目传送门HDU2899

思路:本题的状态函数就是题目中的数学函数,因为是求最小值,所以每次取最小即可,其他细节见代码。

#include<bits/stdc++.h> 
using namespace std;
const double eps=1e-8;//终止温度 
double y;
double fun(double x){ //计算函数结果 
	return 6*pow(x,7.0)+8*pow(x,6.0)+7*pow(x,3.0)+5*pow(x,2.0)-y*x;
}
double solve(){
	double T=100;//初始温度 
	double delta=0.997;//降温系数 
	double x=50.0;//初始x 
	double now=fun(x),ans=now;//当前计算的函数值,ans最后的答案 
	while(T>eps){//如果没有降低终止温度 
		double nx=x+(rand()*2-RAND_MAX)*T;//随T的降低,对x的改变减少 T可能增加,可能减少 
		if(nx>=0&&nx<=100){//nx处于范围内 
			double nt=fun(nx);//计算nx的值 
			ans=min(ans,nt);//取最小 
			if(now-nt>eps){//如果新状态更小 
				x=nx,now=nt;//x取新状态,now=nt 
			}
		}
		T*=delta;//降温 
	}
	return ans;
}
int main(){
	int t;
	cin>>t;
	while(t--){
		cin>>y;
		printf("%.4lf\n",solve());
	}
	return 0;	
}

求平衡点。

题目传送门:P1337

本题的状态函数是:n个点距离答案坐标的距离*各自重量的求和 也是取最小。因为距离乘上重量 越小,代表重物的重力势能越低,也就越平稳。下面上代码。

#include<bits/stdc++.h>
using namespace std;
struct p{
	double x,y,w;
}a[1005];
const double eps=1e-18,delta=0.9986;// delta 玄学delta 一般越接近1得到的可行解精度越高 
int n;
double x,y,now;
double fun(double x,double y){ //状态函数 
	double sum=0,dx,dy;
	for(int i=1;i<=n;i++)
	{
		dx=x-a[i].x,dy=y-a[i].y;
		sum+=sqrt(dx*dx+dy*dy)*a[i].w;
	}
	return sum;
}
void sa(){
	double t=3000;
	while(t>eps){
		double nx=x+(rand()*2-RAND_MAX)*t,ny=y+(rand()*2-RAND_MAX)*t;
		double nt=fun(nx,ny),de=nt-now;
		if(de<0)
		{
			x=nx,y=ny,now=nt;
		}
		else if(exp(-de/t)*RAND_MAX>rand())
				x=nx,y=ny;
		t*=delta;
	}
} 
int main(){
	cin>>n;
	for(int i=1;i<=n;i++) cin>>a[i].x>>a[i].y>>a[i].w,x+=a[i].x,y+=a[i].y;
	x/=n,y/=n;
	now=fun(x,y);
	sa();
	printf("%.3lf %.3lf\n",x,y);
	return 0;
	
}