普及向 ZKW线段树!
啊,是否疲倦了现在的线段树
太弱,还递归!
那我们就欢乐的学习另外一种神奇的线段树吧!(雾
他叫做zkw线段树
这个数据结构灰常好写(虽然线段树本身也特别好写……)
速度快(貌似只在单点更新方面比线段树快……)
是一种自底向上非递归版本的线段树!
首先我们来看一个ppt,《统计的力量》这个是发明人的PPT(啊,ppt内的代码是错的……
好吧,我们来写吧~
首先预备条件:
int M,T[maxn*+];
M指的是什么呢?M就指的是这颗zkw线段树最下面的那个点之前的编号是什么
T数组就是这个zkw线段树的数组,由于zkw线段树是一颗满二叉树,所以直接开两倍就好啦~
接下来我们就来建树吧!
以单点更新,区间查询和作为一个例子~
void build(int x)
{
for(M=;M<=n+;M<<); for(int i=;i<=n;i++)
scanf("%d",&T[i+M]); for(int i=M-;i;i--)
T[i]=T[i<<]+T[i<<|];
}
很显然这是一颗非常弱的从底往上更新的树!
然后updata怎么写呢?qsc仔细想了想,很简单
void updata(int n,int val)
{
T[n+=M]=val;//这个地方是单点修改的哟
for(n>>=;n;n>>=)
T[n]=T[n<<]+T[n<<|];
}
哇,其实和线段树是一个意思,直接顺着节点直接往上爬就是了!
query肿么写呢?还是很简单呀
int query(int l,int r)
{
int ans=;
l=l+M-,r=r+M+;
for(;l^r^;l>>,r>>=)
{
if(~l&)ans+=T[l^];
if(r&) ans+=T[r^];
}
return ans;
}
这个乱七八糟的位运算什么意思呀?
l^r^1的意思,就是左边的这个点和右边这个点是否互为兄弟,或者干脆就是一个点
~l&1 就是判断这个左边这个是否为左儿子,r&1判断这个节点是否为右二子
如果是的话,那就得加上他的兄弟咯~
啊,zkw单点更新区间查询就示范到这儿了,是不是灰常简单呀~
我们首先来搞一道例题:
HDU 1166 敌兵布阵 单点更新,区间查询
//qscqesze
#include <cstdio>
#include <cmath>
#include <cstring>
#include <ctime>
#include <iostream>
#include <algorithm>
#include <set>
#include <vector>
#include <sstream>
#include <queue>
#include <typeinfo>
#include <fstream>
#include <map>
#include <stack>
typedef long long ll;
using namespace std;
//freopen("D.in","r",stdin);
//freopen("D.out","w",stdout);
#define sspeed ios_base::sync_with_stdio(0);cin.tie(0)
#define maxn 200001
#define mod 10007
#define eps 1e-9
//const int inf=0x7fffffff; //无限大
const int inf=0x3f3f3f3f;
/* int buf[10];
inline void write(int i) {
int p = 0;if(i == 0) p++;
else while(i) {buf[p++] = i % 10;i /= 10;}
for(int j = p-1; j >=0; j--) putchar('0' + buf[j]);
printf("\n");
}
*/
//**************************************************************************************
inline ll read()
{
int x=,f=;char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=-;ch=getchar();}
while(ch>=''&&ch<=''){x=x*+ch-'';ch=getchar();}
return x*f;
} ll T[maxn*];
int M,n;
void build()
{
for(M=;M<=n+;M<<=);
for(int i=;i<=n;i++)
T[i+M]=read();
for(int i=M-;i;i--)
T[i]=T[i<<]+T[i<<|];
} void updata(int x,int val)
{
T[x+=M]+=val;
for(x>>=;x>=;x>>=)
{
T[x]=T[x<<]+T[x<<|];
}
} ll query(int l,int r)
{
l=l+M-,r=r+M+;
ll ans=;
for(;l^r^;l>>=,r>>=)
{
if(~l&)ans+=T[l^];
if(r&) ans+=T[r^];
}
return ans;
} int main()
{
int cas=;
int t=read();
string s;
for(int cas=;cas<=t;cas++)
{
memset(T,,sizeof(T));
printf("Case %d:\n",cas);
n=read();
build();
while(cin>>s)
{
if(s[]=='E')
break;
int a,b;
a=read(),b=read();
if(s[]=='A')
updata(a,b);
else if(s[]=='S')
updata(a,-b);
else
printf("%lld\n",query(a,b));
}
}
}
普及向 ZKW线段树!的更多相关文章
- ZKW线段树
简介 zkw线段树虽然是线段树的另一种写法,但是本质上已经和普通的递归版线段树不一样了,是一种介于树状数组和线段树中间的存在,一些功能上的实现比树状数组多,而且比线段树好写且常数小. 普通线段树采用从 ...
- zkw线段树详解
转载自:http://blog.csdn.net/qq_18455665/article/details/50989113 前言 首先说说出处: 清华大学 张昆玮(zkw) - ppt <统计的 ...
- BZOJ3173 TJOI2013最长上升子序列(Treap+ZKW线段树)
传送门 Description 给定一个序列,初始为空.现在我们将1到N的数字插入到序列中,每次将一个数字插入到一个特定的位置.每插入一个数字,我们都想知道此时最长上升子序列长度是多少? Input ...
- 【POJ3468】【zkw线段树】A Simple Problem with Integers
Description You have N integers, A1, A2, ... , AN. You need to deal with two kinds of operations. On ...
- HDU 4366 Successor(树链剖分+zkw线段树+扫描线)
[题目链接] http://acm.hdu.edu.cn/showproblem.php?pid=4366 [题目大意] 有一个公司,每个员工都有一个上司,所有的人呈树状关系,现在给出每个人的忠诚值和 ...
- [SinGuLaRiTy] ZKW线段树
[SinGuLaRiTy-1007] Copyrights (c) SinGuLaRiTy 2017. All Rights Reserved. 关于ZKW线段树 Zkw线段树是清华大学张昆玮发明非递 ...
- 数据结构3——浅谈zkw线段树
线段树是所有数据结构中,最常用的之一.线段树的功能多样,既可以代替树状数组完成"区间和"查询,也可以完成一些所谓"动态RMQ"(可修改的区间最值问题)的操作.其 ...
- 线段树(单标记+离散化+扫描线+双标记)+zkw线段树+权值线段树+主席树及一些例题
“队列进出图上的方向 线段树区间修改求出总量 可持久留下的迹象 我们 俯身欣赏” ----<膜你抄> 线段树很早就会写了,但一直没有总结,所以偶尔重写又会懵逼,所以还是要总结一下. ...
- 『zkw线段树及其简单运用』
阅读本文前,请确保已经阅读并理解了如下两篇文章: 『线段树 Segment Tree』 『线段树简单运用』 引入 这是一种由\(THU-zkw\)大佬发明的数据结构,本质上是经典的线段树区间划分思想, ...
随机推荐
- python 实现字符串转整型
def str2Int(s): l=list(s) if len(l)<=0: return 0 flag=0 sum=0 dict_num={':9} dict_tag={'+':1,'-': ...
- 1-编程基础及Python环境部署
目录 1 编程基础 1.1 基本概念 1.2 语言分类 1.3 高级语言的发展 2 程序 3 python的语言介绍 4 Python的解释器 5 Python版本区别 6 Python安装 6.1 ...
- python并发编程之threading线程(一)
进程是系统进行资源分配最小单元,线程是进程的一个实体,是CPU调度和分派的基本单位,它是比进程更小的能独立运行的基本单位.进程在执行过程中拥有独立的内存单元,而多个线程共享内存等资源. 系列文章 py ...
- Python设计模式中单例模式的实现及在Tornado中的应用
单例模式的实现方式 将类实例绑定到类变量上 class Singleton(object): _instance = None def new(cls, *args): if not isinstan ...
- android sdcard 权限管理策略研究
自从android4.4 以来,第三方应用程序是不能再随便的访问sdcard了,从开发者的角度而言,研究一下android系统到底是怎么样来实现这样的控制还是比较有价值的. 首先分析下现状,现在已知, ...
- [转载]理解Tomcat的Classpath-常见问题以及如何解决
摘自: http://www.linuxidc.com/Linux/2011-08/41684.htm 在很多Apache Tomcat用户论坛,一个问题经常被提出,那就是如何配置Tomcat的cla ...
- C++——map注意事项
1. C++标准模块库STL中提供了两种基本的关联容器:set和map.其内部实现是都是采用红黑树,但不同的是,set中结点存储的是键值,map中结点存储的是<key,value>的键值对 ...
- gan对抗式网络
感觉好厉害,由上图噪声,生成左图噪声生成右图以假乱真的图片, gan网络原理: 本弱又盗了一坨博文,不是我写的,如下:(跪膜各路大神) 前面我们已经讲完了一般的深层网络,适用于图像的卷积神经网络,适用 ...
- js excel导出 前端实现(转载)
本文为转载,本人亲测可用,感谢原作者,原文在这里. 一.由于公司项目需要,需要将表格数据导出为EXCEL表格数据. 环境React+Ant Design 二.安装插件js-export-excel ...
- 洛谷 P1652圆 题解
题目传送门 这道题也就是考你对几何的了解: 圆与圆没有公共点且一个圆在另一个圆外面时,叫做圆与圆相离. 当圆心距大于两圆半径之和时,称为两圆外离: 当圆心距小于两圆半径之差的绝对值时,称为两圆内含. ...