社区讨论

SA 60pts~80pts求调

P2503[HAOI2006] 均分数据参与者 2已保存回复 3

讨论操作

快速查看讨论及其快照的属性,并进行相关操作。

当前回复
3 条
当前快照
1 份
快照标识符
@mhjo0qq8
此快照首次捕获于
2025/11/04 05:41
4 个月前
此快照最后确认于
2025/11/04 05:41
4 个月前
查看原帖
SA的思路是随机将一个数移到随机的组,本质上是将原来的组减去这个数,将新的组加上这个数。
大的小的参数都试过了,最多也是80pts
CPP
#include <bits/stdc++.h>
using namespace std;
int n,m,a[25],sum[10],q[25],v[10];
double ans;
const double bt=1000,et=1e-15,lt=0.9996,tim=0.99;
double work()
{
	double pjs,mans;
	for (int i=1; i<=m;i++)
	{
		pjs+=v[i];
	}
	pjs=pjs/double(m);
	for (int i=1;i <=m; i++)
	{
		mans+=(pjs-double(v[i]))*(pjs-double(v[i]));
	}
	mans=mans/double(m);
	return mans;
}
int main()
{
	srand(time(0));
	srand(rand());
	cin>>n>>m;
	for (int i=1; i<=n ;i++)
	{
		cin>>a[i];
	}
    for (int i=1;i <=n; i++)
    {
    	int minn=INT_MAX;
    	for (int j=1; j<=m; j++)
    	{
    		if (v[j]<minn)
    		{
    			minn=v[j];
    			q[i]=j;
			}
		}
		v[q[i]]+=a[i];
	}
	/*for (int i=1;i<=n; i++)
	{
		cout<<q[i]<<" "<<i<<endl;
	}*/
	ans=work();
	while (clock()/(1.0*CLOCKS_PER_SEC)<=tim)
	{
		double t=bt;
		//int cnt=0;
		while (t>et)
		{
			//cout<<++cnt<<" "<<1<<endl;
			cout<<"";
			int x,y;
			do
			{
				x=rand()%n+1;
				y=rand()%m+1;
			}while (q[x]==y);
			int kl=q[x];
			v[q[x]]-=a[x];
			v[y]+=a[x];
			q[x]=y;
			double nans=work();
			double de=nans-ans;
			if (de<0)
			{
				ans=nans;
			}
			else if (double(rand())/(RAND_MAX*1.0)>exp(double(-de)/bt))
			{
				q[x]=kl;
				v[q[x]]+=a[x];
				v[y]-=a[x];
			}
			t*=lt;
			//cout<<"t"<<" "<<t<<endl;
		}
	}
	cout<<fixed<<setprecision(2)<<sqrt(ans);
}

回复

3 条回复,欢迎继续交流。

正在加载回复...