NOIP 2014 题解

好吧,15年写14年题解,的确够晚的(P.S. 是因为我到现在才搞懂这些题)

正文之前,先膜拜膜拜一位大爷 %%% _debug

多膜拜膜拜 _debug 大爷,打程序时bug会减少哦

 

传送门:

生活大爆炸版石头剪刀布:  http://codevs.cn/problem/3716/

联合权值:         http://codevs.cn/problem/3728/

飞扬的小鸟:        http://codevs.cn/problem/3729/

无线网络发射器选址:    http://codevs.cn/problem/3578/

寻找道路:         http://codevs.cn/problem/3731/

解方程:          http://codevs.cn/problem/3732/

 

这里讲题的顺序:由易到难

 

Day1 T1 与 Day2 T1

这两题若是不会做,建议你先搜搜A+B problem的题解

直接上代码了

P.S. 代码是pascal的 这是我14年考试时的代码 14年我不会用C (QAQ)

 

Day1 T1

 1 var
 2  n,na,nb,i,j,win1,win2,ls1,ls2:longint;
 3  a,b:array[0..200]of longint;
 4  p:array[0..4,0..4]of 0..2;
 5 begin
 6  readln(n,na,nb);
 7  for i:=1 to na do read(a[i]);
 8  for i:=1 to nb do read(b[i]);
 9  p[0,0]:=0; p[0,1]:=2; p[0,2]:=1; p[0,3]:=1; p[0,4]:=2;
10  p[1,0]:=1; p[1,1]:=0; p[1,2]:=2; p[1,3]:=1; p[1,4]:=2;
11  p[2,0]:=2; p[2,1]:=1; p[2,2]:=0; p[2,3]:=2; p[2,4]:=1;
12  p[3,0]:=2; p[3,1]:=2; p[3,2]:=1; p[3,3]:=0; p[3,4]:=1;
13  p[4,0]:=1; p[4,1]:=1; p[4,2]:=2; p[4,3]:=2; p[4,4]:=0;
14  win1:=0; win2:=0;
15  for i:=1 to n do
16  begin
17   ls1:=a[(i-1) mod na+1];
18   ls2:=b[(i-1) mod nb+1];
19   if p[ls1,ls2]=1 then inc(win1);
20   if p[ls1,ls2]=2 then inc(win2);
21  end;
22  writeln(win1,' ',win2);
23 end.
View Code

Day2 T1

 1 var
 2  i,j,l,n,d,ans,x,y,k,ans1,ans2,max1,max2,left,right,up,down:longint;
 3  sum:array[-1..128,-1..128]of longint;
 4 function min(a,b:longint):longint;
 5 begin
 6  if a<b then exit(a) else exit(b);
 7 end;
 8 function max(a,b:longint):longint;
 9 begin
10  if a>b then exit(a) else exit(b);
11 end;
12 begin
13  readln(d); readln(n); max1:=0; max2:=0;
14  for i:=1 to n do
15  begin
16   readln(x,y,k);
17   max1:=max(x,max1);
18   max2:=max(y,max2);
19   for j:=x to 128 do
20   for l:=y to 128 do
21   inc(sum[j,l],k);
22  end;
23  for i:=0 to min(128,max1+2*d) do
24  for j:=0 to min(128,max2+2*d) do
25  begin
26   up:=max(0,i-d);
27   down:=min(128,i+d);
28   left:=max(0,j-d);
29   right:=min(128,j+d);
30   l:=sum[down,right]-sum[down,left-1]-sum[up-1,right]+sum[up-1,left-1];
31   if l>ans2 then begin ans1:=1; ans2:=l; end
32   else if l=ans2 then inc(ans1);
33  end;
34  writeln(ans1,' ',ans2);
35 end.
View Code

 

Day1 T2 联合权值

记录每个点的儿子节点权值的 和(sum[])、最大值(max1[])、次大值(max2[])

令数组w[]记录每个点的权值

令ans1记录联合权值最大值,ans2记录联合权值和

以1为树的根,全部BFS一遍,在BFS中

对于每一次搜到的点 u

ans1=max(ans1,max1[u]*max2[u])

对于每一次搜到的边 u--->v

ans1=max(ans1,w[u]*max1[v])
ans2+=w[v]*(sum[u]-w[v])
ans2+=w[u]*sum[v]*2

大概就是这样了

代码如下

 1 #include<cstdio>
 2 #include<cstring>
 3 #include<cmath>
 4 #include<algorithm>
 5 using namespace std;
 6 #define LL long long
 7 const int maxn=200000+10;
 8 int n,m,i,j,w[maxn],sum[maxn],max1[maxn],max2[maxn];
 9 int aa[maxn],bb[maxn],ans1,ans2;
10 struct node{
11        int v;
12        node *next;
13 };
14 node *g[maxn],*p;
15 void add(int u,int v)
16 {
17      p=new(node);
18      p->v=v;
19      p->next=g[u];
20      g[u]=p;
21 }
22 int dui[maxn],h,t;
23 int max(int a,int b) {return a>b?a:b;}
24 int min(int a,int b) {return a<b?a:b;}
25 int main()
26 {
27     scanf("%d",&n);
28     for (i=1;i<n;i++)
29     {
30         int a,b;
31         scanf("%d%d",&a,&b);
32         if (a>b) {int t=a;a=b;b=t;}
33         add(a,b);
34         aa[i]=a; bb[i]=b;
35     }
36     for (i=1;i<=n;i++) scanf("%d",&w[i]);
37     for (i=1;i<n;i++)
38     {
39         int a=aa[i],b=bb[i];
40         sum[a]+=w[b];
41         sum[a]%=10007;
42         if (w[b]>=max1[a]) max2[a]=max1[a],max1[a]=w[b];
43         else max2[a]=max(max2[a],w[b]);
44     }
45     h=0; t=1; dui[1]=1;
46     while (h!=t)
47     {
48           h++;
49           int u=dui[h];
50           ans1=max(ans1,max2[u]*max1[u]);
51           p=g[u];
52           while (p!=NULL)
53           {
54                 dui[++t]=p->v;
55                 int v=p->v;
56                 ans1=max(ans1,w[u]*max1[v]);
57                 ans2+=w[v]*(sum[u]-w[v]);
58                 ans2+=w[u]*sum[v]*2;
59                 ans2%=10007;
60                 p=p->next;
61           }
62     }
63     ans2%=10007;
64     printf("%d %d\n",ans1,ans2);
65 }

记得ans2要对10007取模

 

Day2 T2寻找道路

傻B题

思路大概是,先把所有边反向,从终点来一次DFS/BFS,标记所有经过了的点

然后会有一些没标记的点,这些就是走不到终点的

在原图上,把指向走不到终点的点的点删掉(这句话有点绕,多绕绕就会绕了)

再从起点来一遍SPFA

好吧,就这么简单

P.S. 去年在考场上时是想到了这个方法的,可惜邻接表不会打,打了个邻接矩阵,70分 (QAQ)

废话不多说,代码如下

 1 #include<cstdio>
 2 #include<cstring>
 3 #include<cmath>
 4 #include<algorithm>
 5 using namespace std;
 6 #define LL long long
 7 const int maxn=10000+10;
 8 const int INF=10000000;
 9 int n,m,i,j,a[maxn],S,T,dis[maxn];
10 int Q[maxn],h,t;
11 int max(int a,int b) {return a>b?a:b;}
12 int min(int a,int b) {return a<b?a:b;}
13 struct node{
14        int v;
15        node *next;
16 };
17 node *g[maxn],*g2[maxn],*p;
18 bool in[maxn];
19 node *add(int u,int v,node *nd)
20 {
21      node *p;
22      p=new(node);
23      p->v=v;
24      p->next=nd;
25      return p;
26 }
27 void spfa()
28 {
29      int i,j;
30      node *p;
31      for (i=0;i<=n;i++) dis[i]=INF,in[i]=0;
32      h=0; t=1; dis[S]=0; in[S]=0; Q[1]=S;
33      while (h!=t)
34      {
35            h++;
36            int u=Q[h];
37            p=g[u];
38            while (p!=NULL)
39            {
40                  int v=p->v;
41                  if (dis[u]+1<dis[v])
42                  {
43                      dis[v]=dis[u]+1;
44                      if (!in[v]) Q[++t]=v,in[v]=1;
45                  }
46                  p=p->next;
47            }
48            in[u]=0;
49      }
50 }
51 int main()
52 {
53     scanf("%d%d",&n,&m);
54     for (i=1;i<=m;i++)
55     {
56         int a,b;
57         scanf("%d%d",&a,&b);
58         g[a]=add(a,b,g[a]);
59         g2[b]=add(b,a,g2[b]);
60     }
61     scanf("%d%d",&S,&T);
62     h=0; t=1; Q[1]=T; in[T]=1;
63     while (h!=t)
64     {
65           h++;
66           int u=Q[h];
67           p=g2[u];
68           while (p!=NULL)
69           {
70                 if (!in[p->v]) Q[++t]=p->v,in[p->v]=1;
71                 p=p->next;
72           }
73     }
74     if (!in[S]) {printf("-1\n"); return 0;}
75     for (i=1;i<=n;i++)
76     if (!in[i])
77     {
78         p=g2[i];
79         while (p!=NULL)
80         {
81               if (p->v==S) {printf("-1\n"); return 0;}
82               g[p->v]=NULL;
83               p=p->next;
84         }
85     }
86     spfa();
87     if (dis[T]==INF) {printf("-1\n"); return 0;}
88     printf("%d\n",dis[T]);
89 }

 

 

Day1 T3 飞扬的小鸟

P.S. 每次看到“飞扬的小鸟”我就觉得我是手残

大爷一眼就能看出要用DP做,唯我第一想法是暴搜

dp[i][j]——小鸟飞到 第i列 高度为j的地方时,最少点击次数

令数组U[i]记录在第i-1列点击一次上升的高度

数组D[i]记录在第i-1列不点击下降的高度

边界条件:dp[0][0]=INF ; dp[0][j]=0 (j=1...m)

转移方程:

1)上升  dp[i][j]=min(dp[i][j],dp[i][j-U[i]]+1,dp[i-1][j-U[i]]+1);

2)下降  dp[i][j]=min(dp[i][j],dp[i-1][j+D[i]]);

3)j=m要特判

 

这一题更加详细的题解我会在以后写出

请关注我的博客http://www.cnblogs.com/Donnie-Darko/

(P.S. 其实现在我的博客里也没什么东西,上面那句话本身想删了的,可是再想一想又觉得,脸皮还是厚点好)

代码如下

 1 #include<cstdio>
 2 #include<cstring>
 3 #include<cmath>
 4 #include<algorithm>
 5 using namespace std;
 6 #define LL long long
 7 const int maxn=10000+10;
 8 const int maxm=1000+10;
 9 const int INF=100000;
10 int n,m,i,j,k,dp[maxn][maxm],U[maxn],D[maxn],ans=0;
11 bool pass[maxn][maxm],conduit[maxn];
12 int max(int a,int b) {return a>b?a:b;}
13 int min(int a,int b) {return a<b?a:b;}
14 int main()
15 {
16     scanf("%d%d%d",&n,&m,&k);
17     for (i=0;i<=n;i++) pass[i][0]=1;
18     for (i=1;i<=n;i++)
19     scanf("%d%d",&U[i],&D[i]);
20     for (i=1;i<=k;i++)
21     {
22         int a,b,c;
23         scanf("%d%d%d",&a,&b,&c);
24         conduit[a]=1;
25         for (j=0;j<=b;j++) pass[a][j]=1;
26         for (j=c;j<=m;j++) pass[a][j]=1;
27     }
28     dp[0][0]=INF;
29     for (i=1;i<=n;i++)
30     {
31         for (j=0;j<=m;j++) dp[i][j]=INF;
32         for (j=U[i]+1;j<m;j++)
33         {
34             dp[i][j]=min(dp[i][j],dp[i][j-U[i]]+1);
35             dp[i][j]=min(dp[i][j],dp[i-1][j-U[i]]+1);
36         }
37         for (j=0;j<=m-D[i];j++)
38         dp[i][j]=min(dp[i][j],dp[i-1][j+D[i]]);
39         for (int o=1;o<=m;o++)
40         {
41             int p=(m-o)/U[i];
42             if ((m-o)%U[i]!=0||p==0) p++;
43             dp[i][m]=min(dp[i][m],dp[i-1][o]+p);
44         }
45         int minl=INF;
46         for (j=0;j<=m;j++) if (pass[i][j]) dp[i][j]=INF;
47         for (j=0;j<=m;j++) minl=min(minl,dp[i][j]);
48         if (minl==INF) break;
49         if (conduit[i]) ans++;
50     }
51     if (ans<k) printf("0\n%d\n",ans);
52     else
53     {
54         ans=INF;
55         for (i=1;i<=m;i++) ans=min(ans,dp[n][i]);
56         printf("1\n%d\n",ans);
57     }
58 }

 

Day2 T3 解方程

每次看到这种数学题就想弃疗

这题,要用模糊一点的方法做

也就是枚举x,算答案时不停对10007取模,若最后的答案是0,就当成正确答案了(连long long都不要用)

当然,天知道出数据的人会不会整个变态数据卡10007

所以我准备了5个模数{11261,19997,22877,21893,14843},只有答案全为0,才认为是正确答案

你来卡我啊,卡我啊,我看你怎么卡

于是这题就愉快的AC了

等等,细心的同学发现了,这样会超时,唉,上面那句话当我没说好了

大家可以发现,在模p意义下

如果x0不是答案

那么在模p意义下,x0+p不是答案,x0+2p也不是答案......

时间复杂度瞬间下降对不对

 

还是那句话,这一题更加详细的题解我会在以后写出

请关注我的博客http://www.cnblogs.com/Donnie-Darko/

(P.S. 记得电视上的广告脸皮厚到能连播三遍,我说两遍又有什么错)

代码如下

 1 #include<cstdio>
 2 #include<cstring>
 3 #include<cmath>
 4 #include<algorithm>
 5 using namespace std;
 6 char s[10010];
 7 int n,m,i,j,a[5][1000010];
 8 bool ok[5][1000010];
 9 int mod[5]={11261,19997,22877,21893,14843};
10 bool pass(char c){return (('0'<=c&&c<='9')||c=='-');}
11 int ans[1000010];
12 int min(int a,int b) {return a<b?a:b;}
13 bool jud(int x)
14 {
15      for(int t=0;t<5;t++)
16      if(ok[t][x%mod[t]]!=0)return 0;
17      return 1;
18 }
19 void R(int p)
20 {
21      scanf("%s",s+1);
22      int len=strlen(s+1);
23      bool flag=0;
24      for (int t=0;t<5;t++)
25      if(s[1]!='-') a[t][i]=s[1]-'0';
26      else a[t][i]=0,flag=1;
27      for (int t=0;t<5;t++)
28      {
29          for (int k=2;k<=len;k++)
30          a[t][i]=(a[t][i]*10+s[k]-'0')%mod[t];
31          if (flag) a[t][i]=-a[t][i];
32      }
33 }
34 int main()
35 {
36     scanf("%d%d",&n,&m);
37     for (i=0;i<=n;i++) R(i);
38     int pre[5][110],res[5];
39     for (i=0;i<=4;i++) pre[i][0]=1;
40     for (int x=1;x<=min(m,22877-1);x++)
41     {
42         for (j=0;j<=4;j++)
43         for (i=1;i<=n;i++)
44         pre[j][i]=(pre[j][i-1]*x)%mod[j];
45         for (i=0;i<=4;i++) res[i]=0;
46         for (i=0;i<=n;i++)
47         for (j=0;j<=4;j++)
48         if (x<mod[j])
49         res[j]+=(pre[j][i]*a[j][i])%mod[j],res[j]%=mod[j];
50         for (i=0;i<=4;i++) if (res[i]&&x<mod[i]) ok[i][x]=1;
51     }
52     for (i=1;i<=m;i++) if (jud(i)) ans[++ans[0]]=i;
53     printf("%d\n",ans[0]);
54     for (i=1;i<=ans[0];i++) printf("%d\n",ans[i]);
55 }

 

posted @ 2015-08-30 15:41  Donnie_Darko  阅读(229)  评论(0编辑  收藏  举报