P8613 [蓝桥杯 2014 省 B] 小朋友排队 题解

复习蓝桥杯的时候碰到了这道题,发现题解有的并没有讲得很清楚,所以想在这里补充一下。

首先我们看一下有关题意:

        简而言之就是给你一个乱序序列,之后让你求最少付出多少代价就可以使得整个序列为有序序列,我们最开始想到的一定是归并排序,很好,这说明我们对这个问题是有所了解的,但是,因为归并排序只是简单求逆序对数量,我们无法直到最终每个小朋友的交换次数,所以这种方法是行不通的。

        那么现在问题的核心在于如何快速求每个小朋友对应的逆序对数量(前面比他高的与后面比他矮的数量之和),遇事不决,肯定可以暴力去做,但是暴力是双重for循环,肯定会超时,所以有没有一种快速求得某个区间中的比某个数字大的个数的方法呢,这时我们可以使用树状数组。

        树状数组适用于求某个区间和,同时可以动态的进行添加,我们结合这两个性质思考一下,由于所有的小朋友数量是固定的,我们可以按照下面的步骤求小朋友数量:

1、首先按顺序读入,每次读入一个,就查询前面比这个小朋友高的数量,之后将这个小朋友的身高添加进去

2、之后逆序读入,每次读入一个,就查询后面比这个小朋友身高矮的数量,之后将这个小朋友的身高添加进去

        按照这样的步骤我们就可以求出每个小朋友的交换次数,之后求1+2+3+...+n即可,这里可以用等差数列性质求解。

下面附上源代码:

#include<iostream>
using namespace std;
#include<cstring>

typedef long long LL;

//本题每个小朋友交换的次数就是每个小朋友前后逆序对的数量
//所以结果就是求出每个小朋友逆序对的数量并进行累加
//但是问题核心就在于如何求出每个小朋友对应的逆序对数量
const int N = 1e6+10;
int tr[N],sum[N],h[N];
//tr为每个小朋友前面比他身高高的人的数量或者后面比他身高矮的数量,会进行实时添加
int n;

//下面三个为树状数组常用函数
int lowbit(int x) {
    return x&-x;
}

void add(int u,int v) {
    for(int i=u;i<N;i+=lowbit(i)) tr[i]+=v;
}

int query(int x){
    int res=0;
    for(int i=x;i>0;i-=lowbit(i)) res+=tr[i];
    return res;
}

int main() {
    cin>>n;
    
    //还有身高为0的小朋友
    for(int i=1;i<=n;i++) scanf("%d",&h[i]),h[i]++;
    
    //先统计前面比他身高高的小朋友数量
    for(int i=1;i<=n;i++){
        //小于最大身高数量-当前小朋友身高数量
        sum[i]+=query(N-1)-query(h[i]);
        add(h[i],1);
    }
    
    memset(tr,0,sizeof(tr));
    //之后统计后面比他身高矮的小朋友
    //这里要清空tr数组
    for(int i=n;i>=1;i--){
        sum[i]+=query(h[i]-1);
        add(h[i],1);
    }
    
    LL res=0;
    //一共交换sum[i]次,等差数列求和公式
    for(int i=1;i<=n;i++) res+=(LL)sum[i]*(sum[i]+1)/2;
    
    printf("%lld",res);
    
    return 0;
}

以上就是全部内容了,通过这道题我们可以总结经验,树状数组不仅仅能够实现区间查询与实时修改,同时还可以求逆序对数量(实际上也是区间查询与实时修改的变种)

  • 8
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
题目描述 如果一个数的十进制表示法中,任何相邻的两个数字均不相同,那么这个数就被称为 “有趣的数”。例如,以下各数都是有趣的数: 1、21、321、4321 但是,以下各数却不是有趣的数: 22、121、3212 给出正整数 n,请你求出长度为 n 的有趣的数的个数,并输出这个值除以 998244353 的余数。 输入格式 输入一行,包含一个整数 n。 输出格式 输出一个整数,表示答案除以 998244353 的余数。 数据范围 1≤n≤1000 输入样例1: 2 输出样例1: 81 输入样例2: 3 输出样例2: 531441 题解 有趣的数 求长度为n的有趣数的个数 思路: 第一位有9种选择,第二位有9种选择,第三位有8种选择(因为只有和前一位不一样才可以) 代码1: 时间复杂度 O(n) C++ 代码 #include <iostream> #include <cstdio> using namespace std; int main() { int n; scanf("%d", &n); int mod = 998244353; long long res = 1; for (int i = 1; i <= n; i ++ ) res = res * (i <= 2 ? 9 : 10 - i % 2) % mod; printf("%lld\n", res); return 0; } 代码2: 时间复杂度 O(n) C++ 代码 #include <iostream> #include <cstdio> using namespace std; const int MOD = 998244353; int main() { int n; scanf("%d", &n); int f[1010][10]; for (int i = 0; i < 10; i ++ ) f[1][i] = 1; for (int i = 2; i <= n; i ++ ) for (int j = 0; j < 10; j ++ ) for (int k = 0; k < 10; k ++ ) if (j != k) f[i][j] = (f[i][j] + f[i - 1][k]) % MOD; int res = 0; for (int i = 1; i < 10; i ++ ) res = (res + f[n][i]) % MOD; printf("%d\n", res); return 0; }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

lian潋湄

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值