首先抛出一个问题:
$logic = ">=";
if ($a . $logic . $b){
}else{
}
假设 $a 与 $b 之间的逻辑判断不能确定,要怎么才能动态添加判断条件呢?
解决方案1: 使用PHP的 eval() 把字符串转化为代码执行
$logic = ">=";
$str = 'return '.$a.' '.$logic.' '.$b.';';
$result = @eval($str);
if ($result ){
}else{
}
- eval()需要添加return才会返回结果,否则会返回null。结束符分号是必须的。
- 如果只是简单的判断,使用eval()确实会比较方便一些,但是当判断的条件复杂了,可读性就会下降。
- eval()可以执行任何php代码,所以会很危险,甚至有些系统关闭了eval()的使用(需要在php.ini中开启)。
解决方案2: 设定好判断的条件类型,封装每种类型对应的函数。判断时,根据类型调用函数进行判断。
function egt($a, $b){
return $a >= $b;
}
function elt($a, $b){
return $a <= $b;
}
...
$logic = "egt";
$result = cal_user_func($logic, $a, $b);
- 多参数也可使用 call_user_func_array()
实例:店铺会员卡等级规则判断的实现
根据上面得到的结果,我们可以用来实现会员等级规则的判断。(使用方案2实现)
上图是自定义规则的设置页面,其中主要关注的是条件的设置,有满足一种条件和满足全部条件两种选项。 根据方案2,首先确定条件的数据结构(json):
{
"name": "满足其中一个条件",
"value": "or",
"condition": [
{
"variable": "charge",
"variable_name": "会籍期间累计储值",
"range": "egt",
"range_name": "大于或等于",
"value": ["1000"]
},
{
"variable": "score",
"variable_name": "会籍期间累计积分",
"range": "egt",
"range_name": "大于或等于",
"value": ["1000"]
},
{
"variable": "night",
"variable_name": "会籍期间累计房晚",
"range": "elt",
"range_name": "小于或等于",
"value": ["10"]
},
{
"variable": "consume",
"variable_name": "累计消费",
"range": "between",
"range_name": "介于之间",
"value": ["2000","3000"]
}
]
}
封装判断函数
function egt($a, $b){
return $a >= $b;
}
function elt($a, $b){
return $a <= $b;
}
function between($a, $b, $c){
return $a >= $b && $a <= $c;
}
public function verify_rule($rule, $charge, $score, $night, $consume){
$rule = json_decode($rule, true);
$res_arr = [];
$condition_num = count($rule['condition']);
foreach ($rule['condition'] as $val){
$params = $val['value'];
if (!isset(${$val['variable']})) continue;
array_unshift($params, ${$val['variable']});
if (function_exists($val['range'])){
$res_arr[] = call_user_func_array($val['range'], $params);
}
}
$result = array_sum($res_arr);
if ($rule['value'] == 'and' && $result == $condition_num){
return true;
}elseif($rule['value'] == 'or' && $result > 0){
return true;
}else{
return false;
}
}
- 根据条件的数据结构,我们需要传入四个比对值(累计储值、累计积分、累计房晚、累计消费)。注意:数据结构中 condition.variable 与传入的参数名需一致,因为在验证方法中,我们使用了可变变量
- 同时还需要保证存在与condition.range同名的函数。
如果想用方案1实现,这里提醒一下,and和or,与&&和||得到的结果是不一样的,拼接字符串的时候最好还是转换一下
|