标题:一列数分两组,使其和的差最小
只看楼主
diaoxue
Rank: 1
等 级:新手上路
帖 子:142
专家分:0
注 册:2007-6-1
 问题点数:0 回复次数:8 
一列数分两组,使其和的差最小
每次都问大家问题
这次贴个我调试好的 呵呵
#include <iostream.h>
#include <math.h>

int t=0;
int minus=1000;
const int n=5;
int a[n]={15,14,11,10,8};
int x[n]={0};
void Disp()
{
    int sleft=0,sright=0;
    for(int i=0;i<n;i++)
    {
        if(x[i]==0)
            sleft+=a[i];
        else if(x[i]==1)
            sright+=a[i];
    }
        int temp=abs(sleft-sright);
        if(temp<minus)
        {
            minus=temp;
            cout<<"The smallest [color=#ff0000]minus is:"[/color];
            cout<<minus<<endl;
            for(int i=0;i<n;i++)
                cout<<x[i]<<"\t";
            cout<<endl;
        }
}
void BackT(int t)
{
    if(t>=n)
        Disp();
    else
        for(int j=0;j<2;j++)
        {
            x[t]=j;
            BackT(t+1);
        }
}
int main(int argc,int *argv)
{
    BackT(t);
    return 0;
}
感觉每次都要计算,没加约束条件
以后再优化吧
高手也可以帮忙优化下啊,谢了
2007-12-04 17:19
nuciewth
Rank: 14Rank: 14Rank: 14Rank: 14
来 自:我爱龙龙
等 级:贵宾
威 望:104
帖 子:9786
专家分:208
注 册:2006-5-23
得分:0 
背包问题:使得得到的一组解的和最接近整个数组元素和的一半.

倚天照海花无数,流水高山心自知。
2007-12-04 19:48
nuciewth
Rank: 14Rank: 14Rank: 14Rank: 14
来 自:我爱龙龙
等 级:贵宾
威 望:104
帖 子:9786
专家分:208
注 册:2006-5-23
得分:0 
其实我还可以这样做.
算出一半后直接按贪心就可以得到结果.

倚天照海花无数,流水高山心自知。
2007-12-04 19:50
nuciewth
Rank: 14Rank: 14Rank: 14Rank: 14
来 自:我爱龙龙
等 级:贵宾
威 望:104
帖 子:9786
专家分:208
注 册:2006-5-23
得分:0 
//做完排序后

void Solve(int *data,int n,int sum)//数组,元素个数,总和
{
    int i,num=0;//当前选择的元素值和
    for(i=0;i<n;i++)
    {
        if(num+data[i]<=sum/2)
        {
            num+=data[i];
            x[i]=1;//解向量
        }
    }
}

倚天照海花无数,流水高山心自知。
2007-12-04 20:09
nuciewth
Rank: 14Rank: 14Rank: 14Rank: 14
来 自:我爱龙龙
等 级:贵宾
威 望:104
帖 子:9786
专家分:208
注 册:2006-5-23
得分:0 
程序代码:
#include <stdlib.h>
#include<stdio.h>
#include<time.h>

int x[100]={0};

void Sort(int data[],int n)
{
    int i,j;
    for(i=0;i<n;i++)
    {
        for(j=0;j<n;j++)
        {
            if(data[i]>data[j])
            {
                int temp=data[i];
                data[i]=data[j],data[j]=temp;
            }
        }
    }
    
}
void Solve(int *data,int n,int sum)
{
    int i,num=0;
    for(i=0;i<n;i++)
    {
        if(num+data[i]<=sum/2)
        {
            num+=data[i];
            x[i]=1;
        }
    }
}
int main()
{
    int n;
    int data[100];
    int sum=0;
    srand(time(NULL));
    printf("输入待排序的个数:");
    scanf("%d",&n);
    for(int i = 0 ;i<n;i++)
    {
        data[i]=rand()%100;
       // scanf("%d",&data[i]);
        printf("%-5d",data[i]);
        sum+=data[i];
    }
    Sort(data,n);
    Solve(data,n,sum);
    //printf("\n\n%d\n\n",sum);
    //sum=0;
    for(int i=0;i<n;i++)
    {
        if(x[i]==1)
        {
            printf("%-5d",data[i]);
           // sum+=data[i];

        }
    }
    //printf("\n\n%d\n\n",sum);
    return 0;
}

倚天照海花无数,流水高山心自知。
2007-12-04 20:11
ranchgirl
Rank: 1
等 级:新手上路
帖 子:11
专家分:0
注 册:2007-12-8
得分:0 
nuciewth's answer is wrong!!!

diaoxue's answer is correct, but exponential in time complexity.


If
nuciewth doesn't believe me, try this data set:
993 588 783 81 374 427 946 697 552 755 831 707

The diff should be 0, but using your algorithm will not get zero, no way!
[color=Black]=====================

[/color][color=Black]I have not found an algorithm not exponential in time complexity yet.
[/color]
Sorry!

[[italic] 本帖最后由 ranchgirl 于 2007-12-8 10:17 编辑 [/italic]]
2007-12-08 10:04
diaoxue
Rank: 1
等 级:新手上路
帖 子:142
专家分:0
注 册:2007-6-1
得分:0 
[url]http://www.[/url]
大家有时间看看啊
谢谢LS

上善若水,水善利万物而不争,处众人之所恶
2007-12-08 15:06
nuciewth
Rank: 14Rank: 14Rank: 14Rank: 14
来 自:我爱龙龙
等 级:贵宾
威 望:104
帖 子:9786
专家分:208
注 册:2006-5-23
得分:0 
The question is that how to get the least  balance by delimiting the list  so that  get two sum.
my answer is wrong,used the greed algorithm indeed here is not advisability.
i'm sorry!
thanks ,ranchgirl .

倚天照海花无数,流水高山心自知。
2007-12-08 15:14
ZaakDov
Rank: 2
等 级:论坛游民
帖 子:7
专家分:26
注 册:2011-6-10
得分:0 
n个数x[0..n-1],假设所有数绝对值的和为S
void resu(int x[],int n){
for(S=0,i=0;i<n;i++) S+=(x[i]<0?-x[i]:x[i]);
for(i=0;i<=S*2;i++) can[0][i]=0;
can[0][S]=1;
for(i=0;i<n;i++){
    for(j=0;j<=S*2;j++) can[(i^1)&1][j]=0;
    for(j=0;j<=S*2;j++) if(can[i&1][j]) can[(i^1)&1][j-x[i]]=1,can[(i^1)&1][j+x[i]]=1;
}
for(i=0;i<=S;i++){
    if(can[n&1][S-i]||can[n&1][S+i]) break;
}
return i;
}
2011-06-10 01:44



参与讨论请移步原网站贴子:https://bbs.bccn.net/thread-190362-1-1.html




关于我们 | 广告合作 | 编程中国 | 清除Cookies | TOP | 手机版

编程中国 版权所有,并保留所有权利。
Powered by Discuz, Processed in 0.839282 second(s), 8 queries.
Copyright©2004-2025, BCCN.NET, All Rights Reserved