[HNOI2019]多边形
注意:n边形里共有n-3条边
最优步数=不与n相连的边数,关键是方案数.
按照处理顺序可以转化为树形结构即二叉树森林,转移方案数用组合数即可
关键是快速处理修改.
1.最优解减少一步,即删掉某棵二叉树的根,合并它的两个儿子.
2.相当于在splay中把它rotate一下,而且不知道为什么它还一定是左儿子
#include<cstdio>
#include<iostream>
#include<cstring>
#include<algorithm>
#include<vector>
#include<map>
#include<cassert>
#define debug(...) fprintf(stderr,__VA_ARGS__)
#define Debug(x) cout<<#x<<"="<<x<<endl
using namespace std;
typedef long long LL;
typedef pair<int,int> pii;
const int INF=1e9+7;
inline LL read(){
register LL x=0,f=1;register char c=getchar();
while(c<48||c>57){if(c=='-')f=-1;c=getchar();}
while(c>=48&&c<=57)x=(x<<3)+(x<<1)+(c&15),c=getchar();
return f*x;
}
const int N=100005;
const int mod=1e9+7;
int size[N],ls[N],rs[N],fa[N],fac[N],ifac[N],inv[N];
int W,n,m,sum,ans=1,Ecnt;
vector <int> E[N];
map <pii,int> M;
inline int add(int x,int y){x+=y;return x>=mod?x-mod:x;}
inline int mul(LL x,int y){x*=y;return x>=mod?x%mod:x;}
inline int cal(int x,int y){return mul(fac[x+y],mul(ifac[x],ifac[y]));}
inline int ical(int x,int y){return mul(ifac[x+y],mul(fac[x],fac[y]));}
inline void dfs(int &x,int l,int r,int pre){
if(l+1==r) return;
x=++Ecnt;///
M[pii(l,r)]=x,fa[x]=pre;
int mid=*upper_bound(E[r].begin(),E[r].end(),l);
dfs(ls[x],l,mid,x);
dfs(rs[x],mid,r,x);
size[x]=size[ls[x]]+size[rs[x]]+1;
ans=mul(ans,cal(size[ls[x]],size[rs[x]]));
}
inline void answer(int x,int y){
if(W==0) printf("%d\n",x);
else printf("%d %d\n",x,y);
}
int main(){
W=read(),n=read();
inv[0]=inv[1]=fac[0]=fac[1]=ifac[0]=ifac[1]=1;
for(int i=2;i<=n;i++) inv[i]=mul(inv[mod%i],mod-mod/i);///
for(int i=2;i<=n;i++) fac[i]=mul(fac[i-1],i),ifac[i]=mul(ifac[i-1],inv[i]);
for(int i=1;i<=n-3;i++){
int x=read(),y=read();
E[x].push_back(y);E[y].push_back(x);
}
for(int i=1;i<n;i++) E[i].push_back(i+1),E[i+1].push_back(i);
E[1].push_back(n),E[n].push_back(1);
for(int i=1;i<=n;i++) sort(E[i].begin(),E[i].end());
for(int i=0;i<(E[n].size()-1);i++){
int tmp=0;
dfs(tmp,E[n][i],E[n][i+1],0);
ans=mul(ans,cal(sum,size[tmp]));
sum+=size[tmp];
}
answer(sum,ans);
m=read();
for(int i=1;i<=m;i++){
int a=read(),b=read(),x=M[pii(a,b)];
int qans=ans,qsum=sum;//-(a==n||b==n);
if(!fa[x]){
//assert(qsum==sum-1);
qsum--;
qans=mul(qans,ical(size[ls[x]],size[rs[x]]));
qans=mul(qans,ical(sum-size[x],size[x]));
qans=mul(qans,cal(sum-size[x],size[ls[x]]));
qans=mul(qans,cal(sum-size[x]+size[ls[x]],size[rs[x]]));
}
else{
int y=fa[x],k=(rs[y]==x);
qans=mul(qans,ical(size[ls[x]],size[rs[x]]));
qans=mul(qans,ical(size[ls[y]],size[rs[y]]));
if(k==0){
qans=mul(qans,cal(size[rs[x]],size[rs[y]]));
qans=mul(qans,cal(size[ls[x]],size[rs[x]]+size[rs[y]]+1));
}
if(k==1){
assert(false);
qans=mul(qans,cal(size[ls[y]],size[ls[x]]));
qans=mul(qans,cal(size[rs[x]],size[ls[y]]+size[ls[x]]+1));
}
}
answer(qsum,qans);
}
}
[HNOI2019]多边形的更多相关文章
- 【BZOJ5491】[HNOI2019]多边形(模拟,组合计数)
[HNOI2019]多边形(模拟,组合计数) 题面 洛谷 题解 突然特别想骂人,本来我考场现切了的,结果WA了几个点,刚刚拿代码一看有个地方忘记取模了. 首先发现终止态一定是所有点都向\(n\)连边( ...
- HNOI2019 多边形 polygon
HNOI2019 多边形 polygon https://www.luogu.org/problemnew/show/P5288 这题镪啊... 首先堆结论: 显然终止状态一定是所有边都连向n了 根据 ...
- luogu P5288 [HNOI2019]多边形
传送门 这是什么神仙操作... 首先要注意一些性质.首先每一个\((x,n)\)的边可以把当前多边形分成两半,这两半的操作是独立的.然后对于某一个没有\((x,n)\)的边的多边形,最优操作是唯一的. ...
- 【洛谷5288】[HNOI2019] 多边形(二叉树模型)
点此看题面 大致题意: 给你一个多边形,用若干不重合.不相交的线段将其划分为若干三角形区域,并定义旋转操作\((a,c)\)为选定\(4\)个点\(a,b,c,d\)满足\(a<b<c&l ...
- [HNOI2019]多边形[二叉树建模、组合计数]
题意 题目链接 分析 不难发现终态一定是 \([2,n-2]\) 中的每个点都与 \(n\) 连边. 关于凸多边形的划分问题,可以将它看作一棵二叉树:每个树点可以看做点可以看做边. 本题中看做点来处理 ...
- 【题解】Luogu P5288 [HNOI2019]多边形
原题传送门 HN的题目就是毒瘤 我们有以下猜想: 1.最后所有的线都连到了n号点上 2.最小步数应该为n-3-已经连到n号点的线段数量 本来有些边\((a_i,n)\)会将整个图分割成很多个区间.对于 ...
- ZJOI2019Day2余姚中学游记(4.23~4.26)
前言 \(Day2\),又是一场噩梦. 前段时间去做了挺多十二省联考和\(HNOI2019\)的题目,还订正掉了\(Day1\)的\(T1\)和\(T2\)(\(T3\)动态\(DP\)完全不想订正啊 ...
- Loj #3056. 「HNOI2019」多边形
Loj #3056. 「HNOI2019」多边形 小 R 与小 W 在玩游戏. 他们有一个边数为 \(n\) 的凸多边形,其顶点沿逆时针方向标号依次为 \(1,2,3, \ldots , n\).最开 ...
- 【loj3056】【hnoi2019】多边形
题目 描述 给出一个 \(n\) 个点的多边形初始的三角剖分: 一次合法的旋转定义为 \((a,b,c,d)\) ,满足 \(a<b<c<d\) : 并且存在边\((a, ...
随机推荐
- null与not null
.not null CREATE TABLE `test` ( `col1` VARCHAR( ) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NUL ...
- [Selenium] 最大化或自定义浏览器的大小
driver.manage().window().maximize(); //将浏览器设置为最大化的状态 driver.manage().window().setSize(new Dimens ...
- dns记录类型(转)
NS:(Domain Name System,域名系统),因特网上作为域名和IP地址相互映射的一个分布式数据库,能够使用户更方便的访问互联网,而不用去记住能够被机器直接读取的IP数串.通过主机名,最终 ...
- HDU 1569 方格取数(2) (最小割)
方格取数(2) Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Su ...
- struct 和union的区别
union ( 共用体):构造数据类型,也叫联合体 用途:使几个不同类型的变量共占一段内存(相互覆盖) struct ( 结构体 ):是一种构造类型 用途: 把不同的数据组合成一个整体——自定义数据 ...
- mongodb-win32-i386-3.0.6 使用
一.下载地址 https://fastdl.mongodb.org/win32/mongodb-win32-i386-3.0.6.zip 二.安装 1. systeminfo OS 名称: Micro ...
- Android-应用安装/替换/卸载/广播监听
在上一篇博客Android-开关机的广播,中介绍了,如何订阅接收者,去接收系统发送的开机/关机广播, 而这篇博客是订阅接收者 去接收应用的(安装/替换/卸载) 三种广播 订阅 接收者 去接收 应用的 ...
- Android-sdcard广播的接收处理
有时候Android手机在开机成功后的那几秒会在状态栏通知,Sdcard开始扫描,Sdcard扫描完成,等信息 当Sdcard的状态发生改变后,系统会自动的发出广播 Sdcard的状态: 1.moun ...
- Android-自定义开关(ViewGroup版)
虽然实现自定义开关,通常情况下都是继承View,比较合理方便快捷一些 但是我今天想去继承ViewGroup来实现自定义开关来玩玩 效果图: 布局代码: <!-- 自定义开关ViewGroup版 ...
- kubernetes 滚动更新发布及回滚
基本命令 记录历史 --record kubectl apply -f **** --record 查看当前状态 kubectl rollout status deployment/demo -w ...