题意:给你n个数, 要求选若干个数, 使得这些数的平均数减去中位数尽量大。
思路:由于该题没有顺序问题, 排好序之后我们可以枚举中位数, 可以证明, 奇数个数一定比偶数优,然后三分中位数左右区间长度x(数的个数), 在中位数的右边选最大的x个数, 在左边也选最大的x个, 这样, 随着区间长度的增加, 平均数将先增大后减小, 或者一直减小,或者一直增大。
为什么呢? 假设第一次的区间长度是1, 那么我们选择了两边最大的两个数, 假设他们加起来取平均大于中位数, 那么对答案有正的影响,提高了平均数的大小, 但是第二次取得两个数加和一定小于等于第一次, 当某一次, 如果加和平均数比中位数小, 那么从此就会不断产生负的影响,拉低了平均数, 所以答案又下降了, 所以是先增加后减小。 那么最大平均数就是先增加后减小的。
#include <set>
#include <map>
#include <stack>
#include <queue>
#include <deque>
#include <cmath>
#include <vector>
#include <string>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
#define L(i) i<<1
#define R(i) i<<1|1
#define INF 0x3f3f3f3f
#define pi acos(-1.0)
#define eps 1e-9
#define maxn 1000100
#define MOD 1000000007
int n,m;
int a[maxn];
long long sum[maxn];
int main()
{
//freopen("in.txt","r",stdin);
//freopen("out.txt","w",stdout);
int t,C = 1;
//scanf("%d",&t);
while(scanf("%d",&n) != EOF)
{
for(int i = 1; i <= n; i++)
scanf("%d",&a[i]);
sort(a+1,a+n+1);
sum[0] = 0;
for(int i = 1; i <= n; i++)
sum[i] = sum[i-1] + a[i];
long long pos = 1,len = 0;
long long ans = 0;
for(int i = 2; i < n; i++)
{
long long l = 1, r = min(i-1, n - i);
long long mid, mmid;
for(int j = 1; j <= 100; j++)
{
mid = (2*l + r) / 3;
mmid = (l + 2*r + 2) / 3;
long long m1 = (sum[i] - sum[i-mid-1] + sum[n] - sum[n-mid]);
long long m2 = (sum[i] - sum[i-mmid-1] + sum[n] - sum[n-mmid]);
if(m2*(mid*2+1) <= m1*(mmid*2+1))
r = mmid - 1;
else
l = mid + 1;
}
double cur = (double)(sum[i] - sum[i-l-1] + sum[n] - sum[n-l]) - (2*l+1)*a[i];
if(ans*(2*l+1) < cur*(2*len+1)) //为了避免精度问题, 转化为整数比较
{
ans = cur;
len = l;
pos = i;
}
}
printf("%lld\n",len*2+1);
for(int i = pos - len; i <= pos; i++)
printf("%d ",a[i]);
for(int i = n-len+1; i <= n; i++)
printf("%d%c",a[i],i==n?'\n':' ');
}
return 0;
}