NOIP 模拟 六十九

0+30+40+90=160

T1 取石子

考试扔了将近两个小时,最后也没有回忆起博弈论的相关内容。。

现在只会50pts。正解待补。

#include<bits/stdc++.h>
using namespace std;
int n,a[500001],ans;
signed main()
{   freopen("stone.in","r",stdin);
    freopen("stone.out","w",stdout);
    scanf("%d",&n);
    for(int i=1;i<=n;++i)scanf("%d",&a[i]);
    for(int i=1;i<=n;++i)
    {   ans=0;
        for(int j=1;j<=n;++j) 
        {   int tmp=a[j];
            ans^=tmp%(i+1);

        }
        if(ans)printf("Alice ");
        else printf("Bob ");
    }
}

T2 大鱼吃小鱼

时间有限,快速拿了暴力30,然后走人。

其实线段树都码好了。在暴跳的过程中我认为就是暴跳,并没有分析出最多跳 \(log\) 次,于是放弃了线段树。

改题时长时间困顿于吃完鱼节点的复原,其实节点不超过 \(log^2\) 个,于是暴力修改即可。

要增强分析时间复杂度的能力。。

回归正题,一个明显的贪心是吃自己能吃的最大的鱼,找到第一个比自己体积大于等于的鱼(吃不了的),开吃,最终体积要先超过它,然后继续跳,直到吃不下了。因为每吃两次鱼体积要翻一倍,所以跳的次数是 \(log\) 的。

线段树二分加上暴跳,复杂度 \(O(nlog^2n)\)

#include<bits/stdc++.h>
#define int long long
#define N 1000500
using namespace std;
int n,q,w[N],san[N<<3],tot,ans,zhi,now,ne,tong1[N<<3];
bool bo=1,vis[N<<3],vvv=1;
struct jj
{int opt,s,k,w;}Q[N];
multiset<int>st;
struct tt{int sum,siz,tag;}tmp[N<<3];
struct zxb
{int sum[N<<3],siz[N<<3],tag[N<<3];
inline void pushup(int x)
{   if(!vis[x] and vvv){tong1[++zhi]=x,tmp[zhi]=(tt){sum[x],siz[x],tag[x]},vis[x]=1;}
    sum[x]=sum[x<<1]+sum[x<<1|1];
    siz[x]=siz[x<<1]+siz[x<<1|1];
    return;
}
inline void pushdown(int x)
{
if(!vis[x<<1]){tong1[++zhi]=x<<1,tmp[zhi]=(tt){sum[x<<1],siz[x<<1],tag[x<<1]},vis[x<<1]=1;}
if(!vis[x<<1|1]){tong1[++zhi]=x<<1|1,tmp[zhi]=(tt){sum[x<<1|1],siz[x<<1|1],tag[x<<1|1]},vis[x<<1|1]=1;}
sum[x<<1]=sum[x<<1|1]=siz[x<<1]=siz[x<<1|1]=0;tag[x<<1]=tag[x<<1|1]=1;tag[x]=0;
}
inline void ins(int x,int l,int r,int pos,int val)
{   if(l==r)
    {   siz[x]+=val;
        sum[x]+=san[l]*val;
        return;
    }
    int mid=(l+r)>>1;
    if(mid<pos)ins(x<<1|1,mid+1,r,pos,val);
    else ins(x<<1,l,mid,pos,val);
    pushup(x);
}
inline int query(int x,int l,int r,int L,int R)
{   if(L>R)return 0;
    if(l>=L and r<=R)return sum[x];
    int mid=(l+r)>>1,res=0;
    if(tag[x])pushdown(x);
    if(mid<R and sum[x<<1|1])res+=query(x<<1|1,mid+1,r,L,R);
    if(mid>=L and sum[x<<1])res+=query(x<<1,l,mid,L,R);
    return res;
}
inline void calc(int x,int l,int r,int L,int R)
{   if(l==r and sum[x]>=ne)
    {   if(!vis[x])
        {tong1[++zhi]=x,tmp[zhi]=(tt){sum[x],siz[x],tag[x]},vis[x]=1;}
        siz[x]-=(ne/san[l])+(ne%san[l]!=0);
        
        ans+=(ne/san[l])+(ne%san[l]!=0);
        now+=((ne/san[l])+(ne%san[l]!=0))*san[l];ne-=((ne/san[l])+(ne%san[l]!=0))*san[l];
        bo=0;
        sum[x]=siz[x]*san[l];
        return;
    }
    if(r<=R)
    {   if(sum[x]<=ne)
        {   ne-=sum[x];now+=sum[x];
            if(ne<=0)bo=1;ans+=siz[x];
            if(!vis[x])
            {tong1[++zhi]=x,tmp[zhi]=(tt){sum[x],siz[x],tag[x]},vis[x]=1;}
            sum[x]=0;siz[x]=0;tag[x]=1;
            return;
        }
    }
    int mid=(l+r)>>1;
    if(tag[x])pushdown(x);
    if(mid>=R){(void)calc(x<<1,l,mid,L,R);pushup(x);return ;}
    if(bo and sum[x<<1|1])calc(x<<1|1,mid+1,r,L,R);
    if(bo and sum[x<<1])calc(x<<1,l,mid,L,R);
    pushup(x);
}
}tree;
inline void work(int s,int k)
{   int need=k-s,val;now=s;ans=0;zhi=0;
    while(need>0 and now<k)
    {   auto it=st.lower_bound(now);
        if(it==st.end())val=999999999999999999;
        else val=*it; 
        ne=min(need,val-now+1);int lst=ne;
        int pos=lower_bound(san+1,san+1+tot,now)-san;bo=1;--pos;
        if(pos)tree.calc(1,1,tot,1,pos);
        else break;
        if(ne>0)break;if(it==st.end())break;
        need-=(lst-ne);
    }
    if(ne>0)puts("-1");
    else printf("%lld\n",ans);
    for(int i=1;i<=zhi;++i)
    {   int x=tong1[i];
        vis[x]=0;
        tree.sum[x]=tmp[i].sum;
        tree.siz[x]=tmp[i].siz;
        tree.tag[x]=0;
    }
}
signed main()
{   freopen("fish.in","r",stdin);
    freopen("fish.out","w",stdout);
    scanf("%lld",&n);
    for(int i=1;i<=n;++i)scanf("%lld",&w[i]),san[++tot]=w[i],st.insert(w[i]);
    scanf("%lld",&q);
    for(int i=1;i<=q;++i)
    {   scanf("%lld",&Q[i].opt);
        if(Q[i].opt==1)scanf("%lld%lld",&Q[i].s,&Q[i].k),san[++tot]=Q[i].s;
        if(Q[i].opt==2)scanf("%lld",&Q[i].w),san[++tot]=Q[i].w;
        if(Q[i].opt==3)scanf("%lld",&Q[i].w),san[++tot]=Q[i].w;
    }
    sort(san+1,san+1+tot);
    tot=unique(san+1,san+1+tot)-san-1;
    san[++tot]=999999999999999999;
    for(int i=1;i<=n;++i)
    {   int pos=lower_bound(san+1,san+1+tot,w[i])-san;
        vvv=0;
        tree.ins(1,1,tot,pos,1);
    }
    vvv=1;
    for(int i=1;i<=q;++i)
    {   if(Q[i].opt==1)
        {   if(Q[i].s>=Q[i].k){puts("0");continue;}
            work(Q[i].s,Q[i].k);
        }
        if(Q[i].opt==2)
        {   vvv=0;
            int pos=lower_bound(san+1,san+1+tot,Q[i].w)-san;
            tree.ins(1,1,tot,pos,1);st.insert(Q[i].w);
            vvv=1;
        }
        if(Q[i].opt==3)
        {   vvv=0;
            int pos=lower_bound(san+1,san+1+tot,Q[i].w)-san;
            tree.ins(1,1,tot,pos,-1);st.erase(st.find(Q[i].w));
            vvv=1;
        }
    }
}

T3 黑客

我竟然错过了一道水题,难受。。。。。

暴力循环做除法。。复杂度 \(O(999^2)\) 。不算gcd。

#include<bits/stdc++.h>
#define int long long
#define mod 1000000007
using namespace std;
int a,b,c,d,ans,maxn;
signed main()
{   freopen("hacker.in","r",stdin);
    freopen("hacker.out","w",stdout);
    scanf("%lld%lld%lld%lld",&a,&b,&c,&d);
    for(int i=2;i<=999;++i)
    {   for(int j=1;j<i;++j)
        {   int base1=j,base2=i-j;
            if(__gcd(base1,base2)!=1)continue;
            int a1=a/base1+(a%base1!=0),b1=b/base1;
            int a2=c/base2+(c%base2!=0),b2=d/base2;
            int l=max(a1,a2),r=min(b1,b2);
            if(l>r)continue;
            ans=(ans+i*(r-l+1)%mod)%mod;
        }
    }
    printf("%lld\n",ans);
}

T4 黑客

这是续集,一眼数位 dp,然后开冲,冲完了一发过小样例,爽!

然后打开大样例,测了一下跑的贼 jb 快,信心倍增,然后查看输出结果,。。。。。。。。。

WOC,md 高精,不是 mod 1e9+7 吗???!!!

然后经历了极其漫长的调试过程。。最后 tmd 0 0 没 jb 输出掉了十分!!

#include<bits/stdc++.h>
using namespace std;
pair<int,int>dp[502][1<<9];
int n,m,k,sta[11],tot=1;
bool f[502][1<<9];
int gao[131002+10][978];
long long work[978];
inline void add(int i1,int i2)
{   for(int i=1;i<=gao[i2][0];++i)gao[i1][i]+=gao[i2][i];
    gao[i1][0]=max(gao[i1][0],gao[i2][0]);
    for(int i=1;i<=gao[i1][0];++i)
    {   if(gao[i1][i]>=10)
        {   if(i==gao[i1][0])++gao[i1][0];
            gao[i1][i+1]+=gao[i1][i]/10;
            gao[i1][i]%=10;
        }
    }
}
inline void add1(int i1)
{   for(int i=1;i<=work[0];++i)gao[i1][i]+=work[i];
    gao[i1][0]=max(1ll*gao[i1][0],work[0]);
    for(int i=1;i<=gao[i1][0];++i)
    {   if(gao[i1][i]>=10)
        {   if(i==gao[i1][0])++gao[i1][0];
            gao[i1][i+1]+=gao[i1][i]/10;
            gao[i1][i]%=10;
        }
    }
}
inline void www(int zhi)
{   int tt=work[0];
    for(int i=1;i<=tt;++i)
       work[i]*=zhi;
    for(int i=1;i<=work[0];++i)
    if(work[i]>=10)
    {   if(work[0]==i)++work[0];
        work[i+1]+=work[i]/10;
        work[i]%=10;
    }
}
inline void caonima(int i1,int base)
{   memset(work,0,sizeof(work));
    for(int i=0;i<=gao[i1][0];++i)work[i]=gao[i1][i];
    if(base)for(int i=work[0];i;--i)work[i+base]=work[i],work[i]=0;
    work[0]+=base;
}
inline pair<int,int>dfs(int x,int lim)
{   if(f[x][lim])return dp[x][lim];
    f[x][lim]=1;
    int tmp1=++tot,tmp2=++tot;
    if(x==n+1) return dp[x][lim]=make_pair(1,0);
   
    for(int i=1;i<=k;++i)
    if(!(lim&(1<<i-1)))
    {   pair<int,int>tmp=dfs(x+1,lim|sta[i]);
        add(tmp1,tmp.first);
        add(tmp2,tmp.second);
        caonima(tmp.first,n-x);
        www(i);
        add1(tmp2);
    }
    return dp[x][lim]=make_pair(tmp1,tmp2);
}
signed main()
{   freopen("hacker2.in","r",stdin);
    freopen("hacker2.out","w",stdout);
    scanf("%d%d%d",&n,&m,&k);
    for(int i=1;i<=m;++i)
    {   int a,b;
        scanf("%d%d",&a,&b);
        sta[a]|=(1<<b-1);
    }
    gao[1][0]=1;gao[1][1]=1;
    pair<int,int>ans=dfs(1,0);
    int id1=ans.first,id2=ans.second;
    if(!gao[id1][0])
    {   cout<<0<<endl<<0<<endl;
        return 0;
    }
    for(int i=gao[id1][0];i;--i)cout<<gao[id1][i];cout<<endl;
    for(int i=gao[id2][0];i;--i)cout<<gao[id2][i];cout<<endl;
}
posted @ 2021-10-06 06:24  -zxb-  阅读(62)  评论(0编辑  收藏  举报