51NOD 1098 最小方差

1098 最小方差
基准时间限制:1 秒 空间限制:131072 KB 分值: 20 难度:3级算法题

若x1,x2,x3......xn的平均数为k。
则方差s^2 = 1/n * [(x1-k)^2+(x2-k)^2+.......+(xn-k)^2] 。
方差即偏离平方的均值,称为标准差或均方差,方差描述波动程度。
给出M个数,从中找出N个数,使这N个数方差最小。
Input1行:2个数M,N,(M > N, M <= 10000)
第2 - M + 1行:M个数的具体值(0 <= Xi <= 10000)

Output

输出最小方差 * N的整数部分。

Input示例

5 3
1
2
3
4
5

Output示例

2

排个序 显然最小方差肯定是排序后连续的n个数
直接穷举 注意会爆掉int

#include<iostream>
#include<stdlib.h>
#include<stdio.h>
#include<string>
#include<vector>
#include<deque>
#include<queue>
#include<algorithm>
#include<set>
#include<map>
#include<stack>
#include<time.h>
#include<math.h>
#include<list>
#include<cstring>
#include<fstream>
//#include<memory.h>
using namespace std;
#define ll long long
#define ull unsigned long long
#define pii pair<int,int>
#define INF 1000000007
#define pll pair<ll,ll>
#define pid pair<int,double>

const int M=10000+3;
int elem[M];

int main()
{
    //freopen("/home/lu/文档/r.txt","r",stdin);
    //freopen("/home/lu/文档/w.txt","w",stdout);
    int m,n;
    while(~scanf("%d%d",&m,&n)){
        for(int i=0;i<m;++i)
            scanf("%d",elem+i);
        sort(elem,elem+m);
        ll sum=0;
        for(int i=0;i<n;++i)
            sum+=elem[i];
        int st=0,end=n;
        ll ans=1ll<<62;
        while(true){
            double k=(double)sum/n;
            double tmp=0;
            for(int i=st;i<end;++i)
                tmp+=(elem[i]-k)*(elem[i]-k);
            ans=min(ans,(ll)tmp);
            if(end==m)
                break;
            sum+=elem[end]-elem[st];
            ++st,++end;
        }
        printf("%lld\n",ans);
    }
    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值