自闭场本来 以为 顶多一些不太会 结果发现 一堆不太会 。

树状数组  感觉 好久没看 了有点遗忘 不过还好 现在我来了。莅临之神将会消灭一切知识点哦。

今天说点不一样东西 树状数组 hh 很有用的东西 现在想想常数可谓是非常的小的吧 因为 查询的时候log 基本上市跑不满的 况且增加的时候log 也很少跑满。

先说构造 构造是最体现水平的东西------lrj 我特别信服这句话 关键不是会用这个东西而是把更深层次的东西挖掘出来,显然的是 发明这个东西比会使用这个东西 来的更难。

构造 树状数组一般采用下标来存储 当然 也是有权值树状数组 那个是后话了。原理 话 我必须得提一下 因为曾经我有一个鬼畜的想法也是正确。

如今 只能靠我自己填坑了 。 一个数字 按照自己的编号lowbit 上去 然后 从一个位置 lowbit 减过去为什么是正确的呢?这点 我想是有必要证明。

因为 考虑 lowbit 上去 你做的一些事情 首先来说不断地二进制位往上进然后 为什么倒着可以累加上去呢 我们感性思考一下(大不了 丢坑

可以证明每个数字我们被我们至多累加一次 且不会漏掉任何一个数字 只要能证明出来这个那么树状数组的原理就是正确的。

首先对于我们最大的二进制位 一个数字下标如果不超过这个二进制位那么 它一定在 我们减剩下 最大的二进制位的时候 被我们累加一次然后结束 累加一次且没漏掉一切这种类型的数字。

然后是对于一个数字的下表没有超过了这个二进制位且大小不超过我们查询下标的大小 那么 此时 显然有这个数字下标不断逐位扩大 扩大至我们当前这个数字二进制位的某一位。也就是这个数字二进制位表示是我们查询的二进制数的子集 这样显然我们询问那个数字在减的同时 一定能到达其子集 如果证明其不会重复呢 这个还是要分两种情况讨论 一个是我们刚好达到累加这个数字起点之时 那么显然一个减一个加 不会再有交集。考虑我们遇到的不是起点而是中间时刻某个点 接下来会不会继续遇到这个问题 仔细思考我们发现中间时刻某点 一定满足x>x1 那么此时我们减去lowbit x这一位 这也同时意味着不可能再有x1这个二进制位了 如果有那么lowbit 的将会是它 但是考虑lowbit的是它下次是否会lowbit x 这位 这显然也是不可能完成累加的因为在上一步想要累加上去必然存在两个数字想等然后一个+一个- 自然不可能出现交集。

证毕。

你是否听说过逆序树状数组的存在 讲操作颠倒的那种 加值得时候向下累加 统计的时候向上统计。这个操作有兴趣 的话可以去研究一下 非常的流弊正序求逆序对。

例题:

bzoj 1878 求区间本质不同的数的个数。离线处理 维护右端点单调递增 然后遇到一个数字当前位置数字 贡献+1 上一次出现这个数字贡献-1 满足贪心的原理 故这个做法是正确的。

//#include<bits/stdc++.h>
#include<iomanip>
#include<iostream>
#include<cstdio>
#include<cstring>
#include<string>
#include<queue>
#include<deque>
#include<cmath>
#include<ctime>
#include<cstdlib>
#include<stack>
#include<algorithm>
#include<vector>
#include<cctype>
#include<utility>
#include<set>
#include<bitset>
#include<map>
#define INF 10000000000000000ll
#define ll long long
#define min(x,y) ((x)>(y)?(y):(x))
#define max(x,y) ((x)>(y)?(x):(y))
#define RI register ll
#define db double
#define EPS 1e-8
using namespace std;
char buf[<<],*fs,*ft;
inline char getc()
{
return (fs==ft&&(ft=(fs=buf)+fread(buf,,<<,stdin),fs==ft))?:*fs++;
}
inline int read()
{
int x=,f=;char ch=getc();
while(ch<''||ch>''){if(ch=='-')f=-;ch=getc();}
while(ch>=''&&ch<=''){x=x*+ch-'';ch=getc();}
return x*f;
}
const int MAXN=,maxn=,MAX=;
int n,m;
int c[maxn],a[maxn],ans[MAXN];
int pre[maxn],last[MAX];
struct wy
{
int l,r;
int id;
int friend operator <(const wy a,const wy b){return a.r<b.r;}
}t[MAXN];
inline void add(int x,int y)
{
while(x<=n)
{
c[x]+=y;
x+=x&(-x);
}
}
inline int ask(int x)
{
int cnt=;
while(x)
{
cnt+=c[x];
x-=x&(-x);
}
return cnt;
}
int main()
{
//freopen("1.in","r",stdin);
n=read();
for(int i=;i<=n;++i)
{
a[i]=read();
pre[i]=last[a[i]];
last[a[i]]=i;
}
m=read();
for(int i=;i<=m;++i)
{
int x,y;
x=read();y=read();
t[i]=(wy){x,y,i};
}
int flag=;
sort(t+,t++m);
for(int i=;i<=n;++i)
{
add(i,);
if(pre[i])add(pre[i],-);
while(t[flag].r==i)
{
ans[t[flag].id]=ask(t[flag].r)-ask(t[flag].l-);
++flag;
}
}
for(int i=;i<=m;++i)printf("%d\n",ans[i]);
return ;
}

然后其他的应用学长讲的很鬼畜 回来再说。

线段树:主要学习了一下标记永久化 因为下传标记 会是一件很麻烦的事情 所以考虑标记不下传等到访问的时候获得标记的价值即可。

bzoj 4636 蒟蒻数列

//#include<bits/stdc++.h>
#include<iomanip>
#include<iostream>
#include<cstdio>
#include<cstring>
#include<string>
#include<queue>
#include<deque>
#include<cmath>
#include<ctime>
#include<cstdlib>
#include<stack>
#include<algorithm>
#include<vector>
#include<cctype>
#include<utility>
#include<set>
#include<bitset>
#include<map>
#define INF 1000000000
#define ll long long
#define min(x,y) ((x)>(y)?(y):(x))
#define max(x,y) ((x)>(y)?(x):(y))
#define RI register ll
#define db double
#define EPS 1e-8
#define l(p) t[p].l
#define r(p) t[p].r
#define mx(p) t[p].mx
using namespace std;
char buf[<<],*fs,*ft;
inline char getc()
{
return (fs==ft&&(ft=(fs=buf)+fread(buf,,<<,stdin),fs==ft))?:*fs++;
}
inline int read()
{
int x=,f=;char ch=getc();
while(ch<''||ch>''){if(ch=='-')f=-;ch=getc();}
while(ch>=''&&ch<=''){x=x*+ch-'';ch=getc();}
return x*f;
}
const int MAXN=;
int n,root,cnt;
struct wy
{
int l,r;
int mx;
}t[MAXN*];
inline void modify(int &p,int l,int r,int L,int R,int z)
{
if(!p)p=++cnt;
if(L<=l&&R>=r){mx(p)=max(mx(p),z);return;}
int mid=(l+r)>>;
if(L<=mid)modify(l(p),l,mid,L,R,z);
if(R>mid)modify(r(p),mid+,r,L,R,z);
}
inline ll ask(int p,int l,int r,int x)
{
if(!p)return ;
int mid=(l+r)>>;
ll sum=;
mx(p)=max(mx(p),x);
if(!l(p))sum+=(ll)mx(p)*(mid-l+);
if(!r(p))sum+=(ll)mx(p)*(r-mid);
sum+=ask(l(p),l,mid,mx(p));
sum+=ask(r(p),mid+,r,mx(p));
return sum;
}
signed main()
{
//freopen("1.in","r",stdin);
n=read();
for(int i=;i<=n;++i)
{
int a,b,k;
a=read();b=read()-;k=read();
if(a>b)continue;
modify(root,,INF,a,b,k);
}
printf("%lld\n",ask(root,,INF,));
return ;
}

其实不需要标记永久化应该也是可以写的不过比较繁琐,支持标记永久化就是支持未来。。

2019 HL SC day4的更多相关文章

  1. 2019 HL SC day1

    今天讲的是图论大体上分为:有向图的强连通分量,有向图的完全图:竞赛图,无向图的的割点,割边,点双联通分量,变双联通分量以及圆方树 2-sat问题 支配树等等. 大体上都知道是些什么东西 但是仍需要写一 ...

  2. 2019 HL SC day10

    10天都过去了 4天都在全程懵逼.. 怎么可以这么难啊 我服了 现在想起依稀只记得一些结论 什么 反演? 什么后缀自动机?什么组合数的应用?什么神仙东西 ,不过讲课人的确都是神仙.(实名羡慕. mzx ...

  3. 2019 HL SC day2

    今天讲的是网络流 大部分题目都写过了 这里 就总结一番. bzoj 1066 裸的最大流 不过需要拆点细节方面有一点坑 剩下的 没什么了. //#include<bits/stdc++.h> ...

  4. N(C)O(S)I(P)P 2019 退役记

    N(C)O(S)I(P)P 2019 退役记 day-4 今天下午老师突然咕了,于是一下午欢乐时光 今天上午考试T3线段树维护个区间加,区间乘 一遍过编译,一遍过样例(第一次,俺比较弱(虽然也发现和暴 ...

  5. Solr分组查询

     项目中需要实时的返回一下统计的东西,因此就要进行分组,在获取一些东西,代码拿不出来,因此分享一篇,还是很使用的. facet搜索 /** * * 搜索功能优化-关键词搜索 * 搜索范围:商品名称.店 ...

  6. Light of future-冲刺集合

    table th:nth-of-type(1) { width: 85px; } table th:nth-of-type(2){ width: 80px; } table th:nth-of-typ ...

  7. Light of future-冲刺总结

    目录 1.凡事预则立.测试博客的链接 2.包含冲刺日志集合随笔的所有内容 3.描述项目预期计划 7.代码仓库地址.测试文档链接地址.PPT链接地址 归属班级 →2019秋福大软件工程实践Z班 作业要求 ...

  8. 线性基求交(2019牛客国庆集训派对day4)

    题意:https://ac.nowcoder.com/acm/contest/1109/C 问你有几个x满足A,B集合都能XOR出x. 思路: 就是线性基求交后,有几个基就是2^几次方. #defin ...

  9. 2019中山纪念中学夏令营-Day4[JZOJ]

    Begin (题目的排序方式:难易程度) 什么是对拍: 对拍是一种在写完程序后,验证自己程序是不是正解的比较方便的方法. 实现过程: 对同一道题,再打一个暴力程序,然后用一些大数据等跑暴力程序来进行验 ...

随机推荐

  1. 状压DP之炮兵阵地

    题目 原题来自:\(NOI 2001\) 司令部的将军们打算在\(N*M\) 的网格地图上部署他们的炮兵部队.一个\(N*M\)的地图由\(N\)行\(M\)列组成,地图的每一格可能是山地(用 H表示 ...

  2. Docker-教你如何通过 Docker 快速搭建各种测试环境

    今天给大家分享的主题是,如何通过 Docker 快速搭建各种测试环境,本文列举的,也是作者在工作中经常用到的,其中包括 MySQL.Redis.Elasticsearch.MongoDB 安装步骤,通 ...

  3. Linux超强截图工具flameshot

    Pop!_OS自带的截屏快捷键如下 但讲道理这个是真的不好用 所以我们借助第三方的截图工具,这里推荐flameshot(火焰截图) 在终端键入以下命令即可安装 sudo apt update sudo ...

  4. for of

    1. 遍历范围 for...of 循环可以使用的范围包括: 数组 Set Map 类数组对象,如 arguments 对象.DOM NodeList 对象 Generator 对象 字符串 2. 优势 ...

  5. jsp页面中同时遍历多个list集合

    在Jsp页面中,我们也许有这样的需求:从后端获取到多个List,但又想将这些List的值同时打印出来 比如, 有用户列表userList,user类有用户ID.用户名.用户性别等基本信息 有用户关系列 ...

  6. 如何在同一台电脑上部署多个tomcat实现多个tomcat在同一台电脑上同时启动

    有时候我们在开发的过程中难免会遇到需要在同一台电脑部署多个tomcat,且还要他们能够都单独同时启动不会对其他的tomcat造成影响 本文就简单记录一下,如何来实现这个骚操作 1. 下载tomcat的 ...

  7. java 面向对象(二十七):注解的使用

    1. 注解的理解① jdk 5.0 新增的功能*② Annotation 其实就是代码里的特殊标记, 这些标记可以在编译, 类加载, 运行时被读取, 并执行相应的处理.通过使用 Annotation, ...

  8. 机器学习实战基础(二十七):sklearn中的降维算法PCA和SVD(八)PCA对手写数字数据集的降维

    PCA对手写数字数据集的降维 1. 导入需要的模块和库 from sklearn.decomposition import PCA from sklearn.ensemble import Rando ...

  9. 数据可视化之分析篇(二)Power BI 数据分析:客户购买频次分布

    https://zhuanlan.zhihu.com/p/100070260 商业数据分析通常都可以简化为对数据进行筛选.分组.汇总的过程,本文通过一个实例来看看PowerBI是如何快速完成整个过程的 ...

  10. Bounding-Box(BB)regression

    最近在学习RCNN,对于Bounding-Box(BB)regression能够提高边界框的精确度,对于其内容产生了很大兴趣. 主要内容学习自大神博客:https://blog.csdn.net/bi ...