传送门

考虑用 \(segment~tree~beats\) 那一套理论,维护区间最小值 \(mn\) 和严格次小值 \(se\)

那么可以直接 \(mlog^2n\) 维护前三个操作

考虑维护历史最小值,先维护历史最小标记

写了写发现 \(max\) 那个修改不好操作

对于 \(max\) 操作来说,只会在 \(mn< v <se\) 的时候打上标记

这就相当于区间内等于 \(mn\) 的权值都要变成 \(v\)

那么 \(max\) 操作就可以变成对区间最小值的加法操作

而 \(v<se\),这样就可以非常方便维护历史最小值了

具体来说,维护下面几个标记

  1. 区间最小值的加法标记
  2. 区间其它值的加法标记
  3. 区间最小值的历史最小的加法标记
  4. 区间其它值的历史最小的加法标记

    下放的时候判断一下是否是区间最小值就好了
# include <bits/stdc++.h>
using namespace std;
typedef long long ll; namespace IO {
const int maxn(1 << 21 | 1); char obuf[maxn], ibuf[maxn], *iS, *iT, c, *oS = obuf, *oT = obuf + maxn - 1, st[60];
int f, tp; inline char Getc() {
return iS == iT ? (iT = (iS = ibuf) + fread(ibuf, 1, maxn, stdin), (iS == iT ? EOF : *iS++)) : *iS++;
} template <class Int> inline void In(Int &x) {
for (c = Getc(), f = 1; c < '0' || c > '9'; c = Getc()) f = c == '-' ? -1 : 1;
for (x = 0; c >= '0' && c <= '9'; c = Getc()) x = (x << 1) + (x << 3) + (c ^ 48);
x *= f;
} inline void Flush() {
fwrite(obuf, 1, oS - obuf, stdout);
oS = obuf;
} inline void Putc(char c) {
*oS++ = c;
if (oS == oT) Flush();
} template <class Int> inline void Out(Int x) {
if (x < 0) Putc('-'), x = -x;
if (!x) Putc('0');
while (x) st[++tp] = x % 10 + '0', x /= 10;
while (tp) Putc(st[tp--]);
}
} using IO :: In;
using IO :: Out;
using IO :: Putc;
using IO :: Flush; const int maxn(2e6 + 5);
const int inf(2e9); struct Min {
int mn1, mn2; inline Min operator +(Min b) const {
Min c;
c.mn1 = min(mn1, b.mn1), c.mn2 = min(mn2, b.mn2);
if (b.mn1 ^ c.mn1) c.mn2 = min(c.mn2, b.mn1);
if (mn1 ^ c.mn1) c.mn2 = min(c.mn2, mn1);
return c;
}
}; Min mn[maxn];
int n, m, hmn[maxn], addmn1[maxn], addmn2[maxn], addhmn1[maxn], addhmn2[maxn]; inline void Update(int x) {
mn[x] = mn[x << 1] + mn[x << 1 | 1], hmn[x] = min(hmn[x << 1], hmn[x << 1 | 1]);
} void Build(int x, int l, int r) {
int mid;
if (l == r) {
In(mn[x].mn1), mn[x].mn2 = inf, hmn[x] = mn[x].mn1;
return;
}
mid = (l + r) >> 1;
Build(x << 1, l, mid), Build(x << 1 | 1, mid + 1, r);
Update(x);
} inline void Puttag(int x, int vmn1, int vmn2, int vhmn1, int vhmn2) {
hmn[x] = min(hmn[x], mn[x].mn1 + vhmn1);
addhmn1[x] = min(addhmn1[x], addmn1[x] + vhmn1);
addhmn2[x] = min(addhmn2[x], addmn2[x] + vhmn2);
addmn1[x] += vmn1, addmn2[x] += vmn2, mn[x].mn1 += vmn1;
if (mn[x].mn2 ^ inf) mn[x].mn2 += vmn2;
} inline void Pushdown(int x) {
if (!addmn1[x] && !addmn2[x] && !addhmn1[x] && !addhmn2[x]) return;
int ls, rs, now;
ls = x << 1, rs = x << 1 | 1, now = min(mn[ls].mn1, mn[rs].mn1);
if (now == mn[ls].mn1) Puttag(ls, addmn1[x], addmn2[x], addhmn1[x], addhmn2[x]);
else Puttag(ls, addmn2[x], addmn2[x], addhmn2[x], addhmn2[x]);
if (now == mn[rs].mn1) Puttag(rs, addmn1[x], addmn2[x], addhmn1[x], addhmn2[x]);
else Puttag(rs, addmn2[x], addmn2[x], addhmn2[x], addhmn2[x]);
addmn1[x] = addmn2[x] = addhmn1[x] = addhmn2[x] = 0;
} void Modify_add(int x, int l, int r, int ql, int qr, int v) {
int mid;
if (ql <= l && qr >= r) {
Puttag(x, v, v, v, v);
return;
}
mid = (l + r) >> 1, Pushdown(x);
if (ql <= mid) Modify_add(x << 1, l, mid, ql, qr, v);
if (qr > mid) Modify_add(x << 1 | 1, mid + 1, r, ql, qr, v);
Update(x);
} void Modify_max(int x, int l, int r, int ql, int qr, int v) {
int mid;
if (mn[x].mn1 >= v) return;
if (ql <= l && qr >= r && mn[x].mn2 > v) {
Puttag(x, v - mn[x].mn1, 0, v - mn[x].mn1, 0);
return;
}
mid = (l + r) >> 1, Pushdown(x);
if (ql <= mid) Modify_max(x << 1, l, mid, ql, qr, v);
if (qr > mid) Modify_max(x << 1 | 1, mid + 1, r, ql, qr, v);
Update(x);
} int Query_min(int x, int l, int r, int ql, int qr) {
int mid, ret;
if (ql <= l && qr >= r) return mn[x].mn1;
mid = (l + r) >> 1, Pushdown(x), ret = inf;
if (ql <= mid) ret = Query_min(x << 1, l, mid, ql, qr);
if (qr > mid) ret = min(ret, Query_min(x << 1 | 1, mid + 1, r, ql, qr));
Update(x);
return ret;
} int Query_hmin(int x, int l, int r, int ql, int qr) {
int mid, ret;
if (ql <= l && qr >= r) return hmn[x];
mid = (l + r) >> 1, Pushdown(x), ret = inf;
if (ql <= mid) ret = Query_hmin(x << 1, l, mid, ql, qr);
if (qr > mid) ret = min(ret, Query_hmin(x << 1 | 1, mid + 1, r, ql, qr));
Update(x);
return ret;
} int main() {
int i, op, l, r, v;
In(n), In(m), Build(1, 1, n);
while (m) {
--m, In(op), In(l), In(r);
if (op == 1) In(v), Modify_add(1, 1, n, l, r, v);
else if (op == 2) In(v), Modify_max(1, 1, n, l, r, v);
else if (op == 3) Out(Query_min(1, 1, n, l, r)), Putc('\n');
else Out(Query_hmin(1, 1, n, l, r)), Putc('\n');
}
return Flush(), 0;
}

UOJ169. 【UR #11】元旦老人与数列的更多相关文章

  1. 2018.07.28 uoj#169. 【UR #11】元旦老人与数列(线段树)

    传送门 线段树好题. 维护区间加,区间取最大值,维护区间最小值,历史区间最小值. 同样先考虑不用维护历史区间最小值的情况,这个可以参考这道题的解法,维护区间最小和次小值可以解决前两个操作,然后使用历史 ...

  2. uoj169:元旦老人与数列

    题意:http://uoj.ac/problem/169 sol  :线段树..........蜜汁TLE了一个点,不管了..... 代码抄snowMyDream的,orz........... 线段 ...

  3. 【UOJ#169】元旦老人与数列

    论文题. 考虑到这题的维护和区间操作是反向的,也就是说无法像V那题快速的合并标记. 我们知道,一个区间的最小值和其他值是可以分开来维护的,因为如果一个区间被整体覆盖,那么最小值始终是最小值. 对于被覆 ...

  4. [UOJ #167]【UR #11】元旦老人与汉诺塔

    题目大意:给你一个有$n$个盘子的汉诺塔状态$S$,问有多少种不同的操作方法,使得可以在$m$步以内到达状态$T$.$n,m\leqslant100$ 题解:首先可以知道的是,一个状态最多可以转移到其 ...

  5. UR11 A.元旦老人与汉诺塔

    题目:http://uoj.ac/contest/23/problem/167 如果我们拿个map来存状态的话.设当前状态是v,下一个状态是s.有f[i+1][s]+=f[i][v]. 初始f[0][ ...

  6. UOJ.52.[UR #4]元旦激光炮(交互 思路)

    题目链接 \(Description\) 交互库中有三个排好序的,长度分别为\(n_a,n_b,n_c\)的数组\(a,b,c\).你需要求出所有元素中第\(k\)小的数.你可以调用至多\(100\) ...

  7. 【uoj#51】[UR #4]元旦三侠的游戏 博弈论+dp

    题目描述 给出 $n$ 和 $m$ ,$m$ 次询问.每次询问给出 $a$ 和 $b$ ,两人轮流选择:将 $a$ 加一或者将 $b$ 加一,但必须保证 $a^b\le n$ ,无法操作者输,问先手是 ...

  8. 网路流 uoj 168 元旦老人与丛林

    http://uoj.ac/problem/168 没想到是网络流 官方题解地址 http://jiry-2.blog.uoj.ac/blog/1115 subtask2告诉我们度数为012的点对答案 ...

  9. uoj167 元旦老人与汉诺塔(记忆化搜索)

    QwQ太懒了,题目直接复制uoj的了 QwQ这个题可以说是十分玄学的一道题了 首先可以暴搜,就是\(dfs\)然后模拟每个过程是哪个柱子向哪个柱子移动 不多解释了,不过实现起来还是有一点点难度的 直接 ...

随机推荐

  1. var 是 Java 开发的好朋友啊!

    简评:Java var != JavaScript var. Java 10 中引入了新的语法用于局部变量类型推断,很多开发者有所疑惑,希望这篇文章能帮到你. 什么是类型推断 其实在 Java 中类型 ...

  2. 38.oracle开篇

    先不聊技术,咱先闷骚一下.刚看完“解忧杂货店”的第二章“深夜的口琴声”,这一章勾起了我万千思绪,小说毕竟是小说,可能与现实不符,但能引发思考,反应一个普遍问题就是好小说.看到一半我还特意去酷狗上搜了一 ...

  3. windows下vim中文乱码处理

    现象:gvim安装后,打开中文utf-8编码的文件中文显示乱码 处理:1.启动gvim8.0,菜单 ”编辑“->"启动设定"在文件最开始处添加如下两行set fileenco ...

  4. 微信小程序授权流程(图)

    看图: 1. 前端H5调用JSSKD的方法  wx.login(), 2. 微信返回 临时登录凭证code 给前端H5 3. 将此Code给到开发者服务器 4. 服务器混同小程序的AppId 和 Ap ...

  5. activity启动模式launchMode区别和优化

    初学android的开发人员,可能会经常忽略这个重要的设置. Activity一共有以下四种launchMode:1.standard2.singleTop3.singleTask4.singleIn ...

  6. spring boot快速入门 6: 表单验证

    廖师兄源码: https://gitee.com/liaoshixiong/girl 样例:拦截所有未满18岁的女生 第一步:在girl实体类中:添加注解 @Min(value=18 ,message ...

  7. 【CSS3】background-origin和background-clip的区别

    background-clip 与 background-origin是css3中引入的两个跟元素背景相关的属性,它们有相同的可选值,即border.padding.content三种,而且这两个属性 ...

  8. 18.async函数

    1.含义 async 函数是什么?一句话,它就是 Generator 函数的语法糖. 前文有一个 Generator 函数,依次读取两个文件. var fs = require('fs'); var ...

  9. Java网络编程(二)关于Socket的一些个人想法

    1.Socket之间是如何通信的? 1.1 通信是要两两之间进行的所以应该有至少一个客户端(Client)和一个服务器端(Server),一般来说都是多个c端对一个s端---c\s 1.2 在客户端: ...

  10. 传统网络栈与InfiniBand栈对比图

    熟悉传统网络协议栈但对InfiniBand协议栈好奇的朋友可以通过下图有一个宏观上的印象. IB实现了"高带宽,低延时"的网络互联,最大的魅力就是支持RDMA,而RDMA就其本质, ...