P1966 火柴排队

题目描述

涵涵有两盒火柴,每盒装有 n 根火柴,每根火柴都有一个高度。 现在将每盒中的火柴各自排成一列, 同一列火柴的高度互不相同, 两列火柴之间的距离定义为: ∑(ai-bi)^2

其中 ai 表示第一列火柴中第 i 个火柴的高度,bi 表示第二列火柴中第 i 个火柴的高度。

每列火柴中相邻两根火柴的位置都可以交换,请你通过交换使得两列火柴之间的距离最小。请问得到这个最小的距离,最少需要交换多少次?如果这个数字太大,请输出这个最小交换次数对 99,999,997 取模的结果。

输入输出格式

输入格式:

输入文件为 match.in。

共三行,第一行包含一个整数 n,表示每盒中火柴的数目。

第二行有 n 个整数,每两个整数之间用一个空格隔开,表示第一列火柴的高度。

第三行有 n 个整数,每两个整数之间用一个空格隔开,表示第二列火柴的高度。

输出格式:

输出文件为 match.out。

输出共一行,包含一个整数,表示最少交换次数对 99,999,997 取模的结果。

输入输出样例

输入样例#1:

  1. 【输入输出样例 1
  2. 4
  3. 2 3 1 4
  4. 3 2 1 4
  5. 【输入输出样例 2
  6. 4
  7. 1 3 4 2
  8. 1 7 2 4
输出样例#1:

  1. 【输入输出样例 1
  2. 1
  3. 【输入输出样例 2
  4. 2

说明

【输入输出样例说明1】

最小距离是 0,最少需要交换 1 次,比如:交换第 1 列的前 2 根火柴或者交换第 2 列的前 2 根火柴。

【输入输出样例说明2】

最小距离是 10,最少需要交换 2 次,比如:交换第 1 列的中间 2 根火柴的位置,再交换第 2 列中后 2 根火柴的位置。

【数据范围】

对于 10%的数据, 1 ≤ n ≤ 10;

对于 30%的数据,1 ≤ n ≤ 100;

对于 60%的数据,1 ≤ n ≤ 1,000;

对于 100%的数据,1 ≤ n ≤ 100,000,0 ≤火柴高度≤ maxlongint

  1. /*
  2. 公式化简后要求Σai*bi的最小值
  3. 根据均值不等式plus plus+直观感受
  4. aibi越接近乘积就越大
  5. 所以考虑a b 一一对应
  6. 离散化后
  7. 先把a排序并记录b对应a的位置
  8. 然后处理b。因为a是有序的b也成为有序的求最小交换次数
  9. 那就是求b的逆序个数咯
  10. */
  11. #include<iostream>
  12. #include<cstdio>
  13. #include<cstring>
  14. #include<algorithm>
  15.  
  16. #define ll long long
  17. #define N 100007
  18. #define mod 99999997
  19. #define inf 2147483647
  20.  
  21. using namespace std;
  22. ll n,m,ans,tot;
  23. ll s[N];
  24. struct ta
  25. {
  26. ll val,id;
  27. bool operator < (const ta &a)const{
  28. return val<a.val;
  29. }
  30. };
  31. ta a[N],b[N];
  32. struct tree
  33. {
  34. ll l,r,sum;
  35. }tr[N<<];
  36.  
  37. inline ll read()
  38. {
  39. ll x=,f=;char c=getchar();
  40. while(c>''||c<''){if(c=='-')f=-;c=getchar();}
  41. while(c>=''&&c<=''){x=x*+c-'';c=getchar();}
  42. return x*f;
  43. }
  44.  
  45. inline void pushup(ll k)
  46. {
  47. tr[k].sum=tr[k<<].sum+tr[k<<|].sum;
  48. }
  49.  
  50. void build(ll k,ll l,ll r)
  51. {
  52. tr[k].l=l;tr[k].r=r;
  53. if(l==r)
  54. {
  55. tr[k].sum=;
  56. return;
  57. }
  58. ll mid=(l+r)>>;
  59. build(k<<,l,mid);build(k<<|,mid+,r);
  60. }
  61.  
  62. void insert(ll k,ll pos)
  63. {
  64. if(tr[k].l==tr[k].r && tr[k].l==pos)
  65. {
  66. tr[k].sum=(tr[k].sum+)%mod;
  67. return;
  68. }
  69. ll mid=(tr[k].l+tr[k].r)>>;
  70. if(pos<=mid) insert(k<<,pos);
  71. if(pos>mid) insert(k<<|,pos);
  72. pushup(k);
  73. }
  74.  
  75. ll query(ll k,ll l,ll r)
  76. {
  77. if(l>r) return ;
  78. if(tr[k].l==l && tr[k].r==r)
  79. return tr[k].sum;
  80. ll mid=(tr[k].l+tr[k].r)>>;
  81. if(r<=mid) return query(k<<,l,r)%mod;
  82. else if(l>mid) return query(k<<|,l,r)%mod;
  83. else return query(k<<,l,mid)%mod+query(k<<|,mid+,r)%mod;
  84. }
  85.  
  86. void love()
  87. {
  88. build(,,tot);ans=;
  89. for(int i=;i<=n;i++)
  90. {
  91. insert(,s[i]);
  92. ans+=query(,s[i]+,tot)%mod;
  93. ans%=mod;
  94. }
  95. }
  96.  
  97. int main()
  98. {
  99. n=read();tot=;
  100. for (int i=;i<=n;i++){a[i].val=read();a[i].id=i;}
  101. for (int i=;i<=n;i++){b[i].val=read();b[i].id=i;tot=max(tot,b[i].val);}
  102. sort(a+,a+n+);
  103. sort(b+,b+n+);
  104. for (int i=;i<=n;i++) s[a[i].id]=b[i].id;
  105. love();
  106. printf("%lld\n",ans);
  107. return ;
  108. }

P1966 火柴排队(逆序对)的更多相关文章

  1. luogu P1966 火柴排队 (逆序对)

    luogu P1966 火柴排队 题目链接:https://www.luogu.org/problemnew/show/P1966 显然贪心的想,排名一样的数相减是最优的. 证明也很简单. 此处就不证 ...

  2. P1966 火柴排队——逆序对(归并,树状数组)

    P1966 火柴排队 很好的逆序对板子题: 求的是(x1-x2)*(x1-x2)的最小值: x1*x1+x2*x2-2*x1*x2 让x1*x2最大即可: 可以证明将b,c数组排序后,一一对应的状态是 ...

  3. 洛谷P1966 火柴排队(逆序对)

    题意 题目链接 Sol 不算很难的一道题 首先要保证权值最小,不难想到一种贪心策略,即把两个序列中rank相同的数放到同一个位置 证明也比较trivial.假设\(A\)中有两个元素\(a, b\), ...

  4. NOIP2013火柴排队[逆序对]

    题目描述 涵涵有两盒火柴,每盒装有 n 根火柴,每根火柴都有一个高度. 现在将每盒中的火柴各自排成一列, 同一列火柴的高度互不相同, 两列火柴之间的距离定义为: ∑(ai-bi)^2 其中 ai 表示 ...

  5. [NOIP2013提高&洛谷P1966]火柴排队 题解(树状数组求逆序对)

    [NOIP2013提高&洛谷P1966]火柴排队 Description 涵涵有两盒火柴,每盒装有 n 根火柴,每根火柴都有一个高度. 现在将每盒中的火柴各自排成一列, 同一列火柴的高度互不相 ...

  6. 【刷题】洛谷 P1966 火柴排队

    题目描述 涵涵有两盒火柴,每盒装有 n 根火柴,每根火柴都有一个高度. 现在将每盒中的火柴各自排成一列, 同一列火柴的高度互不相同, 两列火柴之间的距离定义为: ∑(ai-bi)^2 其中 ai 表示 ...

  7. 洛谷 P1966 火柴排队 解题报告

    P1966 火柴排队 题目描述 涵涵有两盒火柴,每盒装有 \(n\) 根火柴,每根火柴都有一个高度. 现在将每盒中的火柴各自排成一列, 同一列火柴的高度互不相同, 两列火柴之间的距离定义为: \(\s ...

  8. 洛谷——P1966 火柴排队&&P1774 最接近神的人_NOI导刊2010提高(02)

    P1966 火柴排队 这题贪心显然,即将两序列中第k大的数的位置保持一致,证明略: 树状数组求逆序对啦 浅谈树状数组求逆序对及离散化的几种方式及应用 方法:从前向后每次将数插入到bit(树状数组)中, ...

  9. 洛谷P1966 火柴排队 贪心+离散化+逆序对(待补充QAQ

    正解: 贪心+离散化+逆序对 解题报告: 链接在这儿呢quq 这题其实主要难在想方法吧我觉得?学长提点了下说用贪心之后就大概明白了,感觉没有很难 但是离散化这里还是挺有趣的,因为并不是能很熟练地掌握离 ...

随机推荐

  1. redis键的过期和内存淘汰策略

    键的过期时间 设置过期时间 Redis可以为存储在数据库中的值设置过期时间,作为一个缓存数据库,这个特性是很有帮助的.我们项目中的token或其他登录信息,尤其是短信验证码都是有时间限制的. 按照传统 ...

  2. 【LeetCode】1、Two Sum

    题目等级:Easy 题目描述:   Given an array of integers, return indices of the two numbers such that they add u ...

  3. Array.prototype.slice.call()的理解

    最近在看廖雪峰的JS课程,浏览器中的操作DOM的那一章,有这样一道题. JavaScript Swift HTML ANSI C CSS DirectX <!-- HTML结构 --> & ...

  4. uva1584 Circular Sequence(Uva-1584)

    vj:https://vjudge.net/problem/UVA-1584 这个题讲的是一个圆环,圆环上面有一堆字母,找出字典序最小的那一圈 这个题我觉得直接用c语言的strcmp那一套感觉真是用不 ...

  5. TypeError与ValueError的区别

    typeerror:函数或方法接受了不适当的[类型]的参数,比如sum('nick'),sum函数不接受字符串类型:valueerror:函数或方法虽然接受了正确的[类型]的参数,但是该参数的[值]不 ...

  6. 洛谷 2042 BZOJ 1500 NOI 2005 维护数列

    [题意概述] 维护一个数列,要求支持以下6种操作: [题解] 大Boss...可以用Treap解决 需要用到垃圾回收.线性建树. #include<cstdio> #include< ...

  7. BZOJ 4278 [ONTAK2015]Tasowanie (后缀数组)

    题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=4278 题解: 居然把后缀数组写成n^2的..我真厉害.. 想了无数种方法,最后发现就是 ...

  8. 【Codeforces 27A】Next Test

    [链接] 我是链接,点我呀:) [题意] 让你求没出现过的最小值 [题解] 模拟..for一下就好 [代码] import java.io.*; import java.util.*; public ...

  9. 【codeforces 755F】PolandBall and Gifts

    [题目链接]:http://codeforces.com/contest/755/problem/F [题意] n个人; 计划是每个人都拿一个礼物来送给一个除了自己之外的人; 且如果一个人没有送出礼物 ...

  10. 【codeforces 514A】Chewbaсca and Number

    [题目链接]:http://codeforces.com/contest/514/problem/A [题意] 允许你把每个数字翻转 ->x变成9-x 然后问你能够变成的最小的数字是什么; 不能 ...