Codechef MGCHGYM Misha and Gym 容斥、背包、Splay
简化题意:给定一个长度为\(N\)的数列,\(Q\)个操作:
\(1\,x\,a\)、将数列中第\(x\)个元素改为\(a\)
\(2\,l\,r\)、反转子序列\([l,r]\)
\(3\,l\,r\,w\)、询问区间\([l,r]\)中是否存在若干个数和为\(w\),一个数只能取一次
注意:在整个过程中,在数列中出现过的数的种数不会超过\(K(K \leq 10)\)。
注意到最后一个条件很奇怪……
考虑询问实际上是:最开始给出不超过\(10\)个数\(a_1,...,a_{10}\),每一次给出\(a_1,...,a_{10}\)分别最多能够取的次数,问是否能够取出若干使得和为\(w\);而前两个操作只是在改变这个能够取的最多次数。
不妨更进一步想,试着求能够取的方案数……
是不是想到了……
那么我们可以直接按照硬币购物的方法去做
先用\(a_1\)到\(a_{10}\)跑完全背包,对于每一次询问进行容斥,强制令某一些数字超出使用次数并计算答案。那么每一次询问的复杂度是\(2^{10}\)的。
最后使用\(Splay\)维护一下前两个修改操作,题目就做完了。
关于完全背包存不下那么多方案数的问题……直接模\(10^9+7\)
#include<bits/stdc++.h>
#define lch Tree[x].ch[0]
#define rch Tree[x].ch[1]
#define root Tree[0].ch[0]
//This code is written by Itst
using namespace std;
inline int read(){
int a = 0;
char c = getchar();
bool f = 0;
while(!isdigit(c) && c != EOF){
if(c == '-')
f = 1;
c = getchar();
}
if(c == EOF)
exit(0);
while(isdigit(c)){
a = a * 10 + c - 48;
c = getchar();
}
return f ? -a : a;
}
const int MAXN = 1e5 + 10 , MOD = 1e9 + 7;
int dp[MAXN] , dir[11] , *cnt , N , Q , cntN , cntL;
map < int , int > lsh;
struct node{
int ch[2] , sz , fa , val , sum[11];
bool mark;
}Tree[MAXN];
struct query{
int ind , a , b , c;
}que[MAXN];
inline int getL(int x){
if(!lsh.count(x)){
lsh[x] = ++cntL;
dir[cntL] = x;
}
return lsh[x];
}
inline bool son(int x){
return Tree[Tree[x].fa].ch[1] == x;
}
inline void pushup(int x){
for(int i = 1 ; i <= 10 ; ++i)
Tree[x].sum[i] = Tree[lch].sum[i] + Tree[rch].sum[i] + (Tree[x].val == i);
Tree[x].sz = Tree[lch].sz + Tree[rch].sz + 1;
}
inline void rotate(int x){
bool f = son(x);
int y = Tree[x].fa , z = Tree[y].fa , w = Tree[x].ch[f ^ 1];
Tree[x].fa = z;
Tree[z].ch[son(y)] = x;
Tree[x].ch[f ^ 1] = y;
Tree[y].fa = x;
Tree[y].ch[f] = w;
if(w)
Tree[w].fa = y;
pushup(y);
}
inline void Splay(int x , int tar){
while(Tree[x].fa != tar){
if(Tree[Tree[x].fa].fa != tar)
rotate(son(x) == son(Tree[x].fa) ? Tree[x].fa : x);
rotate(x);
}
pushup(x);
}
inline void mark(int x){
if(!x)
return;
swap(lch , rch);
Tree[x].mark ^= 1;
}
inline void pushdown(int x){
if(Tree[x].mark){
mark(lch);
mark(rch);
Tree[x].mark = 0;
}
}
void insert(int &x , int rk , int val , int fa){
if(!x){
x = ++cntN;
Tree[x].fa = fa;
Tree[x].sz = 1;
Tree[x].val = val;
Splay(x , 0);
return;
}
if(Tree[lch].sz >= rk)
insert(lch , rk , val , x);
else
insert(rch , rk - 1 - Tree[lch].sz , val , x);
}
void findKth(int x , int rk , int tar){
pushdown(x);
if(Tree[lch].sz == rk)
Splay(x , tar);
else
if(Tree[lch].sz > rk)
findKth(lch , rk , tar);
else
findKth(rch , rk - Tree[lch].sz - 1 , tar);
}
inline void modify(int x , int val){
findKth(root , x , 0);
--Tree[root].sum[Tree[root].val];
++Tree[root].sum[Tree[root].val = val];
}
inline void rev(int l , int r){
findKth(root , l - 1 , 0);
findKth(root , r + 1 , root);
mark(Tree[Tree[root].ch[1]].ch[0]);
}
inline void query(int l , int r){
findKth(root , l - 1 , 0);
findKth(root , r + 1 , root);
cnt = Tree[Tree[Tree[root].ch[1]].ch[0]].sum;
}
void init(){
dp[0] = 1;
for(int i = 1 ; i <= cntL ; ++i)
for(int j = dir[i] ; j <= 1e5 ; ++j)
dp[j] = (dp[j] + dp[j - dir[i]]) % MOD;
}
int dfs(int x , int sum , int flg){
if(sum < 0)
return 0;
if(x > cntL)
return flg * dp[sum];
return (dfs(x + 1 , sum , flg) + dfs(x + 1 , sum - (cnt[x] + 1) * dir[x] , flg * -1) + 1ll * MOD) % MOD;
}
int main(){
#ifndef ONLINE_JUDGE
freopen("in","r",stdin);
freopen("out","w",stdout);
#endif
N = read();
Q = read();
insert(root , 0 , 0 , 0);
for(int i = 1 ; i <= N ; ++i)
insert(root , i , getL(read()) , 0);
insert(root , N + 1 , 0 , 0);
for(int i = 1 ; i <= Q ; ++i){
que[i].ind = read();
que[i].a = read();
que[i].b = read();
if(que[i].ind == 3)
que[i].c = read();
if(que[i].ind == 1)
que[i].b = getL(que[i].b);
}
init();
for(int i = 1 ; i <= Q ; ++i)
switch(que[i].ind){
case 1:
modify(que[i].a , que[i].b);
break;
case 2:
rev(que[i].a , que[i].b);
break;
case 3:
query(que[i].a , que[i].b);
puts(dfs(1 , que[i].c , 1) ? "Yes" : "No");
}
return 0;
}
Codechef MGCHGYM Misha and Gym 容斥、背包、Splay的更多相关文章
- BZOJ 1042: [HAOI2008]硬币购物 容斥+背包
1042: [HAOI2008]硬币购物 Description 硬币购物一共有4种硬币.面值分别为c1,c2,c3,c4.某人去商店买东西,去了tot次.每次带di枚ci硬币,买si的价值的东西.请 ...
- Atcoder Grand Contest 038 E - Gachapon(Min-Max 容斥+背包)
Atcoder 题面传送门 & 洛谷题面传送门 我竟然能独立做出 Ag 的 AGC E,incredible!更新了 Atcoder 做题难度上限( 首先按照套路 Min-Max 容斥,\(a ...
- 洛谷 P4707 - 重返现世(扩展 Min-Max 容斥+背包)
题面传送门 首先看到这种求形如 \(E(\max(T))\) 的期望题,可以套路地想到 Min-Max 容斥 \(\max(S)=\sum\limits_{T\subseteq S}(-1)^{|T| ...
- 牛客练习赛64 D【容斥+背包】
牛客练习赛64 D.宝石装箱 Description \(n\)颗宝石装进\(n\)个箱子使得每个箱子中都有一颗宝石.第\(i\)颗宝石不能装入第\(a_i\)个箱子.求合法的装箱方案对\(99824 ...
- P1450 [HAOI2008]硬币购物(完全背包+容斥)
P1450 [HAOI2008]硬币购物 暴力做法:每次询问跑一遍多重背包. 考虑正解 其实每次跑多重背包都有一部分是被重复算的,浪费了大量时间 考虑先做一遍完全背包 算出$f[i]$表示买价值$i$ ...
- ARC 101E.Ribbons on Tree(容斥 DP 树形背包)
题目链接 \(Description\) 给定一棵\(n\)个点的树.将这\(n\)个点两两配对,并对每一对点的最短路径染色.求有多少种配对方案使得所有边都至少被染色一次. \(n\leq5000\) ...
- BZOJ-1042:硬币购物(背包+容斥)
题意:硬币购物一共有4种硬币.面值分别为c1,c2,c3,c4.某人去商店买东西,去了tot次.每次带di枚ci硬币,买si的价值的东西.请问每次有多少种付款方法. 思路:这么老的题,居然今天才做到. ...
- Luogu-P1450 [HAOI2008]硬币购物-完全背包+容斥定理
Luogu-P1450 [HAOI2008]硬币购物-完全背包+容斥定理 [Problem Description] 略 [Solution] 上述题目等价于:有\(4\)种物品,每种物品有\(d_i ...
- BZOJ 1042 [HAOI2008]硬币购物(完全背包+容斥)
题意: 4种硬币买价值为V的商品,每种硬币有numi个,问有多少种买法 1000次询问,numi<1e5 思路: 完全背包计算出没有numi限制下的买法, 然后答案为dp[V]-(s1+s2+s ...
随机推荐
- python自动化开发-7
socket编程 Socket是应用层与TCP/IP协议族通信的中间软件抽象层,它是一组接口.在设计模式中,Socket其实就是一个门面模式,它把复杂的TCP/IP协议族隐藏在Socket接口后面,对 ...
- 二层协议--STP协议总结
生成树协议的技术实现与配置注意点 一.stp协议的用途 二.stp协议的运行机制 三.stp协议规范
- SpringBoot整合Rabbitmq设置消息请求头
String str = "{\"origin\":\"BBC\",\"origin_coupon_id\":51,\" ...
- 第五章 绘图基础(BEZIER)
/*----------------------------- BEZIER.C -- Bezier Splines Demo (c) Charles Petzold, 1998 ---------- ...
- Win7系统system进程句柄数一直增加解决方案
公司内部最近有个服务端的同事电脑句柄数一开机就一直增加 一台Windows7x64系统16G 其实物理内存使用情况在开机后并没有太大的变化,但虚拟内存占用明显在不停的增加. 我通过“任务管理器”一直也 ...
- Hadoop2.7.6_05_mapreduce-Yarn
1. MAPREDUCE原理 Mapreduce是一个分布式运算程序的编程框架,是用户开发“基于hadoop的数据分析应用”的核心框架: Mapreduce核心功能是将用户编写的业务逻辑代码和自带默认 ...
- January 25th, 2018 Week 04th Thursday
What made something precious? Losing it and finding it. 一件东西怎样才会变得珍贵无比?先弄丢了它,然后又找到了. A short time ag ...
- if语句的嵌套以及条件运算符和条件表达式(初学者)
1.当if语句中的执行语句又是if语句时,则构成了if语句的嵌套情形. 其一般形式可表示为: if() { if()……; } 或: if() if()语句1: else 语句2: else if() ...
- Beta冲刺! Day1 - 磨刀
Beta冲刺! Day1 - 磨刀 今日已完成 晨瑶:罗列Beta计划.和新人交接.任务安排 昭锡:无 永盛:服务器出现一些 mysql 的问题,伟鹏的爬取脚本没办法远程链接到服务器,在修 立强:学习 ...
- Java面试——微服务
1.什么是微服务? 就目前而言,对于微服务业界并没有一个统一的,标准的定义. 但通常而言,微服务架构是一种架构模式或者说是一种架构风格,它提倡将单一应用程序划分一组小的服务,每个服务运行在其独立 ...