zoukankan      html  css  js  c++  java
  • [HNOI2019]多边形

    https://www.luogu.org/problemnew/show/P5288

    题解

    非常有意思的一道题。

    首先观察可得最终的不能继续操作的状态一定是所有边都连向(n)的。

    我们还可以发现最优的操作一定是依次将每条边调整为连向(n)的。

    所以最小的操作次数就是(n-3-)连向(n)的边的条数。

    我们假设现在拿着一个多边形,我们把n这个点作为这个多边形的顶端,那么每条边都会有一个管辖区间就是它下面的点的集合。

    这些区间只会有包含关系,我们可以把它整成一棵树。

    于是我们发现,最优情况下每次操作一定是将一个根进行操作。

    那么如果要求合法操作序列的话就把这颗树搞出来,(dfs)一遍用插板法就能算出来了。

    对于修改,我们可以手动模拟一下,发现这个和(splay)(rotate)差不多,那么就按照(rotate)的过程把贡献算一下就好了。

    代码

    #include<bits/stdc++.h>
    #define N 200009
    #define mm make_pair
    #define ls ch[cnt][0]
    #define rs ch[cnt][1]
    using namespace std;
    typedef long long ll;
    const int mod=1e9+7;
    int ans1,n,W,size[N],ch[N][2],fa[N],rt[N],tot;
    ll ans2,jie[N],ni[N];
    map<pair<int,int>,int>mp;
    vector<int>vec[N];
    inline ll rd(){
      ll x=0;char c=getchar();bool f=0;
      while(!isdigit(c)){if(c=='-')f=1;c=getchar();}
      while(isdigit(c)){x=(x<<1)+(x<<3)+(c^48);c=getchar();}
      return f?-x:x;
    }
    inline ll power(ll x,ll y){
      ll ans=1;
      while(y){if(y&1)ans=ans*x%mod;x=x*x%mod;y>>=1;}
      return ans;
    }
    inline ll C(int n,int m){return jie[n]*ni[m]%mod*ni[n-m]%mod;}
    inline ll _C(int n,int m){return ni[n]*jie[m]%mod*jie[n-m]%mod;}
    inline ll calc(int x,int y){return C(x+y,y);}
    inline ll _calc(int x,int y){return _C(x+y,y);}
    inline bool ge(int x){return ch[fa[x]][1]==x;}
    inline void build(int &cnt,int l,int r,int _fa){
        if(r-l<=1)return;
        cnt=++tot;
        fa[cnt]=_fa;
        size[cnt]=1;
        mp[mm(l,r)]=cnt;
        int p=*lower_bound(vec[r].begin(),vec[r].end(),l+1);
        build(ch[cnt][0],l,p,cnt);build(ch[cnt][1],p,r,cnt);
        size[cnt]+=size[ls]+size[rs];
        ans2=ans2*calc(size[ls],size[rs])%mod;
    }
    inline void print(){
      if(!W){printf("%d
    ",ans1);}
      else printf("%d %lld
    ",ans1,ans2);
    }
    inline void print(int ans1,ll ans2){
      if(!W){printf("%d
    ",ans1);}
      else printf("%d %lld
    ",ans1,ans2);
    }
    inline void yu(int n){
      jie[0]=1;
      for(int i=1;i<=n;++i)jie[i]=jie[i-1]*i%mod;
      ni[n]=power(jie[n],mod-2);
      for(int i=n-1;i>=0;--i)ni[i]=ni[i+1]*(i+1)%mod;
    }
    int main(){
      W=rd();
      n=rd();
      yu(2*n);
      ans1=n-3;
      int x,y;
      for(int i=1;i<=n-3;++i){
        x=rd();y=rd();
        if(x>y)swap(x,y);
        if(y==n)ans1--;
        vec[x].push_back(y);
        vec[y].push_back(x);
      }
      ans2=1;
      for(int i=1;i<n;++i)vec[i].push_back(i+1),vec[i+1].push_back(i);
      vec[1].push_back(n);vec[n].push_back(1);
      for(int i=1;i<=n;++i)sort(vec[i].begin(),vec[i].end());
      int sum=0;
      for(int i=0;i<vec[n].size()-1;++i){
        build(rt[i],vec[n][i],vec[n][i+1],0);
        ans2=ans2*calc(size[rt[i]],sum)%mod;
        sum+=size[rt[i]];
      }
      print();
      int m=rd();
      while(m--){
         x=rd();y=rd();
         if(x>y)swap(x,y);
         int id=mp[mm(x,y)];
         int _ans1=ans1;
         if(!fa[id])_ans1--;
         ll _ans2=ans2;
         if(fa[id]){
            int f=fa[id],o=ge(id);
            _ans2=_ans2*_calc(size[ch[id][0]],size[ch[id][1]])%mod;
            _ans2=_ans2*_calc(size[ch[f][0]],size[ch[f][1]])%mod;
            _ans2=_ans2*calc(size[ch[id][o^1]],size[ch[f][o^1]])%mod;
            _ans2=_ans2*calc(size[ch[id][o]],size[ch[id][o^1]]+size[ch[f][o^1]]+1)%mod;
         }
         else{
            _ans2=_ans2*_calc(size[ch[id][0]],size[ch[id][1]])%mod;
            _ans2=_ans2*_calc(sum-size[id],size[id])%mod;
            _ans2=_ans2*calc(sum-size[id],size[ch[id][0]])%mod;
            _ans2=_ans2*calc(sum-size[id]+size[ch[id][0]],size[ch[id][1]])%mod;
         }
         print(_ans1,_ans2);
      }
      return 0;
    }
    
  • 相关阅读:
    Head First Servlets & JSP 学习笔记 第七章 —— 作为JSP
    Head First Servlets & JSP 学习笔记 第六章 —— 会话状态
    八大基本排序--基数排序
    Stack类常用api
    3.从尾到头打印链表
    八大基本排序--归并排序
    八大基本排序--选择排序
    八大基本排序--希尔排序
    八大基本排序--插入排序
    八大基本排序--冒泡排序
  • 原文地址:https://www.cnblogs.com/ZH-comld/p/10909785.html
Copyright © 2011-2022 走看看