Problem H:Boring Counting

Time Limit : 6000/3000ms (Java/Other)   Memory Limit : 65535/32768K (Java/Other)

Problem Description

In this problem you are given a number sequence P consisting of N integer and Pi is the ith element in the sequence. Now you task is to answer a list of queries, for each query, please tell us among [L, R], how many Pi is not less than A and not greater than B( L<= i <= R). In other words, your task is to count the number of Pi (L <= i <= R,  A <= Pi <= B).

Input

In the first line there is an integer T (1 < T <= 50), indicates the number of test cases.
        For each case, the first line
contains two numbers N and M (1 <= N, M <= 50000), the size of sequence
P, the number of queries. The second line contains N numbers Pi(1
<= Pi <= 10^9), the number sequence P. Then there are M lines,
each line contains four number L, R, A, B(1 <= L, R <= n, 1 <= A, B
<= 10^9)

Output

For each case, at first output a line
‘Case #c:’, c is the case number start from 1. Then for each query output a
line contains the answer.

Sample Input

1

13
5

6
9 5 2 3 6 8 7 3 2 5 1 4

1
13 1 10

1
13 3 6

3
6 3 6

2
8 2 8

1
9 1 9

Sample Output

Case
#1:

13

7

3

6

9

给你n个数和m条询问,每次查询区间[l,r]中满足A<=x<=B的x的个数。

感觉很不错的一道题,考察主席树的应用。

关于主席树,请参考:http://seter.is-programmer.com/posts/31907.html

建立n棵线段树,线段树root[i]代表区间[1,i]之间一段数字区间出现的次数。

对于每一棵线段树,每个节点表示一个区间[a,b],记录满足a<=x<=b的x的个数。

因为线段树root[i]是在线段树root[i-1]的基础上增加了一个数得到的,所以可以由root[i-1]得到root[i]。

由于每棵线段树的大小形态都是一样的,而且初始值全都是0,那每个线段树都初始化不是太浪费了?所以一开始只要建一棵空树即可。

然后是在某棵树上修改一个数字,由于和其他树相关联,所以不能在原来的树上改,必须弄个新的出来。难道要弄一棵新树?不是的,由于一个数字的更改只影响了一条从这个叶子节点到根的路径,所以只要只有这条路径是新的,另外都没有改变。比如对于某个节点,要往右边走,那么左边那些就不用新建,只要用个指针链到原树的此节点左边就可以了,这个步骤的前提也是线段树的形态一样。

n是数字个数,这个步骤的空间复杂度显然是O(logn)。

所有树节点的空间消耗为:O(n*4+nlogn)

预处理得到所有的root[i]。

查询时在两棵线段树root[R]和root[L-1]中分别查询区间[A,B]中数的个数,相减即是结果。

本题的时间复杂度:建树:O(n)+预处理O(nlogn)+查询O(logn)

 #include <cstdio>
#include <cstring>
#include <cstdlib>
#include <algorithm> using namespace std; #define lson l, m, rt->left
#define rson m + 1, r, rt->right const int MAXN = ;
const int INF = (1e9)+; struct Node
{
int sum; //存储区间[A, B]之间一共有多少个数
Node *left, *right;
}; Node *root[MAXN];
Node ChairTree[ MAXN * ];
Node *idx; int pos[MAXN]; //处于位置i的数排第几
int num[MAXN]; //排好序并去重的原始数,相当于把所有数离散化之后的结果
int p[MAXN]; //排第i的是哪个数
int cnt, n, Q; bool cmp( int a, int b )
{
return pos[a] < pos[b];
} Node *nextNode()
{
idx->sum = ;
idx->left = idx->right = NULL;
return idx++;
} Node *copyNode( Node *ori )
{
idx->sum = ori->sum;
idx->left = ori->left;
idx->right = ori->right;
return idx++;
} void PushUp( Node *rt )
{
rt->sum = rt->left->sum + rt->right->sum;
return;
} void build( int l, int r, Node* rt ) //建立一个空树
{
if ( l == r ) return; int m = ( l + r ) >> ; rt->left = nextNode();
rt->right = nextNode(); build( lson );
build( rson ); return;
} void init()
{
scanf( "%d%d", &n, &Q ); for ( int i = ; i <= n; ++i )
{
scanf( "%d", &pos[i] );
p[i] = i;
}
sort( p + , p + + n, cmp ); int pre = -INF;
cnt = ;
for ( int i = ; i <= n; ++i ) //离散化+去重
{
if ( pos[ p[i] ] == pre )
pos[ p[i] ] = cnt;
else
{
pre = pos[ p[i] ];
num[ ++cnt ] = pos[ p[i] ];
pos[ p[i] ] = cnt;
}
}
return;
} Node *add( int val, int l, int r, Node* rt ) //根据上一棵树构造下一棵树
{
Node *temp = copyNode( rt ); if ( l == r )
{
temp->sum += ;
return temp;
}
int m = ( l + r ) >> ;
if ( val <= m ) temp->left = add( val, lson );
else temp->right = add( val, rson );
PushUp( temp ); return temp;
} int Query( int L, int R, int l, int r, Node* treeL, Node* treeR )
{
if ( L <= l && r <= R )
{
return treeR->sum - treeL->sum;
}
int res = ; int m = ( l + r ) >> ;
if ( L <= m ) res += Query( L, R, l, m, treeL->left, treeR->left );
if ( R > m ) res += Query( L, R, m + , r, treeL->right, treeR->right );
return res;
} void chuli()
{
idx = ChairTree;
root[] = nextNode();
build( , cnt, root[] ); //建立一颗空树 for ( int i = ; i <= n; ++i )
root[i] = add( pos[i], , cnt, root[i - ] ); //根据第i-1棵树构造第i棵树
return;
} int BiSearch1( int tar ) //查询最左边的 >= x 的数
{
int low = , high = cnt;
int mid, ans = -;
while ( low <= high )
{
mid = ( low + high ) >> ;
if ( num[mid] >= tar )
{
ans = mid;
high = mid - ;
}
else low = mid + ;
}
return ans;
} int BiSearch2( int tar ) //查询最右边的 <= x 的数
{
int low = , high = cnt;
int mid, ans = -;
while ( low <= high )
{
mid = ( low + high ) >> ;
if ( num[mid] <= tar )
{
ans = mid;
low = mid + ;
}
else high = mid - ;
}
return ans;
} int main()
{
int T, cas = ;
scanf( "%d", &T );
while ( T-- )
{
init();
chuli(); printf( "Case #%d:\n", ++cas );
while ( Q-- )
{
int l, r, a, b;
scanf("%d%d%d%d", &l, &r, &a, &b );
int u = BiSearch1( a );
int v = BiSearch2( b );
if ( u == - || v == - )
{
puts("");
continue;
}
printf( "%d\n", Query( u, v, , cnt, root[l - ], root[r] ) );
}
}
return ;
}

UPC 2224 / “浪潮杯”山东省第四届ACM大学生程序设计竞赛 1008 Boring Counting 主席树的更多相关文章

  1. “浪潮杯”山东省第五届ACM大学生程序设计竞赛(总结贴)

    第一次參加省赛有点小激动,尽管是作为打星队參赛,但心情却是上下起伏. 5月9号晚上11点多到威海,有点略冷.可是空气比淄博好多了,大家到了旅馆的时候都非常晚了,抱怨了一下三星级的酒店的待遇,喝杯咖啡早 ...

  2. 2012年"浪潮杯"山东省第三届ACM大学生程序设计竞赛--n a^o7 ! 分类: 比赛 2015-06-09 17:16 14人阅读 评论(0) 收藏

    n a^o7 ! Time Limit: 1000ms   Memory limit: 65536K  有疑问?点这里^_^ 题目描述 All brave and intelligent fighte ...

  3. 山东省第四届ACM大学生程序设计竞赛解题报告(部分)

    2013年"浪潮杯"山东省第四届ACM大学生程序设计竞赛排名:http://acm.upc.edu.cn/ranklist/ 一.第J题坑爹大水题,模拟一下就行了 J:Contes ...

  4. Alice and Bob(2013年山东省第四届ACM大学生程序设计竞赛)

    Alice and Bob Time Limit: 1000ms   Memory limit: 65536K 题目描述 Alice and Bob like playing games very m ...

  5. 2013年山东省第四届ACM大学生程序设计竞赛-最后一道大水题:Contest Print Server

    点击打开链接 2226: Contest Print Server Time Limit: 1 Sec  Memory Limit: 128 MB Submit: 53  Solved: 18 [Su ...

  6. sdut Mountain Subsequences 2013年山东省第四届ACM大学生程序设计竞赛

    Mountain Subsequences 题目描述 Coco is a beautiful ACMer girl living in a very beautiful mountain. There ...

  7. 2013年山东省第四届ACM大学生程序设计竞赛J题:Contest Print Server

    题目描述     In ACM/ICPC on-site contests ,3 students share 1 computer,so you can print your source code ...

  8. 2013年山东省第四届ACM大学生程序设计竞赛 Alice and Bob

      Alice and Bob Time Limit: 1000ms   Memory limit: 65536K 题目描述 Alice and Bob like playing games very ...

  9. 2013年山东省第四届ACM大学生程序设计竞赛E题:Alice and Bob

    题目描述 Alice and Bob like playing games very much.Today, they introduce a new game. There is a polynom ...

随机推荐

  1. About Curah

    相信下列场景对您来说一点都不陌生:您遇到一个问题,花了好几个小时在网上搜寻解答和可靠的技术内容.即使前往许多技术博客和论坛翻箱倒柜后,还是无法确定要相信谁,也不知道该选哪个答案. Curah! 网站就 ...

  2. Redis集群明细文档

    Redis目前版本是没有提供集群功能的,如果要实现多台Redis同时提供服务只能通过客户端自身去实现(Memchached也是客户端实现分布式).目前根据文档已经看到Redis正在开发集群功能,其中一 ...

  3. ENVI 5.1操作心得

    1.ENVI中计算的NDVI导出ARCGIS中识别的float数据类型,选择File——save as——erdas img.在ARCGIS中就能统计出NDVI信息 2.如何去掉Nan值从影像中心位置 ...

  4. Python学习指南

    学习python书籍&资料: 1. Python v2.7.5 documentation 2. [Python参考手册(第4版)].(美)比兹利.扫描版.pdf 3. [Python技术手册 ...

  5. 【Nhibernate】入门 踩雷篇

    总结(喜欢写在前面,记性不好老忘记解决问题时的思路): 使用框架一般不会完整的看文档,直接上来就搞,踩雷是必须的,重要的是遇到雷的时候要快速变换思路,是不是姿势不对(文件位置不对) 提高解决问题的速度 ...

  6. MVC 删除文件

    protected void DeleteTempFiles(string iFileName) { FileInfo f = new FileInfo(iFileName); DirectoryIn ...

  7. ThinkPHP学习笔记 实例化模型的四种方法

    创建Action类   [php]   <?php   class NewObjectAction extends Action{       public function index(){ ...

  8. Linux "ls -l"文件列表权限详解

    ls Linux "ls -l"文件列表权限详解 1.使用 ls -l 命令 执行结果如下(/var/log) : drwxr-x--- root adm -- : apache2 ...

  9. 【转载】Spring加载resource时classpath*:与classpath:的区别

    免责声明:     本文转自网络文章,转载此文章仅为个人收藏,分享知识,如有侵权,请联系博主进行删除.     原文作者:kyfxbl     原文地址: spring配置中classpath和cla ...

  10. oracle——session

    一.解释session web应用中,session是服务器段保存用户信息的一个对象,cookie是浏览器端保存用户信息的对象.今天了解了oracle也有session对象,那么什么是oracle的s ...