首页 > 其他分享 >洛谷题单指南-分治与倍增-P1966 [NOIP2013 提高组] 火柴排队

洛谷题单指南-分治与倍增-P1966 [NOIP2013 提高组] 火柴排队

时间:2024-09-12 09:47:01浏览次数:14  
标签:NOIP2013 idx int 洛谷题 value b1 b2 序列 P1966

原题链接:https://www.luogu.com.cn/problem/P1966

题意解读:计算两个序列∑(ai​−bi​)^2的最小值,对10^8-3取模。

解题思路:

1、贪心思路

要使得两个序列对应位置元素之差的平方和最小,必须满足两个序列相对排序是一致的,什么意思?

设a序列有两个元素:a1,a2,b序列有两个元素b1,b2

当a1<a2,b1<b2时,(a1-b1)^2 + (a2-b2)^2是最小的!

为什么?

可以分析,如果a1<a2 b1>b2,

令:A = (a1-b1)^2 + (a2-b2)^2,B = (a1-b2)^2 + (a2-b1)^2

将A,B展开,相减

B - A = 2a1b1+2a2b2-2a1b2-2a2b1 = 2a1(b1-b2)-2a2(b1-b2)=2(a1-a2)(b1-b2) < 0

所以B更小,也就是b1,b2交换位置之后,再a1,a2计算距离会更小

所以得出结论:两个序列的相对顺序保持一致是,对应元素之差的平方和最小。

2、逆序对求解

有了以上结论,就可以固定一个序列为标准顺序,然后求将另外一个序列转换成标准顺序需要交换的次数

标准顺序要定义为升序1~n的,另外一个序列转成1~n需要的交换次数就是其逆序对个数。

下面模拟样例:

序列1:1 3 4 2

序列2:1 7 2 4

如果以序列1为基准,先要将序列1的元素进行离散化,这里正好是1~4四个数字,离散化之后也保持不变,我们对每一个元素给定一个序号,已序号的位置为标准顺序

1 3 4 2
序号 1 2 3 4

再对序列2进行离散化处理

原值 1 7 2 4
离散化后值 1 4 2 3
序号 1 2 3 4
离散化后值在序列1中的序号 1 3 4 2

接下来,我们的目标是要将

转换为

只需要计算1 3 4 2的逆序对数,即为2。

100分代码:

#include <bits/stdc++.h>
using namespace std;

const int N = 100005, MOD = 1e8 - 3;

struct node 
{
    int value, idx;
};
int n;
long long ans;
node a[N], b[N];
int h[N], c[N];

bool cmp_value(node x, node y)
{
    return x.value < y.value;
}

bool cmp_idx(node x, node y)
{
    return x.idx < y.idx;
}

void merge(int s1, int e1, int s2, int e2)
{
    int i = s1, j = s2;
    int tmp[e2 - s1 + 1], cnt = 0;
    while(i <= e1 && j <= e2)
    {
        if(c[i] <= c[j]) tmp[++cnt] = c[i++];
        // 如果c[i] > c[j],则c[i]~c[e1]都会比c[j]大,对逆序对的贡献增加了e1 - i + 1个
        else tmp[++cnt] = c[j++], ans += e1 - i + 1; 
    } 
    while(i <= e1) tmp[++cnt] = c[i++];
    while(j <= e2) tmp[++cnt] = c[j++];
    for(int k = 1; k <= cnt; k++) c[k + s1 - 1] = tmp[k];
}

void merge_sort(int l, int r)
{
    if(l >= r) return;
    int mid = (l + r) / 2;
    merge_sort(l, mid);
    merge_sort(mid + 1, r);
    merge(l, mid, mid + 1, r);
}

int main()
{
    cin >> n;
    for(int i = 1; i <= n; i++) 
    {
        cin >> a[i].value;
        a[i].idx = i;
    }
    sort(a + 1, a + n + 1, cmp_value);
    for(int i = 1; i <= n; i++) a[i].value = i; //把数值按顺序离散化处理
    
    for(int i = 1; i <= n; i++) h[a[i].value] = a[i].idx; //保存value-idx的映射关系

    for(int i = 1; i <= n; i++)
    {
        cin >> b[i].value;
        b[i].idx = i;
    } 
    sort(b + 1, b + n + 1, cmp_value);
    for(int i = 1; i <= n; i++) b[i].value = i; //把数值按顺序离散化处理
    sort(b + 1, b + n + 1, cmp_idx); //还原为原来的顺序
    for(int i = 1; i <= n; i++) c[i] = h[b[i].value]; //c数组是将b的value替换为同样数值在a中对应的idx

    //计算c中的逆序对
    merge_sort(1, n);
    cout << ans % MOD;

    return 0;
}

 

标签:NOIP2013,idx,int,洛谷题,value,b1,b2,序列,P1966
From: https://www.cnblogs.com/jcwy/p/18408141

相关文章

  • 洛谷题单指南-分治与倍增-P1908 逆序对
    原题链接:https://www.luogu.com.cn/problem/P1908题意解读:求序列逆序对数。解题思路:1、暴力法对于每一个数,寻找后面有多少数比其小,或者采用冒泡排序,交换的次数即逆序对的个数,复杂度为O(n^2)2、归并排序法在归并排序过程中,会进行有序序列的合并,设两部分连续的有序序列为a[s1,......
  • 洛谷题单指南-常见优化技巧-P2880 [USACO07JAN] Balanced Lineup G
    原题链接:https://www.luogu.com.cn/problem/P2880题意解读:在若干个不定长区间里,求区间最大值与最小值之差解题思路:对于区间求最值,通常有几种方式:1、暴力法,通过枚举所有的区间来计算区间最值2、单调队列,针对区间长度固定的情况3、ST表,针对区间长度不固定且元素不会发生改变的......
  • 洛谷题单指南-常见优化技巧-P1886 滑动窗口 /【模板】单调队列
    原题链接:https://www.luogu.com.cn/problem/P1886题意解读:单调队列模版题。解题思路:采用双端队列维护单调的序列,单调队列三部曲:1、去头,当窗口内元素个数超过k,队头出队2、去尾,当要加入的元素会破坏单调性,队尾出队3、入队,将元素的下标存入队列每一次入队后,队头元素即为窗口最......
  • 洛谷题单指南-常见优化技巧-P3467 [POI2008] PLA-Postering
    原题链接:https://www.luogu.com.cn/problem/P3467题意解读:用长方形的海报覆盖建筑的侧面,最少需要的海报数如上图,左边最少需要3张,右边最少需要4张解题思路:可以看出,需要海报数与建筑宽度无关,只与高度有关。当建筑高度与之前不同时,肯定需要增加一张海报;当建筑高度与之前有相同......
  • 洛谷题单指南-常见优化技巧-P3143 [USACO16OPEN] Diamond Collector S
    原题链接:https://www.luogu.com.cn/problem/P3143题意解读:找到两个不相交的最长连续序列,使得序列最大值和最小值差不超过k,求两个最长的序列长度和。解题思路:先将所有数从小到大排序,记为a[]要找到两个不相交的最长连续序列,可以采用下面技巧:设b[i]表示i之前“差值在k之内的连续......
  • 洛谷题单指南-常见优化技巧-P4653 [CEOI2017] Sure Bet
    原题链接:https://www.luogu.com.cn/problem/P4653题意解读:选中的灯泡中,某一类较少的总权值减去灯泡数量所得到的收益最大值。解题思路:注意,此题关键是:要使得较少的收益最大化1、要最大化,意味着每次应该选择尽可能大权值的灯泡2、要使A、B类中较少的收益最大化,意味着每次应该优......
  • 洛谷题单指南-常见优化技巧-唯一的雪花 Unique Snowflakes
    原题链接:https://www.luogu.com.cn/problem/UVA11572题意解读:本质上是要计算最长连续不重复子序列的长度,典型的双指针应用。解题思路:通过双指针来枚举子序列,右指针指向的元素每次记录元素出现的次数,可以借助hash数组h[]如果枚举到的元素出现次数超过1,则表示左、右指针之间的子......
  • 洛谷题单指南-常见优化技巧-P2216 [HAOI2007] 理想的正方形
    原题链接:https://www.luogu.com.cn/problem/P2216题意解读:在矩阵中找n*n正方形里最大值和最小值差值的最小值。解题思路:1、枚举法直接枚举所有n*n的正方形的位置,然后在遍历求最大值、最小值,复杂度为O(n^4),显然不能通过。2、二维单调队列既然是求正方形范围内的最值,看起来是......