首页 > 其他分享 >【题解】CF193D Two Segments

【题解】CF193D Two Segments

时间:2023-05-20 20:55:34浏览次数:59  
标签:mid int 题解 Segments 段数 Two pos 值域 区间

题意

给定一个\(1\sim N\)的排列,在这个排列中选出两段互不重叠的区间,求使选出的元素排序后构成公差为1的等差数列的方案数。选出的两段区间中元素构成的集合相同时视为同一种方案。\(1\le N\le 3\times 10^5\)。

传送门

分析

如果考虑怎么优化枚举的两个区间的话,发现不太好搞(反正我只会暴力)。

于是考虑枚举连续的值域区间,再判断一下连续的值域区间是由原排列中几段连续的区间构成,如果 \(\le 2\),就是可行的方案。

对于这种区间问题,一般套路是确定一个点,然后对其他点算贡献。

设\(f[l][r]\) 表示值域 \([l,r]\) 是由几段构成的,\(pos[i]\) 表示 \(i\) 这个值在原序列的位置,我们从 \(1\) 到 \(n\) 依次枚举右端点 \(i\),考虑从 \(i - 1\) 转移到 \(i\),那如何在 \(O(i)\) 的时间内转移呢?可以找到如下规律:

  • 如果原序列中 \(i\) 在的位置左右两个数都 \(\le i\) ,那么肯定在之前加入了,而现在加入 \(i\) 会使 \([l, i], l \in [1, \min(a[pos[i]-1], a[pos[i] + 1])]\) 值域的段数 \(-1\),\([l, i], l \in (\min(a[pos[i]-1], a[pos[i] + 1]), \max(a[pos[i]-1], a[pos[i] + 1])]\)值域的段数不变,\([l, i], l \in (\max(a[pos[i]-1], a[pos[i] + 1]), i - 1]\) 值域的段数 \(+1\)。
  • 如果只有一个,设那个数的位置为 \(x\),那么对于 \([l, i] ,l\in [1, x]\)值域的段数不变,\([l, i], l \in (x, i - 1]\) 的段数 \(+1\)。
  • 如果没有,那么 \([l, i] ,l\in i - 1\)的值域的段数会 \(+1\)。

这样是 \(O(n ^ 2)\),区间加,区间减,很容易想到用线段树优化。

设枚举\(i\),线段树的区间 \([l,r]\),表示 \([x, i], x\in [l,r]\) 的各种信息。

我们需要线段树维护区间的最小段数的值,是这个最小段数的值的区间个数,和是次小段数这个值的区间个数。

最后询问$1∼i−1 $需要分的段数是否小于等于 \(2\) 即可

如何维护详见代码注释。

#include<bits/stdc++.h>
#define N 300005
#define int long long
#define ls u << 1
#define rs u << 1 | 1 
using namespace std;
int read(){
	int x = 0, f = 1; char ch = getchar();
	while(ch < '0' || ch > '9'){if(ch == '-') f = -f; ch = getchar();}
	while(ch >= '0' && ch <= '9'){x = (x << 1) + (x << 3) + (ch ^ 48); ch = getchar();}
	return x * f;
}
int n, ans; 
int a[N], pos[N];
int minn[N << 2], cnt0[N << 2], cnt1[N << 2], lazy[N << 2];
struct Segment{
	void pushup(int u){
		minn[u] = min(minn[ls], minn[rs]);
		cnt0[u] = (minn[ls] == minn[u]) * cnt0[ls] + (minn[rs] == minn[u]) * cnt0[rs];
		cnt1[u] = (minn[ls] == minn[u]) * cnt1[ls] + (minn[ls] == minn[u] + 1) * cnt0[ls]; 
		cnt1[u] += (minn[rs] == minn[u]) * cnt1[rs] + (minn[rs] == minn[u] + 1) * cnt0[rs];
		//如果左/右区间的最小值等于整个区间的最小值,那么左/右区间次小值就是整个区间的次小值,统计个数
		//如果左/右区间的最小值等于整个区间的最小值 + 1,那么最小值就是整个区间的次小值,因为每次枚举 $i$ 时,
		//值的变化最多加减1,所以次小值就是最小值 + 1。 
	}
	void pushdown(int u){
		minn[ls] += lazy[u], lazy[ls] += lazy[u];
		minn[rs] += lazy[u], lazy[rs] += lazy[u];
		lazy[u] = 0;
	}
 	void build(int u, int l, int r){
		if(l == r) return cnt0[u] = 1, void(); //初始都为 1 
		int mid = (l + r) >> 1;
		build(ls, l, mid), build(rs, mid + 1, r);
		pushup(u);
	}
	void update(int u, int l, int r, int L, int R, int val){
		if(L <= l && r <= R) return minn[u] += val, lazy[u] += val, void();
		pushdown(u);
		int mid = (l + r) >> 1;
		if(L <= mid) update(ls, l, mid, L, R, val);
		if(R > mid) update(rs, mid + 1, r, L, R, val);
		pushup(u);
	}
	int query(int u, int l, int r, int L, int R){
		if(L <= l && r <= R) return cnt0[u] * (minn[u] <= 2) + cnt1[u] * (minn[u] <= 1);
		//如果最小值小于等于 2 ,说明最小值是符合的,统计进去。
		//如果最小值小于等于 1 , 次小值 = 最小值 + 1 , 次小值也符合。 
		pushdown(u);
		int mid = (l + r) >> 1, res = 0;
		if(L <= mid) res += query(ls, l, mid, L, R);
		if(R > mid) res += query(rs, mid + 1, r, L, R);
		return res; 
	}
}tr;
signed main(){
	n = read();
	for(int i = 1; i <= n; ++i) a[i] = read(), pos[a[i]] = i;
	tr.build(1, 1, n);
	for(int i = 1; i <= n; ++i){
		tr.update(1, 1, n, 1, i, 1);
		if(a[pos[i] - 1] < i && a[pos[i] - 1]) tr.update(1, 1, n, 1, a[pos[i] - 1], -1);
		if(a[pos[i] + 1] < i && a[pos[i] + 1]) tr.update(1, 1, n, 1, a[pos[i] + 1], -1); 
		if(i >= 2) ans += tr.query(1, 1, n, 1, i - 1);//[i,i]这段是不能算进去的 
	}
	printf("%lld\n", ans);
	return 0;
}

标签:mid,int,题解,Segments,段数,Two,pos,值域,区间
From: https://www.cnblogs.com/jiangchen4122/p/17417767.html

相关文章

  • [USACO08JAN]Cell Phone Network G
    题意:给出由n个点和(n-1)条边构成的树,每个点可以覆盖每个相邻点,求把树上所有点覆盖完成至少需要挑出多少点来做覆盖操作思路:先明确用树形dp来做解答,用dp[i][]来表示覆盖对应点和其下方所有节点的最小花费对于要覆盖的每个点,我们可以有三种选择:1.自己覆盖自己:这时字节......
  • jre jdk更改目录后Java无法运行问题解决方案
    问题:在将Java文件(包含jdkjre)由C盘直接剪贴到D盘后,所有Java程序无法运行,且其Java图标不再显示。解决方案:首先更改环境变量。当我们单纯地将Java文件更改位置后,我们计算机的环境变量仍未改变,依旧是当时安装Java时的配置。步骤:控制面板—>系统和安全—>系统—>高级系统设置—>环境......
  • P5179 Fraction 题解
    题目描述给你四个正整数\(a,\,b,\,c,\,d\),求一个最简分数\(\frac{p}{q}\)满足\(\frac{a}{b}<\frac{p}{q}<\frac{c}{d}\)。若有多组解,输出\(q\)最小的一组,若仍有多组解,输出\(p\)最小的一组。前置知识:Stern-Brocot树首先引入分数逼近。这里的分数逼近是指用用一个......
  • CF1781F题解
    \(\text{link}\)。也是一道非常巧妙的\(\texttt{dp}\)。容易想到把括号变成\(\pm1\)。考虑括号序列合法等价于前缀和\(\ge0\),我们可以想加入\(()\)或\()(\)对前缀的影响。设加入的位置的前一位前缀和为\(x\),则加入\(()\)相当于把\(x\)替换为\([x,x+1,x]\),加入......
  • CSP-J2021试题题解
    1.分糖果原题:https://www.luogu.com.cn/problem/P7909原代码:#include<bits/stdc++.h>#definelllonglongusingnamespacestd;lln,l,r;intmain(){ cin>>n>>l>>r; if(r%n==n-1)cout<<n-1; elseif(l%n==n-1)cout<<n-1; elseif(......
  • 问题解一
    (1)用pycharm连接mysql时,需要安装一个驱动,点击连接界面,如果TestConnection不可用,点击右边的按钮即可(2)如果要将同一网站不同链接的数据存入数据库,可考虑重写start_requestseg:forindex,hrefinenumerate(self.start_urls):ifindex==0:yieldscrapy.Request(......
  • 僵尸进程问题解决
    1何为僵尸进程僵尸进程是当子进程比父进程先结束,而父进程又没有回收子进程,释放子进程占用的资源,此时子进程将成为一个僵尸进程。如果父进程先退出,子进程被init接管,子进程退出后init会回收其占用的相关资源。在UNIX系统中,一个进程结束了,但是它的父进程没有等待(调用wait/wai......
  • 【P4331 [BalticOI 2004]】Sequence 数字序列 题解(左偏树维护动态区间中位数)
    左偏树维护动态区间中位数。传送门P4331BalticOI2004Sequence数字序列。Solution1我的思路和题解前半部分完全重合了((如果按照单调不增去分割\(a\)序列的话,对于每一段我们能很简单地得出它的最佳答案:中位数。发现严格单调很难做,很难拿捏,考虑对\(a\)序列的每一项都进......
  • CSP-J2022山东补赛题解
    1.植树节原题:https://www.luogu.com.cn/problem/U285015代码:#include<bits/stdc++.h>#definelllonglongusingnamespacestd;constintN=1e6+255;inta[N],n,x,y,maxb=-1e9,ans=-1e9;intmain(){ cin>>n; for(inti=1;i<=n;i++){ cin>>x&g......
  • CSP-J2019试题题解
    1.数字游戏原题:https://www.luogu.com.cn/problem/P5660代码:#include<bits/stdc++.h>#include<iostream>#include<cstdio>#include<cstring>#include<algorithm>usingnamespacestd;strings;intmain(){ cin>>s;intnum=0; fo......