CF1621G Weighted Increasing Subsequences(离散化+树状数组优化dp+栈维护后缀最大值+计数)

problem

luogu-link

solution

显然单独考虑每个 iii 的贡献,即被多少个合法上升子序列包含。

x=max⁡{j∣j>i∧aj>ai}x=\max\{j\ |\ j>i\wedge a_j>a_i\}x=max{j  j>iaj>ai},则包含 iii 的合法子序列的结尾元素最远只能到 x−1x-1x1

发现,重要的不是值本身,而是大小关系。所有可以离散化从小到大标号 1∼n1\sim n1n,权值小的放前面;权值相同下标大的放前面

下面的 aaa 均表示离散化后的序列。

同时将合法上升子序列拆分成正着以 iii 结尾的上升子序列和倒着以 iii 结尾的下降子序列两个部分,分别求出方案数后乘法原理即可。

1∼i1\sim i1iiii 结尾的上升子序列,在已经离散化后,预处理就非常简单了。树状数组维护 fi:if_i:ifi:i 结尾的上升子序列个数。

然后考虑 x−1∼ix-1\sim ix1iiii 结尾的下降子序列。

因为每个 iii 对应的 xxx 不一样,不能像预处理一样只扫描一遍整个 1∼n1\sim n1n,也不可能每次都去遍历这个区间。

不妨考虑转化成n∼in\sim iniiii 结尾的下降子序列个数减去以 xxx 开头以 iii 结尾的个数

n∼in\sim iniiii 结尾的下降子序列个数,预处理方法同上。

最后就只剩下以 xxx 开头以 iii 结尾的下降子序列个数的计算了,开头和结尾都已经定了,是可做的。

y:ay=max⁡{aj∣x<j≤n}y:a_y=\max\{a_j\ |\ x<j\le n\}y:ay=max{aj  x<jn},即 xxx 以后的后缀最大值的下标。

必然有 ay<ai<axa_y<a_i<a_xay<ai<ax

也就是说,当我们利用维护出后缀最大值序列后,这个下降子序列的所有元素 kkk 都满足ay<ak<axa_y<a_k<a_xay<ak<ax

那么,我们可以对于维护出来的后缀最大值上的每个节点开一个 vector\text{vector}vector

把所有满足上述限制的 kkk,全都挂在一个节点上。

暴力地按照预处理的计算方式,计算下降子序列个数。

因为每个 iii 只会挂在一个 xxx 上面,所以时间复杂度是可以保证的。

code

#include <bits/stdc++.h>
using namespace std;
#define mod 1000000007
#define int long long
#define maxn 300005
vector < int > v[maxn];
int T, n, top;
int id[maxn], t[maxn], a[maxn], f[maxn], g[maxn], h[maxn], s[maxn];void clear() { for( int i = 1;i <= n;i ++ ) t[i] = 0; }
void Add( int i, int x ) { for( ;i <= n;i += i & -i ) ( t[i] += x ) %= mod; }
int Ask( int i ) { int ans = 0; for( ;i;i -= i & -i ) ( ans += t[i] ) %= mod; return ans; }signed main() {scanf( "%lld", &T );while( T -- ) {top = 0; for( int i = 1;i <= n;i ++ ) v[i].clear();scanf( "%lld", &n );for( int i = 1;i <= n;i ++ ) scanf( "%lld", &a[i] ), id[i] = i;sort( id + 1, id + n + 1, []( int x, int y ) { return a[x] == a[y] ? x > y : a[x] < a[y]; } );for( int i = 1;i <= n;i ++ ) a[id[i]] = i; //离散化clear();for( int i = 1;i <= n;i ++ ) Add( a[i], f[i] = Ask( a[i] ) + 1 );//f[i]:1~i以i结尾的上升子序列个数 //f[i]=1+\sum_{j=1}^{i-1}f[j](a_j<a_i)clear();for( int i = n;i;i -- ) Add( n - a[i] + 1, g[i] = Ask( n - a[i] + 1 ) + 1 );//g[i]:n~i以i结尾的下降子序列个数//g[i]=1+\sum_{j=n}^{i+1}g[j](a_j>a_i)for( int i = n;i;i -- ) if( a[i] > a[s[top]] ) s[++ top] = i;//后缀最大值的有用点for( int i = n;i;i -- ) { //必须倒序int l = 1, r = top, pos;while( l <= r ) {int mid = ( l + r ) >> 1;if( a[i] <= a[s[mid]] ) r = mid - 1, pos = mid;else l = mid + 1;}if( i ^ s[pos] ) v[pos].push_back( i );//将满足a_{s[pos-1]}<a_i<a_{s[pos]}的i归属于s[pos]集合}for( int i = 1;i <= n;i ++ ) t[i] = 0;for( int i = 1;i <= top;i ++ ) {Add( n - a[s[i]] + 1, h[s[i]] = 1 );//h[k]:k~k隶属的s[i] 以s[i]开始k结尾的下降子序列//h[k]=\sum_{j=k+1}^{s[i]}h[j](a_j>a_k)for( int k : v[i] ) Add( n - a[k] + 1, h[k] = Ask( n - a[k] + 1 ) );//所有下标>k的j都已经加入了 这也是为什么上面必须倒序储存//相当于clear() 但每次k数量很少 从1~n都清一遍浪费时间for( int k : v[i] ) Add( n - a[k] + 1, -h[k] );Add( n - a[s[i]] + 1, -1 );}int ans = 0;//g[i]-h[i]:合法的n~i以i结尾的下降子序列数量 f[i]:合法的1~i以i结尾的上升子序列数量for( int i = 1;i <= n;i ++ ) ( ans += ( g[i] - h[i] ) % mod * f[i] ) %= mod;printf( "%lld\n", ( ans + mod ) % mod );}return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/316600.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Acwing 232. 守卫者的挑战

Acwing 232. 守卫者的挑战 题意&#xff1a; 有n个挑战&#xff0c;一开始背包容量为k&#xff0c;每次挑战有p[i]的概率成功&#xff0c;成功的话会得到一个大小为1的地图碎片或者是提升背包容量X&#xff0c;所有的地图碎片必须装在包里&#xff0c;问最后带地图离开的概率…

IdentityServer4-前后端分离之Vue

前言之前文章讲到如何使用Node.jsExpress构建JavaScript客户端&#xff0c;实现前后端分离。本节将介绍如何使用Vue实现前后端分离&#xff0c;文中介绍Vue的知识比较基础&#xff0c;适合新手学习。一、搭建Vue项目前提条件&#xff1a;安装nodejs、webpack和vue-cli。这个网上…

P1850 [NOIP2016 提高组] 换教室

P1850 [NOIP2016 提高组] 换教室 题意&#xff1a; 有2n个课安排在n个时间段上&#xff0c;每个时间段上都有两个一样的课同时在不同地方上&#xff0c;起初牛牛被所有课都被安排在Ci上课&#xff0c;另一节课在Di上课。牛牛现在想跟换到Di位置&#xff0c;它最多可以申请m节…

守列划分问题(圆排列+排列dp+结论)

problem 将正整数 1∼n1\sim n1∼n 任意划分成 mmm 个非空集合 A1,...,AmA_1,...,A_mA1​,...,Am​。 一个划分是守序的&#xff0c;当且仅当存在一个环排列 (p1,...,pm)(p_1,...,p_m)(p1​,...,pm​)&#xff0c;使得 max⁡Api>min⁡Api−1\max A_{p_i}>\min A_{p_{i-…

ASP.NET Core应用程序容器化、持续集成与Kubernetes集群部署(三

在上文ASP.NET Core应用程序容器化、持续集成与Kubernetes集群部署&#xff08;二&#xff09;中&#xff0c;我介绍了如何使用Azure DevOps为ASP.NET Core应用程序案例&#xff1a;tasklist搭建持续集成环境。在持续集成的过程中&#xff0c;Azure DevOps的Build Pipeline会下…

ASP.NET Core开源Web应用程序框架ABP

"作为面向服务架构(SOA)的一个变体,微服务是一种将应用程序分解成松散耦合服务的新型架构风格. 通过细粒度的服务和轻量级的协议,微服务提供了更多的模块化,使应用程序更容易理解,开发,测试,并且更容易抵抗架构侵蚀. 它使小型团队能够开发,部署和扩展各自的服务,实现开发的…

CodeForces 1517G Starry Night Camping(网络流最小割)

CodeForces 1517G Starry Night Camping problem 洛谷链接 solution 这个平行四边形的脑洞我™真的长见识了 本题最离谱的要求就是&#xff1a;平行四边形的一条边平行于 xxx 轴。 而往往这种离谱要求就是正解的途径。(((φ(◎ロ◎;)φ))) 首先不观察也能知道&#xff0c…

Acwing 307. 连通图

Acwing 307. 连通图 题意&#xff1a; 求 N 个节点的无向连通图有多少个&#xff0c;节点有标号&#xff0c;编号为 1∼N。 例如下列图示&#xff0c;三个节点的无向连通图共 4 个。 题解: 用py写 代码&#xff1a; def c(n, m):n int(n)m int(m)ret 1for i in range(…

Acwing 309. 装饰围栏

Acwing 309. 装饰围栏 题意&#xff1a; 有n个模板&#xff0c;长度分别是1到N&#xff0c;现在按照高低交错的方式排列模板&#xff0c;能到的很多种排列的方案。 每个方案都可以写作一个长度为N的序列&#xff0c;序列中的个元素是木板的长度&#xff0c;把这些序列按照字典…

CodeForces 901D Weighting a Tree(结论)

problem 洛谷链接 注意&#xff1a;保证 C[v]C[v]C[v] 的奇偶性与顶点 vvv 的度的奇偶性相同。 solution 先考虑树的情况。这是个经典问题了&#xff0c;从叶子往上推&#xff0c;对于一个点还差的边权就有这个点和其父亲的边来补足。最后判断根是否满足条件即可。 那么怎…

CF938E Max History

CF938E Max History 题意&#xff1a; 我们定义f(a)为&#xff1a; 1、开始时&#xff0c;f(a)0,M1。 2、对于每个2<i<n,如果a[M]<a[i],那么f(a)f(a)a[M],Mi。 现在对于一个给定的数组a&#xff0c;求其所有排列的f(a)之和&#xff0c;答案对1e97取模。 题解&am…

最大限度地降低多线程 C# 代码的复杂性

分支或多线程编程是编程时最难最对的事情之一。这是由于它们的并行性质所致&#xff0c;即要求采用与使用单线程的线性编程完全不同的思维模式。对于这个问题&#xff0c;恰当类比就是抛接杂耍表演者&#xff0c;必须在空中抛接多个球&#xff0c;而不要让它们相互干扰。这是一…

CodeForces 1491G Switch and Flip(结论)

problem 洛谷链接 solution 弱化版&#xff1a;如果不考虑翻面&#xff0c;那就是转化为若干个环的问题了。 这里我们尝试用同样的思路解决。 首先明确几个硬币一次交换后的等价情况&#xff0c;如图&#xff08;灰色表示反面&#xff09; &#xff08;u→vu\rightarrow …

ASP.NET Core 借助 K8S 玩转容器编排

Production-Grade Container Orchestration - Automated container deployment, scaling, and management. 生产级别的容器编排系统——自动化的容器部署、扩展和管理。1. 引言由于最近在学习微服务&#xff0c;所以就基于之前docker的基础上把玩一下k8s&#xff08;Kubernetes…

CF1009E Intercity Travelling

CF1009E Intercity Travelling 题意&#xff1a; 有一段路程&#xff0c;路程可以看作是从0到n的一条直线 如果从起点出发或者从休息点出发&#xff0c;连续驾驶k千米&#xff0c;则需要消耗的体能为a1…ak 每个整点都有可能拥有一个休息点&#xff0c;每个休息点存在或者不存…

CodeForces 1396E Distance Matching(构造+树的重心+dfs+set)

problem 洛谷链接 solution 这种要求值和恰好为 kkk 的题目&#xff0c;一般要先明确值和的取值范围。 所以我们先来确定一下值和的最小值和最大值。 将一条路径拆成若干条边&#xff0c;单独计算每条边对路径的贡献。 假设一条边将树划分成 S,TS,TS,T 集合。因为 nnn 为…

程序员修神之路--

写在开始一般来说有两种策略用来在并发线程中进行通信&#xff1a;共享数据和消息传递。使用共享数据方式的并发编程面临的最大的一个问题就是数据条件竞争。处理各种锁的问题是让人十分头痛的一件事。传统多数流行的语言并发是基于多线程之间的共享内存&#xff0c;使用同步方…

Lawn of the Dead

Lawn of the Dead 题意&#xff1a; 有一个N * M的方格&#xff0c;我们从(1,1)出发&#xff0c;只能向右走或者向下走&#xff0c;存在一些障碍&#xff0c;问有多少格子是我们所能到达的 2<n,m,k<1e5 题解&#xff1a; 所有的点减去不能到达的点的个数&#xff0c;…

CodeForces 1622F Quadratic Set(结论+异或哈希+散列表)

problem 洛谷链接 solution 最后子集大小一定 ≥n−3\ge n-3≥n−3&#xff0c;下面考虑证明这个结论。 假设 n2kn2kn2k。 ∏i1n(i!)∏i1k(2i−1)!(2i)!∏i1k(((2i−1)!)22i)∏i1k((2i−1)!)2⋅∏i1k2i∏i1k((2i−1)!)2⋅2k⋅k!\prod_{i1}^n(i!)\prod_{i1}^{k}(2i-1)!(2i)!\…

.Netcore 2.0 Ocelot Api网关教程(8)- 缓存

Ocelot中使用 CacheManager 来支持缓存&#xff0c;官方文档中强烈建议使用该包作为缓存工具。以下介绍通过使用CacheManager来实现Ocelot缓存。1、通过Nuget添加 Ocelot.Cache.CacheManager 包在OcelotGetway项目中添加引用&#xff1a;2、修改 Startup 中的 ConfigureService…