题目分享H 二代目
题意:有m个限制,每个限制l1,r1,l2,r2四个数,限制了一个长度为n的数第l1到r1位要与第l2到r2相同,保证r1-l1=r2-l2,求在限制下一共有多少种数
分析:
暴力的话肯定是从l1-r1扫一遍用并查集,但显然时间和空间都是不允许的
但再一想,这是不是相当于区间并?操作
看到区间的东西,我直接就往线段树去想了
#include<cstdio>
#include<map>
#include<algorithm>
using namespace std; #define pii pair<int,int>
#define ll long long const int mod=1e9+; map<pii,pii> fa;
int n; pii find(pii x)
{
while(x!=fa[x]) x=fa[x];
return x;
} pii ya(pii x)
{
pii fax=find(x),nowfa;
while(x!=fax)
{
nowfa=fa[x];
fa[x]=fax;
x=nowfa;
}
return fax;
} void bing(pii x,pii y)
{
pii fax=ya(x),fay=ya(y);
fa[fay]=fax;
} void down(int l,int r)
{
int mid=l+r>>;
int p=ya(make_pair(l,r)).first-l;
bing(make_pair(l,mid),make_pair(l+p,mid+p));
bing(make_pair(mid+,r),make_pair(mid++p,r+p));
} void dfs(int l,int r,int left,int right,int p)
{
if(l>=left&&r<=right)
{
bing(make_pair(l,r),make_pair(l+p,r+p));
return;
}
down(l,r);
int mid=l+r>>;
if(left<=mid) dfs(l,mid,left,right,p);
if(mid>right) dfs(mid+,r,left,right,p);
} void mem(int l,int r)
{
fa[make_pair(l,r)]=make_pair(l,r);
if(l==r) return;
int mid=l+r>>;
mem(l,mid);mem(mid+,r);
} void get_ans(int l,int r)
{
if(l==r) return;
down(l,r);
int mid=l+r>>;
get_ans(l,mid);get_ans(mid+,r);
} ll pow(int x,int y)
{
ll now=(ll)x,ans=1ll;
while(y)
{
if(y&) ans*=now,ans%=(ll)mod;
y>>=;now*=now,now%=(ll)mod;
}
return ans;
} int main()
{
int m,l1,r1,l2,r2;
scanf("%d%d",&n,&m);
mem(,n);
while(m--)
{
scanf("%d%d%d%d",&l1,&r1,&l2,&r2);
dfs(,n,l1,r1,l2-l1);
}
get_ans(,n);
int ans=-;
for(int i=;i<=n;i++) if(fa[make_pair(i,i)]==make_pair(i,i)) ans++;
printf("%d",(int)(pow(,ans)*%mod));
return ;
}
然鹅,这个pair可能有点不好开,导致全员MLE
不过处理区间的东西,ST表也是有地位的
于是就能往ST表那里去想
首先ST表传统艺能肯定是f[x][k]表示x到x+(1<<k)左闭右开怎么怎么样,
那么这里就应该表示x到x+(1<<k)左闭右开的父亲是谁(因为要搞并查集)
而这里的父亲应该对应另一个y到y+(1<<k)的区间,因为我们的k是已知的,所以其实f[x][k]只需要记录这个y即可,也就是对应区间的左端点
而这m组将两个区间并起来的操作时,也应该像lca向上跳一样,从maxlg往下慢慢找符合的k,然后再讲x跳到其对应的x+(1<<k)位置上
这里有一点值得注意,其实我们这里也可以按照平日ST表那样只将l到l+(1<<k)与 r-(1<<k)+1,r+1这两个区间与对应的区间合并(这里k应该是小于等于r-l的最大的k)
但我认为这样不如像lca那样快,如果有和我意见不相同的可以讨论一波
但你肯定马上就会想,这只合并两个,而lca合并很多个区间,但这些合并其实都要下放的
为了方便统计,最后要干的是将所有的结果下传,达到每个单个的位置对应一个位置的效果,
这又类似线段树了,下传标记
这应该很好理解,ya()后得到的是f[x][y]所对应的根节点区间的左端点,这里bing传的三个参数分别为l1,l2,k,也就是把l1到l1+(1<<k)左闭右开与l2到l2+(1<<k)左闭右开合并
最后得到每个单个位置对应的位置,只需要数数有多少个f[x][0]=x即可(f[x][0]=x表示这个x是这个并查集的根节点)
对了,还有一个小问题,你统计出来的这个值是指能自由取数的个数,但显然这其中一定有一个位置与首位相关,这个位置只能取1-9,别的位置则能取0-9
所以最后结果应该是9*10^(count-1)
代码:
#include<cstdio>
#include<map>
#include<algorithm>
using namespace std; #define ll long long const int mod=1e9+;
const int maxn=1e5+;
const int maxlg=2e1+; int fa[maxn][maxlg];
int n; int find(int x,int y)
{
while(x!=fa[x][y]) x=fa[x][y];
return x;
} int ya(int x,int y)
{
int fax=find(x,y),nowfa;
while(x!=fax)
{
nowfa=fa[x][y];
fa[x][y]=fax;
x=nowfa;
}
return fax;
} void bing(int x1,int x2,int y)
{
int fax=ya(x1,y),fay=ya(x2,y);
fa[fay][y]=fax;
} void down(int x,int y)
{
int nowx=ya(x,y);
bing(x,nowx,y-);
bing(x+(<<y-),nowx+(<<y-),y-);
} void get_ans()
{
for(int i=maxlg-;i;i--) for(int j=;j<=n;j++) if(j+(<<i)<=n+) down(j,i);
} void mem()
{
for(int i=;i<maxlg;i++) for(int j=;j<=n;j++) if(j+(<<i)<=n+) fa[j][i]=j;
} ll pow(int x,int y)
{
ll now=(ll)x,ans=1ll;
while(y)
{
if(y&) ans*=now,ans%=(ll)mod;
y>>=;now*=now,now%=(ll)mod;
}
return ans;
} int main()
{
int m,l1,r1,l2,r2;
scanf("%d%d",&n,&m);
mem();
while(m--)
{
scanf("%d%d%d%d",&l1,&r1,&l2,&r2);
int now=r1-l1+;
for(int i=maxlg-;i>=;i--)
{
if((<<i)<=now)
{
bing(l1,l2,i);now-=(<<i);
l1+=(<<i);l2+=(<<i);
}
}
}
get_ans();
int ans=-;
for(int i=;i<=n;i++) if(fa[i][]==i) ans++;
printf("%d",(int)(pow(,ans)*%mod));
return ;
}
题目分享H 二代目的更多相关文章
- 题目分享E 二代目
题意:一棵点数为n的树,每个节点有点权,要求在树中中找到一个最小的x,使得存在一个点满足max(该点点权,该点相邻的点的点权+1,其他点的点权+2)=x 分析:首先要能把题目转化为上述题意 首先题目让 ...
- 题目分享D 二代目
题意:给定一个T条边的无向图,求S到E恰好经过N条边的最短路径 T≤100 N≤1000000 分析:(据说好像假期学长讲过) 首先很容易想到的是dp[i][j][k]表示从i到j经过k条边的最短路径 ...
- 题目分享C 二代目
题意:一个数列是由 1 1 2 1 2 3 1 2 3 4 1 2 3 4 5 1 2 3 4 5 6.....组成,也就是1-1,1-2,1-3......并且如果遇到多位数也要拆成数字比如1-10 ...
- 题目分享G 二代目
题意:有n组数,每组包含两个数,问在每组只能取一个的前提下能组成的最长的从1开始的连续自然数有几个? 分析:刚学了差分约束系统,很容易往转换成图的方向去想 将他读入的这n组数当成边读入 很容易会得到一 ...
- 题目分享F 二代目
题意:T个点R种双向边,P种单向边,求点S到每个点的最短距离 分析:(这再看不出来是spfa就该**了) 首先,这题能否用spfa就看他是否有负环呗,显然,双向边的权值非负,单向边还有个啥政策,总之显 ...
- 题目分享V
题意:现在两个人做游戏,每个人刚开始都是数字1,谁赢了就能乘以k^2,输的乘以k(k可以是任意整数,每次不一定相同)现在给你最终这两个人的得分,让你判断是否有这个可能,有可能的话Yes,否则No. 分 ...
- 2019年腾讯PHP程序员面试题目分享
有需要学习交流的友人请加入交流群的咱们一起,有问题一起交流,一起进步!前提是你是学技术的.感谢阅读! 点此加入该群jq.qq.com 1. php 的垃圾回收机制 PHP 可以自动进行内存管理,清除 ...
- 20190924-LeetCode解数独题目分享
解决数独 题目描述 编写一个程序,通过已填充的空格来解决数独问题. 一个数独的解法需遵循如下规则: 数字 1-9 在每一行只能出现一次. 数字 1-9 在每一列只能出现一次. 数字 1-9 在每一个以 ...
- 题目分享X
题意:一张票有n位数,如果这张票的前一半数字的和等于后一半数字的和(n一定是偶数),就称这张票为快乐票.有些数被擦除了,标记为’?’(’?‘的个数也是偶数),现在Monocarp 和 Bicarp 进 ...
随机推荐
- MD5中使用16进制
MD5中使用16进制消息摘要 分类: java_secruity2012-12-28 13:11 719人阅读 评论(0) 收藏 举报 消息摘要 由于数据在计算机中的表示,最终以二进制的形式存在,所以 ...
- C#多线程(4):进程同步Mutex类
Mutex 类 构造函数和方法 系统只能运行一个程序的实例 解释一下上面的示例 接替运行 进程同步示例 另外 Mutex 类 Mutex 中文为互斥,Mutex 类叫做互斥锁.它还可用于进程间同步的同 ...
- HTTP 405 的错误提示:消息 JSP 只允许 GET、POST 或 HEAD。Jasper 还允许 OPTIONS 的解决方法
如果项目是运行在 Tomcat 8 及以上,会发现发出的 PUT 请求和 DELETE 请求可以被控制其接收到,但是返回页面时(forward)会报HTTP 405 的错误提示:"消息 JS ...
- 绕过XSS过滤姿势总结
0x01 弹窗关键字检测绕过 基本WAF都针对常用的弹窗函数做了拦截,如alert().prompt().confirm(),另外还有代码执行函数eval(),想要绕过去也比较简单,我们以alert( ...
- R - Cow and Message CodeForces - 1307C
思路对了,但是不会写. 等差数列长度不是1就是2,所以不是一个字母就是俩字母,一开始写的时候直接枚举两个字母,然后让次数相乘.这样是不对的,比如abaabb,字母ab的个数应该是3+2+2,因该是每一 ...
- MySQL系列操作
Linux环境下安装使用MySQL Portal 数据备份&恢复 Portal
- async,await执行流看不懂?看完这篇以后再也不会了
昨天有朋友在公众号发消息说看不懂await,async执行流,其实看不懂太正常了,因为你没经过社会的毒打,没吃过牢饭就不知道自由有多重要,没生过病就不知道健康有多重要,没用过ContinueWith就 ...
- vue2.x学习笔记(二十一)
接着前面的内容:https://www.cnblogs.com/yanggb/p/12632730.html. 可复用性&结合-混入 基础 混入(mixin)提供了一种非常灵活的方式,来分发v ...
- Java讲解RPC的基本实现
RPC远程过程调用可以说是分布式系统的基础,本文将通过Java演示一次普通的rpc调用到底发生了什么. 我曾经在网上看到有人提问,为什么RPC要叫作远程过程调用,而不叫作RMC远程方法调用.个人认为R ...
- OAuth - 四种方式
OAuth 2.0 的标准是 RFC 6749 文件.该文件先解释了 OAuth 是什么. OAuth 引入了一个授权层,用来分离两种不同的角色:客户端和资源所有者.......资源所有者同意以后,资 ...