zoukankan      html  css  js  c++  java
  • 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
  • 相关阅读:
    双线性过滤
    textureView
    cubemap
    selfshadow
    cbuffer padding
    异常
    Python深浅拷贝
    数据类型分类
    集合类型内置方法
    字典数据类型内置方法
  • 原文地址:https://www.cnblogs.com/hzoi-lsc/p/11287851.html
Copyright © 2011-2022 走看看