【agc023E】Inversions(线段树,动态规划)

题面

AT

给定\(a_i\),求所有满足\(p_i\le a_i\)的排列\(p\)的逆序对数之和。

题解

首先如何计算排列\(p\)的个数。

设\(cnt[i]\)表示\(a_k\ge i\)的个数,那么满足条件的\(p\)的总数就是\(\prod cnt[i]-(n-i)\)

大概就是从\(n\)开始填数,对于每个数字\(i\)而言,它一共有\(cnt[i]\)个位置可以填,但是后面的数字一共占用了\(n-i\)个位置,所以还剩下\(cnt[i]-(n-i)\)个位置可以填。

先讲讲\(O(n^2log)\)的做法。

枚举任意两个位置\(i,j,i\lt j\),考虑\(i,j\)之间形成逆序对的贡献。

分成三种情况讨论。

1.\(a_i=a_j\)

显然对于任意一种满足条件的方案,要么\(p_i>p_j\)要么\(p_i<p_j\),并且两两对称,所以就是总方案除\(2\)。

2.\(a_i<a_j\)

当\(p_j\)取\([a_i+1,a_j]\)这部分的值的时候,显然不会产生贡献,而当\(p_j\in[1,a_i]\)时,则和上面是一样的,但是注意一下,此时我们直接把\(a_j\)变成了\(a_i\),会对于总方案产生影响,\([a_i+1,a_j]\)这一段的\(cnt\)减小了,所以用一个线段树维护一下区间积,就可以方便的维护当前状态下的总方案,那么这一部分也很好算。

3.\(a_i>a_j\)

很麻烦,但是我们可以正难则反来算,用总方案减去\(p_i<p_j\)的方案数,转化成了第二种情况,也就是将\(a_i\)直接变成\(a_j\)。

这样子用线段树维护,时间复杂度\(O(n^2logn)\),用前后缀之类的东西维护可以做到\(O(n^2)\)。

代码如下:

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<vector>
using namespace std;
#define ll long long
#define MAX 200200
#define MOD 1000000007
#define inv2 500000004
#define lson (now<<1)
#define rson (now<<1|1)
inline int read()
{
int x=0;bool t=false;char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=true,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return t?-x:x;
}
int fpow(int a,int b)
{
int s=1;
while(b){if(b&1)s=1ll*s*a%MOD;a=1ll*a*a%MOD;b>>=1;}
return s;
}
int n,a[MAX],tot,ans;
int cnt[MAX];
int t[2][MAX<<2];
void Build(int now,int l,int r)
{
if(l==r){t[1][now]=cnt[l]-1-(n-l);t[0][now]=cnt[l]-(n-l);return;}
int mid=(l+r)>>1;
Build(lson,l,mid);Build(rson,mid+1,r);
t[0][now]=1ll*t[0][lson]*t[0][rson]%MOD;
t[1][now]=1ll*t[1][lson]*t[1][rson]%MOD;
}
int Query(int now,int l,int r,int L,int R,int c)
{
if(L<=l&&r<=R)return t[c][now];
int mid=(l+r)>>1,ret=1;
if(L<=mid)ret=1ll*ret*Query(lson,l,mid,L,R,c)%MOD;
if(R>mid)ret=1ll*ret*Query(rson,mid+1,r,L,R,c)%MOD;
return ret;
}
int main()
{
n=read();tot=1;
for(int i=1;i<=n;++i)++cnt[a[i]=read()];
for(int i=n-1;i;--i)cnt[i]+=cnt[i+1];
for(int i=1;i<=n;++i)tot=1ll*tot*(cnt[i]-(n-i))%MOD;
if(!tot){puts("0");return 0;}
Build(1,1,n);
for(int i=1;i<=n;++i)
for(int j=i+1;j<=n;++j)
{
if(a[i]==a[j])ans=(ans+1ll*tot*inv2)%MOD;
else if(a[i]<a[j])
{
int d=1ll*tot*fpow(Query(1,1,n,a[i]+1,a[j],0),MOD-2)%MOD;
d=1ll*d*Query(1,1,n,a[i]+1,a[j],1)%MOD;
ans=(ans+1ll*d*inv2)%MOD;
}
else
{
int d=1ll*tot*fpow(Query(1,1,n,a[j]+1,a[i],0),MOD-2)%MOD;
d=1ll*d*Query(1,1,n,a[j]+1,a[i],1)%MOD;
ans=(ans+MOD-1ll*d*inv2%MOD+tot)%MOD;
}
}
printf("%d\n",ans);
return 0;
}

而这种方法的复杂度瓶颈在于枚举任意两个位置之间逆序对的贡献。

考虑这个能否优化。

首先我们记\(D_i=\frac{cnt[i]-1-(n-i)}{cnt[i]-(n-i)}\),如果要修改某个区间的总方案数,等价于用全局的总方案数乘上某一段区间。设\(S\)为全局总方案数,那么强制求改\(a[i],a[j]\)成一样的总方案数就是:\(S\times \prod_{i=min+1}^{max}D_i\)。这个东西很显然可以写成前缀的形式,也就是\(S=\frac{\prod_{i=1}^{max}D_i}{\prod_{i=1}^{min}D_i}\)。

这样子只需要维护前缀积然后求个和就好了。

注意下\(D_i\)可能为\(0\),所以求的时候要分段计算一下贡献就好了,具体实现见代码。

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<vector>
using namespace std;
#define ll long long
#define MAX 200200
#define MOD 1000000007
#define inv2 500000004
inline int read()
{
int x=0;bool t=false;char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=true,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return t?-x:x;
}
int fpow(int a,int b)
{
int s=1;
while(b){if(b&1)s=1ll*s*a%MOD;a=1ll*a*a%MOD;b>>=1;}
return s;
}
int n,a[MAX],tot,ans;
int cnt[MAX];
int c1[MAX],c2[MAX];
int lb(int x){return x&(-x);}
void add(int x,int w){while(x<=n)c1[x]=(c1[x]+w)%MOD,++c2[x],x+=lb(x);}
int getsum1(int x){int ret=0;while(x)ret=(ret+c1[x])%MOD,x-=lb(x);return ret;}
int getsum2(int x){int ret=0;while(x)ret+=c2[x],x-=lb(x);return ret;}
int D[MAX],invD[MAX],zero[MAX],S[MAX];
int main()
{
n=read();tot=1;
for(int i=1;i<=n;++i)++cnt[a[i]=read()];
for(int i=n-1;i;--i)cnt[i]+=cnt[i+1];
for(int i=1;i<=n;++i)tot=1ll*tot*(cnt[i]-=(n-i))%MOD;
if(!tot){puts("0");return 0;}
S[0]=D[0]=1;
for(int i=1;i<=n;++i)
{
int x=1ll*(cnt[i]-1)*fpow(cnt[i],MOD-2)%MOD;
if(!x)S[zero[i]=zero[i-1]+1]=i,D[i]=D[i-1];
else zero[i]=zero[i-1],D[i]=1ll*D[i-1]*x%MOD;
invD[i]=fpow(D[i],MOD-2);
}
for(int i=1;i<=n;++i)
{
ans=(ans+1ll*(getsum1(a[i])-getsum1(S[zero[a[i]]]-1)+MOD)*D[a[i]]%MOD*tot%MOD*inv2%MOD)%MOD;
add(a[i],invD[a[i]]);
}
for(int i=1;i<=n;++i)c1[i]=c2[i]=0;
for(int i=n;i;--i)
{
ans-=1ll*(getsum1(a[i]-1)-getsum1(S[zero[a[i]]]-1)+MOD)*D[a[i]]%MOD*tot%MOD*inv2%MOD;
ans=(ans+1ll*getsum2(a[i]-1)*tot)%MOD;ans=(ans+MOD)%MOD;
add(a[i],invD[a[i]]);
}
printf("%d\n",ans);
return 0;
}

【agc023E】Inversions(线段树,动态规划)的更多相关文章

  1. POJ 2374 Fence Obstacle Course(线段树+动态规划)

    Fence Obstacle Course Time Limit: 3000MS   Memory Limit: 65536K Total Submissions: 2524   Accepted:  ...

  2. Codeforces 675E Trains and Statistic - 线段树 - 动态规划

    题目传送门 快速的vjudge通道 快速的Codeforces通道 题目大意 有$n$个火车站,第$i$个火车站出售第$i + 1$到第$a_{i}$个火车站的车票,特殊地,第$n$个火车站不出售车票 ...

  3. Codeforces 750E New Year and Old Subsequence - 线段树 - 动态规划

    A string t is called nice if a string "2017" occurs in t as a subsequence but a string &qu ...

  4. ACM学习历程—HDU5696 区间的价值(分治 && RMQ && 线段树 && 动态规划)

    http://acm.hdu.edu.cn/showproblem.php?pid=5696 这是这次百度之星初赛2B的第一题,但是由于正好打省赛,于是便错过了.加上2A的时候差了一题,当时有思路,但 ...

  5. 线段树详解 (原理,实现与应用)(转载自:http://blog.csdn.net/zearot/article/details/48299459)

    原文地址:http://blog.csdn.net/zearot/article/details/48299459(如有侵权,请联系博主,立即删除.) 线段树详解    By 岩之痕 目录: 一:综述 ...

  6. BZOJ_1672_[Usaco2005 Dec]Cleaning Shifts 清理牛棚_动态规划+线段树

    BZOJ_1672_[Usaco2005 Dec]Cleaning Shifts 清理牛棚_动态规划+线段树 题意:  约翰的奶牛们从小娇生惯养,她们无法容忍牛棚里的任何脏东西.约翰发现,如果要使这群 ...

  7. 【arc073f】Many Moves(动态规划,线段树)

    [arc073f]Many Moves(动态规划,线段树) 题面 atcoder 洛谷 题解 设\(f[i][j]\)表示第一个棋子在\(i\),第二个棋子在\(j\)的最小移动代价. 发现在一次移动 ...

  8. 【BZOJ5469】[FJOI2018]领导集团问题(动态规划,线段树合并)

    [BZOJ5469][FJOI2018]领导集团问题(动态规划,线段树合并) 题面 BZOJ 洛谷 题解 题目就是让你在树上找一个最大的点集,使得两个点如果存在祖先关系,那么就要满足祖先的权值要小于等 ...

  9. Codeforces 834D The Bakery - 动态规划 - 线段树

    Some time ago Slastyona the Sweetmaid decided to open her own bakery! She bought required ingredient ...

随机推荐

  1. Scikit-learn数据变换

    转载自:https://blog.csdn.net/Dream_angel_Z/article/details/49406573 本文主要是对照scikit-learn的preprocessing章节 ...

  2. 用 Delphi 7 实现基于 FFMS2 的视频转 GIF 工具 [原创]

    儿子经常要把自拍的视频(ts格式)转成表情包,下载了几个工具都不大好用,更多的还要收费.那就想自己写一个吧,没想到这一下断断续续地,居然 3 个月过去了.现在总算弄出个作品来了,结个贴吧.唉,天资愚钝 ...

  3. CHAPTER 25 The Greatest Show on Earth 第25章 地球上最壮观的演出

    CHAPTER 25 The Greatest Show on Earth 第25章 地球上最壮观的演出 Go for a walk in the countryside and you will f ...

  4. 【视频编解码·学习笔记】12. 图像参数集(PPS)介绍

    一.PPS相关概念: 除了序列参数集SPS之外,H.264中另一重要的参数集合为图像参数集Picture Paramater Set(PPS). 通常情况下,PPS类似于SPS,在H.264的裸码流中 ...

  5. dumpe2fs命令详解

    基础命令学习目录首页   dumpe2fs 显示ext2.ext3.ext4文件系统的超级快和块组信息.此命令的适用范围:RedHat.RHEL.Ubuntu.CentOS.SUSE.openSUSE ...

  6. 将项目托管到GitHub实现步骤

    修改于:2017.1.14 第一步:先注册一个Github的账号 注册地址:Github官网注册入口 第二步:准备工作 gitHub网站使用Git版本管理工具来对仓库进行管理,但是它们并不等同. gi ...

  7. Scrum立会报告+燃尽图(十二月五日总第三十六次):Final阶段分配任务

    此作业要求参见:https://edu.cnblogs.com/campus/nenu/2018fall/homework/2284 项目地址:https://git.coding.net/zhang ...

  8. Daily Scrumming* 2015.11.1(Day 13)

    一.今明两天任务表 Member Today’s Task Tomorrow’s Task 江昊 实现API小的更改 实现前后端整合 杨墨犁 实现首页 修改首页 付帅 实现创建编辑登录登出 测试修改 ...

  9. c++课的圆周面积

    又回顾了一下一两个月没动过的类,似乎又有点手生了,不过还好还可以做. 在栋哥的推荐下下载了一个vs2015,表示从dev的白鼠形式的简单操作缓过来还有些不习惯呢,不过有些功能,例如诊断还是挺好用的 这 ...

  10. SpringMVC(四)-- springmvc的系统学习之文件上传、ajax&json处理

    资源:尚学堂 邹波 springmvc框架视频 一.文件上传 1.步骤: (1)导入jar包 commons-fileupload,commons-io (2)在springmvc的配置文件中配置解析 ...