首页 技术 正文
技术 2022年11月8日
0 收藏 305 点赞 1,362 浏览 1832 个字

Description

quailty和tangjz正在玩一个关于线段的游戏。在平面上有n条线段,编号依次为1到n。其中第i条线段的两端点坐标分别为(0,i)和(1,p_i),其中p_1,p_2,…,p_n构成了1到n的一个排列。quailty先手,他可以选择一些互不相交的线段,将它们拿走,当然他也可以一条线段也不选。然后tangjz必须拿走所有剩下的线段,若有两条线段相交,那么他就输了,否则他就赢了。注意若quailty拿走了全部线段,那么tangjz也会胜利。quailty深深喜欢着tangjz,所以他不希望tangjz输掉游戏,请计算他有多少种选择线段的方式,使得tangjz可以赢得游戏。

Input

第一行包含一个正整数n(1<=n<=100000),表示线段的个数。第二行包含n个正整数p_1,p_2,…,p_n(1<=p_i<=n),含义如题面所述。

Output

输出一行一个整数,即tangjz胜利的方案数,因为答案很大,请对998244353取模输出。

问题可以转化为:
给定一个序列,求将这个序列分成两个上升子序列的方案数.
令 $f_{i,j}$ 表示一个序列选 $i$,令一个序列的最大值为 $p_{j}$ 的方案数 ($j<i$).
考虑转移:

  • $f_{i,j}=f_{i-1,j},j<i-1$ 且 $p_{i-1}<p_{i}$
  • $f_{i,i-1}=\sum f_{i-1,k},p_{k}<p_{i}$

$f_{i,j}$ 的转移只和 $f_{i-1,?}$ 的转移有关,所以可以将第一维滚掉.
而每一次 $f$ 的更新只有两种:区间清零,用一个区间和来更新单点的点值.
这些操作可以用线段树来实现.

#include <cstdio>
#include <algorithm>
#define N 100006
#define mod 998244353
#define setIO(s) freopen(s".in", "r" , stdin)
using namespace std;
int a[N], n ;
struct Node
{
int sum, lazy ;
}t[N << 2];
inline void mark(int now)
{
t[now].sum = 0, t[now].lazy = 1;
}
inline void pushdown(int l, int r, int now)
{
int mid = (l + r) >> 1;
if(t[now].lazy)
{
mark(now << 1);
if(r > mid) mark(now << 1 | 1);
t[now].lazy = 0;
}
}
inline void pushup(int l, int r, int now)
{
int mid = (l + r) >> 1;
t[now].sum = t[now << 1].sum ;
if(r > mid) t[now].sum += t[now << 1 | 1].sum, t[now].sum %= mod ;
}
inline void update(int l, int r, int now, int p, int v)
{
if(l == r)
{
t[now].sum += v, t[now].sum %= mod;
return ;
}
int mid = (l + r) >> 1;
pushdown(l, r, now);
if(p <= mid) update(l, mid, now << 1, p, v);
else update(mid + 1, r, now << 1 | 1, p, v);
pushup(l, r, now);
}
inline int query(int l, int r, int now, int L, int R)
{
if(l >= L && r <= R) return t[now].sum ;
pushdown(l, r, now);
int mid = (l + r) >> 1, re = 0;
if(L <= mid) re += query(l, mid, now << 1, L, R);
if(R > mid) re += query(mid + 1, r, now << 1 | 1, L, R);
return re % mod;
}
int main()
{
// setIO("input");
int i , j ;
scanf("%d", &n);
for(i = 1 ; i <= n ; ++ i) scanf("%d", &a[i]);
update(0, n, 1, 0, 1);
for(i = 2 ; i <= n ; ++ i)
{
int sum = query(0, n , 1, 0, a[i] - 1);
if(a[i] < a[i - 1]) mark(1);
update(0, n, 1, a[i - 1], sum);
}
printf("%d\n", query(0, n, 1, 0, n) * 2 % mod);
return 0;
}

  

相关推荐
python开发_常用的python模块及安装方法
adodb:我们领导推荐的数据库连接组件bsddb3:BerkeleyDB的连接组件Cheetah-1.0:我比较喜欢这个版本的cheeta…
日期:2022-11-24 点赞:878 阅读:9,087
Educational Codeforces Round 11 C. Hard Process 二分
C. Hard Process题目连接:http://www.codeforces.com/contest/660/problem/CDes…
日期:2022-11-24 点赞:807 阅读:5,562
下载Ubuntn 17.04 内核源代码
zengkefu@server1:/usr/src$ uname -aLinux server1 4.10.0-19-generic #21…
日期:2022-11-24 点赞:569 阅读:6,412
可用Active Desktop Calendar V7.86 注册码序列号
可用Active Desktop Calendar V7.86 注册码序列号Name: www.greendown.cn Code: &nb…
日期:2022-11-24 点赞:733 阅读:6,185
Android调用系统相机、自定义相机、处理大图片
Android调用系统相机和自定义相机实例本博文主要是介绍了android上使用相机进行拍照并显示的两种方式,并且由于涉及到要把拍到的照片显…
日期:2022-11-24 点赞:512 阅读:7,821
Struts的使用
一、Struts2的获取  Struts的官方网站为:http://struts.apache.org/  下载完Struts2的jar包,…
日期:2022-11-24 点赞:671 阅读:4,905