社区讨论
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 条回复,欢迎继续交流。
正在加载回复...