ZR1153
ZR1153
首先我们可以发现一个比较简单的容斥做法
直接暴力枚举\(2^m\)个限制强制不合法,算贡献
注意如果两个限制冲突那么答案为0
直接暴力差分就好了
这样就有了快乐的\(30\)分了
接下来考虑对容斥进行DP
把所有点区间按照右端点排序,如果出来两个颜色相同的区间一个包含了另外一个,那么大区间是没有用的,因为小区间满足条件大区间一定满足
我们设\(f_{i}\)表示满足第\(i\)个限制的带容斥系数的方案数
那么转移我们就枚举上一个没有交的区间
\]
其中\(g_x\)表示\(1-x\)位置满足\(1-x\)的所有容斥之后的限制的前缀和
也就是说
\]
就是看看第\(i\)位置上的限制满足还是不满足综合考虑的前缀和
继续回到求\(f_i\)的式子
既然\(i\)的这个限制要容斥,那么强制他不被满足,前面就是对所有和他没有交的限制求一个总的容斥
后面算有交的部分的贡献,必须满足和当前限制的颜色相同,
我们排序之后,有交的集合是一个区间,我们二分找到对应位置维护前缀和即可
#include<cstdio>
#include<iostream>
#include<queue>
#include<algorithm>
#include<cstring>
#include<cctype>
#include<vector>
#include<ctime>
#include<cmath>
#include<set>
#include<map>
#define LL long long
#define pii pair<int,int>
#define mk make_pair
#define fi first
#define se second
using namespace std;
const int N = 4e5 + 3;
const LL mod = 998244353;
struct seg{
int li,ri;
int xi;
}a[N],b[N];
vector <pii> co[N];
vector <LL> h[N];
LL f[N],g[N];
int n,m,s,cnt;
inline int read(){
int v = 0,c = 1;char ch = getchar();
while(!isdigit(ch)){
if(ch == '-') c = -1;
ch = getchar();
}
while(isdigit(ch)){
v = v * 10 + ch - 48;
ch = getchar();
}
return v * c;
}
inline bool cmp(seg x,seg y){
return x.ri < y.ri || (x.ri == y.li && x.li > y.li);
}
inline LL find(int x,int rr){
// printf("%d %d\n",x,rr);
int l = 0,r = h[x].size() - 1,ans = -1;
if(r < 0) return 0;
while(l <= r){
int mid = (l + r) >> 1;
if(co[x][mid].se < rr) l = mid + 1,ans = mid;
else r = mid - 1;
}
// printf("%d %d %d %lld\n",l,r,ans,ans == -1 ? h[x].back() : h[x].back() - h[x][ans]);
return ans == -1 ? h[x].back() : h[x].back() - h[x][ans];
}
inline LL mo(LL x){
if(x >= mod) x-= mod;
return x;
}
int main(){
n = read(),m = read(),s = read();
for(int i = 1;i <= m;++i){
a[i].li = read();
a[i].ri = read();
a[i].xi = read();
}
sort(a + 1,a + m + 1,cmp);
for(int i = 1;i <= m;++i){
bool flag = 0;
if(!co[a[i].xi].size()) co[a[i].xi].push_back(mk(a[i].li,a[i].ri));
else{
pii x = co[a[i].xi].back();
if(x.fi >= a[i].li && x.se <= a[i].ri) flag = 1;
else co[a[i].xi].push_back(mk(a[i].li,a[i].ri));
}
if(!flag) b[++cnt] = a[i];
}
m = cnt;
memcpy(a,b,sizeof(a));
// puts("new::");
// for(int i = 1;i <= m;++i) cerr << a[i].li << " " << a[i].ri << " " << a[i].xi << endl;
// puts("next::");
int now = 1;
f[0] = g[0] = 1;
for(int i = 1;i <= m;++i){
// cerr << "dsdas::"<< a[i].li << " " << a[i].ri << " " << a[i].xi << endl;
for(;now < a[i].ri;now++) g[now] = (g[now] + g[now - 1] * s) % mod;
f[i] = (-g[a[i].li - 1] + mod);
f[i] -= find(a[i].xi,a[i].li);
if(f[i] < 0) f[i] += mod;
LL gg = h[a[i].xi].empty() ? 0 : h[a[i].xi].back();
h[a[i].xi].push_back(mo(gg + f[i]));
g[a[i].ri] = mo(g[a[i].ri] + f[i]);
}
for(;now <= n;++now) g[now] = (g[now] + g[now - 1] * s) % mod;
printf("%lld\n",g[n]);
return 0;
}
ZR1153的更多相关文章
随机推荐
- Effective Modern C++:01类型推导
C++的官方钦定版本,都是以ISO标准被接受的年份命名,分别是C++98,C++03,C++11,C++14,C++17,C++20等.C++11及其后续版本统称为Modern C++. C++11之 ...
- JavaScript--天猫数量输入框
<!DOCTYPE html> <head> <meta charset="utf-8" /> <title>无标题文档</t ...
- 从零学React Native之14 网络请求
通过HTTP或者HTTPS协议与网络侧服务器交换数据是移动应用中常见的通信方式. node-fetch是RN推荐的请求方式. React Native框架在初始化项目时, 引入了node-fetch包 ...
- Linux的 crontab定时任务小记
编辑任务 crontab -e 查看任务 crontab -l 任务配置基本格式:* * * * * command分(0-59) 时(0-23) 天(1-31) 月(1-12) 周(0-6,0代 ...
- 「BZOJ3694」「FJ2014集训」最短路
「BZOJ3694」「FJ2014集训」最短路 首先树剖没得说了,这里说一下并查集的做法, 对于一条非树边,它会影响的点就只有u(i),v(i)到lca,对于lca-v的路径上所有点x,都可通过1-t ...
- AtCoder Regular Contest 058
这个应该是第一场有英文的atcoder吧??不过题解却没有英文的... 从前往后慢慢做... C こだわり者いろはちゃん / Iroha's Obsession 数据范围这么小,直接暴力 #inclu ...
- C++第5次作业
运算符重载 定义 - 运算符重载是对已有的运算符赋予多重含义,使同一个运算符作用于不同类型的数据时导致不同行为. 运算符重载规则 - C++运算符除了少数几个,全都可以重载,且只能重载C++已有的运算 ...
- logging.basicConfig函数
在UI自动化应用中,经常会出错,打log就是一个很重要的环节,python的logging.basicConfig函数 真是既方便,又简单,每次粘贴到用例前,就可以打log了. logging模块是 ...
- oracle多个平等的索引
当SQL语句的执行路径可以使用分布在多个表上的多个索引时, ORACLE会同时使用多个索引并在运行时对它们的记录进行合并, 检索出仅对全部索引有效的记录. 在ORACLE选择执行路径时,唯一性索引的等 ...
- POJ2185 Milking Grid 题解 KMP算法
题目链接:http://poj.org/problem?id=2185 题目大意:求一个二维的字符串矩阵的最小覆盖子矩阵,即这个最小覆盖子矩阵在二维空间上不断翻倍后能覆盖原始矩阵. 题目分析:next ...