php 全组合算法,PHP算法可从单个集合生成特定大小的所有组合

我试图推导一种算法,该算法生成特定大小的所有可能组合,例如一个函数,该函数接受一个chars和size数组作为其参数,并返回一个组合数组。

例:

假设我们有一组字符:

设置A = {A,B,C}

a)大小2的所有可能组合:(3 ^ 2 = 9)

AA, AB, AC

BA, BB, BC

CA, CB, CC

b)大小3的所有可能组合:(3 ^ 3 = 27)

AAA, AAB, AAC,

ABA, ABB, ACC,

CAA, BAA, BAC,

.... ad so on total combinations = 27

请注意,线对的大小可以大于对等的总大小。 例如 如果set包含3个字符,则我们还可以创建大小为4的组合。

编辑:

另请注意,这与置换不同。 在置换中,我们不能有重复的字符,例如,如果使用置换算法,AA就不会出现。 在统计中,这称为抽样。

我会使用递归函数。这是一个带有注释的(有效)示例。希望这对您有用!

function sampling($chars, $size, $combinations = array()) {

# if it's the first iteration, the first set    # of combinations is the same as the set of characters    if (empty($combinations)) {

$combinations = $chars;

}

# we're done if we're at size 1    if ($size == 1) {

return $combinations;

}

# initialise array to put new values in    $new_combinations = array();

# loop through existing combinations and character set to create strings    foreach ($combinations as $combination) {

foreach ($chars as $char) {

$new_combinations[] = $combination . $char;

}

}

# call same function again for the next iteration    return sampling($chars, $size - 1, $new_combinations);

}

// example

$chars = array('a', 'b', 'c');

$output = sampling($chars, 2);

var_dump($output);

/*

array(9) {

[0]=>

string(2)"aa"

[1]=>

string(2)"ab"

[2]=>

string(2)"ac"

[3]=>

string(2)"ba"

[4]=>

string(2)"bb"

[5]=>

string(2)"bc"

[6]=>

string(2)"ca"

[7]=>

string(2)"cb"

[8]=>

string(2)"cc"

}

*/

除了使用double foreach之外,您还可以编写自己的笛卡尔乘积函数,但是对于本示例来说,这似乎有些过头了。

这不是迭代函数。它是递归的,因为它显然一直在呼唤自己。

对。偶然写错了字。

对于不希望每个组合中的字符存在一次以上的用户,请将最后一个foreach循环更改为:if(strpos($ combination,$ char)=== false){$ new_combinations [] = $ combination。 $ char;}

可以在新版本的PHP 7.2中执行此功能,还是没有消息优化新版本中的当前功能?@JoelHinz?

@AndreasHunter我看不出为什么该代码无法在PHP> = 7.2中工作,尽管我确定可以对其进行优化-这只是作为其工作方式的一个示例,而不是作为优化。

@JoelHinz我不是在告诉您您的代码不适用于该语言的较新版本。您的代码可在新版本和旧版本的PHP上运行,我亲自对其进行了测试。我只是想问一问,是否可以通过使用新的语言功能来优化生成时间。谢谢您的回复。

不要以为任何人都有很大的字符串长度来处理这个问题(就内存而言)?确切来说,长度为20的4个可能字符。我只需要运行一次并存储输出。

您可以递归执行此操作。请注意,按照您的定义,长度n+1的"组合"可以通过采用长度n的每种组合并在集合中附加一个字母来从长度n的组合中生成。如果您愿意,可以通过数学归纳法证明这一点。

因此,例如对于一组{A,B,C},长度1的组合为:

A, B, C

因此,长度2的组合为

(A, B, C) + A = AA, BA, CA

(A, B, C) + B = AB, BB, BC

(A, B, C) + C = AC, CB, CC

这将是代码,并在ideone上

function comb ($n, $elems) {

if ($n > 0) {

$tmp_set = array();

$res = comb($n-1, $elems);

foreach ($res as $ce) {

foreach ($elems as $e) {

array_push($tmp_set, $ce . $e);

}

}

return $tmp_set;

}

else {

return array('');

}

}

$elems = array('A','B','C');

$v = comb(4, $elems);

是的,这是正确的,但是如何在算法中将其推广以创建n个大小的组合

@ asim-ishaq这是由于我描述的此属性对于所有n都成立。病态编辑。

@ asim-ishaq已更新代码。

可能的算法是:

$array_elems_to_combine = array('A', 'B', 'C');

$size = 4;

$current_set = array('');

for ($i = 0; $i < $size; $i++) {

$tmp_set = array();

foreach ($current_set as $curr_elem) {

foreach ($array_elems_to_combine as $new_elem) {

$tmp_set[] = $curr_elem . $new_elem;

}

}

$current_set = $tmp_set;

}

return $current_set;

基本上,您将要做的是获取当前集合的每个元素,然后追加元素数组的所有元素。

第一步:结果为('a', 'b', 'c'),秒数步骤之后为:('aa', 'ab', 'ac', 'ba', 'bb', 'bc', 'ca', 'cb', 'cc'),依此类推。

我正在尝试测试。什么是$ arra_of_elem,并且在第二和第三循环中使用foreach代替for

@ asim-ishaq这是您要组合元素的数组或集合。在您的情况下:Array(A,B,C)

运作不正常。对于任何给定的大小,它将生成大小为3的组合

@ asim-ishaq我刚刚在writecodeonline.com/php中测试了上面的代码,更改了print_r的返回值,它对于4个元素的组合非常有效

使用数字基数转换的另一个想法

$items = ['a', 'b', 'c', 'd'];

$length = 3;

$numberOfSequences = pow(count($items), $length);

for ($i = 0; $i < $numberOfSequences; $i++) {

$results[] = array_map(function ($key) use ($items) {

return $items[base_convert($key, count($items), 10)];

}, str_split(str_pad(base_convert($i, 10, count($items), $length, 0, STR_PAD_LEFT)));

}

return $results;

警告,items数组中的元素不应超过base_convert参数可以处理的元素:该数字为36

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值