题意

题目链接

Sol

好的又是神仙题。。。

我的思路:对于区间分两种情况讨论,一种是完全包含,另一种是部分包含。
第一种情况非常好判断,至于计算对于一个区间[l, r]的$\sum a[i]$就可以了,但是后两种呢?qwq。想了半天也没想出来。
看了下题解,果然还有更高端的操作!

首先这题可以看是二分图匹配,最暴力的写法是对于每个a[i],直接拆成a[i]个点,然后分别向$[l_i, r_i]$连边,最后看是否能完全匹配。

有一个专门判断这玩意儿的定理:

Hall定理:
二部图G中的两部分顶点组成的集合分别为$X, Y$, $X = \{X1, X2, X3,X4,.........,Xm\},$Y=\{y1, y2, y3, y4 ,.........,yn\},G中有一组无公共点的边,一端恰好为组成X的点的充分必要条件是:
X中的任意k个点至少与Y中的k个点相邻。(1≤k≤m)

对于此题来说,直接应用Hall定理得到的推论为:对于任意的x个人,都至少对应x条边与其相连

然而这样好像还是不好搞,考虑一步步推广

1、对于任意一个询问$[l, r], a_i$,若$a_i$满足要求,那么任意的$x <= a_i$,都满足要求。

这是显然的,因为每个$a_i$连的点都是相同的

2、对于任意的区间$[l, r]$,若他们包含的$a[i]$, $\sum a[i] <= r - l + 1$满足条件,则去掉任意的$a[i]$后,该区间仍然满足条件。

同样显然。

这样我们就把给出的问题转化为:判断对于任意$[l_j, r_i]$,是否满足条件

对所有询问按右端点排序后线段树维护

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<map>
#include<vector>
#include<cstring>
#define Pair pair<int, int>
#define MP(x, y) make_pair(x, y)
#define fi first
#define se second
#define LL long long
using namespace std;
const int MAXN = * 1e6 + , mod = 1e9 + ;
inline LL read() {
char c = getchar(); LL x = , f = ;
while(c < '' || c > '') {if(c == '-') f = -; c = getchar();}
while(c >= '' && c <= '') x = x * + c - '', c = getchar();
return x * f;
}
#define ls k << 1
#define rs k << 1 | 1
int T, N, M;
int mx[MAXN], f[MAXN], date[MAXN];
struct Qu {
int l, r, a;
bool operator < (const Qu &rhs) const {
return r == rhs.r ? l < rhs.l : r < rhs.r;
}
}q[MAXN];
void update(int k) {
mx[k] = max(mx[ls], mx[rs]);
}
void add(int k, int val) {
mx[k] += val, f[k] += val;
}
void pushdown(int k) {
if(f[k]) add(ls, f[k]), add(rs, f[k]), f[k] = ;
}
void IntAdd(int k, int ll, int rr, int l, int r, int val) {
if(ll <= l && r <= rr) {add(k, val); return ;}
int mid = l + r >> ;
pushdown(k);
if(ll <= mid) IntAdd(ls, ll, rr, l, mid, val);
if(rr > mid) IntAdd(rs, ll, rr, mid + , r, val);
update(k);
}
int Query(int k, int ll, int rr, int l, int r) {
if(ll <= l && r <= rr) return mx[k];
int mid = l + r >> ;
pushdown(k);
if(ll > mid) return Query(rs, ll, rr, mid + , r);
else if(rr <= mid) return Query(ls, ll, rr, l, mid);
else return max(Query(ls, ll, rr, l, mid), Query(rs, ll, rr, mid + , r));
}
main() {
T = read();
while(T--) {
memset(mx, , sizeof(mx));
memset(f, , sizeof(f));
N = read(); M = read();
int cnt = , tot = ;
LL sum = ;
for(int i = ; i <= N; i++) {
q[++cnt].l = read(), q[cnt].r = read(), q[cnt].a = read();
sum += q[cnt].a;
if(q[cnt].l > q[cnt].r) q[cnt].r += M;
else if(q[cnt].r < M) q[cnt + ] = (Qu) {q[cnt].l + M, q[cnt].r + M, q[cnt].a}, cnt++;
}
if(sum > M) {puts("No"); continue;}
for(int i = ; i <= cnt; i++) q[i].l++, q[i].r++, date[++tot] = q[i].l, date[++tot] = q[i].r;
for(int i = ; i <= * N; i++) date[++tot] = i; sort(q + , q + cnt + );
sort(date + , date + tot + );
tot = unique(date + , date + tot + ) - date - ; int cur = , flag = ;
for(int i = ; i <= cnt; i++) {
int l = q[i].l, r = q[i].r;
l = lower_bound(date + , date + tot + , l) - date;
r = lower_bound(date + , date + tot + , r) - date;
while(cur < r) cur++, IntAdd(, cur, cur, , tot, date[cur] - );
IntAdd(, , l, , tot, q[i].a);
int val = Query(, , r, , tot);
if(val > date[r]) {puts("No"); flag = ; break;}
}
if(!flag) puts("Yes"); }
return ;
}

最新文章

  1. io.js入门(一)—— 初识io.js
  2. 《storm实战-构建大数据实时计算读书笔记》
  3. MVC3 数据验证用法之密码验证设计思路
  4. 快消零售行业怎么用K2做开关店管理?
  5. paper 61:计算机视觉领域的一些牛人博客,超有实力的研究机构等的网站链接
  6. Win8驱动测试模式
  7. 在eclipse中调试web项目的时候如何把web项目分配给配置好的服务器
  8. nginx之如何获取真实客户端ip
  9. js动态加载控件jsp页面
  10. B - ACboy needs your help(动态规划,分组背包)
  11. OC比C中,新增的数据类型
  12. Charles使用(一)
  13. acffo的开源项目汇总
  14. H5_0006:JS判断PC,平板,手机平台的方法
  15. 2014年蓝桥杯省赛A组c++第2题(推公式)
  16. spring security文档地址
  17. 利用 jrebel 热部署\远程调试\远程热部署 springboot项目 服务器上的代码
  18. c++11 时间类 std::chrono
  19. Gulp命令自动生成精灵图
  20. rpm包的安装,查询,卸载,升级,校验,数据库重建,验证数据包

热门文章

  1. spider(一)
  2. sklearn正规化(Normalization或者scale)
  3. CODING 代码托管架构升级之路
  4. 剑指offer(65):获取数据流中的中位数
  5. 获得Servlet的初始化参数
  6. hdu1099
  7. hdu1073
  8. 剑指Spring源码(三)俯瞰Spring的Bean的生命周期(大众版)
  9. 【转】C# 使用正则表达式去掉字符串中的数字,或者去掉字符串中的非数字
  10. [Xcode 实际操作]五、使用表格-(6)UITableView滑动到指定单元格