[SCOI2016]萌萌哒

题目描述

一个长度为n的大数,用S1S2S3...Sn表示,其中Si表示数的第i位,S1是数的最高位,告诉你一些限制条件,每个条件表示为四个数,l1,r1,l2,r2,即两个长度相同的区间,表示子串Sl1Sl1+1Sl1+2...Sr1与Sl2Sl2+1Sl2+2...Sr2完全相同。

比如n=6时,某限制条件l1=1,r1=3,l2=4,r2=6,那么123123,351351均满足条件,但是12012,131141不满足条件,前者数的长度不为6,后者第二位与第五位不同。问满足以上所有条件的数有多少个。

输入输出格式

输入格式:

第一行两个数n和m,分别表示大数的长度,以及限制条件的个数。

接下来m行,对于第i行,有4个数li1,ri1,li2,ri2,分别表示该限制条件对应的两个区间。1<=n<=10^5,1<=m<=10^5,1<=li1,ri1,li2,ri2<=n;并且保证ri1-li1=ri2-li2。

输出格式:

一个数,表示满足所有条件且长度为n的大数的个数,答案可能很大,因此输出答案模10^9+7的结果即可。

输入输出样例

输入样例#1: 复制
4 2
1 2 3 4
3 3 3 3
输出样例#1: 复制
90
用并查集维护连通关系
设logn个并查集
第i个并查集的j点与k点连通表示[j~j+2^i]=[k~k+2^i]
于是就可以从后往前,把第i个并查集的信息传到第i-1个
设第0个并查集有cnt个连通块
那么答案就是$9*10^{cnt-1}$
 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cstring>
 4 #include<algorithm>
 5 #include<cmath>
 6 using namespace std;
 7 int set[25][100011],ans,Mod=1e9+7,n,m,Log[100011],cnt;
 8 int find(int t,int x)
 9 {
10   if (set[t][x]!=x) set[t][x]=find(t,set[t][x]);
11   return set[t][x];
12 }
13 void merge(int t,int x,int y)
14 {
15   int p=find(t,x);
16   int q=find(t,y);
17   if (p!=q)
18     {
19       set[t][p]=q;
20     }
21 }
22 int main()
23 {int i,j,l1,l2,r1,r2,len;
24   cin>>n>>m;
25   for (i=2;i<=n;i++)
26     {
27       Log[i]=Log[i/2]+1;
28     }
29   for (i=0;i<=Log[n];i++)
30     for (j=1;j<=n;j++)
31       set[i][j]=j;
32   for (i=1;i<=m;i++)
33     {
34       scanf("%d%d%d%d",&l1,&r1,&l2,&r2);
35       len=Log[r1-l1+1];
36       merge(len,l1,l2);
37       merge(len,r1-(1<<len)+1,r2-(1<<len)+1);
38     }
39   for (i=Log[n];i>=1;i--)
40     {
41       for (j=1;j+(1<<i)-1<=n;j++)
42     {
43       int k=find(i,j);
44       if (j==k) continue;
45       merge(i-1,j,k);
46       merge(i-1,j+(1<<i-1),k+(1<<i-1));
47     }
48     }
49   ans=9;
50   for (i=1;i<=n;i++)
51     if (find(0,i)==i) cnt++;
52   for (i=1;i<cnt;i++)
53     ans=1ll*10*ans%Mod;
54   cout<<ans;
55 }

 

转载于:https://www.cnblogs.com/Y-E-T-I/p/8969786.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
中描述了一个幼儿园里分配糖果的问题,每个小朋友都有自己的要求。问题的输入包括两个整数NN和KK,表示幼儿园里的小朋友数量和要满足的要求数量。接下来的KK行表示小朋友们的要求,每行有三个数字,XX,AA,BB。如果X=1,表示第AA个小朋友分到的糖果必须和第BB个小朋友分到的糖果一样多;如果X=2,表示第AA个小朋友分到的糖果必须少于第BB个小朋友分到的糖果;如果X=3,表示第AA个小朋友分到的糖果必须不少于第BB个小朋友分到的糖果;如果X=4,表示第AA个小朋友分到的糖果必须多于第BB个小朋友分到的糖果;如果X=5,表示第AA个小朋友分到的糖果必须不多于第BB个小朋友分到的糖果。这个问题可以被看作是一个差分约束系统的问题。 具体地说,可以使用差分约束系统来解决这个问题。差分约束系统是一种通过给变量之间的关系添加约束来求解最优解的方法。对于这个问题,我们需要根据小朋友们的要求建立约束条件,并通过解决这个约束系统来得出最小的糖果数量。 在问题的输入中,X的取值范围为1到5,分别对应不同的关系约束。根据这些约束,我们可以构建一个差分约束图。图中的节点表示小朋友,边表示糖果数量的关系。根据不同的X值,我们可以添加相应的边和权重。然后,我们可以使用SPFA算法(Shortest Path Faster Algorithm)来求解这个差分约束系统,找到满足所有约束的最小糖果数量。 需要注意的是,在读取输入时需要判断X和Y是否合法,即是否满足X≠Y。如果X=Y,则直接输出-1,因为这种情况下无法满足约束条件。 综上所述,为了满足每个小朋友的要求,并且满足所有的约束条件,我们可以使用差分约束系统和SPFA算法来求解这个问题。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [【差分约束系统】【SCOI2011】糖果 candy](https://blog.csdn.net/jiangzh7/article/details/8872699)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *2* [P3275 [SCOI2011]糖果(差分约束板子)](https://blog.csdn.net/qq_40619297/article/details/88678605)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值