AC题目简解-线段树
线段树:
http://www.notonlysuccess.com/index.php/segment-tree-complete/
鉴于notonlysuccess大牛的博客对于题目的思路写的很简陋,我就稍微补充下。
线段树的基本内容,是通过建二叉树来实现段的存储,最下面的叶子节点是每个值,左孩子和右孩子的父亲则是这个段的信息,依次推上去,实现从1..n的存储。可以通过程序带入样例来模拟这个过程体会。
建树:非叶子节点存储的都是段的值,而每个段,都有一个左边界和右边界。那么就建到叶子节点为止。每次都二分这个段,最后肯定会左边界等于右边界。
更新:如果是点更新则判断点即可,只是在判断该点在哪个区间的时候注意,一下。
查询:查询和更新类似
注意每一个return。
敌兵布阵:很简单的查询和单点更新
#include<cstdio> #include<cstring> #include<algorithm> using namespace std; ; int n; struct node{ int data,left,right; void init(int aleft,int aright){ left=aleft; right=aright; data=; } }tree[MAXN<<]; void pushup(int id){ tree[id].data=tree[id<<].data+tree[id<<|].data; } void build(int left,int right,int id){ tree[id].init(left,right); if(left==right){ scanf("%d",&tree[id].data); return;//remember here! } ; build(left,mid,id<<); build(mid+,right,id<<|); pushup(id); } int query(int left,int right,int id){ if(left==tree[id].left&&right==tree[id].right)return tree[id].data; ; ); else |); else )+query(mid+,right,id<<|); } void update(int p,int vadd,int id){ if(tree[id].left==tree[id].right){ tree[id].data+=vadd; return ; } ; ); |); pushup(id); } int main(){ int t,a,b; scanf("%d",&t); ]; ;i<=t;i++){ printf("Case %d:\n",i); scanf("%d",&n); build(,n,); ]!='E'){ scanf("%d%d",&a,&b); ]){ ));break; );break; );break; } } } ; }
hdu1754 I Hate It :单点替换,注意pushup上去
#include<cstdio> #include<cstring> #include<algorithm> using namespace std; ; struct node{ int data,left,right; void init(int aleft,int aright){ left=aleft;right=aright; data=; } }; node tree[MAXN<<]; inline void pushup(int id){ tree[id].data=max(tree[id<<].data,tree[id<<|].data); } void build(int left,int right,int id){ tree[id].init(left,right);//this if(left==right){ scanf("%d",&tree[id].data); return; } ; build(left,mid,id<<); build(mid+,right,id<<|); pushup(id); } void update(int p,int value,int id){ if(tree[id].left==tree[id].right){ tree[id].data=value; return; } ; ); |); pushup(id); } int query(int left,int right,int id){ if(left==tree[id].left&&right==tree[id].right)return tree[id].data; ; ); else |); else ),query(mid+,right,id<<|)); } int main(){ int n,m,a,b; ]; while(scanf("%d%d",&n,&m)!=EOF){ build(,n,); ;i<m;i++){ scanf("%s%d%d",&op,&a,&b); ]==)); ); } } ; }
hdu1394 Minimum Inversion Number 题意:求Inversion后的最小逆序数可以看看这篇博客
http://www.cnblogs.com/ziyi--caolu/archive/2013/01/15/2860768.html
#include<cstdio> #include<cstring> #include<algorithm> using namespace std; ; struct node{ int left,right,data; void init(int aleft,int aright){ left=aleft; right=aright; data=;//the num of the value before it } }; node tree[MAXN<<]; int n,a[MAXN]; inline void pushup(int id){ tree[id].data=tree[id<<].data+tree[id<<|].data; } void build(int left,int right,int id){ tree[id].init(left,right); if(left==right)return; ; build(left,mid,id<<); build(mid+,right,id<<|); } void update(int p,int id){ if(tree[id].left==tree[id].right){ tree[id].data=; return; } ; ); |); pushup(id); } int query(int k,int id){ ; if(k<=tree[id].left)return tree[id].data; ; )+query(k,id<<|); } int main(){ while(scanf("%d",&n)!=EOF){ build(,n-,); ; ;i<n;i++){ scanf("%d",&a[i]); ans+=query(a[i],); update(a[i],); } int minx=ans; ;i<n;i++){ ans+=n-*a[i]-; if(ans<minx)minx=ans; } printf("%d\n",minx); } ; }
AC题目简解-线段树的更多相关文章
- AC题目简解-数据结构
A - Japan POJ 3067 要两条路有交叉,(x1,y1)(x2,y2)那么需要满足:(x1-x2)*(y1-y2)<0判断出这是求逆序的问题 树状数组求逆序,先通过自定义的比较器实 ...
- AC题目简解-dp
dp类:A - Bridging signals ZOJ 3627 POJ1631 HDU1950给出一个从1-n的数字排列,求最长上升子序列长度.直接说解法吧.新开一个数组d,d[i]表示的是能构成 ...
- AC题目简解-数论
反素数: HDU2521定义对于任何正整数x,其约数的个数记做g(x).例如g(1)=1,g(6)=4.如果某个正整数x满足:对于任意i(0<i<x),都有g(i)<g(x),则称x ...
- [zoj3813]Alternating Sum 公式化简,线段树
题意:给一个长度不超过100000的原串S(只包含数字0-9),令T为将S重复若干次首尾连接后得到的新串,有两种操作:(1)修改原串S某个位置的值(2)给定L,R,询问T中L<=i<=j& ...
- 线段树详解 (原理,实现与应用)(转载自:http://blog.csdn.net/zearot/article/details/48299459)
原文地址:http://blog.csdn.net/zearot/article/details/48299459(如有侵权,请联系博主,立即删除.) 线段树详解 By 岩之痕 目录: 一:综述 ...
- hdu4973 线段树(题目不错,用了点,段,更新查找还有DFS)
题意: 给你一个初始序列,初始序列长度n,分别为1 2 3 4 5 ....n,有两种操作 (1)D l r 把l_r之间的数据都复制一遍 1 2 3 4 5 6 D 2 4 = 1 2 ...
- [Swust OJ 746]--点在线上(线段树解法及巧解)
题目链接:http://acm.swust.edu.cn/problem/746/ Time limit(ms): 1000 Memory limit(kb): 65535 fate是一个数学大牛 ...
- 题解报告:hdu 1754 I Hate It(线段树)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1754 Problem Description 很多学校流行一种比较的习惯.老师们很喜欢询问,从某某到某 ...
- POJ 3667 Hotel(线段树 区间合并)
Hotel 转载自:http://www.cnblogs.com/scau20110726/archive/2013/05/07/3065418.html [题目链接]Hotel [题目类型]线段树 ...
随机推荐
- for xml path以及sql合并查询
sql中for xml path的用法. http://www.cnblogs.com/yanghaibo/archive/2010/06/04/1751405.html
- Struct是干什么的
对于结构(Struct)这一看起来比较特殊的东西(用的比较少,只好用东西来形容了),真心用得少,只有在被问起的时候,才会想起,看看它到底是什么吧. 先给一个链接:http://www.cnblogs. ...
- WPF之旅(一)- 概述
WPF与之前出现的其他技术相比引入了“内置硬件加速”和“分辨率无关”等创新功能.WPF是一种成熟的技术,它是几个已经发布的.NET平台的一部分,并通过几个版本不断地进行完善(WPF3.0 -> ...
- 表达式语言之ongl表达式
OGNL的全称是Object Graph Navigation Language(对象图导航语言),它是一种强大的表达式语言,让你通过简单一致的表达式语法来读取和设置Java对象的属性值,调用对象的方 ...
- C#——中文转化成拼音
在KS系统中用到了中文转化成拼音的功能.通过查阅资料为下面是代码. /// <summary> /// MyConvert 的摘要说明 /// </summary> publi ...
- .NET基础之:i++和i=i+1和++i的区别
i++ 一定等同于 i=i+1吗? 大家都知道,i++通常情况都是等同于i=i+1,在编译时,clr会自动向上转换 比如说 int t =1; t++; 等同于 t=t+1; //1 是 ...
- 将eclipse的应用程序打包成.exe
转自:http://blog.163.com/loveshijie_1991/blog/static/1760553112012102573437156/ 参考:http://blog.csdn.ne ...
- Linux 系统 网络配置
Linux 系统 网络配置 配置Linux系统网络的方法有几种,这里介绍本人常用的两种. 第一种:使用命令ifconfig配置,具体用法:Ipconfig ethx x.x.x.x net ...
- JPA学习---第六节:大数据字段映射与字段延迟加载
1.大数据字段所需的注解 @Lob ,例如: @Lobprivate String info; 在mysql中映射产生的字段的类型是longtext:在oracle中是 CLOB @Lobpriva ...
- Sponsored Feature: Common Performance Issues in Game Programming
转自http://www.gamasutra.com/view/feature/132084/sponsored_feature_common_.php?print=1 By Becky Heinem ...