PHP排列组合算法

本文介绍如何使用PHP实现排列和组合的算法,包括计算排列和组合的数量,并通过递归方式生成具体的排列组合序列。通过实例演示了如何计算从指定数量的数字中选取特定数量数字的所有可能情况。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

排列:从M个数字中选择N个数字,数字可以重复
组合:从M个数字中选择N个数字,数字不能重复

<?php
// 阶乘
function factorial($n)
{
    return array_product(range(1, $n));
}

// 计算排列个数
function A($n, $m)
{
    return factorial($n) / factorial($n - $m);
}

// 计算组合个数
function C($n, $m)
{
    return A($n, $m) / factorial($m);
}

// 排列
function arrangement($a, $m)
{
    $r = array();
    $n = count($a);
    if ($m <= 0 || $m > $n) {
        return $r;
    }
    for ($i = 0; $i < $n; $i++) {
        $b = $a;
        $t = array_splice($b, $i, 1);
        if ($m == 1) {
            $r[] = $t;
        } else {
            $c = arrangement($b, $m - 1);
            foreach ($c as $v) {
                $r[] = array_merge($t, $v);
            }
        }
    }

    return $r;
}

// 组合
function combination($a, $m)
{
    $r = array();
    $n = count($a);
    if ($m <= 0 || $m > $n) {
        return $r;
    }
    for ($i = 0; $i < $n; $i++) {
        $t = array($a[$i]);
        if ($m == 1) {
            $r[] = $t;
        } else {
            $b = array_slice($a, $i + 1);
            $c = combination($b, $m - 1);
            foreach ($c as $v) {
                $r[] = array_merge($t, $v);
            }
        }
    }

    return $r;
}

// ====== 测试 ======
$arr = array(1, 2, 3, 4, 5);//可用数字
$arr_len = count($arr);
$maxlength = 3;    //选择几位

$r = A($arr_len, $maxlength);
echo "排列 P({$arr_len}, {$maxlength})={$r}<br>";

$r = arrangement($arr, $maxlength);
foreach ($r as $v) {
    echo join('', $v).' ';
}
echo '<hr>';
$r = C($arr_len, $maxlength);
echo "组合 C({$arr_len}, {$maxlength})={$r}<br>";

$r = combination($arr, $maxlength);
foreach ($r as $v) {
    echo join('', $v).' ';
}

排列 P(5, 3)=60
123 124 125 132 134 135 142 143 145 152 153 154 213 214 215 231 234 235 241 243 245 251 253 254 312 314 315 321 324 325 341 342 345 351 352 354 412 413 415 421 423 425 431 432 435 451 452 453 512 513 514 521 523 524 531 532 534 541 542 543
组合 C(5, 3)=10
123 124 125 134 135 145 234 235 245 345
 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值