[CF351E]Jeff and Permutation——贪心

题目大意:

给出数组a ,你可以改变每个数的正负,求逆序对数最少是多少

思路:

考虑一个数对 ( a i , a j ) (a_i,a_j) (ai,aj)的贡献之和绝对值大的那个数字有关。
于是我们把每个数对的计算放在绝对值较大的那个数上面,不难发现这个位置产生的逆序对只和它自己本身的正负有关,直接BIT贪心就好了。

#include<bits/stdc++.h>

#define REP(i,a,b) for(int i=a,i##_end_=b;i<=i##_end_;++i)
#define DREP(i,a,b) for(int i=a,i##_end_=b;i>=i##_end_;--i)
#define debug(x) cout<<#x<<"="<<x<<endl
#define fi first
#define se second
#define mk make_pair
#define pb push_back
typedef long long ll;

using namespace std;

void File(){
	freopen("inverse.in","r",stdin);
	freopen("inverse.out","w",stdout);
}

template<typename T>void read(T &_){
	_=0; T f=1; char c=getchar();
	for(;!isdigit(c);c=getchar())if(c=='-')f=-1;
	for(;isdigit(c);c=getchar())_=(_<<1)+(_<<3)+(c^'0');
	_*=f;
}

const int maxn=1e6+10;
const int N=1e6;
int n,a[maxn];
ll ans;

struct BIT{
	int sum[maxn];
	int lowbit(int x){return x&(-x);}
	void add(int p,int x){for(;p<=N;p+=lowbit(p))sum[p]+=x;}
	int query(int p){int ret=0;for(;p>=1;p-=lowbit(p))ret+=sum[p];return ret;}
}T1,T2;

int main(){
	File();
	read(n);
	REP(i,1,n)read(a[i]),a[i]=abs(a[i])+1;
	REP(i,1,n)T1.add(a[i],1);
	REP(i,1,n){
		ans+=min(T1.query(a[i]-1),T2.query(a[i]-1));
		T1.add(a[i],-1);
		T2.add(a[i],1);
		//printf("%d\n",T1.query(N)+T2.query(N));
	}
	printf("%lld\n",ans);
	return 0;
}

对于 `next_permutation` 函数来说,不能使用 `break` 语句直接退出循环。因为 `next_permutation` 是一个算法,它的作用是生成下一个排列。`next_permutation` 函数会不断生成下一个排列,直到所有的排列都被生成完毕或者满足某个条件时停止。 如果你想在某个条件成立时退出循环,你可以使用 `std::find_if` 算法来查找满足条件的元素,并将其作为终止条件传递给 `next_permutation` 函数。这样,当满足条件的元素被找到时,`next_permutation` 函数会在下一次迭代中返回 `false`,从而退出循环。 以下是一个示例代码,展示了如何在满足某个条件时退出 `next_permutation` 循环: ```cpp #include <iostream> #include <algorithm> #include <vector> bool condition_met(const std::vector<int>& permutation) { // 满足某个条件的判断逻辑 // 返回 true 表示满足条件,退出循环 // 返回 false 表示不满足条件,继续迭代 // 这里只是一个示例,实际情况根据具体需求来定义 return permutation[0] == 1 && permutation[1] == 2; } int main() { std::vector<int> numbers = {1, 2, 3}; do { // 处理当前排列 // ... if (condition_met(numbers)) { break; // 满足条件时退出循环 } } while (std::next_permutation(numbers.begin(), numbers.end())); // 继续执行其他操作 // ... return 0; } ``` 在上述示例中,`condition_met` 函数用于判断当前排列是否满足条件。当满足条件时,使用 `break` 语句退出 `next_permutation` 循环。如果不满足条件,则继续迭代生成下一个排列。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值