shaker.php 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200
  1. <?php
  2. /**
  3. * Created by PhpStorm.
  4. * User: stanley-king
  5. * Date: 16/6/21
  6. * Time: 下午1:07
  7. */
  8. namespace bonus;
  9. use \Exception;
  10. use \Db;
  11. class shaker
  12. {
  13. const max_strength = 5;
  14. const min_amount = 0.01;
  15. const direct_asc = 0;
  16. const direct_dec = 1;
  17. //$binded 只能表示状态是领取还没充值到个人账户中的状态
  18. public function reassign($type_id, $bonus_id, $bonus_status, &$bonus_valin, $strength, $direction)
  19. {
  20. try
  21. {
  22. $bonus_status = intval($bonus_status);
  23. if($bonus_status != 1 && $bonus_status != 2) return false;
  24. $bonus_val = $bonus_valin;
  25. $trans = new trans_wapper(null,__METHOD__);
  26. $mod_bonus = Model('user_bonus');
  27. $min_amount = self::min_amount;
  28. $items = $mod_bonus->get(array('type_id' => $type_id,'bonus_status' => 0,'bonus_value' => array('gt',$min_amount)));
  29. if(!empty($items))
  30. {
  31. $total_num = $this->weight_num($strength,count($items));
  32. $total_amount = 0;
  33. $index = 0;
  34. $selitems = array();
  35. foreach ($items as $bonus)
  36. {
  37. $total_amount += intval($bonus['bonus_value'] * 100 + 0.5);
  38. array_push($selitems,$bonus);
  39. if(++$index >= $total_num) {
  40. break;
  41. }
  42. }
  43. $total_amount = doubleval($total_amount / 100);
  44. $ret = $this->separate($bonus_val,$total_amount,$total_num,$direction,$strength);
  45. if($ret == true)
  46. {
  47. $type = type::create_by_id($type_id);
  48. $type_max_value = $type->max_amount();
  49. $alloc = new allocator();
  50. $moneys = $alloc->separate_money(self::min_amount,$total_amount,$total_num,$min_val,$max_val);
  51. $ret = $mod_bonus->where(array('type_id' => $type_id,'bonus_id' => $bonus_id))->update(array('bonus_value' => $bonus_val));
  52. if($ret == false || $mod_bonus->affected_rows() <= 0) {
  53. throw new Exception();
  54. }
  55. $index = 0;
  56. $max_value = 0.00;
  57. foreach ($selitems as &$item) {
  58. $item['bonus_value'] = $moneys[$index];
  59. $item['remain_amount'] = $moneys[$index];
  60. $max_value = $this->max($max_val,$item['bonus_value']);
  61. ++$index;
  62. }
  63. $max_value = $this->max($type_max_value,$max_value);
  64. $remain_change = $bonus_val - $bonus_valin;
  65. if($bonus_status === 2) {
  66. $data[] = array('max_amount' => $max_value,'remain_amount' => array('exp', "remain_amount-{$remain_change}"));
  67. Model('bonus_type')->edit(array('type_id' => $type_id),$data);
  68. }
  69. else
  70. {
  71. if($this->equal($type_max_value,$max_value) == false) {
  72. Model('bonus_type')->edit(array('type_id' => $type_id),array('max_amount' => $max_value));
  73. }
  74. }
  75. $ret = $mod_bonus->insertAll($selitems,array(),true);
  76. if($ret == false && $mod_bonus->affected_rows() <= 0) {
  77. throw new Exception();
  78. }
  79. }
  80. }
  81. $trans->commit();
  82. $bonus_valin = $bonus_val;
  83. return true;
  84. }
  85. catch (Exception $ex) {
  86. $trans->rollback();
  87. return false;
  88. }
  89. }
  90. private function equal($left,$right) {
  91. $left = intval($left * 100 + 0.5);
  92. $right = intval($right * 100 + 0.5);
  93. return ($left == $right);
  94. }
  95. private function max($left,$right)
  96. {
  97. $left = intval($left * 100 + 0.5);
  98. $right = intval($right * 100 + 0.5);
  99. return $left > $right ? $left / 100 : $right / 100;
  100. }
  101. private function delta_asc($total,$num,$strength)
  102. {
  103. if($strength == 5) {
  104. $mul = 2;
  105. } elseif ($strength == 4) {
  106. $mul = 1.6;
  107. } elseif ($strength == 3) {
  108. $mul = 1.2;
  109. } elseif ($strength == 2) {
  110. $mul = 0.8;
  111. } elseif ($strength == 1) {
  112. $mul = 0.4;
  113. } else {
  114. return 0;
  115. }
  116. $avg = doubleval($total / $num);
  117. $avg = intval($avg * $mul + 0.5);
  118. return mt_rand(0,$avg);
  119. }
  120. private function delta_dec($total,$strength)
  121. {
  122. if($strength == 5) {
  123. $mul = 1;
  124. } elseif ($strength == 4) {
  125. $mul = 0.8;
  126. } elseif ($strength == 3) {
  127. $mul = 0.6;
  128. } elseif ($strength == 2) {
  129. $mul = 0.4;
  130. } elseif ($strength == 1) {
  131. $mul = 0.2;
  132. } else {
  133. return 0;
  134. }
  135. $avg = intval($total * $mul + 0.5);
  136. return mt_rand(0,$avg);
  137. }
  138. private function weight_num($strength,$total_num)
  139. {
  140. if($strength == 0) {
  141. $strength = 1;
  142. }
  143. $weight = intval(doubleval($total_num / self::max_strength) * $strength + 0.5);
  144. return ($weight > $total_num) ? $total_num : $weight;
  145. }
  146. private function separate(&$bonus_out,&$total_amount,$total_num,$direction,$strength)
  147. {
  148. $total_tmp = intval($total_amount * 100 + 0.5);
  149. $bonus_val = intval($bonus_out * 100 + 0.5);
  150. $min_amount = 1;
  151. $base_amount = $min_amount * $total_num;
  152. if($base_amount == $total_tmp) {
  153. return false;
  154. }
  155. if($direction == self::direct_asc) { //金额变大
  156. $delta = $this->delta_asc($total_tmp - $base_amount,$total_num + 1, $strength);
  157. $bonus_val += $delta;
  158. $total_tmp = $total_tmp - $delta;
  159. } else { //金额变小
  160. $delta = $this->delta_dec($bonus_val - self::min_amount, $strength);
  161. $bonus_val = $bonus_val - $delta + self::min_amount;
  162. $total_tmp = $total_tmp + $delta;
  163. }
  164. $total_amount = $total_tmp / 100;
  165. $bonus_out = $bonus_val / 100;
  166. return true;
  167. }
  168. }