题目连接ios
思惟+贪心c++
时间复杂度O(nlogn)
算法
1.这道题的题意主要就是让你对一个数组进行一种特殊的排序,使得数组中相邻的两个数的差的绝对值成非递减趋势;数组
2.刚开始对这道题老是固执于两个相等的数在不一样位置,如何把它们放到前面
这个问题,由于路走歪了,最终无果,没有思路。后来看了一些关于这道题的解题博客,豁然开朗。spa
3.使得数组中相邻的两个数的差的绝对值成非递减趋势,怎么想呢。单纯想怎么从差的绝对值最小到最大变化不太容易想,咱们能够反过来想,怎么由差的绝对值最大到最小变化。何时差的绝对值最大,固然是数组中的最小值和最大值之间的差的绝对值最大。最小值和次大值之间的差的绝对值大
仍是最小值和次小值之间的差的绝对值大
(或者最大值和次小值的差的绝对值大
仍是最大值和次大值的绝对值大
),固然是前者。code
4.而后在想接下来可能再小的是什么,固然是次小值和次大值之间的差的绝对值。以此类推,能够的出下面这个式子。排序
最小值、最大值、次小值、次大值、第三小值、第三大值、... 或者 最大值、最小值、次大值、次小值、第三大值、第三小值、...
注意上面的式子只是为了好理解才按照这个顺序写的,最终输出的时候不要忘了把它倒过来(不知道为啥请看题意)。ci
5.列出了式子后,那么思考一下何时才到头呢,即到哪里结束呢?get
对于偶数个数的数组来说,即最终达处处于中间的那两个数;博客
对于奇数个数的数组来说,即最终到达处于中间的那一个数。
因此要特判一下。这也是为何前面一开始就说要将题意倒过来想,不然直接想出从中间向两边展开这个思路不太容易。
6.因此最终得出的思路是先对数组排序,而后从中间向两边展开输出。
#include<iostream> #include<algorithm> using namespace std; const int N = 1e5 + 10; int t, n; int a[N]; int main() { cin >> t; while(t--) { cin >> n; for(int i = 0; i < n; i++) cin >> a[i]; sort(a, a + n); /* n = 5 0,1,2,3,4 n = 4 0,1,2,3 */ int l, r; if(n % 2 == 1) { cout << a[n/2] << " "; l = n / 2 - 1, r = n / 2 + 1; } else l = n / 2 - 1, r = n / 2; while(l >= 0 && r < n) { //cout << a[r] << " " << a[l] << " "; cout << a[l] << " " << a[r] << " "; //上面这两个式子用哪一个均可以 ++r; --l; } puts(""); } return 0; }