蓝桥杯历年试题 小朋友排队

问题描述

  n 个小朋友站成一排。现在要把他们按身高从低到高的顺序排列,但是每次只能交换位置相邻的两个小朋友。

  每个小朋友都有一个不高兴的程度。开始的时候,所有小朋友的不高兴程度都是0。

  如果某个小朋友第一次被要求交换,则他的不高兴程度增加1,如果第二次要求他交换,则他的不高兴程度增加2(即不高兴程度为3),依次类推。当要求某个小朋友第k次交换时,他的不高兴程度增加k。

  请问,要让所有小朋友按从低到高排队,他们的不高兴程度之和最小是多少。

  如果有两个小朋友身高一样,则他们谁站在谁前面是没有关系的。

输入格式

  输入的第一行包含一个整数n,表示小朋友的个数。
  第二行包含 n 个整数 H1 H2 … Hn,分别表示每个小朋友的身高。

输出格式

  输出一行,包含一个整数,表示小朋友的不高兴程度和的最小值。

样例输入

3
3 2 1

样例输出

9

样例说明

  首先交换身高为3和2的小朋友,再交换身高为3和1的小朋友,再交换身高为2和1的小朋友,每个小朋友的不高兴程度都是3,总和为9。

数据规模和约定

  对于10%的数据, 1<=n<=10;
  对于30%的数据, 1<=n<=1000;
  对于50%的数据, 1<=n<=10000;
  对于100%的数据,1<=n<=100000,0<=Hi<=1000000。

思路:

逆序数的变形,不仅要求逆序数的对数,还要把对数明确到每一个单位,在求的过程中要注意排除相等的情况(如果有更方便的排除法麻烦留个言,感觉自己写的判断有点麻烦),用数组数组求也需要判断。

使用了归并排序求逆序数的方法,把模板改了一下。

#include <iostream>
#include <cstdio>
#include <string>
#include <cmath>
#include <map>
#include <algorithm>
#define LL long long int
#define N 100005
using namespace std;
int cmp(int a,int b)
{
    if(a==b) return 0;
    return a<b?-1:1;
}
struct node
{
    int num;
    int ang;

};
LL ans[N];
void MergeArr(node num[], int left, int mid, int right)
{
    node AL[N], AR[N];
    int lenl=mid-left+1,lenr=right-mid;
    for (int i = 0;i < lenl;i++)
        AL[i] = num[left + i];
    for (int i = 0;i < lenr;i++)
        AR[i] = num[mid + 1 + i];
    int j = 0, k = 0,pos=left;
    int tempNum=-1,tempCnt=0;
    while (j < lenl&&k < lenr)
    {
        int fix=cmp(AL[j].num, AR[k].num);
        if (fix<=0)
        {
            num[pos++] = AL[j++],num[pos-1].ang+=k;
            if(num[pos-1].num==tempNum)
                num[pos-1].ang-=tempCnt;
        }
        //else if(fix==-1) num[pos++] = AL[j++];
        else
            num[pos++] = AR[k++],num[pos-1].ang+=mid-(left+j)+1;//关键步骤
        if(fix>0)
        {
            if(num[pos-1].num==tempNum)
                tempCnt++;
            else
                tempNum=num[pos-1].num,tempCnt=1;
        }

    }
    while (j < lenl)
    {
        num[pos++] = AL[j++];
        num[pos-1].ang+=k;
        if(num[pos-1].num==tempNum)
            num[pos-1].ang-=tempCnt;
    }
    while (k < lenr)
        num[pos++] = AR[k++];
}
void MergeSort(node num[], int left, int right)
{
    if (left < right)
    {
        int mid = (right + left) / 2;
        MergeSort(num, left, mid);
        MergeSort(num, mid+1, right);
        MergeArr(num, left, mid, right);
    }
}
LL mat(LL num)
{
    return num*(num+1)/2;
}
node num[N];
int n;
int main()
{
    cin.sync_with_stdio(false);
    while(cin>>n)
    {
        LL fuck=0;
        for(int i=0;i<n;i++)
            cin>>num[i].num,num[i].ang=0;
        MergeSort(num,0,n-1);
        for(int i=0;i<n;i++)
            fuck+=mat(num[i].ang);

        cout<<fuck<<endl;
    }
    return 0;
}
时间: 2024-10-03 16:28:19

蓝桥杯历年试题 小朋友排队的相关文章

蓝桥杯 历届试题 小朋友排队 树状数组实现

历届试题 小朋友排队 时间限制:1.0s   内存限制:256.0MB 问题描述 n 个小朋友站成一排.现在要把他们按身高从低到高的顺序排列,但是每次只能交换位置相邻的两个小朋友. 每个小朋友都有一个不高兴的程度.开始的时候,所有小朋友的不高兴程度都是0. 如果某个小朋友第一次被要求交换,则他的不高兴程度增加1,如果第二次要求他交换,则他的不高兴程度增加2(即不高兴程度为3),依次类推.当要求某个小朋友第k次交换时,他的不高兴程度增加k. 请问,要让所有小朋友按从低到高排队,他们的不高兴程度之和

蓝桥杯 历届试题 小朋友排队 【树状数组】+【逆序数】

历届试题 小朋友排队 时间限制:1.0s   内存限制:256.0MB 问题描述 n 个小朋友站成一排.现在要把他们按身高从低到高的顺序排列,但是每次只能交换位置相邻的两个小朋友. 每个小朋友都有一个不高兴的程度.开始的时候,所有小朋友的不高兴程度都是0. 如果某个小朋友第一次被要求交换,则他的不高兴程度增加1,如果第二次要求他交换,则他的不高兴程度增加2(即不高兴程度为3),依次类推.当要求某个小朋友第k次交换时,他的不高兴程度增加k. 请问,要让所有小朋友按从低到高排队,他们的不高兴程度之和

蓝桥杯 历届试题 小朋友排队

历届试题 小朋友排队 时间限制:1.0s   内存限制:256.0MB 问题描述 n 个小朋友站成一排.现在要把他们按身高从低到高的顺序排列,但是每次只能交换位置相邻的两个小朋友. 每个小朋友都有一个不高兴的程度.开始的时候,所有小朋友的不高兴程度都是0. 如果某个小朋友第一次被要求交换,则他的不高兴程度增加1,如果第二次要求他交换,则他的不高兴程度增加2(即不高兴程度为3),依次类推.当要求某个小朋友第k次交换时,他的不高兴程度增加k. 请问,要让所有小朋友按从低到高排队,他们的不高兴程度之和

【蓝桥杯】历届试题 小朋友排队(运行超时)

  历届试题 小朋友排队   时间限制:1.0s   内存限制:256.0MB 问题描述 n 个小朋友站成一排.现在要把他们按身高从低到高的顺序排列,但是每次只能交换位置相邻的两个小朋友. 每个小朋友都有一个不高兴的程度.开始的时候,所有小朋友的不高兴程度都是0. 如果某个小朋友第一次被要求交换,则他的不高兴程度增加1,如果第二次要求他交换,则他的不高兴程度增加2(即不高兴程度为3),依次类推.当要求某个小朋友第k次交换时,他的不高兴程度增加k. 请问,要让所有小朋友按从低到高排队,他们的不高兴

蓝桥杯 历届试题 题目总结

后天就是蓝桥杯省赛了,今天总结一下这段时间做的蓝桥杯历届试题,还是一个一个题目的来吧!!!!!! 1,历届试题 矩阵翻硬币 这个题目说真的,我不会,在网上看了某神牛的题解答案为 ans=sqrt(n)*sqrt(m),具体怎么证明的我也不知道 2,历届试题 兰顿蚂蚁 这个题目怎么说呢,应该是送分题,直接模拟就可以了,这里就不说了. 3, 历届试题 分糖果 这个题目好像之前在哪里做过,也是一道模拟题,弄两个数组搞一下就可以了 下面是代码 #include<bits/stdc++.h> using

蓝桥杯——历年真题之带分数

问题描述 100 可以表示为带分数的形式:100 = 3 + 69258 / 714. 还可以表示为:100 = 82 + 3546 / 197. 注意特征:带分数中,数字1~9分别出现且只出现一次(不包含0). 类似这样的带分数,100 有 11 种表示法. 输入格式 从标准输入读入一个正整数N (N<1000*1000) 输出格式 程序输出该数字用数码1~9不重复不遗漏地组成带分数表示的全部种数. 注意:不要求输出每个表示,只统计有多少表示法! 样例输入1 100 样例输出1 11 样例输入

《拼音字母》 蓝桥杯复试试题

题目描述 [编程题](满分19分) 在很多软件中,输入拼音的首写字母就可以快速定位到某个词条.比如,在铁路售票软件中,输入: "bj"就可以定位到"北京".怎样在自己的软件中实现这个功能呢?问题的关键在于:对每个汉字必须能计算出它的拼音首字母. GB2312汉字编码方式中,一级汉字的3755个是按照拼音顺序排列的.我们可以利用这个特征,对常用汉字求拼音首字母. GB2312编码方案对每个汉字采用两个字节表示.第一个字节为区号,第二个字节为区中的偏移号.为了能与已有的

《DNA比对》蓝桥杯复赛试题

题目描述 脱氧核糖核酸即常说的DNA,是一类带有遗传信息的生物大分子.它由4种主要的脱氧核苷酸(dAMP.dGMP.dCMT和dTMP)通过磷酸二酯键连接而成.这4种核苷酸可以分别记为:A.G.C.T. DNA携带的遗传信息可以用形如:AGGTCGACTCCA.... 的串来表示.DNA在转录复制的过程中可能会发生随机的偏差,这才最终造就了生物的多样性. 为了简化问题,我们假设,DNA在复制的时候可能出现的偏差是(理论上,对每个碱基被复制时,都可能出现偏差): 1. 漏掉某个脱氧核苷酸.例如把

蓝桥杯历届试题 地宫取宝 dp or 记忆化搜索

问题描述 X 国王有一个地宫宝库.是 n x m 个格子的矩阵.每个格子放一件宝贝.每个宝贝贴着价值标签. 地宫的入口在左上角,出口在右下角. 小明被带到地宫的入口,国王要求他只能向右或向下行走. 走过某个格子时,如果那个格子中的宝贝价值比小明手中任意宝贝价值都大,小明就可以拿起它(当然,也可以不拿). 当小明走到出口时,如果他手中的宝贝恰好是k件,则这些宝贝就可以送给小明. 请你帮小明算一算,在给定的局面下,他有多少种不同的行动方案能获得这k件宝贝. 输入格式 输入一行3个整数,用空格分开:n