题目意思:
有n颗珍珠,要求每颗珍珠达到预定颜色,每次操作可以选一连续区间的珍珠,让它们达到预定颜色,花费为该区间不同颜色种数的平方。求完成任务的最少花费。
n<=5*10^4
解题思路:
dp+数据结构
o(n^2)肯定会超时.考虑花费最多为n,且最大的种数为sqrt(n),可以一种一种的往前扫(不是一个一个的),注意如果后面已经选了某种,则前面的该种不用扫,直接连到前面去,用一个并查集维护可以连续跳的最前面的那个,直到开辟一种新的颜色。根表示可以跳到的最前的位置。
代码:
//#include<CSpreadSheet.h>
#include<iostream>
#include<cmath>
#include<cstdio>
#include<sstream>
#include<cstdlib>
#include<string>
#include<string.h>
#include<cstring>
#include<algorithm>
#include<vector>
#include<map>
#include<set>
#include<stack>
#include<list>
#include<queue>
#include<ctime>
#include<bitset>
#include<cmath>
#define eps 1e-6
#define INF 0x3f3f3f3f
#define PI acos(-1.0)
#define ll __int64
#define LL long long
#define lson l,m,(rt<<1)
#define rson m+1,r,(rt<<1)|1
#define M 1000000007
//#pragma comment(linker, "/STACK:1024000000,1024000000")
using namespace std;
#define Maxn 51000
int Fa[Maxn],nu[Maxn],sa[Maxn],temp[Maxn],la[Maxn];
int n,dp[Maxn];
int Find(int x)
{
while(Fa[x]!=x)
x=Fa[x];
return x;
}
void Unio(int a,int b)
{
int x=Find(a),y=Find(b);
if(x!=y)
Fa[y]=x; //注意是后一个连前一个
}
int main()
{
//freopen("in.txt","r",stdin);
//freopen("out.txt","w",stdout);
while(~scanf("%d",&n))
{
Fa[0]=0;
for(int i=1;i<=n;i++)
{
scanf("%d",&sa[i]);
temp[i]=sa[i];
Fa[i]=i;
}
sort(temp+1,temp+n+1);
int cnt=unique(temp+1,temp+n+1)-temp;
map<int,int>myp;
for(int i=1;i<=cnt;i++)
{
myp[temp[i]]=i; //标记某数是第几大
la[i]=-1;
}
for(int i=1;i<=n;i++)
nu[i]=myp[sa[i]]; //标记该数的大小序
dp[0]=0;
for(int i=1;i<=n;i++)
{
if(la[nu[i]]!=-1)
Unio(la[nu[i]]-1,la[nu[i]]);
la[nu[i]]=i;
dp[i]=i;
int num=0;
for(int j=i;j>0;j=Find(j-1)) //一种一种的往前移动,如果后面选了,可以连续往前跳
{
num++;
if(num*num>=dp[i])
break;
int nn=Find(j-1);
dp[i]=min(dp[i],dp[nn]+num*num);
}
}
printf("%d\n",dp[n]);
}
return 0;
}
09 hdu 5015 233 Matrix
题目意思:
告诉一个矩阵的第一行和第一列,按照sa[i][j]=sa[i-1][j]+sa[i][j-1]的运算规则求出sa[n][m].
n<=10 m<=10^9
解题思路:
矩阵快速幂。
一列一列的往前推。构造递推矩阵。
矩阵构造示意图:
代码:
//#include<CSpreadSheet.h>
#include<iostream>
#include<cmath>
#include<cstdio>
#include<sstream>
#include<cstdlib>
#include<string>
#include<string.h>
#include<cstring>
#include<algorithm>
#include<vector>
#include<map>
#include<set>
#include<stack>
#include<list>
#include<queue>
#include<ctime>
#include<bitset>
#include<cmath>
#define eps 1e-6
#define INF 0x3f3f3f3f
#define PI acos(-1.0)
#define ll __int64
#define LL long long
#define lson l,m,(rt<<1)
#define rson m+1,r,(rt<<1)|1
#define M 10000007
//#pragma comment(linker, "/STACK:1024000000,1024000000")
using namespace std;
#define Maxn 15
int n,m;
struct Mar
{
int row,col;
ll s[Maxn][Maxn];
void init(int a,int b)
{
row=a,col=b;
memset(s,0,sizeof(s));
}
};
Mar operator * (const Mar &a,const Mar &b)
{
Mar res;
res.init(a.row,b.col);
for(int k=1;k<=a.col;k++)
{
for(int i=1;i<=res.row;i++)
{
if(a.s[i][k]==0)
continue;
for(int j=1;j<=res.col;j++)
{
if(b.s[k][j]==0)
continue;
res.s[i][j]=(a.s[i][k]*b.s[k][j]+res.s[i][j])%M;
}
}
}
return res;
}
int main()
{
while(~scanf("%d%d",&n,&m))
{
if(!n&&!m)
{
printf("0\n");
continue;
}
Mar ba;
ba.init(n+2,n+2);
ba.s[1][1]=10;
ba.s[1][n+2]=3;
for(int i=2;i<=n+1;i++)
for(int j=1;j<=i;j++)
ba.s[i][j]=1;
ba.s[n+2][n+2]=1;
Mar ans;
ans.init(n+2,1);
ans.s[1][1]=233;
for(int i=2;i<=n+1;i++)
scanf("%I64d",&ans.s[i][1]);
ans.s[n+2][1]=1;
int k=m;
while(k>0)
{
if(k&1)
ans=ba*ans;
k>>=1;
ba=ba*ba;
}
if(n==0)
printf("%I64d\n",ans.s[n+1][1]/10);
else
printf("%I64d\n",ans.s[n+1][1]);
}
return 0;
}
题目意思:
求在给定椭圆面的一点,使它到原点的距离最小。
解题思路:
概率-模拟退火
开始任找一点,然后8个方向改变x,y,根据椭圆方程求出z。越精确,改变的幅度越小。
代码:
#include<cstdio>
#include<cmath>
#include<cstdlib>
#define INF 1e8
#define eps 1e-10
int dir[8][2]={{-1,0},{0,1},{1,0},{0,-1},{-1,1},{1,1},{1,-1},{-1,-1}};
double a,b,c,d,e,f;
double dis(double x,double y,double z)
{
return sqrt(x*x+y*y+z*z);
}
double cal(double x,double y)
{
double A=c,B=d*y+e*x,C=f*x*y+a*x*x+b*y*y-1;
double temp=B*B-4*A*C;
if(temp<0)
return 1e9;
temp=sqrt(temp);
double z1=(-B+temp)/A/2;
double z2=(-B-temp)/A/2;
if(dis(x,y,z1)<dis(x,y,z2))
return z1;
return z2;
}
double solve()
{
double x=0,y=0,z=sqrt(1.0/c);
double be=1,ne=0.99;
while(be>eps)
{
for(int i=0;i<8;i++)
{
double xx=x+be*dir[i][0],yy=y+be*dir[i][1]; //ÿŽÎÒª³ËÒÔžÅÂÊ£¬ÕâÑùÔœµœºóÃ棬·¶Î§±ãԜС£¬Ôœ×ŒÈ·à¶
double zz=cal(xx,yy);
if(zz>INF)
continue;
if(dis(xx,yy,zz)<dis(x,y,z))
{
x=xx;
y=yy;
z=zz;
}
}
be*=ne;
}
return dis(x,y,z);
}
int main(int argc, const char *argv[])
{
while(~scanf("%lf%lf%lf%lf%lf%lf",&a,&b,&c,&d,&e,&f))
printf("%.10f\n",solve());
return 0;
}