[FROM LUOGU]火柴排队

传送门

为什么是提高+/省选- 啊(

SOL
其实怎样的顺序最优是很显然的,不过这里我还是给出一个证明
要证明 a i ai ai b i bi bi按大小顺序一一对应为最优,我们只需证明一个子问题即可
设在排过序后的火柴序列中选取 i < j i<j i<j,那么 a i < a j , b i < b j ai<aj,bi<bj ai<aj,bi<bj,若调换其位置,那么答案的变化即为:
d e l t a = ( a i − b i ) 2 + ( a j − b j ) 2 − ( a i − b j ) 2 − ( a j − b i ) 2 delta=(ai-bi)^2+(aj-bj)^2-(ai-bj)^2-(aj-bi)^2 delta=(aibi)2+(ajbj)2(aibj)2(ajbi)2
化简可得
d e l t a = 2 ( b j − b i ) ∗ ( a j − a i ) > 0 delta=2(bj-bi)*(aj-ai)>0 delta=2(bjbi)(ajai)>0
所以调换后的答案更大,显然不优
利用数学归纳法可以证明整个序列具有同样的性质
证毕

接下来的操作就比较容易了
由于火柴只能相邻的直接交换,因此每根火柴的交换次数即为其逆序对个数,于是树状数组维护即可

代码:

#include<bits/stdc++.h>
#define re register
using namespace std;
inline int rd(){
	int data=0;static char ch=0;ch=getchar();
	while(!isdigit(ch))ch=getchar();
	while(isdigit(ch))data=(data<<1)+(data<<3)+(ch^48),ch=getchar();
	return data;
}
const int mod=99999997,N=1e5+5;
struct matches{int hi,id,pa;}a[N],b[N];
inline bool cmp1(const matches&x,const matches&y){return x.hi<y.hi;}
inline bool cmp2(const matches&x,const matches&y){return x.id<y.id;}
int n,c[N],ans;
#define lb(x) (x&-x)
inline void edit(int p,int k){for(;p<=n;p+=lb(p))c[p]+=k;}
inline int ask(int p,int ret=0){for(;p;p-=lb(p))ret+=c[p];return ret;}
signed main(){
	n=rd();
	for(int re i=1;i<=n;++i)a[i].hi=rd(),a[i].id=i;
	for(int re i=1;i<=n;++i)b[i].hi=rd(),b[i].id=i;
	sort(a+1,a+n+1,cmp1),sort(b+1,b+n+1,cmp1);
	for(int re i=1;i<=n;++i)b[i].pa=a[i].id;
	sort(a+1,a+n+1,cmp2),sort(b+1,b+n+1,cmp2);
	for(int re i=1;i<=n;++i)(ans+=i-1-ask(b[i].pa))%=mod,edit(b[i].pa,1);
	cout<<ans,exit(0);
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值