原题:

校门外有很多树,有苹果树,香蕉树,有会扔石头的,有可以吃掉补充体力的……
如今学校决定在某个时刻在某一段种上一种树,保证任一时刻不会出现两段相同种类的树,现有两个操作:
K=1,K=1,读入l、r表示在区间[l,r]中种上一种树,每次操作种的树的种类都不同
K=2,读入l,r表示询问l~r之间能见到多少种树
(l,r>0)

输入样例:

5 4
1 1 3
2 2 5
1 2 4
2 3 5

样例输出:

1
2

曾经,校门外的树是几行的暴力染色

然后,校门外的树是长长的线段树

后来啊,校门外的树是几行的树状数组

判断询问的区间与之前多少个已知区间有交集

如果将之前的已知区间双关键字排序

再做二分查找

很快就能得到有交集的区间数量

那么现在只有两个事情要办:

1、排序,要让已知的区间始终是有序的

2、查找,以logn的效率迅速找有交集的已知区间

可以用来联系线段树,也能作为树状数组的尝试

步骤(原来在写线段树的,写到一半突然发现树状数组可解,于是直接删代码写树状数组):

1:如果当前根不为空:得到一个区间信息,从根开始,如果该区间比根小,则把左子节点当成根做下一次操作的根,比根大则把右子节点作为下一次操作的根

2:如果当前根为空:愉快地将该区间信息放在根位置上

3:返回第一步

我的程序:

 #include <stdio.h>
int h[],t[];
int n,k;
void add(int a[],int k)
{
while(k<=n){
a[k]++;
k+=k&(-k);
}
}
int search(int a[],int k)
{
int tot=;
while(k){
tot+=a[k];
k-=k&(-k);
}
return tot;
}
int main()
{ scanf("%d%d",&n,&k);
for(int i=;i<=k;i++)
{
int a,b,c;
scanf("%d%d%d",&a,&b,&c);
if(a==){
add(h,b);
add(t,c);
}
else printf("%d\n",search(h,c)-search(t,b-));
}
}

PS:博客园第一文

最新文章

  1. PHP写文件函数
  2. java poi ppt操作示例
  3. iOS-协议与代理&lt;转&gt;
  4. python模拟浏览器保存Cookie进行会话
  5. ubuntu下phpstorm无法输入中文的解决办法
  6. jsp中文件下载的实现
  7. 你不需要jQuery(二)
  8. Android 数字签名学习笔记
  9. java覆写hashcode方法
  10. 作业还是作孽?——Leo鉴书79
  11. Codeforces #377 Div2
  12. sizeof计算空间大小的总结
  13. dijit.byId(&quot;grid&quot;) is undefined
  14. What are the differences between struct and class in C++?
  15. Hive的join表连接查询的一些注意事项
  16. Git创建新项目
  17. Python爬虫学习——光学字符识别
  18. 帝国CMS 列表模板list.var支持程序代码
  19. MySQL的知识海洋
  20. Spring Cloud Config配置中心的使用

热门文章

  1. WinAPI: GetClassName - 获取指定窗口的类名
  2. HTTP协议缓存策略深入详解之ETAG妙用
  3. Spark算子总结及案例
  4. 关于solaris中 crontab -e 出现数字0的解决办法
  5. Java 字符终端上获取输入三种方式
  6. [TPYBoard-Micropython教程之1] 运行第一个脚本——点亮LED
  7. CodeForces757B
  8. 简单js
  9. .Net程序员学用Oracle系列(12):增删改查
  10. 简述jpg、gif、png-8、png-24的区别,分别使用场景