编程之美2.18 数组分割

来自《编程之美》中的一道题:有一个没有排序、元素个数为2n的正整数数组,要求:把这个数组分割为元素个数为n的两个数组,并使两个数组的和最接近。
---------------------------------------------------------------------------------------------------------------------------------
本题与01背包问题有些类似,只不过最优化条件变成了:“必须挑选出n个数”与"n个数的和要最接近2n个数和的一半"。
比较理想的情况是遍历一次原始数组就能求解出答案,假设2n个数的和为sum,用数学归纳法来描述:
归纳假设:已知如何在m个数中挑选1到min(m,n)个数,并使其和小于等于sum/2;
以此对m+1进行归纳,终止条件为m=2n,然后检查n个数的所有可能的和,并找到最接近sum/2的结果即可,当然还需要用某种方法记录生成此和的n个数。

以m=3为例,我们已知3个数的所有排列组合的和,那么就有3种1个数的组合、3种2个数的组合、1种3个数的组合,我们需要把这些组合及其和保存下来,以方便将问题规模归纳扩展到m+1。

#include <iostream>
#include <vector>
#include <assert.h>
using namespace std;
typedef unsigned int UINT;
vector<UINT> arraySplit(const vector<UINT>& a)
{
    UINT sum =0;
    for(UINT i=0; i < a.size(); ++i){sum += (a[i]);}
    sum /= 2;
    
    size_t cnt = a.size()/2;
    vector<vector<UINT> > ele(sum+1,vector<UINT>());
    vector<vector<vector<UINT> > > table(cnt,ele);
    for (UINT i=0; i < cnt*2; ++i)
    {
        UINT begin = (i < cnt) ? i : cnt-1;
        for(UINT k = begin; k > 0; --k)
        {
            for (UINT j=0; j <= sum; ++j)
            {   
                if (table[k-1][j].size() == 0){continue;}
                UINT val = j+ a[i];
                if (val > sum){continue;}
                if (table[k][val].size() == 0)
                {
                    table[k][val].insert(table[k][val].end(),table[k-1][j].begin(),table[k-1][j].end()); 
                    table[k][val].push_back(a[i]);
                }
            }
        }
        if (table[0][a[i]].size() == 0){table[0][a[i]].push_back(a[i]);}
    }
    
    for(int i=sum; i >=0; --i)
    {
        if (table[cnt-1][i].size() > 0)
        {
            return table[cnt-1][i];
        }
    }
    assert(false && "error");
    return vector<UINT>();
}

void doTestArraySplit()
{
    vector<UINT>a;
    for (int i=0; i < 10; ++i)
    {
        a.push_back(UINT(double(rand())/RAND_MAX*10.0));
    }
    
    vector<UINT> result = arraySplit(a);
    assert(result.size() == a.size()/2 && "invalid cnt");
    for (size_t i=0; i < a.size(); ++i){cout<<a[i]<<",";}
    UINT sum =0;
    for(UINT i=0; i < a.size(); ++i){sum += (a[i]);}
    cout<<"sum="<<sum<<endl;
    for(size_t j=0; j < result.size(); ++j){cout<<result[j]<<",";}

    sum=0;
    for(UINT i=0; i < result.size(); ++i){sum += (result[i]);}
    cout<<"sum="<<sum<<endl;
}

void testArraySplit()
{
    srand(2);
    doTestArraySplit();
	srand(4);
    doTestArraySplit();
    srand(7);
    doTestArraySplit();
}

void main()
{
	testArraySplit();
}


测试结果:

1,5,7,8,9,6,3,11,20,17,sum=87
1,5,8,9,20,sum=43


参考文献: http://poplars.blog.163.com/blog/static/139422174201155762242/
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值