Vijos1448校门外的树 题解

描述:
校门外有很多树,有苹果树,香蕉树,有会扔石头的,有可以吃掉补充体力的……
如今学校决定在某个时刻在某一段种上一种树,保证任一时刻不会出现两段相同种类的树,现有两个操作:
K=1,K=1,读入l、r表示在区间[l,r]中种上一种树,每次操作种的树的种类都不同
K=2,读入l,r表示询问l~r之间能见到多少种树
(l,r>0)
 
输入格式:
第一行n,m表示道路总长为n,共有m个操作
接下来m行为m个操作
 
输出格式:
对于每个k=2输出一个答案
 
样例输入:
5 4
1 1 3
2 2 5
1 2 4
2 3 5
 
样例输出:
1
2
 
数据范围:
20%的数据保证,n,m<=100
60%的数据保证,n <=1000,m<=50000
100%的数据保证,n,m<=50000
 
 
~~~~~~~~~~~~~~~~~~~~~~~~~~~分割线~~~~~~~~~~~~~~~~~~~~~~~~~~~
 
分析:
这道题是一个典型的区间问题,考虑到数据量较大,使用线段树完成这个操作。由于树的种类很多,不难想到用线段树暴力维护的方法。但是暴力维护一定会超时,那么这么解决这个问题呢?
这里介绍一种十分机智的想法——括号序列
假设有一个长度为10的数轴,我们要将区间[ 2 , 5 ]中种树,这时,我们将 2 处放一个左括号 " ( "  ,5处放一个 " )"  ,表示区间 [ 2 , 5 ]种了树。
查询某个区间树的种类,如区间[ 3 , 10],只需统计10之前(包括10)有多少个‘(’,统计3之前有多少个‘)’,(不包括3)。  
如下图所示:

以上就是括号序列的过程。简单的说,就是更新区间[a,b]时,点a记录左括号数,点b记录右括号数,查询区间[a,b]时,即为b之前(包括b)的左括号数-a之前的右括号数。
 
下面贴注释代码:

 #include "bits/stdc++.h"
#define maxN 50010 using namespace std ;
typedef long long QAQ ; struct Tree
{
int l , r ;
QAQ liml , limr ;//左括号右括号
}; Tree tr[maxN << ]; void Build_Tree ( int x , int y , int i )//建树
{
tr[i].l = x ;
tr[i].r = y ;
if( x == y )return ;
else
{
QAQ mid = (tr[i].l + tr[i].r) >> ;
Build_Tree ( x , mid , i << );
Build_Tree ( mid + , y , i << | );
return ;
}
} void Update_left ( int w , int i )
{
if( w == tr[i].l && w == tr[i].r )tr[i].liml++;//找到目标节点
else
{
QAQ mid = (tr[i].l + tr[i].r) >> ;
if( w > mid )Update_left( w , i << | );//找右儿子
else if( w <= mid)Update_left( w , i << );//找左儿子
tr[i].liml = tr[i << ].liml + tr[i << | ].liml ;//回溯更新
}
} void Update_right ( int w , int i )//同Update_left
{
if( w == tr[i].l && w == tr[i].r )tr[i].limr++;
else
{
QAQ mid = (tr[i].l + tr[i].r) >> ;
if( w > mid )Update_right( w , i << | );
else if( w <= mid)Update_right( w , i << );
tr[i].limr = tr[i << ].limr + tr[i << | ].limr ;
}
} QAQ Query_left ( int q , int w , int i )//同Query_right
{
if( q <= tr[i].l && w >= tr[i].r )return tr[i].liml ;
else
{
QAQ mid = (tr[i].l + tr[i].r) >> ;
if ( q > mid )return Query_left ( q , w , i << | );
else if ( w <= mid ) return Query_left ( q , w , i << );
else return Query_left ( q , w , i << | ) + Query_left ( q , w , i << );
}
} QAQ Query_right ( int q , int w , int i )
{
if( q <= tr[i].l && w >= tr[i].r )return tr[i].limr ;//找到目标区间直接返回
else
{
QAQ mid = (tr[i].l + tr[i].r) >> ;
if ( q > mid )return Query_right ( q , w , i << | );//找右儿子
else if ( w <= mid ) return Query_right ( q , w , i << );//找左儿子
else return Query_right ( q , w , i << | ) + Query_right ( q , w , i << );//左右儿子都查找
}
} int main()
{
int N, M, op, ll, rr ;
scanf("%d %d", &N, &M);
Build_Tree ( , N , ) ;//建树
while(M--)
{
scanf("%d%d%d", &op, &ll, &rr);
if( op == )
{
Update_left ( ll , );//添加左括号
Update_right ( rr , );//添加右括号
}
else
{
QAQ ans = Query_left( , rr , );
if (ll != )ans -= Query_right( , ll - , );//当ll不等于1时再相减,否则栈会炸
printf("%I64d\n", ans);
}
}
return ;
}
 PS: 本题也可以用树状数组完成,代码量较少,容易实现。
 
 (完)
 

Vijos1448校门外的树 题解的更多相关文章

  1. Vijos1448 校门外的树 [树状数组]

    题目传送门 校门外的树 描述 校门外有很多树,有苹果树,香蕉树,有会扔石头的,有可以吃掉补充体力的……如今学校决定在某个时刻在某一段种上一种树,保证任一时刻不会出现两段相同种类的树,现有两个操作:K= ...

  2. vijos1448校门外的树

    描述 校门外有很多树,有苹果树,香蕉树,有会扔石头的,有可以吃掉补充体力的……如今学校决定在某个时刻在某一段种上一种树,保证任一时刻不会出现两段相同种类的树,现有两个操作:K=1,K=1,读入l.r表 ...

  3. 洛谷P1047校门外的树题解

    题目 此题是一个模拟题,但需要注意的一点就是它的树是从数轴的0开始,所以我们也要从0开始,这样才能实现代码. 代码: #include<iostream> using namespace ...

  4. 洛谷 P1047 校门外的树 题解

    Case 1. 本题其实不难,直接模拟就可以了.时间复杂度: \(O(L \times M)\) Case 2. 考虑一个简单的增强:把原来的: \[L \leq 10^4,M \leq 10^2 \ ...

  5. >题解< 校门外的树

    题目描述 某校大门外长度为L的马路上有一排树,每两棵相邻的树之间的间隔都是 11 米.我们可以把马路看成一个数轴,马路的一端在数轴 00 的位置,另一端在 LL 的位置:数轴上的每个整数点,即 0,1 ...

  6. JDOJ 2197: 校门外的树

    JDOJ 2197: 校门外的树 题目传送门 Description 某校大门外长度为L的马路上有一排树,每两棵相邻的树之间的间隔都是1米. 我们可以把马路看成一个数轴,马路的一端在数轴1的位置,另一 ...

  7. NC16649 [NOIP2005]校门外的树

    NC16649 [NOIP2005]校门外的树 题目 题目描述 某校大门外长度为 \(L\) 的马路上有一排树,每两棵相邻的树之间的间隔都是 \(1\) 米.我们可以把马路看成一个数轴,马路的一端在数 ...

  8. P1047 校门外的树

    P1047 校门外的树 题目描述 某校大门外长度为L的马路上有一排树,每两棵相邻的树之间的间隔都是1米.我们可以把马路看成一个数轴,马路的一端在数轴0的位置,另一端在L的位置:数轴上的每个整数点,即0 ...

  9. OpenJudge计算概论-校门外的树

    /*======================================================================== 校门外的树 总时间限制: 1000ms 内存限制: ...

随机推荐

  1. C-线性顺序表的增删改查

    闲来无事,练练手,写点C代码,对于线性表的简单操作.编辑工具Notpad++,编译工具tcc. /* *the sequence of the list *author:JanneLee *data: ...

  2. 【网络资料】Astar算法详解

    关于A*算法,很早就想写点什么,可是貌似天天在忙活着什么,可事实又没有做什么,真是浮躁啊!所以今晚还是来写一下总结吧! A*算法是很经典的只能启发式搜索算法,关于只能搜索算法和一般的搜索算法(例如DF ...

  3. Solr入门之(1)前言与概述

    一.前言:为何选择Solr 由于搜索引擎功能在门户社区中对提高用户体验有着重在门户社区中涉及大量需要搜索引擎的功能需求,目前在实现搜索引擎的方案上有几种方案可供选择: 1. 基于Lucene自己进行封 ...

  4. cordova+angularJS+ionic

    1.创建项目 2.路由 angular.module("starter",['ionic']) // 依赖 ionic 提供的ui-router .config(function ...

  5. linux的<pthread.h>

    转自:http://blog.sina.com.cn/s/blog_66cc44d00100in5b.html Linux系统下的多线程遵循POSIX线程接口,称为pthread.编写Linux下的多 ...

  6. [unity3d插件]2dtoolkit系列一 创建精灵

    从今天开始要做一个2d游戏,由于之前都是做cocos2dx的,然后接触了一段时间的unity3d,都是做3D方面的东西,得知要做2d游戏还是有点开心的,或许因为不想丢失之前的2d游戏的一些思想,然后接 ...

  7. C语言字符串长度(转)

    C语言字符串长度的计算是编程时常用到的,也是求职时必考的一项. C语言本身不限制字符串的长度,因而程序必须扫描完整个字符串后才能确定字符串的长度. 在程序里,一般会用strlen()函数或sizeof ...

  8. gdb调试小结

    gdb最基本的调试命令. 1以调试程序test.cpp为例: 进入调试环境 gdb test 2.b 12 在文件的第12行设置断点. 删除断点: info b 列出所有的断点信息 (gdb) inf ...

  9. HDU 4812 D Tree 树分治+逆元处理

    D Tree Problem Description   There is a skyscraping tree standing on the playground of Nanjing Unive ...

  10. 【HTML5 video】video标签的部分属性解析

    转自:http://www.cnblogs.com/kiter/archive/2013/02/25/2932157.html 现在如果要在页面中使用video标签,需要考虑三种情况,支持Ogg Th ...