03 hdu 5009 Paint Pearls
题目意思:
有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; }
题目意思:
告诉一个矩阵的第一行和第一列,按照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; }
11 hdu 5017 Ellipsoid
题目意思:
求在给定椭圆面的一点,使它到原点的距离最小。
解题思路:
概率-模拟退火
开始任找一点,然后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; }