题面

洛谷P4229 某位歌姬的故事

\(T\) 组测试数据。有 \(n\) 个音节,每个音节 \(h_i\in[1,A]\),还有 \(m\) 个限制 \((l_i,r_i,g_i)\) 表示 \(\max_{k=l_i}^{r_i}h_k=g_i\)。求满足条件的 \(h_i\) 的方案数膜 \(998244353\)。

数据范围:\(1\le T\le 20\),\(1\le l_i\le r_i\le n\le 9\cdot 10^8\),\(1\le g_i\le A\le 9\cdot 10^8\),\(1\le m\le 500\)。


蒟蒻语

这是一眼题,蒟蒻 \(7\) 个小时就秒掉了。

题解做法时间复杂度 \(\Theta(Tm\log m)\)。


蒟蒻解

转化条件:\(\forall k\in[l_i,r_i],h_k\le g_i\) 并且 \(\exists k\in [l_i,r_i],h_k\ge g_i\)。

可以离散化,所以下文中可以把 \(n\) 看成与 \(m\) 相等的范围。

很明显对于 \(g_i<g_j\) 如果一个区间满足了 \(i\) 条件就无法满足 \(j\) 条件了,所以可以按 \(g\) 从小到大排序条件,每次处理完以后把区间删除;但是又会发现对于 \(g\) 相等的区间情况很复杂,所以可以排序后对每个 \(g\) 分开处理覆盖部分答案求积。删除区间可以用并查集(参考),然后最后对于每个没删除的音节答案乘以 \(A\) 即可。

然后只需假设 \(g\) 一定,并且区间覆盖全部音节。这时有多个棘手的条件,根据条件的形式容易想到子集反演

设 \(p=(\forall k\in[l_i,r_i],h_k\le g_i)\),\(q=(\exists k\in [l_i,r_i],h_k\ge g_i)\),所以:

\[p(S)\&q(S)=\sum_{T\in S}(-1)^{|T|}p(S-T)\neg q(T),\neg q(T)=(\forall k\in[l_i,r_i],h_k<g_i)
\]

这个子集反演是可以用 \(\rm dp\) 维护的,\(f(i,j)\) 表示 \(\forall k\in[i,j),h_k<g_i\) 前 \(i\) 个音节的方案数。

\[\forall j\le i,f(i,j)\leftarrow g\cdot f(i-1,j)
\]
\[\forall j> i,f(i,j)\leftarrow (g-1)\cdot f(i-1,j)
\]

假设有个以 \(i\) 为左端点的条件 \([i,t)\),可以用 \(\rm dp\) 维护反演老套路打个负标记并用类似 \(\rm min-max\) 卷积的方法:

\[\forall j,f(i,\max(j,t))\leftarrow f(i,\max(j,t))-f(i,j)
\]

操作比看起来简单:把 \(j\ge t\) 的 \(f(i,j)\) 直接置为 \(0\),然后 \(f(i,t)\leftarrow\sum_{j<t} f(i,j)=\sum f(i,j)\)。

然后是一波骚操作:把 \(i\) 这维滚掉(用离散化和并查集),然后线段树维护 \(j\) 这一维(啊!\(\rm dp\) 沦陷了!)。

最后还有个小细节:要把 \(g\) 相等但是 \([l_i,r_i]\in[l_j,r_j]\) 的 \(j\) 条件不放到 \(\rm dp\) 里,最后删除区间时统计。

时间复杂度 \(\Theta(Tm\log m)\)。 蒟蒻的题解都写成这样了,有人看得懂就怪了。


代码

#include <bits/stdc++.h>
using namespace std; //Start
typedef long long ll;
typedef double db;
#define mp(a,b) make_pair((a),(b))
#define x first
#define y second
#define be(a) (a).begin()
#define en(a) (a).end()
#define sz(a) int((a).size())
#define pb(a) push_back(a)
#define R(i,a,b) for(int i=(a),I=(b);i<I;i++)
#define L(i,a,b) for(int i=(b)-1,I=(a)-1;i>I;i--)
const int iinf=0x3f3f3f3f;
const ll linf=0x3f3f3f3f3f3f3f3f; //Data
const int Q=501,N=Q<<1,mod=998244353;
int Pow(int a,int x){
if(!a) return 0; int res=1;
for(;x;a=1ll*a*a%mod,x>>=1)if(x&1) res=1ll*a*res%mod;
return res;
}
int n,m,h,l[Q],r[Q],g[Q],dn,d[N],te[N+1];
int find(int u){return u==te[u]?u:(te[u]=find(te[u]));}
map<int,vector<int>> mp; //SegmentTree
const int T=N<<2; int val[T],ml[T];
#define mid ((l+r)>>1)
void clear(int k=0,int l=0,int r=dn){
val[k]=0,ml[k]=1; if(r-l==1) return;
clear(k*2+1,l,mid),clear(k*2+2,mid,r);
}
void pushup(int k){(val[k]=val[k*2+1]+val[k*2+2])%=mod;}
void pushmul(int k,int v){val[k]=1ll*val[k]*v%mod,ml[k]=1ll*ml[k]*v%mod;}
void pushdown(int k){if(ml[k]^1) pushmul(k*2+1,ml[k]),pushmul(k*2+2,ml[k]),ml[k]=1;}
void add(int x,int v,int k=0,int l=0,int r=dn){
if(r<=x||x+1<=l) return; if(r-l==1) return void((val[k]+=v)%=mod);
pushdown(k),add(x,v,k*2+1,l,mid),add(x,v,k*2+2,mid,r),pushup(k);
}
void mul(int x,int y,int v,int k=0,int l=0,int r=dn){
if(r<=x||y<=l) return; if(x<=l&&r<=y) return pushmul(k,v);
pushdown(k),mul(x,y,v,k*2+1,l,mid),mul(x,y,v,k*2+2,mid,r),pushup(k);
} //DP
int dp(int x,vector<int> a){
mul(0,dn,0),add(0,1); vector<int> b;
for(int i:a) l[i]=find(l[i]),r[i]=find(r[i]);
sort(be(a),en(a),[&](int p,int q){return l[p]!=l[q]?l[p]<l[q]:r[p]<r[q];});
for(int i:a){while(sz(b)&&r[b.back()]>=r[i]) b.pop_back();b.pb(i);}
R(t,0,sz(b)){
int i=find(l[b[t]]),j=find(r[b[t]]); if(i>=j) return 0;
int ni=j;if(t+1<sz(b)) ni=min(ni,find(l[b[t+1]]));
mul(j,dn,0),add(j,(mod-val[0])%mod);
for(int k=i;k<ni;k=te[k]=find(k+1))
mul(0,k+1,Pow(x,d[k+1]-d[k])),mul(k+1,dn,Pow(x-1,d[k+1]-d[k]));
}
int res=val[0];
for(int i:a) for(int k=find(l[i]);k<find(r[i]);k=te[k]=find(k+1)) res=1ll*res*Pow(x,d[k+1]-d[k])%mod;
return res;
} //Main
int Main(){
cin>>n>>m>>h,dn=0,d[dn++]=0,d[dn++]=n,mp.clear();
R(i,0,m) cin>>l[i]>>r[i]>>g[i],--l[i],d[dn++]=l[i],d[dn++]=r[i];
sort(d,d+dn),dn=unique(d,d+dn)-d,clear();
R(i,0,m) l[i]=lower_bound(d,d+dn,l[i])-d,r[i]=lower_bound(d,d+dn,r[i])-d,mp[g[i]].pb(i);
iota(te,te+dn+1,0); int res=1;
for(auto u:mp) res=1ll*res*dp(u.x,u.y)%mod;
for(int k=find(0);k<dn-1;k=te[k]=find(k+1)) res=1ll*res*Pow(h,d[k+1]-d[k])%mod;
return res;
}
int main(){
ios::sync_with_stdio(0);
cin.tie(0),cout.tie(0);
int ta; cin>>ta; while(ta--) cout<<Main()<<'\n';
return 0;
}

祝大家学习愉快!

最新文章

  1. swf格式文件如何修改里面的动作路径或者动作脚本(没有源文件的情况)
  2. 史上最全的html标签属性用法对照表
  3. Ubuntu下搭建NodeJS+Express WEB开发框架
  4. knockout.js简单实用教程1
  5. tomcat之 JDK8.0安装、tomcat-8.5.15安装
  6. APP测试 - android os6,7 新增特性测试
  7. Pivot Table系列之切片器 (Slicer)
  8. 2017年总结的前端文章——CSS高级技巧汇总
  9. PKUWC 2018 滚粗记
  10. 关机充电如何实现短按pwrkey灭屏
  11. REST(Representational state transfer)的四个级别以及HATEOAS介绍
  12. Judy Beta Postmortem
  13. 深入理解Redis Cluster
  14. Codeforces 875F Royal Questions (看题解)
  15. 使用 Weinre 调试移动网站
  16. Hibernate中查询优化策略
  17. CentOS7.1 Liberty云平台之Dashboard篇(7)
  18. com.thoughtworks.xstream.converters.ConversionException
  19. bzoj 1854 构图 并查集
  20. Keil出错解决方法

热门文章

  1. lseek系统调用
  2. binary hacks读数笔记(file命令与magic file)
  3. 记录一次读取hdfs文件时出现的问题java.net.ConnectException: Connection refused
  4. Flink处理函数实战之一:深入了解ProcessFunction的状态(Flink-1.10)
  5. 创建一个自定义名称的Ceph集群
  6. RestPack Java实现Html转PDF文件
  7. Spring源码理论
  8. 《Spring Boot 实战纪实》之需求管理
  9. SSL加密原理
  10. Linux提权(持续更新)