Skip to content

Instantly share code, notes, and snippets.

@SamZhangQingChuan
Created June 27, 2020 14:24
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save SamZhangQingChuan/7791013ca026965d7c5d9456aad865bb to your computer and use it in GitHub Desktop.
Save SamZhangQingChuan/7791013ca026965d7c5d9456aad865bb to your computer and use it in GitHub Desktop.
#pragma comment(linker, "/stack:200000000")
#pragma GCC optimize("Ofast")
//#pragma GCC optimize(3)
//#pragma GCC target("sse,sse2,sse3,ssse3,sse4,popcnt,abm,mmx,avx,tune=native")
//#pragma GCC target("sse3","sse2","sse")
//#pragma GCC target("avx","sse4","sse4.1","sse4.2","ssse3")
//#pragma GCC target("f16c")
//#pragma GCC optimize("inline","fast-math","unroll-loops","no-stack-protector")
//#pragma GCC diagnostic error "-fwhole-program"
//#pragma GCC diagnostic error "-fcse-skip-blocks"
//#pragma GCC diagnostic error "-funsafe-loop-optimizations"
//#pragma GCC diagnostic error "-std=c++14"
#include "bits/stdc++.h"
#include "ext/pb_ds/tree_policy.hpp"
#include "ext/pb_ds/assoc_container.hpp"
#define PB push_back
#define PF push_front
#define LB lower_bound
#define UB upper_bound
#define fr(x) freopen(x,"r",stdin)
#define fw(x) freopen(x,"w",stdout)
#define REP(x, l, u) for(ll x = l;x<u;x++)
#define RREP(x, l, u) for(ll x = l;x>=u;x--)
#define complete_unique(a) a.erase(unique(begin(a),end(a)),end(a))
#define mst(x, a) memset(x,a,sizeof(x))
#define all(a) begin(a),end(a)
#define rall(a) rbegin(a),rend(a)
#define PII pair<int,int>
#define PLL pair<ll,ll>
#define MP make_pair
#define lowbit(x) ((x)&(-(x)))
#define bitcnt(x) (__builtin_popcountll(x))
#define lson (ind<<1)
#define rson (ind<<1|1)
#define se second
#define fi first
#define sz(x) ((int)x.size())
#define EX0 exit(0);
typedef long long ll;
typedef unsigned long long ull;
typedef double db;
typedef long double ld;
using namespace __gnu_pbds; //required
using namespace std;
template<typename T> using ordered_set = tree<T, null_type, less<T>, rb_tree_tag, tree_order_statistics_node_update>;
typedef vector<ll> VLL;
typedef vector<int> VI;
const ll mod = 998244353;
string to_string (string s) { return '"' + s + '"'; }
string to_string (const char *s) { return to_string ((string) s); }
string to_string (bool b) { return (b ? "true" : "false"); }
template<typename A, typename B>
string to_string (pair<A, B> p) { return "(" + to_string (p.first) + ", " + to_string (p.second) + ")"; }
template<typename A>
string to_string (A v) {
bool first = true;
string res = "{";
for (const auto &x : v) {
if (!first) { res += ", "; }
first = false;
res += to_string (x);
}
res += "}";
return res;
}
void debug_out () { cerr<<endl; }
template<typename Head, typename... Tail>
void debug_out (Head H, Tail... T) {
cerr<<" "<<to_string (H);
debug_out (T...);
}
#ifdef LOCAL
#define dbg(...) cerr << "[" << #__VA_ARGS__ << "]:", debug_out(__VA_ARGS__)
#else
#define dbg(...) {}
#endif
template<typename T, typename S>
inline bool upmin (T &a, const S &b) { return a > b ? a = b, 1 : 0; }
template<typename T, typename S>
inline bool upmax (T &a, const S &b) { return a < b ? a = b, 1 : 0; }
ull twop (ll x) { return 1ULL<<x; }
ll MOD (ll a, ll m) {
a %= m;
if (a < 0)a += m;
return a;
}
ll inverse (ll a, ll m) {
a = MOD (a, m);
if (a <= 1)return a;
return MOD ((1 - inverse (m, a) * m) / a, m);
}
template<typename T>
T sqr (T x) { return x * x; }
ll gcd (ll a, ll b) {
a = abs (a), b = abs (b);
while (b != 0) {
a %= b;
swap (a, b);
}
return a;
}
ll fast (ll a, ll b, ll mod) {
a %= mod;
if (b < 0)a = inverse (a, mod), b = -b;
ll ans = 1;
while (b) {
if (b & 1)ans = ans * a % mod;
a = a * a % mod;
b /= 2;
}
return ans % mod;
}
const ll maxn = 1000100;
ll fac[maxn], facinv[maxn], inv[maxn];
void init () {
fac[0] = 1;
REP(i, 1, maxn)fac[i] = i * (fac[i - 1]) % mod;
facinv[maxn - 1] = inverse (fac[maxn - 1], mod);
RREP(i, maxn - 2, 0) {
facinv[i] = facinv[i + 1] * (i + 1) % mod;
}
inv[1] = 1;
for (int i = 2; i < maxn; i++)
inv[i] = (mod - (ll) mod / i * inv[mod % i] % mod);
}
ll combi (int n, int m) {
if (n < 0 || m < 0 || n < m)return 0;
return fac[n] * facinv[m] % mod * facinv[n - m] % mod;
}
ll catalan (ll n) {
if (!n)return 1;
return MOD (combi (2 * n, n) - combi (2 * n, n + 1), mod);
}
struct SegTree {
static const int maxn = 500010;
struct node {
int l, r;
ll val;
};
node no[maxn * 4];
void push_up (int ind) {
no[ind].val = 1ll * no[lson].val * no[rson].val % mod;
}
void push_down (int ind) {
}
void build (int l, int r, int ind) {
no[ind].l = l;
no[ind].r = r;
no[ind].val = 1;
if (l == r) {
} else {
int mid = (l + r) / 2;
build (l, mid, lson);
build (mid + 1, r, rson);
push_up (ind);
}
}
void update (int l, int r, int ind, ll val) {
if (l > no[ind].r || r < no[ind].l)return;
if (l <= no[ind].l && no[ind].r <= r) {
no[ind].val = 1ll * no[ind].val * val % mod;
} else {
push_down (ind);
update (l, r, lson, val);
update (l, r, rson, val);
push_up (ind);
}
}
void query (int l, int r, int ind, ll &ans) {
if (l > no[ind].r || r < no[ind].l)return;
if (l <= no[ind].l && no[ind].r <= r) {
ans = 1ll * ans * no[ind].val % mod;
} else {
push_down (ind);
query (l, r, lson, ans);
query (l, r, rson, ans);
push_up (ind);
}
}
void set (int l, int r, int ind, ll val) {
if (l > no[ind].r || r < no[ind].l)return;
if (l <= no[ind].l && no[ind].r <= r) {
no[ind].val = val;
} else {
push_down (ind);
set (l, r, lson, val);
set (l, r, rson, val);
push_up (ind);
}
}
};
namespace SOLVE {
const ll N = 500010;
ll cat[N], invcat[N], val[N];
vector<PII > query[N];
SegTree stkTree, buryTree;
int res[N];
void main () {
init ();
stkTree.build (0, 500000, 1);
buryTree = stkTree;
REP(i, 0, N)cat[i] = catalan (i);
REP(i, 0, N)invcat[i] = inverse (cat[i], mod);
int n, q;
cin>>n;
REP(i, 1, n + 1)cin>>val[i];
cin>>q;
REP(i, 0, q) {
int l, r;
cin>>l>>r;
query[r].PB (MP (l, i));
}
vector<VI> stk;
REP(i, 1, n + 1) {
while (sz(stk) and val[stk.back ().front ()] > val[i]) {
auto pos = stk.back ();
stk.pop_back ();
int cnt = 1;
for (int i = sz(pos) - 1; i >= 0; i--) {
buryTree.update (pos[i], pos[i], 1, cat[cnt] * invcat[cnt - 1] % mod);
cnt++;
}
}
if (sz(stk) and val[stk.back ().front ()] == val[i]) {
stk.back ().PB (i);
const int cnt = sz(stk.back ());
stkTree.update (sz(stk) - 1, sz(stk) - 1, 1, cat[cnt] * invcat[cnt - 1] % mod);
} else {
stk.PB (VI (1, i));
stkTree.set (sz(stk) - 1, sz(stk) - 1, 1, 1);
}
for (auto q:query[i]) {
int l, r, id;
r = i;
tie (l, id) = q;
auto find_seg = [&] () {
int ql = 0, qr = sz(stk) - 1;
while (ql < qr) {
int mid = (ql + qr) / 2;
if (stk[mid].back () < l) ql = mid + 1;
else qr = mid;
}
return ql;
};
int ind = find_seg ();
ll ans = 1;
buryTree.query (l, r, 1, ans);
dbg(ans);
dbg(cat[4]);
stkTree.query (ind + 1, sz(stk) - 1, 1, ans);
auto cnt = stk[ind].end () - LB (all(stk[ind]), l);
dbg(stk);
ans = 1ll * ans * cat[cnt] % mod;
res[id] = ans;
}
}
REP(i, 0, q)cout<<res[i]<<"\n";
}
}
signed main () {
#ifdef LOCAL
fr("/Users/zhangqingchuan/Desktop/cp/cp/input.txt");
fw("/Users/zhangqingchuan/Desktop/cp/cp/output.txt");
#endif
ios::sync_with_stdio (false);
cin.tie (nullptr);
cout.tie (nullptr);
int t = 1;
// cin >> t;
for (int i = 1; i <= t; i++) {
// cout<<"Case #"<<i<<": ";
SOLVE::main ();
}
// clock_t st = clock();
// while(clock() - st < 3.0 * CLOCKS_PER_SEC){
//
// }
return 0;
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment