给你一系列的正整数, 例如 1, 4 , 100, 1000000000, 如果利用线段树求解的话,很明显
会导致内存的耗尽。所以我们做一个映射关系,将范围很大的数据映射到范围很小的数据上
1---->1 4----->2 100----->3 1000000000----->4
int tree[M*16];//一共有M*2个端点,一个线段映射到四个点,左右端点, 左端点-1, 右端点+1, 数组的大小是线段树最底层数据个数的4倍
int insert(int p, int lr, int rr, int ld, int rd){
if(tree[p] && lr<=ld && rd<=rr)//如果当前的区间[ld, rd]被包含在[lr, rr]中,而且[lr, rr]的区间已经被覆盖
else if(lr==ld && rr==rd){
f1=insert(p<<1, lr, mid, ld, rd);
f2=insert(p<<1|1, mid+1, rr, ld, rd);
f3=insert(p<<1, lr, mid, ld, mid);
f4=insert(p<<1|1, mid+1, rr, mid+1, rd);
tree[p]=tree[p<<1] && tree[p<<1|1];//两个子树都被覆盖的时候父类才会被覆盖
scanf("%d%d", &edge[i].ld, &edge[i].rd);
maxR=unique(p, p+maxR)-p;//元素去重
for(i=0, nm=0; i<maxR; ++i){
memset(tree, 0, sizeof(tree));//初始值是所有的点都没有被覆盖
for(i=n-1; i>=0; --i){//由外向里看真是个不错的主意
if(!insert(1, 1, nm, hash[edge[i].ld], hash[edge[i].rd]))