分享實作PHP紅包演算法的想法(附開發程式碼)

藏色散人
發布: 2023-04-10 16:28:01
轉載
3775 人瀏覽過
根据很多需求的使用场景,如发红包、砍价类需求,这两个功能都有一个同样的特点,如下:
登入後複製

紅包

#1.總金額
2.紅包數
3.最小紅包數量

砍價

1. 砍價總金額
2.要多少人完成砍價(人數依需求而定)

  • 固定砍價人數
  • 隨機砍價人數
  • 指定隨機砍價人數
    第2點三個規則都需要依照規則得出一個人數

3.最小砍價金額

開發想法

#驗證參數

最小金額不允許小於0
總金額不允許大於數量乘最小金額

分配金額

取得平均金額(總金額/剩餘數量)

分配金額 平均金額小於等於最金額時直接分配最小金額
取得金額幅度比例 最小值不允許小於-1 最大值不允許大於1##得出分配金額 幅度計算(平均值*(1
幅度比例)) 分配金額判斷
分配金額小於最小金額分配金額大於可領取最大金額 ((最小金額剩餘總金額)-  (剩餘數量×最小金額))時重新分配金額剩餘最後一個則剩餘所有金額都分配

開發代碼

<?php

/**
 * 发送红包
 * Class sandRed
 */
class sandRed
{
    #红包金额
    protected $amount;
    #红包个数
    protected $num;
    #领取的红包最小金额
    protected $minAmount;
    #红包分配结果
    protected $amountArr = [];

    public function __construct($amount, $num = 1, $minAmount = 1)
    {
        $this->amount = $amount;
        $this->num = $num;
        $this->minAmount = $minAmount;
    }

    /**
     * 处理返回
     * @return array
     * @throws Exception
     */
    public function handle()
    {
        # 验证
        if ($this->amount < $validAmount = $this->minAmount * $this->num) {
            throw new Exception('红包总金额必须≥'.$validAmount.'元');
        }
        # 分配红包
        $this->allot();
        return $this->amountArr;
    }

    /**
     * 分配红包
     */
    protected function allot()
    {
        # 剩余可分配的红包个数
        $num = $this->num;

        # 剩余可领取的红包金额
        $amount = $this->amount;
        while ($num >= 1) {
            if ($num == 1) {
                # 剩余一个的时候,直接取剩余红包
                $coupon_amount = $this->formattingAmount($amount);
            } else {

                # 平均金额
                $avgAmount = $this->formattingAmount($amount / $num);

                # 分配金额
                $countAllotAmount = $this->countAllotAmount($avgAmount, $amount, $num);

                # 剩余的红包的平均金额
                $coupon_amount = $this->formattingAmount($countAllotAmount);
            }
            # 追加分配金额
            $this->amountArr[] = $coupon_amount;

            # 计算剩余金额
            $amount -= $coupon_amount;

            $num--;
        }
        # 随机打乱
        // shuffle($this->amountArr);
    }

    /**
     * 计算分配的红包金额
     * @param float $avgAmount 每次计算的平均金额
     * @param float $amount 剩余可领取金额
     * @param int $num 剩余可领取的红包个数
     * @return float
     */
    protected function countAllotAmount($avgAmount, $amount, $num)
    {
        # 如果平均金额小于等于最低金额,则直接返回最低金额
        if ($avgAmount <= $this->minAmount) {
            return $this->minAmount;
        }
        # 浮动比率
        $floatingRate = $this->floatingRate();

        # 分配金额
        $allotAmount = $avgAmount * (1 + $floatingRate);

        # 浮动计算
        $coupon_amount = $this->formattingAmount($allotAmount);

        # 如果低于最低金额或超过可领取的最大金额,则重新获取
        if ($coupon_amount < $this->minAmount || $coupon_amount > $this->canReceiveMaxAmount($amount, $num)) {
            return $this->countAllotAmount($avgAmount, $amount, $num);
        }
        return $coupon_amount;
    }

    /**
     * 计算分配的红包金额-可领取的最大金额
     * @param $amount
     * @param $num
     * @return float|int
     */
    protected function canReceiveMaxAmount($amount, $num)
    {
        return $this->minAmount + $amount - $num * $this->minAmount;
    }

    /**
     * 红包金额浮动比例
     * @return float|int
     */
    protected function floatingRate()
    {
        # 60%机率获取剩余平均值的大幅度红包(可能正数、可能负数)
        if (rand(1, 100) <= 60) {
            # 上下幅度70%
            return rand(-70, 70) / 100;
        }
        # 其他情况,上下浮动30%;
        return rand(-30, 30) / 100;
    }

    /**
     * 格式化金额,保留2位
     * @param $amount
     * @return string
     */
    protected function formattingAmount($amount)
    {
        return sprintf(&#39;%01.2f&#39;, round($amount, 2));
    }
}
登入後複製

    總金額
  • $amount = 1;
    登入後複製
    分配數量
  • $num = 10;
    登入後複製
    #最小金額
  • $minAmount = 0.01;
    
    $red = new sandRed($amount, $num, $minAmount);
    
    $res = $red->handle();
    print_r($res);
    登入後複製
      輸出結果 [0.10,0.04,0.08,0.04,0.16,0.14,0.11,0.13,0.11,0.09]
    echo array_sum($res);
    登入後複製
    # #輸出結果1
    推薦學習:《
PHP影片教學

以上是分享實作PHP紅包演算法的想法(附開發程式碼)的詳細內容。更多資訊請關注PHP中文網其他相關文章!

相關標籤:
php
來源:segmentfault.com
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
熱門教學
更多>
最新下載
更多>
網站特效
網站源碼
網站素材
前端模板
關於我們 免責聲明 Sitemap
PHP中文網:公益線上PHP培訓,幫助PHP學習者快速成長!