用並查集合並相同的點,答案爲9*10^{並查集的塊數-1}。c++
由「區間對區間的」能夠聯想到線段樹優化之類的方法,換成倍增會更簡單~優化
#include <bits/stdc++.h> #define ll long long using namespace std; const int N=1e5+10; const int mod=1e9+7; int n,m,L[N],f[20][N]; int find(int i,int x) {return f[i][x]==x?x:f[i][x]=find(i,f[i][x]);} void merge(int i,int x,int y) {f[i][find(i,x)]=find(i,y);} int main() { scanf("%d%d",&n,&m); for(int i=2; i<=n; ++i) L[i]=L[i>>1]+1; for(int i=0; i<=L[n]; ++i) for(int j=1; j<=n; ++j) f[i][j]=j; for(int l1,r1,l2,r2; m--; ) { scanf("%d%d%d%d",&l1,&r1,&l2,&r2); int d=L[r1-l1+1]; merge(d,l1,l2); merge(d,r1-(1<<d)+1,r2-(1<<d)+1); } for(int i=L[n]; i; --i) for(int j=1; j+(1<<i)-1<=n; ++j) { int k=find(i,j); if(k==j) continue; merge(i-1,j,k); merge(i-1,j+(1<<(i-1)),k+(1<<(i-1))); } ll ans=9,tot=0; for(int i=1; i<=n; ++i) if(find(0,i)==i) tot++; for(int i=1; i<tot; ++i) ans=ans*10%mod; printf("%lld\n",ans); return 0; }