PHP 实现微信红包拆分算法
编辑时间:2020-04-21 17:32:54 作者:666666
源自 https://github.com/flc1125

/**

 * 红包分配算法

 *

 * example

 *      $coupon = new Coupon(200, 5);

 *      $res = $coupon->handle();

 *      print_r($res);

 *

 * @author Flc <2018-04-06 20:09:53>

 * @see http://flc.ren | http://flc.io | https://github.com/flc1125

 */

class Coupon

{

    /**

     * 红包金额

     *

     * @var float

     */

    protected $amount;



    /**

     * 红包个数

     *

     * @var int

     */

    protected $num;



    /**

     * 领取的红包最小金额

     *

     * @var float

     */

    protected $coupon_min;



    /**

     * 红包分配结果

     *

     * @var array

     */

    protected $items = [];



    /**

     * 初始化

     *

     * @param float $amount     红包金额(单位:元)最多保留2位小数

     * @param int   $num        红包个数

     * @param float $coupon_min 每个至少领取的红包金额

     */

    public function __construct($amount, $num = 1, $coupon_min = 0.01)

    {

        $this->amount = $amount;

        $this->num = $num;

        $this->coupon_min = $coupon_min;

    }



    /**

     * 处理返回

     *

     * @return array

     */

    public function handle()

    {

        // A. 验证

        if ($this->amount < $validAmount = $this->coupon_min * $this->num) {

            throw new Exception('红包总金额必须≥'.$validAmount.'元');

        }



        // B. 分配红包

        $this->apportion();



        return [

            'items' => $this->items,

        ];

    }



    /**

     * 分配红包

     */

    protected function apportion()

    {

        $num = $this->num;  // 剩余可分配的红包个数

        $amount = $this->amount;  //剩余可领取的红包金额



        while ($num >= 1) {

            // 剩余一个的时候,直接取剩余红包

            if ($num == 1) {

                $coupon_amount = $this->decimal_number($amount);

            } else {

                $avg_amount = $this->decimal_number($amount / $num);  // 剩余的红包的平均金额



                $coupon_amount = $this->decimal_number(

                    $this->calcCouponAmount($avg_amount, $amount, $num)

                );

            }



            $this->items[] = $coupon_amount; // 追加分配



            $amount -= $coupon_amount;

            --$num;

        }



        shuffle($this->items);  //随机打乱

    }



    /**

     * 计算分配的红包金额

     *

     * @param float $avg_amount 每次计算的平均金额

     * @param float $amount     剩余可领取金额

     * @param int   $num        剩余可领取的红包个数

     *

     * @return float

     */

    protected function calcCouponAmount($avg_amount, $amount, $num)

    {

        // 如果平均金额小于等于最低金额,则直接返回最低金额

        if ($avg_amount <= $this->coupon_min) {

            return $this->coupon_min;

        }



        // 浮动计算

        $coupon_amount = $this->decimal_number($avg_amount * (1 + $this->apportionRandRatio()));



        // 如果低于最低金额或超过可领取的最大金额,则重新获取

        if ($coupon_amount < $this->coupon_min

            || $coupon_amount > $this->calcCouponAmountMax($amount, $num)

        ) {

            return $this->calcCouponAmount($avg_amount, $amount, $num);

        }



        return $coupon_amount;

    }



    /**

     * 计算分配的红包金额-可领取的最大金额

     *

     * @param float $amount

     * @param int   $num

     */

    protected function calcCouponAmountMax($amount, $num)

    {

        return $this->coupon_min + $amount - $num * $this->coupon_min;

    }



    /**

     * 红包金额浮动比例

     */

    protected function apportionRandRatio()

    {

        // 60%机率获取剩余平均值的大幅度红包(可能正数、可能负数)

        if (rand(1, 100) <= 60) {

            return rand(-70, 70) / 100; // 上下幅度70%

        }



        return rand(-30, 30) / 100; // 其他情况,上下浮动30%;

    }



    /**

     * 格式化金额,保留2位

     *

     * @param float $amount

     *

     * @return float

     */

    protected function decimal_number($amount)

    {

        return sprintf('%01.2f', round($amount, 2));

    }

}



// 例子

$coupon = new Coupon(200, 5, 30);

$res = $coupon->handle();

print_r($res);


 点赞 0
 收藏 0
 分享
来说两句吧
最新评论
    暂无评论
天气预报
万年历
2015年
7月
返回今天

博客声明

本博客属个人所有,不涉及商业目的。遵守中华人民共和国法律法规、中华民族基本道德和基本网络道德规范,尊重有节制的言论自由和意识形态自由,反对激进、破坏、低俗、广告、投机等不负责任的言行。所有转载的文撰写页面章、图片仅用于说明性目的,被要求或认为适当时,将标注署名与来源。避免转载有明确“不予转载”声明的作品。若不愿某一作品被转用,请及时通知本人。对于无版权或自由版权作品,本博客有权进行修改和传播,一旦涉及实质性修改,本博客将对修改后的作品享有相当的版权。二次转载者请再次确认原作者所给予的权力范围。

本博客所有原创作品,包括文字、资料、图片、网页格式,转载时请标注作者与来源。非经允许,不得用于赢利目的。本博客受中国知识产权、互联网法规和知识共享条例保护和保障,任何人不得进行旨在破坏或牟取私利的行为。本博客声明以简体中文版为准,不对其他语言版本负责。

如有侵权请及时联系我进行处理。邮箱youseeim666@163.com