[BZOJ2138]stone(Hall定理,线段树)
Description
话说Nan在海边等人,预计还要等上M分钟。为了打发时间,他玩起了石子。Nan搬来了N堆石子,编号为1到N,每堆
包含Ai颗石子。每1分钟,Nan会在编号在\([L_i,R_i]\)之间的石堆中挑出任意Ki颗扔向大海(好疼的玩法),如果\([L_i,R_i]\)剩下石子不够\(K_i\)颗,则取尽量地多。为了保留扔石子的新鲜感,Nan保证任意两个区间\([L_i,R_i]\)和\([L_j,R_j]\),不会存在\(L_i<=L_j\& R_j<=R_i\)的情况,即任意两段区间不存在包含关系。可是,如果选择不当,可能无法扔出最多的石子,这时Nan就会不高兴了。所以他希望制定一个计划,他告诉你他m分钟打算扔的区间\([L_i,R_i]\)以及\(K_i\)。现在他想你告诉他,在满足前i-1分钟都取到你回答的颗数的情况下,第i分钟最多能取多少个石子。
\(n\le 40000\)
Solution
Hall定理:
设二分图中\(\text{G=< V1,V2,E >}\)中 \(\text{|V1|=m<=|V2|=n}\) ,\(\text{G}\)中存在从 \(\text{V1}\)到\(\text{V2}\)的完全匹配当且仅当\(\text{V1}\)中任意\(\text{k(k=1,2,...,m)}\)个顶点至少与\(\text{V2}\)中\(\text{k}\)个顶点是相邻的。
建立二分图匹配模型,左部节点为石头(每个点拆为A[i]个点),右部节点为需求(每个点拆为K[i]个点),从小到大依次加入右部节点,然后询问在之前需求点匹配数不变的情况下该点最多能匹配的点数。
不难发现最后是一部分完美匹配+一部分不完美匹配,根据hall定理,一些点具有完美匹配要求任取左部区间,该区间石头数大于等于包含在区间内的需求数。
那么设 \(a[i]\) 表示右端点 \(\le i\) 的区间需求量, \(b[i]\) 为左端点 \(\le i\) 的区间的需求量。
那么区间 \([l,r]\) :
石子数 \(=s[r] - s[l - 1]\)
需求数 \(=a[r] - b[l - 1]\)
那么只要满足 \(\forall 0 \le i < j \le n\),
\]
设 \(f[i] = s[i] - a[i], g[i] = s[i] - b[i]\)。
考虑区间 \([l,r]\) 满足 \(k\) 的需求量造成的影响。
g[l...n] -= k
\]
接下来分类讨论,列出不等式:
f[j] - k - g[i] \geq 0\\
k\le min(f[j]) - max(g[i])\\
k\le min(f[r+1...n]) - max(g[1...l-1])
\]
当 \(k\) 取 \(min(f[r+1...n]) - max(g[0...l-1])\) 时最优。
用线段树维护即可。
Code
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <fstream>
typedef long long LL;
typedef unsigned long long uLL;
#define SZ(x) ((int)x.size())
#define ALL(x) (x).begin(), (x).end()
#define MP(x, y) std::make_pair(x, y)
#define DE(x) cerr << x << endl;
#define debug(...) fprintf(stderr, __VA_ARGS__)
#define GO cerr << "GO" << endl;
#define rep(i, a, b) for (register int (i) = (a); (i) <= (b); ++(i))
using namespace std;
inline void proc_status()
{
ifstream t("/proc/self/status");
cerr << string(istreambuf_iterator<char>(t), istreambuf_iterator<char>()) << endl;
}
inline int read()
{
register int x = 0; register int f = 1; register char c;
while (!isdigit(c = getchar())) if (c == '-') f = -1;
while (x = (x << 1) + (x << 3) + (c xor 48), isdigit(c = getchar()));
return x * f;
}
template<class T> inline void write(T x)
{
static char stk[30]; static int top = 0;
if (x < 0) { x = -x, putchar('-'); }
while (stk[++top] = x % 10 xor 48, x /= 10, x);
while (putchar(stk[top--]), top);
}
template<typename T> inline bool chkmin(T &a, T b) { return a > b ? a = b, 1 : 0; }
template<typename T> inline bool chkmax(T &a, T b) { return a < b ? a = b, 1 : 0; }
const int maxN = 4e4;
int n, m;
int s[maxN + 2], K[maxN + 2];
#define ls (x << 1)
#define rs (x << 1 | 1)
#define Rson rs, mid + 1, r
#define Lson ls, l, mid
int f[maxN << 2], g[maxN << 2], tagG[maxN << 2], tagF[maxN << 2];
void pushup(int x)
{
f[x] = min(f[ls], f[rs]);
g[x] = max(g[ls], g[rs]);
}
void build(int x, int l, int r)
{
if (l == r) { f[x] = g[x] = s[l]; return; }
int mid = l + r >> 1;
build(Lson); build(Rson);
pushup(x);
}
void pushG(int x, int k)
{
tagG[x] += k;
g[x] += k;
}
void pushF(int x, int k)
{
tagF[x] += k;
f[x] += k;
}
void pushdown(int x)
{
if (tagG[x] != 0)
{
pushG(ls, tagG[x]);
pushG(rs, tagG[x]);
tagG[x] = 0;
}
if (tagF[x] != 0)
{
pushF(ls, tagF[x]);
pushF(rs, tagF[x]);
tagF[x] = 0;
}
}
void addF(int x, int l, int r, int L, int R, int k)
{
if (L <= l and r <= R) { return pushF(x, k); }
int mid = l + r >> 1;
pushdown(x);
if (L <= mid) addF(Lson, L, R, k);
if (R > mid) addF(Rson, L, R, k);
pushup(x);
}
void addG(int x, int l, int r, int L, int R, int k)
{
if (L <= l and r <= R) { return pushG(x, k); }
int mid = l + r >> 1;
pushdown(x);
if (L <= mid) addG(Lson, L, R, k);
if (R > mid) addG(Rson, L, R, k);
pushup(x);
}
int queryG(int x, int l, int r, int L, int R)
{
if (L <= l and r <= R) { return g[x]; }
int mid = l + r >> 1;
pushdown(x);
int ans = -0x3f3f3f3f;
if (L <= mid) chkmax(ans, queryG(Lson, L, R));
if (mid < R) chkmax(ans, queryG(Rson, L, R));
return ans;
}
int queryF(int x, int l, int r, int L, int R)
{
if (L <= l and r <= R) { return f[x]; }
int mid = l + r >> 1;
pushdown(x);
int ans = 0x3f3f3f3f;
if (L <= mid) chkmin(ans, queryF(Lson, L, R));
if (mid < R) chkmin(ans, queryF(Rson, L, R));
return ans;
}
int main()
{
#ifndef ONLINE_JUDGE
freopen("stone.in", "r", stdin);
freopen("stone.out", "w", stdout);
#endif
int x, y, z, P;
scanf("%d", &n);
scanf("%d%d%d%d", &x, &y, &z, &P);
for (int i = 1; i <= n; ++i)
s[i] = s[i - 1] + ((i - x) * (i - x) + (i - y) * (i - y) + (i - z) * (i - z)) % P;
scanf("%d", &m);
scanf("%d%d%d%d%d%d", &K[1], &K[2], &x, &y, &z, &P);
for (int i = 3; i <= m; ++i)
K[i] = (1ll * x * K[i - 1] + 1ll * y * K[i - 2] + z) % P;
build(1, 0, n);
for (int i = 1; i <= m; ++i)
{
int L, R, k;
scanf("%d%d", &L, &R);
printf("%d\n", k = min(queryF(1, 0, n, R, n) - queryG(1, 0, n, 0, L - 1), K[i]));
addF(1, 0, n, R, n, -k);
addG(1, 0, n, L, n, -k);
}
return 0;
}
[BZOJ2138]stone(Hall定理,线段树)的更多相关文章
- 【BZOJ2138】stone Hall定理+线段树
[BZOJ2138]stone Description 话说Nan在海边等人,预计还要等上M分钟.为了打发时间,他玩起了石子.Nan搬来了N堆石子,编号为1到N,每堆包含Ai颗石子.每1分钟,Nan会 ...
- [BZOJ2138]stone[霍尔定理+线段树]
题意 一共有 \(n\) 堆石子,每堆石子有一个数量 \(a\) ,你要进行 \(m\) 次操作,每次操作你可以在满足前 \(i-1\) 次操作的回答的基础上选择在 \([L_i,R_i]\) 区间中 ...
- BZOJ.3693.圆桌会议(Hall定理 线段树)
题目链接 先考虑链.题目相当于求是否存在完备匹配.那么由Hall定理,对于任意一个区间[L,R],都要满足[li,ri]完全在[L,R]中的ai之和sum小于等于总位置数,即R-L+1.(其实用不到H ...
- LOJ.6062.[2017山东一轮集训]Pair(Hall定理 线段树)
题目链接 首先Bi之间的大小关系没用,先对它排序,假设从小到大排 那么每个Ai所能匹配的Bi就是一个B[]的后缀 把一个B[]后缀的匹配看做一条边的覆盖,设Xi为Bi被覆盖的次数 容易想到 对于每个i ...
- loj#6062. 「2017 山东一轮集训 Day2」Pair hall定理+线段树
题意:给出一个长度为 n的数列 a和一个长度为 m 的数列 b,求 a有多少个长度为 m的连续子数列能与 b匹配.两个数列可以匹配,当且仅当存在一种方案,使两个数列中的数可以两两配对,两个数可以配对当 ...
- BZOJ3693: 圆桌会议(Hall定理 线段树)
题意 题目链接 Sol 好的又是神仙题... 我的思路:对于区间分两种情况讨论,一种是完全包含,另一种是部分包含.第一种情况非常好判断,至于计算对于一个区间[l, r]的$\sum a[i]$就可以了 ...
- 模拟赛 怨灵退治 题解(Hall定理+线段树)
题意: 有 n 群怨灵排成一排,燐每秒钟会选择一段区间,消灭至多 k 只怨灵. 如果怨灵数量不足 k,则会消灭尽量多的怨灵. 燐作为一只有特点的猫,它选择的区间是不会相互包含的.它想要知道它每秒最多能 ...
- Codeforces 338E - Optimize!(Hall 定理+线段树)
题面传送门 首先 \(b_i\) 的顺序肯定不会影响匹配,故我们可以直接将 \(b\) 数组从小到大排个序. 我们考虑分析一下什么样的长度为 \(m\) 的数组 \(a_1,a_2,\dots,a_m ...
- BZOJ1135 LYZ(POI2009) Hall定理+线段树
做这个题之前首先要了解判定二分图有没有完备匹配的Hall定理: 那么根据Hell定理,如果任何一个X子集都能连大于等于|S|的Y子集就可以获得完备匹配,那么就是: 题目变成只要不满足上面这个条件就能得 ...
随机推荐
- MTCNN 人脸检测
demo.py import cv2 from detection.mtcnn import MTCNN # 检测图片中的人脸 def test_image(imgpath): mtcnn = MTC ...
- ASE Beta Sprint - backend scrum 1
本次scrum于2019.12.2与前端组和模型组一起在sky garden进行,持续50分钟. 参与人: Xin Kang, Zhikai Chen, Lihao Ran, Hao Wang 请假: ...
- 并发工具CountDownLatch源码分析
CountDownLatch的作用类似于Thread.join()方法,但比join()更加灵活.它可以等待多个线程(取决于实例化时声明的数量)都达到预期状态或者完成工作以后,通知其他正在等待的线程继 ...
- nginx+flask+gevent+uwsgi实现websocket
Websocket简介 WebSocket是HTML5开始提供的一种在单个 TCP 连接上进行全双工通讯的协议.在WebSocket API中,浏览器和服务器只需要做一个握手的动作,然后,浏览器和服务 ...
- centos6 安装mysql5.77(开发版)
1. 配置yum源: [root@yyf ~]#rpm -Uvh http://dev.mysql.com/get/mysql-community-release-el6-5.noarch.rpm [ ...
- JAVA泛型通配符T,E,K,V区别,T以及Class<T>,Class<?>的区别
1. 先解释下泛型概念 泛型是Java SE 1.5的新特性,泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数.这种参数类型可以用在类.接口和方法的创建中,分别称为泛型类.泛型接口.泛 ...
- [BZOJ2600] ricehub
问题描述 乡间有一条笔直而长的路称为"米道".沿着这条米道上 R 块稻田,每块稻田的坐标均为一个 1 到 L 之间(含 1 和 L)的整数.这些稻田按照坐标以不减的顺序给出,即对于 ...
- 对props的研究
Vue.component('my-component', { props: { // 基础的类型检查 (`null` 匹配任何类型) propA: Number, // 多个可能的类型 propB: ...
- Python_014(面向对象之继承)
一.面向对象之继承 1.初始继承 引入:面向对象的三大特性:继承,多态,封装 a.继承是创建新类的一种方式,在python中,新建的类可以继承一个或多个父类,父类又可称为基类或超类;新建的类称为派生类 ...
- 【HDOJ6659】Acesrc and Good Numbers(dfs)
题意:定义f(n,d)为数码d在1到n中出现的次数,其中d=0..9 如果f(d,k)=k,则称k是d好数 给定x和d,求不大于x的最大的d好数 x<=1e18 思路:考虑f的增长率主要和位数有 ...