A. Artwork

倒过来并查集维护即可。

  1. #include<cstdio>
  2. #include<algorithm>
  3. using namespace std;
  4. const int N=1111;
  5. int n,m,q,i,j,ce;
  6. bool black[N][N];
  7. bool v[N*N];
  8. int f[N*N];
  9. int res;
  10. int ans[N*N],id[N][N],cnt;
  11. struct P{
  12. int x,y;
  13. P(){}
  14. P(int _x,int _y){x=_x,y=_y;}
  15. }e[2222222];
  16. void gao(){
  17. int A,B,C,D;
  18. scanf("%d%d%d%d",&A,&B,&C,&D);
  19. if(A==C){
  20. if(B>D)swap(B,D);
  21. for(int i=B;i<=D;i++)if(!black[A][i])e[++ce]=P(A,i),black[A][i]=1;
  22. }else{
  23. if(A>C)swap(A,C);
  24. for(int i=A;i<=C;i++)if(!black[i][B])e[++ce]=P(i,B),black[i][B]=1;
  25. }
  26. }
  27. int F(int x){return f[x]==x?x:f[x]=F(f[x]);}
  28. inline void merge(int x,int y){
  29. if(!x||!y)return;
  30. if(!v[x]||!v[y])return;
  31. if(F(x)!=F(y))res--,f[f[x]]=f[y];
  32. }
  33. inline void wake(int x,int y){
  34. v[id[x][y]]=1;
  35. res++;
  36. merge(id[x][y],id[x-1][y]);
  37. merge(id[x][y],id[x+1][y]);
  38. merge(id[x][y],id[x][y-1]);
  39. merge(id[x][y],id[x][y+1]);
  40. }
  41. int main(){
  42. scanf("%d%d%d",&n,&m,&q);
  43. for(i=1;i<=q;i++){
  44. gao();
  45. e[++ce]=P(0,i);
  46. }
  47. for(i=1;i<=n;i++)for(j=1;j<=m;j++){
  48. id[i][j]=++cnt;
  49. f[cnt]=cnt;
  50. }
  51. for(i=1;i<=n;i++)for(j=1;j<=m;j++)if(!black[i][j])wake(i,j);
  52. for(i=ce;i;i--){
  53. if(e[i].x)wake(e[i].x,e[i].y);
  54. else ans[e[i].y]=res;
  55. }
  56. for(i=1;i<=q;i++)printf("%d\n",ans[i]);
  57. }

  

B. Bless You Autocorrect!

将字典和询问串都插入Trie中,建好图然后BFS即可。

  1. #include<cstdio>
  2. #include<cstring>
  3. #include<algorithm>
  4. using namespace std;
  5. typedef long long ll;
  6. const int N=1100010;
  7. int n,m,i,j,x,pos[111111],ask[111111];
  8. int tot=1,f[N],id[N],son[N][26];
  9. int d[N],q[N],h,t;
  10. char s[N];
  11. inline int ins0(int p){
  12. scanf("%s",s);
  13. int len=strlen(s);
  14. int x=1,i=0,w;
  15. for(i=0;i<len;i++){
  16. w=s[i]-'a';
  17. if(!son[x][w]){
  18. son[x][w]=++tot;
  19. f[tot]=x;
  20. id[tot]=p;
  21. }
  22. x=son[x][w];
  23. }
  24. return x;
  25. }
  26. inline int ins1(){
  27. scanf("%s",s);
  28. int len=strlen(s);
  29. int x=1,i=0,w;
  30. for(i=0;i<len;i++){
  31. w=s[i]-'a';
  32. if(!son[x][w]){
  33. son[x][w]=++tot;
  34. f[tot]=x;
  35. }
  36. x=son[x][w];
  37. }
  38. return x;
  39. }
  40. inline void ext(int x,int y){
  41. if(!x||d[x])return;
  42. d[x]=y;
  43. q[++t]=x;
  44. }
  45. int main(){
  46. scanf("%d%d",&n,&m);
  47. for(i=1;i<=n;i++){
  48. pos[i]=ins0(i);
  49. }
  50. for(i=1;i<=m;i++){
  51. ask[i]=ins1();
  52. }
  53. for(i=1;i<=tot;i++)id[i]=pos[id[i]];
  54. h=1,t=0;
  55. ext(1,1);
  56. while(h<=t){
  57. x=q[h++];
  58. ext(f[x],d[x]+1);
  59. ext(id[x],d[x]+1);
  60. for(i=0;i<26;i++)ext(son[x][i],d[x]+1);
  61. }
  62. for(i=1;i<=m;i++)printf("%d\n",d[ask[i]]-1);
  63. }

  

C. Card Hand Sorting

枚举花色的顺序以及升降序,那么此时最小移动次数$=n-LIS$。

  1. #include<bits/stdc++.h>
  2. using namespace std;
  3. typedef pair<int,int>pi;
  4. int n;
  5. string hs="shdc";
  6. pi a[77],b[77];
  7. int rev[333];
  8. int idx[66];
  9. int p[10];
  10. int done[66];
  11. int cmp(int x,int y){
  12. return b[x]<b[y];
  13. }
  14. int dp[555];
  15. int solve(int mask){
  16. for(int i=0;i<n;i++){
  17. b[i]=a[i];
  18. if(mask>>b[i].first&1){
  19. b[i].second=16-b[i].second;
  20. }
  21. b[i].first=p[b[i].first];
  22. }
  23. for(int i=0;i<n;i++)idx[i]=i;
  24. sort(idx,idx+n,cmp);
  25. int ret=0;
  26.  
  27. for(int i=0;i<n;i++){
  28. dp[i]=0;
  29. for(int j=0;j<i;j++)if(idx[j]<idx[i])dp[i]=max(dp[i],dp[j]);
  30. dp[i]++;
  31. ret=max(ret,dp[i]);
  32. }
  33.  
  34. /*for(int i=0;i<n;i++)done[i]=0;
  35. for(int i=0,cur=0;i<n;i++){
  36. while(cur<n&&done[cur])cur++;
  37. if(cur==idx[i]){
  38. cur++;
  39. }
  40. else{
  41. ret++;
  42. }
  43. done[idx[i]]=1;
  44. }*/
  45. return n-ret;
  46. }
  47. int getnum(char c){
  48. if(c>='2'&&c<='9')return c-'0';
  49. if(c=='T')return 10;
  50. if(c=='J')return 11;
  51. if(c=='Q')return 12;
  52. if(c=='K')return 13;
  53. return 14;
  54. }
  55. int main(){
  56. for(int i=0;i<4;i++)rev[hs[i]]=i;
  57. while(scanf("%d",&n)!=EOF){
  58. for(int i=0;i<n;i++){
  59. char ss[10];
  60. scanf("%s",ss);
  61. a[i]=pi(rev[ss[1]],getnum(ss[0]));
  62. }
  63. int ans=1e9;
  64. for(int mask=0;mask<1<<4;mask++){
  65. for(int i=0;i<4;i++)p[i]=i;
  66. do{
  67. ans=min(ans,solve(mask));
  68. }while(next_permutation(p,p+4));
  69. }
  70. printf("%d\n",ans);
  71. }
  72. }

  

D. Daydreaming Stockbroker

设$f[i][j]$表示第$i$天结束时有$j$个商品,手上最多有多少钱,要么什么也不做,要么全部卖掉,要么全部买入。

  1. #include<cstdio>
  2. #include<algorithm>
  3. using namespace std;
  4. typedef long long ll;
  5. const int N=100010;
  6. const ll inf=1LL<<60;
  7. int n,m,i,j,x;ll f[N],g[N];
  8. inline void up(ll&a,ll b){if(a<b)a=b;}
  9. int main(){
  10. scanf("%d",&n);
  11. m=100000;
  12. for(i=1;i<=m;i++)f[i]=-inf;
  13. f[0]=100;
  14. while(n--){
  15. scanf("%d",&x);
  16. for(i=0;i<=m;i++)g[i]=f[i];
  17. for(i=0;i<=m;i++)if(f[i]>=0){
  18. //all sell
  19. up(g[0],f[i]+x*i);
  20. //all buy
  21. ll t=min(f[i]/x,100000LL-i);
  22. up(g[i+t],f[i]-x*t);
  23. }
  24. for(i=0;i<=m;i++)f[i]=g[i];
  25. }
  26. printf("%lld",f[0]);
  27. }

  

E. Exponial

根据欧拉定理迭代计算即可。

  1. #include<cstdio>
  2. #include<vector>
  3. using namespace std;
  4. const int Maxn=1000020;
  5.  
  6. int powmod(int x,int y,int mod){
  7. int ret=1%mod;
  8. while(y){
  9. if(y&1){
  10. ret=1LL*ret*x%mod;
  11. }
  12. y>>=1;
  13. x=1LL*x*x%mod;
  14. }
  15. return ret;
  16. }
  17. int rel(int x){
  18. if(x==4)return powmod(4,9,10000000);
  19. if(x==3)return 9;
  20. if(x==2)return 2;
  21. return 1;
  22. }
  23. bool isp[Maxn];
  24. vector<int>pri;
  25. void pre(){
  26. for(int i=2;i<Maxn;i++){
  27. if(!isp[i])pri.push_back(i);
  28. for(int j=0;j<pri.size();j++){
  29. if(1LL*pri[j]*i>=Maxn)break;
  30. isp[pri[j]*i]=1;
  31. if(i%pri[j]==0)break;
  32. }
  33. }
  34. }
  35. int phi(int x){
  36. if(x==1)return 1;
  37. int cur=x;
  38. for(int i=0;i<pri.size()&&1LL*pri[i]*pri[i]<=cur;i++){
  39. if(cur%pri[i]==0){
  40. // printf("p=%d\n",pri[i]);
  41. x=x/pri[i]*(pri[i]-1);
  42. while(cur%pri[i]==0)cur/=pri[i];
  43. }
  44. }
  45. if(cur>1)x=x/cur*(cur-1);
  46. return x;
  47. }
  48. int solve(int n,int m){
  49. if(n==1)return 1%m;
  50. if(m==1)return 0;
  51. if(n<=5){
  52. return powmod(n,rel(n-1),m);
  53. }
  54. int tmp=phi(m);
  55.  
  56. // printf("m=%d phi=%d\n",m,tmp);
  57. return powmod(n,solve(n-1,tmp)+tmp,m);
  58. }
  59. int main(){
  60. pre();
  61. int n,m;
  62. while(scanf("%d%d",&n,&m)!=EOF){
  63. printf("%d\n",solve(n,m));
  64. }
  65. }

  

F. Fleecing the Raffle

从小到大枚举作弊的票数,一旦发现解变劣则退出。

  1. #include<bits/stdc++.h>
  2. using namespace std;
  3. typedef pair<int,int>pi;
  4.  
  5. int n , p ;
  6.  
  7. int main () {
  8. scanf ( "%d%d" , &n , &p ) ;
  9. double ans = 1.0 * p / ( n + 1 ) ;
  10. for ( int i = 2 ; ; ++ i ) {
  11. double tmp = ans * i / ( i - 1 ) * ( n - p + i ) / ( n + i ) ;
  12. if ( tmp < ans ) break ;
  13. ans = tmp ;
  14. }
  15. printf ( "%.10f\n" , ans ) ;
  16. return 0 ;
  17. }

  

G. Game Rank

按题意模拟即可。

  1. #include<bits/stdc++.h>
  2. using namespace std;
  3. typedef pair<int,int>pi;
  4. char s[20000];
  5. int tot[111];
  6. int main(){
  7. for(int i=1;i<=10;i++){
  8. tot[i]=5;
  9. }
  10. for(int i=11;i<=15;i++){
  11. tot[i]=4;
  12. }
  13.  
  14. for(int i=16;i<=20;i++){
  15. tot[i]=3;
  16. }
  17.  
  18. for(int i=21;i<=25;i++){
  19. tot[i]=2;
  20. }
  21. while(scanf("%s",s)!=EOF){
  22. int level=25;
  23. int star=0;
  24. int lx=0;
  25. for(int i=0;s[i];i++){
  26. if(level==0)continue;
  27. if(s[i]=='W'){
  28. lx++;
  29. int bonus=0;
  30. if(level>=6&&level<=25&&lx>=3)bonus=1;
  31.  
  32. star++;
  33. if(star>tot[level]){
  34. star=1;
  35. level--;
  36. }
  37. if(bonus&&level>0){
  38. star++;
  39. if(star>tot[level]){
  40. star=1;
  41. level--;
  42. }
  43. }
  44. }
  45. else{
  46. lx=0;
  47. if(((level!=20)||(star!=0))&&level<=20){
  48. star--;
  49. if(star<0){
  50. level++;
  51. star=tot[level]-1;
  52. }
  53. }
  54. }
  55. }
  56. if(level==0)puts("Legend");
  57. else
  58. printf("%d\n",level);
  59. }
  60. }

  

H. Highest Tower

等价于给每个矩形确定一个独一无二的底边长,最大化高的和。

对于每个矩形$(a,b)$,在$a-b$之间建一条边,若方向是$a->b$则代表底边是$a$,高是$b$。

那么一组可行解中每个点最多只有一条出边。

考虑每个连通块,首先每个点会贡献$(deg[i]-1)\times val[i]$,其次若这个连通块是棵树,那么选取$val$最大的点作为根可以额外得到$val$的收益。

时间复杂度$O(n\log n)$。

  1. #include<cstdio>
  2. #include<algorithm>
  3. #include<map>
  4. using namespace std;
  5. const int N=500010,inf=~0U>>1;
  6. int n,m,i,x,y;
  7. int val[N],d[N],g[N],v[N],nxt[N],ed,ma,sum;bool vis[N];
  8. long long ans;
  9. map<int,int>idx;
  10. inline void add(int x,int y){d[x]++;v[++ed]=y;nxt[ed]=g[x];g[x]=ed;}
  11. void dfs(int x){
  12. if(vis[x])return;
  13. vis[x]=1;
  14. ma=max(ma,val[x]);
  15. ans+=1LL*val[x]*(d[x]-1);
  16. sum+=d[x]-2;
  17. for(int i=g[x];i;i=nxt[i])dfs(v[i]);
  18. }
  19. int main(){
  20. scanf("%d",&n);
  21. for(i=1;i<=n;i++){
  22. scanf("%d%d",&x,&y);
  23. if(!idx[x]){
  24. idx[x]=++m;
  25. val[m]=x;
  26. }
  27. if(!idx[y]){
  28. idx[y]=++m;
  29. val[m]=y;
  30. }
  31. x=idx[x];
  32. y=idx[y];
  33. add(x,y);
  34. add(y,x);
  35. }
  36. for(i=1;i<=m;i++)if(!vis[i]){
  37. ma=sum=0;
  38. dfs(i);
  39. if(sum<0)ans+=ma;
  40. }
  41. printf("%lld",ans);
  42. }

  

I. Interception

留坑。

J. Jumbled Compass

按题意模拟即可。

  1. #include<bits/stdc++.h>
  2. using namespace std;
  3. int main(){
  4. int a,b;
  5. while(scanf("%d%d",&a,&b)!=EOF){
  6. int t1=b-a;
  7. if(t1<0)t1+=360;
  8.  
  9. int t2=a-b;
  10. if(t2<0)t2+=360;
  11. if(t1<=t2){
  12. printf("%d\n",t1);
  13. }
  14. else{
  15. printf("%d\n",-t2);
  16. }
  17. }
  18. }

K. Keeping the Dogs Apart

按照到达转折点的时刻将时间分为$O(n+m)$段区间,在每段中用$B$的速度减去$A$的速度,然后求$A$到线段$B$的最短距离即可。

时间复杂度$O(n)$。

  1. #include<cstdio>
  2. #include<cmath>
  3. #include<algorithm>
  4. using namespace std;
  5. const int N=100010;
  6. int n,m,i;
  7. double ans;
  8. const double eps=1e-9;
  9. struct P{
  10. double x,y;
  11. P(){}
  12. P(double _x,double _y){x=_x,y=_y;}
  13. P operator+(const P&b){return P(x+b.x,y+b.y);}
  14. P operator-(const P&b){return P(x-b.x,y-b.y);}
  15. double len(){return sqrt(x*x+y*y);}
  16. void read(){
  17. scanf("%lf%lf",&x,&y);
  18. }
  19. P operator*(double b){return P(x*b,y*b);}
  20. P operator/(double b){return P(x/b,y/b);}
  21. double operator*(const P&b){return x*b.x+y*b.y;}
  22. }a[N],b[N];
  23. inline int sgn(double x){
  24. if(x>eps)return 1;
  25. if(x<-eps)return -1;
  26. return 0;
  27. }
  28. inline double dis(P a,P b){return (a-b).len();}
  29. inline double cross(P a,P b){return a.x*b.y-a.y*b.x;}
  30. inline double ask(P p,P a,P b){
  31. if((b-a).len()>eps&&sgn((p-a)*(b-a))>=0&&sgn((p-b)*(a-b))>=0)
  32. return fabs(cross(p-a,b-a)/(b-a).len());
  33. return min((p-a).len(),(p-b).len());
  34. }
  35. inline void cal(P A,P B,P C,P D,double z){//z is distance
  36. if(z<eps)return;
  37. P va=B-A;
  38. va=va/va.len();
  39. P vb=D-C;
  40. vb=vb/vb.len();
  41. vb=vb-va;
  42. ans=min(ans,ask(A,C,C+(vb*z)));
  43. }
  44. P lerp(P a,P b,double t){return a*(1.0-t)+b*t;}
  45. void work(){
  46. int i=2,j=2;//nxt
  47. P A=a[1],B=b[1];
  48. while(i<=n&&j<=m){
  49. ans=min(ans,dis(A,B));
  50. double x=dis(A,a[i]),y=dis(B,b[j]);
  51. cal(A,a[i],B,b[j],min(x,y));
  52. if(sgn(x-y)==0){
  53. A=a[i++];
  54. B=b[j++];
  55. continue;
  56. }
  57. if(x<y){
  58. A=a[i++];
  59. B=lerp(B,b[j],x/y);
  60. }else{
  61. B=b[j++];
  62. A=lerp(A,a[i],y/x);
  63. }
  64. }
  65. }
  66. int main(){
  67. scanf("%d",&n);
  68. for(i=1;i<=n;i++)a[i].read();
  69. scanf("%d",&m);
  70. for(i=1;i<=m;i++)b[i].read();
  71. ans=dis(a[1],b[1]);
  72. work();
  73. printf("%.13f",ans);
  74. }

  

Urozero Autumn 2016. NCPC 2016的更多相关文章

  1. 【计算几何】【预处理】【枚举】Urozero Autumn Training Camp 2016 Day 5: NWERC-2016 Problem K. Kiwi Trees

    发现由于角的度数和边的长度有限制,那俩圆如果放得下的话,必然是塞在两个角里. 于是预处理n个圆心的位置(注意要判断那个圆会不会和其他的边界相交),然后n^2枚举俩角即可. #include<cs ...

  2. 【枚举】【SPFA】Urozero Autumn Training Camp 2016 Day 5: NWERC-2016 Problem I. Iron and Coal

    那个人派出的队伍的行走的路径一定前半程是重合的,后半程分叉开来. 于是预处理每个点离1号点的最短路,到最近的铁的最短路,到最近的煤的最短路.(三次BFS / SPFA)然后枚举分岔点,尝试更新答案即可 ...

  3. 【二分】Urozero Autumn Training Camp 2016 Day 5: NWERC-2016 Problem C. Careful Ascent

    二分Vx即可. #include<cstdio> #include<algorithm> using namespace std; #define EPS 0.00000000 ...

  4. 【强连通分量缩点】【DFS】【动态规划】Urozero Autumn Training Camp 2016 Day 5: NWERC-2016 Problem B. British Menu

    有向图,不经过重复点的最长链,强连通分量大小不超过5. 每个强连通分量内部暴力预处理任意两对点之间的最长路,外面DAG上dp. 不是很好写,但是预处理完了之后,可以重构每个强连通分量内部的结构,然后整 ...

  5. PHPStorm 2016.2 - 2016.3许可证服务器

    最快,最安全的选择,以激活您的PHPStorm 2016.2 - 2016.3,这是足够的激活服务器,软件将自动激活.该过程将不断更新,如果不工作评价写入,如果有,以激活没有列出的服务器也可以说. 通 ...

  6. Windows Server 2016 + SCO 2016 安装及配置介绍

    Windows Server 2016 + SCO 2016 安装及配置介绍 高文龙关注1人评论6332人阅读2017-02-26 23:23:02 Windows Server 2016 + SCO ...

  7. Urozero Autumn 2016. BAPC 2016

    A. Airport Logistics 根据光路最快原理以及斯涅尔定律,可以得到从定点$P$进入某条直线的最佳入射角. 求出每个端点到每条线段的最佳点,建图求最短路即可. 时间复杂度$O(n^2\l ...

  8. Urozero Autumn 2016. UKIEPC 2016

    B. Build a Boat 首先求出每块船舱的面积$S$,然后进行$m$次二分,得到每个切割线的位置. 为了计算某个切割线形成的区域的面积,需要将多边形整理成上边界和下边界,分别二分出断点位置,中 ...

  9. Game Rank(NCPC 2016 大模拟)

    题目: The gaming company Sandstorm is developing an online two player game. You have been asked to imp ...

随机推荐

  1. 使用mysqlbinlog对主库binlog进行同步

    #!/bin/bash BASEDIR="/usr/local/mysql" BIN="$BASEDIR/bin" MYSQLBINLOG="$BIN ...

  2. python14 1.带参装饰器 | wrapper 了了解 # 2.迭代器 ***** # 可迭代对象 # 迭代器对象 # for迭代器 # 枚举对象

    ## 复习 '''函数的嵌套定义:在函数内部定义另一个函数 闭包:被嵌套的函数 -- 1.外层通过形参给内层函数传参 -- 2.验证执行 开放封闭原则: 功能可以拓展,但源代码与调用方式都不可以改变 ...

  3. 2017-12-19python全栈9期第四天第二节之列表的增删查改之公共方法len和count和index

    #!/user/bin/python# -*- coding:utf-8 -*-li = ['zs','ls','ww','zl','xx']l = len(li) #总数print(l)num = ...

  4. org.hibernate.ObjectNotFoundException: No row with the given identifier exists解决办法

    hibernate-取消关联外键引用数据丢失抛异常的设置@NotFound hibernate项目里面配了很多many-to-one的关联,后台在查询数据时已经作了健全性判断,但还是经常抛出对象找不到 ...

  5. 在鼠标右键上加入使用notepad++编辑【转】

    我们在安装完notepad++文本编辑器之后,在一个文本文件上右键有时候并没有出现“使用notepad++编辑的选项”,我们可以通过简单地修改注册表文件来增加这样的功能: 1.  首先打开注册表,wi ...

  6. 如何解决failed to push some refs to git

    $ git push -u origin master To git@github.com:yangchao0718/cocos2d.git ! [rejected]        master -& ...

  7. Windows系统盘符错乱导致桌面无法加载。

    问题如下 : 同事有台笔记本更换SSD硬盘,IT职员帮他将新硬盘分好区后再将系统完整Ghost过来,然后装到笔记本上.理论上直接就可以使用了!但结果开机后登陆用户桌面无法显示,屏幕黑屏什么都没有. 问 ...

  8. MarkDown 的两种页内跳转方法!!!!!

    页面内跳转就是点击某个文本,能够跳转到页面里指定的其他地方,经常用于目录中. 第一种是利用Html5 比如点击Feature, 跳转到features中 MarkDown: [Feature](#1) ...

  9. Flink学习(二)Flink中的时间

    摘自Apache Flink官网 最早的streaming 架构是storm的lambda架构 分为三个layer batch layer serving layer speed layer 一.在s ...

  10. vue之生命周期函数例子

    执行代码看生命周期函数的执行顺序 <!-- 根组件 --> <!-- vue的模板内,所有内容要被一个根节点包含起来 App.vue --> <template> ...