求一算法,各位大侠帮帮忙!

hx_li 2005-01-28 01:25:18
我有1,2,3,4,5,6,7,8,9共9个数,要列出所有和为10的组合
...全文
224 21 打赏 收藏 转发到动态 举报
写回复
用AI写文章
21 条回复
切换为时间正序
请发表友善的回复…
发表回复
EATSAHARA 2005-01-28
  • 打赏
  • 举报
回复
赫赫,恭喜
hx_li 2005-01-28
  • 打赏
  • 举报
回复
在各位大侠的提点下,终于将算法搞掂,现将代码贴出来,与大家分享

using System;
using System.Drawing;
using System.Collections;
using System.ComponentModel;
using System.Windows.Forms;
using System.Data;

namespace NumSum
{
/// <summary>
/// Form1 的摘要说明。
/// </summary>
public class Form1 : System.Windows.Forms.Form
{
/// <summary>
/// 必需的设计器变量。
/// </summary>
private System.ComponentModel.Container components = null;

public Form1()
{
//
// Windows 窗体设计器支持所必需的
//
InitializeComponent();

//
// TODO: 在 InitializeComponent 调用后添加任何构造函数代码
//
}

/// <summary>
/// 清理所有正在使用的资源。
/// </summary>
protected override void Dispose( bool disposing )
{
if( disposing )
{
if (components != null)
{
components.Dispose();
}
}
base.Dispose( disposing );
}

#region Windows 窗体设计器生成的代码
/// <summary>
/// 设计器支持所需的方法 - 不要使用代码编辑器修改
/// 此方法的内容。
/// </summary>
private void InitializeComponent()
{
//
// Form1
//
this.AutoScaleBaseSize = new System.Drawing.Size(6, 14);
this.ClientSize = new System.Drawing.Size(292, 273);
this.Name = "Form1";
this.Text = "Form1";
this.Load += new System.EventHandler(this.Form1_Load);

}
#endregion

/// <summary>
/// 应用程序的主入口点。
/// </summary>
[STAThread]
static void Main()
{
Application.Run(new Form1());
}

private void Form1_Load(object sender, System.EventArgs e)
{
string strSTime=DateTime.Now.ToString();
int[] array=new int[20]{1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20};//原始数组
int[] flag=new int[20]{0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0}; //标志数组,表示哪个数字被选中(1,表示被选中.0,表示未被选中)
int sum=25; //和

string strMsg="";

for(int i=0;i<array.Length-1;i++)
{
for(int j=i+1;j<array.Length;j++)
{
flag=new int[20]{0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0};
flag[i]=1;

plus(ref strMsg,array[i],j,array,flag,sum); //函数返回值

}
}

string strETime=DateTime.Now.ToString();
MessageBox.Show(strSTime+"||"+strETime);
MessageBox.Show(strMsg);

}

/// <summary>
/// 递归算法
/// </summary>
/// <param name="strMsg">输出结果</param>
/// <param name="iCurSum">当前计算的和</param>
/// <param name="iSecIndex">相加数的索引</param>
/// <param name="array">数值数组</param>
/// <param name="flag">标志数组</param>
/// <param name="sum">和</param>
private void plus(ref string strMsg,int iCurSum,int iSecIndex,int[] array,int[] flag,int sum)
{
iCurSum=iCurSum+array[iSecIndex];//当前和
if(iCurSum<sum)
{
flag[iSecIndex]=1;
if(++iSecIndex<array.Length)
{
int k=iSecIndex;
for(;k<array.Length;k++)
{
if(iCurSum+array[k]==sum)
{
flag[k]=1;
strMsg+=plusResult(array,flag); //得到一个成功组合
flag[k]=0;
break;
}
}
if(k!=iSecIndex)
{
plus(ref strMsg,iCurSum,iSecIndex,array,flag,sum); //递归调用
}
}
}
else if(iCurSum==sum)
{
flag[iSecIndex]=1;
strMsg+=plusResult(array,flag); //得到一个成功组合
flag[iSecIndex]=0;

}
else if(iCurSum>sum)
{
flag[iSecIndex]=0;
}
}

/// <summary>
/// 输出一个满足条件的组合
/// </summary>
/// <param name="array"></param>
/// <param name="flag"></param>
/// <returns></returns>
private string plusResult(int[] array,int[] flag)
{
string strMsg="";
for(int m=0;m<flag.Length;m++)
{
if(flag[m]==1)
{
strMsg+=array[m].ToString()+",";
}
}
return strMsg+"||";
}
}
}
sexfreebird 2005-01-28
  • 打赏
  • 举报
回复
高手!!!
帮顶
jackie615 2005-01-28
  • 打赏
  • 举报
回复
up
ylzxg 2005-01-28
  • 打赏
  • 举报
回复
递归拆数
EATSAHARA 2005-01-28
  • 打赏
  • 举报
回复
用二维数组,和求最短路径的floyd算法,一定可以,时间复杂度是o(n^3)
jialiang 2005-01-28
  • 打赏
  • 举报
回复
原理:
1:将这些数字从小到大排序,按顺序累加求出所需的最大位数的组合;
如此例:1+2+3+4>=10 ,即可得到最多是4个数字相加,可以得到10,最少为两个数字相加得到10
2:根据此依次求出2个数累加=10的组合,3个数=10,4个数=10.....

根据以上思路大概写算法如下
1到10个数
for (i=1;i<=n;i++)
2到4个数的和的组合
for (s=2;s<=4;k++)
sum(sums,a[i],2,i,s)
//递归函数:如求2位数时逐个相加即可,3为数时相加后要递归一次,4为数要递归两次,
//即递归次数等于s-2
//sums记录字符串,val已经加过的和,count是加了几位数,
j是加到数组的第几个,kinds是求几位数的和
private void sum(ref string sums,int val,ref int count,int j,int kinds )
{
int i;
if (kinds==count)
{
for (i=j;i<n;i++)
if ((val+a[i])==10)
sums=sums+i.ToString();
}
else
{
for (i=j;i<n;i++)
if (val+a[i]<10)
{
count=count+1;
sum(ref sums,val+a[i],count,i,kinds);
}
}

}
EATSAHARA 2005-01-28
  • 打赏
  • 举报
回复
1.将这些数字从小到大排序;
2.将前两个数字相加,看是否大于10,若小于10,将前三个数字相加,若... 直至大于10(此步骤用以确定求和组合中最大元素数);
3.从2到最大元素数执行类似我上述所说递归
DeltaCat 2005-01-28
  • 打赏
  • 举报
回复
^_^
LoveCherry 2005-01-28
  • 打赏
  • 举报
回复
那就递归
landlordh 2005-01-28
  • 打赏
  • 举报
回复
http://book.78cm.com/article.php?articleid=4086
hx_li 2005-01-28
  • 打赏
  • 举报
回复
能不能给一个通用的算法,假如我有30个数,要找出和为定值的数的组合,且要求组合中的每一个数都不相同?
LoveCherry 2005-01-28
  • 打赏
  • 举报
回复
int [] arr=new int[9]{1,2,3,4,5,6,7,8,9};
foreach(int a in arr)
{
foreach(int b in arr)
{
foreach(int c in arr)
{
foreach(int d in arr)
{
if(a+b+c+d==10&&a!=b&&b!=c&&c!=d)Console.WriteLine(a+"+"+b+"+"+c+"+"+d+"=10");
}
if(a+b+c==10&&a!=b&&b!=c)Console.WriteLine(a+"+"+b+"+"+c+"=10");
}
if(a+b==10&&a!=b)Console.WriteLine(a+"+"+b+"=10");
}
}
hx_li 2005-01-28
  • 打赏
  • 举报
回复
现在的问题值不值200分呢?
hx_li 2005-01-28
  • 打赏
  • 举报
回复
我要的结果是所有的组合,如:1+2+3+4=10,1+2+7=10
suro 2005-01-28
  • 打赏
  • 举报
回复
这个问题值200分......
楼上楼上楼上正解
tongcheng 2005-01-28
  • 打赏
  • 举报
回复
看看.....
hs_tang7788 2005-01-28
  • 打赏
  • 举报
回复
这个问题值200分......
楼上正解
EATSAHARA 2005-01-28
  • 打赏
  • 举报
回复
只要算法是不是?

1.将这些数字从小到大排序;
2.从第一个数字到倒数第二个数字,与从这个数字到数列末尾居中的数字相加,得出结果与10比较,若大于10,则取前部居中的数字与与第一个数字相加,若小于10,取后部居中的数字,递归此过程。
LoveCherry 2005-01-28
  • 打赏
  • 举报
回复
int [] arr=new int[9]{1,2,3,4,5,6,7,8,9};
foreach(int a in arr)
{
foreach(int b in arr)
{
if(a+b==10)Console.WriteLine(a+"+"+b+"=10");
}
}
Console.ReadLine();
加载更多回复(1)

110,538

社区成员

发帖
与我相关
我的任务
社区描述
.NET技术 C#
社区管理员
  • C#
  • Web++
  • by_封爱
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告

让您成为最强悍的C#开发者

试试用AI创作助手写篇文章吧