php排列組合 1到9數字相加都等於20

<?php
set_time_limit(0);
/*
函數說明:huoqu_zhuhe($eq,$jiashu,$isone=0)
參數說明:$eq---幾個數相加的總和;
  $jiashu-------加數數組:$jiashu=array(1,2,3,4,5,6,7,8,9),可以使用的加數;
  $isone---是否要每次使用不同的加數,唯一性,1是 0 不,默認1

返回類型:數組,數字以+相連的字符串:[0] => 3+8+9 [1] => 4+7+9

測試效果:1:對於加數數組比較小的,速度可以,過大的話,有些慢;2:每次可以使用不同的加數的,處理會變慢
採用的方法是:生成所有可能排列,對排列處理過濾重複的,得到組合
*/

function huoqu_zhuhe($eq,$jiashu,$isone=1)
{if(empty($jiashu)||!is_array($jiashu)){echo 'error:加數必須數組';return false;}
$feishu=0;
for($i=0;$i<count($jiashu);$i++){
if(!is_numeric($jiashu[$i])){$feishu=1;break;}
}


if($feishu==1){echo 'error;數組中必須是合法的數字';return false;}
$lian=$jiashu;
$savearr=array();
while(!empty($lian)){
//echo 1;
$newarr=array();
$k=0;
for($i=0;$i<count($lian);$i++){

$lianstr=$lian[$i];
$arr=explode('+',$lianstr);
$nowhe=array_sum($arr);
//echo $nowhe;
for($j=0;$j<count($jiashu);$j++){

$savestr=$lianstr.'+'.$jiashu[$j];
if($isone==1&&in_array($jiashu[$j],$arr))continue;
if(($nowhe+$jiashu[$j])>$eq)break;
else if(($nowhe+$jiashu[$j])==$eq){
$savearr[]=$savestr;

}
else{$newarr[$k]=$savestr;$k++;}

}//end for($j=0;$j<count($jiashu)

}// end for($i=0;$i

$lian=$newarr;

}//end while(!empty($lian))

 

//print_r($savearr);


//生成組合部分,過濾重複,2個數組以一個爲參考,看另一個是否能通過移動達到匹配,可以,過濾

$isguolu=array();//存儲對應的id的取捨 0取 1舍
for($i=0;$i<count($savearr);$i++){
$isguolu[]=0;

}//初始化全部0


for($i=0;$i<count($savearr);$i++){

$arr1=explode('+',$savearr[$i]);
$len1=count($arr1);
for($j=$i+1;$j<count($savearr);$j++){
$arr2=explode('+',$savearr[$j]);
$len2=count($arr2);
if($len1!=$len2)continue;
if($isguolu[$j]==1)continue;
//比較$arr1和$arr2開始
$jishu=0;
for($i1=0;$i1<count($arr1);$i1++){

$a=$arr1[$i1];
$isyou=0;

for($i2=$i1;$i2<count($arr2);$i2++){
if($a==$arr2[$i2]){
$jishu++;
$isyou=1;
$t=$arr2[$i1];
$arr2[$i1]=$arr2[$i2];
$arr2[$i2]=$t;
break;
}

}//end for($i2=0

if($isyou==0)break;

}// end for($i1=0;$i1<count($arr1);

if($jishu==$len1)$isguolu[$j]=1;

}//end for($j=$i+1;

}//end for($i=0;$i<count($savearr);$i++)

//print_r($isguolu);


//根據過濾數組選擇
$newarr=array();
for($i=0;$i<count($savearr);$i++){
if($isguolu[$i]==0)$newarr[]=$savearr[$i];
}


//print_r($newarr);

return $newarr;
}

//下面是一個測試
//取用1,2,3,4,5,6,7,8,9相加所有等於20的組合
$jiashu=array(1,2,3,4,5,6,7,8,9);

$eq=20;
if($jieguo=huoqu_zhuhe($eq,$jiashu,1))print_r($jieguo);

 

?>

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章