8.1 NOIP模拟11

8.1 NOIP模拟 11

今天上午返校之后,颓了一会,然后下午就开始考试,中午睡着了,然后刚开始考试的时候就困的一匹,我一看T1,woc,这不是之前线段树专题的题啊,和那道题差不多,所以我......想起来了,我当时没有做,完了,这就是之前坑哇的太大的缘故。我的内心当时就在滴血,心想推正解还不如先打个暴力,然后愉快的5分钟吗了一个暴力,整个就是俩sort,又花了五分钟改成了桶排,然后就愉快的交上去了,然后我T1还是没有思路,就15分钟的时候转了T2,T2一看就开始推组合数的柿子,但是由于我的误判成数学题,就完犊子了,那还等啥,一看20%的数据,就果断摔了一个dfs上去,然后就转T3,(此时内心虚的一批),T3一看,先想了一个啥都没有的暴力,然后又想到了前缀和优化,就优化一下,自己有造了几个极限数据,发现20%和40%应该都能过(当时心花怒放,但是其实60是不存在的),然后开考50分钟,打完了三个暴力,就歇菜了,剩下的时间就开始刚正解,最后的提交我也是挂着暴力交上去的,所以最后我的分数并没有因为我的覆盖而降低,但是也没有因为我的“正解”而分数增高,刚考完我就知道我这次凉凉,只打了3个暴力,肯定又垫底了,但是竟然还有暴力都没打对的大佬然后,我就和13个同学并列rank7,100分,虽然rank7,但是这是在别人失误的情况下得的,所以如果别人都没失误,那我这次就又垫底了,所以这次还是失败的考试。

T1 string

这这,一看就是之前学长讲过的那道用线段树维护的题,好像还带了二分,但是仔细一看略有不同,之前的那个是指询问地q个数,而这个是询问整个结果,所以我就(缨缨婴)只能打暴力,但是好像全场都没几个打正解的。在下面改题的过程中我就想到了一种奇爬的方法,就是维护一个每个字母出现次数的前缀和,然后通过一系列操作实现对字符串的变化,但是时间复杂度是$ O(n*m*26) $交上去T40,和我的暴力跑的时间一样,然后我就想怎么优化,有这么一种思路,就是我把前缀和的图像画在了之上,然后发现他是单调上升函数(这不是废话吗!),那么我就想到使用链表吧图像的拐点存下来,这样就可以知道了一些信息,而且可以 $ O(1) $计算出一些信息,但是需要 $ O(m*26) $的复杂度进行转移,所以应该比正解跑的还要快,但是男就难在了这个思路不好实现,所以我还是老老实实的打了线段树,最后还是没有忍住话了一点时间打了一下,发现确实很难条,我就暂且哇个坑,有神犇愿意为我条一下的也可以,后来我看cbx的T1快到飞起,其实他的思路和我的这个思路是一样的,就不赘述了;

我的T40暴力代码:

 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cstring>
 4 #include<cmath>
 5 #include<cstdlib>
 6 #include<algorithm>
 7 using namespace std;
 8 int read()
 9 {
10     int x=0,f=1;char cc;cc=getchar();
11     while(cc>'9'||cc<'0'){if(cc=='-')f=-1;cc=getchar();}
12     while(cc>='0'&&cc<='9'){x=(x<<3)+(x<<1)+cc-'0';cc=getchar();}
13     return x*f;
14 }
15 char s[100005];
16 int n,m,l,r,x;
17 inline bool cmp1(char a,char b){return a<b;}
18 inline bool cmp2(char a,char b){return a>b;}
19 int main()
20 {
21     //freopen("cnm.txt","r",stdin);
22     n=read();m=read();
23     scanf("%s",s+1);
24     for(int i=1;i<=m;i++)
25     {
26         l=read(),r=read(),x=read();
27         if(x==1)sort(s+l,s+r+1,cmp1);
28         else sort(s+l,s+r+1,cmp2);
29     }
30     for(int i=1;i<=n;i++)
31     printf("%c",s[i]);
32     puts("");
33     return 0;
34 }
T40

我的AC线段树代码:

  1 #include<cstdio>
  2 #include<cstring>
  3 #include<iostream>
  4 #include<cstdlib>
  5 #include<algorithm>
  6 const int maxn=100005;
  7 using namespace std;
  8 #define re register
  9 #define debug(x) cout<<x<<" debug!"<<endl;
 10 inline int read()
 11 {
 12     re int x=0,f=1;char cc;cc=getchar();
 13     while(cc>'9'||cc<'0'){if(cc=='-')f=-1;cc=getchar();}
 14     while(cc>='0'&&cc<='9'){x=(x<<3)+(x<<1)+cc-'0';cc=getchar();}
 15     return x*f;
 16 }
 17 int m,n,k,t,sum[27];
 18 char s[maxn];
 19 struct tree{    
 20     int l,r,vv,lev;    
 21 }tr[maxn<<2];
 22 inline void pushup(int x)
 23 {
 24     if(tr[x<<1].vv==tr[x<<1|1].vv) 
 25         tr[x].vv=tr[x<<1].vv;
 26     else tr[x].vv=-1;
 27 }
 28 inline void pushdown(int x)
 29 {
 30     if(tr[x].vv!=-1) 
 31         tr[x<<1].vv=tr[x<<1|1].vv=tr[x].vv;
 32     else tr[x].vv=-1;
 33 }
 34 inline void build(int x,int l,int r)
 35 {
 36     tr[x].l=l, tr[x].r=r; 
 37         tr[x].vv=-1;
 38     if(tr[x].l==tr[x].r)
 39     {    
 40         tr[x].vv=s[l]-'a'; 
 41         return;    
 42     }
 43     int mid=(tr[x].l+tr[x].r)>>1;
 44     build(x<<1,l,mid); build(x<<1|1,mid+1,r);
 45     pushup(x);
 46 }    
 47 inline void query(int x)
 48 {
 49     if(tr[x].vv!=-1)
 50     {
 51         for(register int i=1;i<=tr[x].r-tr[x].l+1;++i)
 52             putchar(tr[x].vv+'a');
 53            return;
 54     }
 55     query(x<<1);
 56     query(x<<1|1);
 57 }    
 58 inline void make(int x,int l,int r)
 59 {
 60     if(l<=tr[x].l&&tr[x].r<=r)
 61     {
 62         if(tr[x].vv!=-1)
 63         {
 64             sum[tr[x].vv]+=tr[x].r-tr[x].l+1; 
 65             return;
 66          }
 67     }
 68     pushdown(x);
 69     int mid=(tr[x].l+tr[x].r)>>1;
 70     if(l<=mid)make(x<<1,l,r);
 71     if(mid+1<=r)make(x<<1|1,l,r);
 72 }
 73 inline void change(int x,int l,int r,int vv)
 74 {
 75     if(l>r) return;
 76     if(l<=tr[x].l&&tr[x].r<=r)
 77     {    
 78         tr[x].vv=vv; 
 79         return;    
 80     } 
 81     int mid=(tr[x].l+tr[x].r)>>1;
 82     if(l<=mid)change(x<<1,l,r,vv);
 83     if(mid+1<=r)change(x<<1|1,l,r,vv);
 84     pushup(x);
 85 }
 86 int main()
 87 {
 88     //freopen("1.txt","r",stdin);
 89     n=read(),m=read();
 90     scanf("%s",s+1); 
 91     build(1,1,n);
 92     while(m--)
 93     {
 94         memset(sum,0,sizeof(sum));
 95         int l,r,x;
 96         scanf("%d%d%d",&l,&r,&x);
 97         make(1,l,r);
 98         if(x==1)
 99         {
100             int L=l;
101             for(register int i=0;i<26;++i)
102             {
103                 change(1,L,L+sum[i]-1,i);
104                 L+=sum[i];
105             }
106         }
107         if(x==0)
108         {
109             int L=l;
110             for(register int i=25;i>=0;--i)
111             {
112                 change(1,L,L+sum[i]-1,i);
113                 L+=sum[i];
114             }
115         }
116     }
117     query(1);
118     puts("");
119 }
AC

还有cbx的代码就不放了,本人懒得征求版权了!

T2 Matrix

这到题是本人感觉正常最难的题,虽然看起来像组合数学(而且我也是按组合数学打的,但是并不是)

 这是题解,其实就是一个贼神仙的dp,由于时间限制以及任务繁多,本人扔下代码就跑:

 1 #include<iostream>
 2 #include<cstdio>
 3 using namespace std;
 4 #define LL long long
 5 #define re register
 6 inline LL read()
 7 {
 8     re LL x=0,f=1;char cc;cc=getchar();
 9     while(cc>'9'||cc<'0'){if(cc=='-')f=-1;cc=getchar();}
10     while(cc>='0'&&cc<='9'){x=(x<<3)+(x<<1)+cc-'0';cc=getchar();}
11     return x*f;
12 }
13 const LL p=998244353;
14 LL n,m,dp[3005][3005],pw[3005],inv[3005],ipw[3005],l[3005],r[3005];
15 int main()
16 {
17     n=read(),m=read();
18     for(re LL i=1;i<=n;++i)++l[read()],++r[read()];
19     for(re LL i=1;i<=m;++i)l[i]+=l[i-1],r[i]+=r[i-1];
20     dp[0][0]=1;
21     for(re LL i=1;i<=m;++i)
22     {    
23         dp[i][0]=dp[i-1][0];
24         for(re LL j=1;j<=r[i];++j)
25             dp[i][j]=(dp[i-1][j]+dp[i-1][j-1]*(r[i]-j+1)%p)%p;
26         for(re LL j=l[i-1];j<l[i];++j)
27             for(re LL k=0;k<=i-j;++k) 
28                 dp[i][k]=dp[i][k]*(i-k-j)%p;
29     }
30     printf("%lld
",dp[m][n]);
31 }
T2

T3 big

本人也是先哇下坑,先沽了;

 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cstring>
 4 using namespace std;
 5 #define LL long long
 6 #define re register
 7 LL ch[4000005][2],tot;
 8 LL num[10000005],n,m;
 9 LL ans,cnt,sum,pre[1000005],x;
10 LL head[300005],nxt[300005],ver[300005],tot2,vvt[300005];
11 const int pc=29987;
12 inline LL read()
13 {
14     re LL x=0,f=1;char cc;cc=getchar();
15     while(cc>'9'||cc<'0'){if(cc=='-')f=-1;cc=getchar();}
16     while(cc>='0'&&cc<='9'){x=(x<<3)+(x<<1)+cc-'0';cc=getchar();}
17     return x*f;
18 }
19 void inser(LL x)
20 {
21     LL h=x%pc;
22     for(int i=head[h];i;i=nxt[i])
23         if(ver[i]==x)++vvt[i];
24     ver[++tot2]=x;
25     nxt[tot2]=head[h];
26     head[h]=tot2;
27     vvt[tot2]++;
28 }
29 LL query(LL x)
30 {
31     LL h=x%pc;
32     LL op=0;
33     for(int i=head[h];i;i=nxt[i])
34         if(ver[i]==x)
35             op=vvt[i];
36     return op;
37 }
38 LL insert(LL x)
39 {
40      LL u=0,wc=0,xx=x;
41      for(register int i=n-1;i>=0;--i)
42      {
43          wc=(xx&(1<<i))?1:0;
44          if(!ch[u][wc]) 
45              ch[u][wc]=++tot;
46          u=ch[u][wc];
47      }
48 }
49 inline void dfs(int x,int dep,int sum)
50 {
51      if(dep==-1)
52      {
53          if(sum>=ans) ans=sum,inser(ans);
54          return;
55      }
56      if(!ch[x][0]) dfs(ch[x][1],dep-1,sum^(1<<dep));
57      if(!ch[x][1]) dfs(ch[x][0],dep-1,sum^(1<<dep));
58      if(ch[x][0]&&ch[x][1])
59      {
60          dfs(ch[x][0],dep-1,sum);
61          dfs(ch[x][1],dep-1,sum);
62      }
63  }
64 int main()
65 {
66      n=read(),m=read();
67      for(register int i=1;i<=m;++i) 
68          x=read(),pre[i]=pre[i-1]^x;    
69      for(register int i=0,x;i<=m;++i)
70      {
71          x=((2*pre[i])/(1<<n)+2*pre[i])%(1<<n);
72          insert(x^pre[i]^pre[m]);
73      }
74      dfs(0,n-1,0);
75      LL uu=query(ans);
76      printf("%lld
%lld
",ans,uu);
77 }
T3
原文地址:https://www.cnblogs.com/hzoi-lsc/p/11287851.html