thinkPHP为什么显示不出验证码
本文实例讲述了thinkPHP显示不出验证码的原因与解决方法。分享给大家供大家参考,具体如下:
今天到公司,svn update代码后,在浏览器上输入域名后,在验证码那块显示不出,找了半个上午,后来仔细看了下apache的配置文件
代码如下 | 复制代码 |
ServerName admin.exam.com DocumentRoot E:/www/exam/trunk/server/Admin/ <DirectoryE:/www/exam/trunk/server/apps/Admin/> Options FollowSymLinks AllowOverride all Order deny,allow allow from all Satisfy all
|
原来是DocumentRoot和Directory不一致
代码如下 | 复制代码 |
ServerName admin.exam.com DocumentRoot E:/www/exam/trunk/server/Admin/ Options FollowSymLinks AllowOverride all Order deny,allow allow from all Satisfy all
|
应该是再添加后台的时候,配置文件时复制里一个模块的,粗心了。
另外:
今天一同事又遇到类似的问题,我以为用上面的方法就可以解决呢,可是试了下不行,本着求真务实的态度我就去瞅了下生成验证码的源文件,终于发现了,原来是那个crystal.ttf字体文件不显示,难怪就只显示一条线,把字体文件移到源文件所在的目录就ok了。
这篇文章主要介绍了php分页查询mysql结果的base64处理方法,涉及php+mysql数据库查询及base64编码转换相关操作技巧,需要的朋友可以参考下本文实例讲述了php分页查询mysql结果的base64处理方法。分享给大家供大家参考,具体如下:
代码如下 | 复制代码 |
publicfunctionpublic_about(){ $sql="SELECTc.catid,c.catname,c.catdir,FROM_BASE64(p.content) FROM v9_page p JOINv9_category c ON c.catid=p.catid WHERE c.parentdir = 'jmwm/'"; if(isset($_REQUEST['biaoshi'])){ $sql="SELECTc.catid,c.catname,c.catdir,p.content FROM v9_page p JOIN v9_category c ONc.catid=p.catid WHERE c.catdir LIKE '{$_REQUEST['biaoshi']}%'"; } $mydb= pc_base::load_model('mymodel'); $mpages=$mydb->query_listinfo($sql); $this->array_to_base64($mpages); // $article['content']=base64_encode(toUtf8($article['content'])); $msg=array(); $msg['returncode']="0"; $msg['msg']= togbk("关于联系"); $msg['comment']=$mpages; echojsonFormat($msg); // return_Msg(0,"关于联系",$mpages); } publicfunctionarray_to_base64(&$array){ foreach($arrayas$key=>$value) { if(is_array($value)) { $this->array_to_base64($array[$key]); }elseif($key=="content"){ $array[$key]=base64_encode(toUtf8($value)); }else{ $array[$key]=togbk($value); } } } |
下面小编就为大家带来一篇PHP面向对象的解释器模式。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
最近在看 “深入PHP面向对象模式与实践” ,学习书中的内容后瞬间觉得自己有点高大上了,哈 ! 其实还是个菜。相信也会有新手朋友在看这本(我自己也是新手),对书中我个人认为比较难的内容的学习心得就想拿出来分享和交流,1是希望对自己所学知识能够起到巩固和加深理解的作用 2是希望对看到本文且感兴趣的新手朋友一些帮助。
这部分内容看了好几遍了代码也跟着敲了几遍,估计本文想要实现的功能大概就是用户在web页面上输入一些内容,然后通过后台程序解析后进行回复(感觉就是在废话)。例如我在前台web页面输入框里输入:
$input="4";
$inputequals"4"or$inputequals"four";
然后提交,系统就会回复类似 “条件成立” 或者 “条件不成立”的结果(有点类似直接在前台写代码并运行,后台解析后会返回一个结果。原书中虽然没有讲解整个前台输入到后台解析的过程但我猜这个后台解析应该还有一个使用正则表达式提取类似上面2行代码中关键字的过程)
上面这二行代码虽然是作者发明的语言,但根据字面含义也不难理解,第一行是定义一个变量并赋值,第二行是对变量进行一个判断(变量等于4或者等于four)。
废话不多说来看看这个模式定义的这几个类 (类图请自行看原文):
一、interpreterContext 这个类就像一个容器主要是用来存放和获取需要进行比较的值和比较的结果的,例如上述代码中的4, four,和比较结果 “true”或“false”,保存的形式是数组即类的属性$expressionstore,代码如下:
classInterpreterContext{
private$expressionstore=array();//存放比较的值和结果
functionreplace(Expression$exp,$value){ // 设置值
$this->expressionstore[$exp->getKey()] =$value;
}
functionlookup(Expression$exp){ //获取值
return$this->expressionstore[$exp->getKey()];
}
}
这个类就像一个工具,供其他类来使用(它和其他类不存在继承、组合或聚合的关系)。
二、Expression 这是一个表达式的抽象类,定义了抽象方法interpret() 和方法getKey()
代码如下:
abstractclassExpression {
privatestatic$keycount= 0; //计数用的
private$key; //存放一个唯一值
//主要实现将前台获取到的数据存放到上述InterpreterContext类中的功能,看到下面的内容就会发现继承他的类调用了InterpreterContext类的replace()方法
abstractfunctioninterpret (InterpreterContext$context);
//获取一个唯一值
functiongetKey(){
if(!isset($this->key)){
self::$keycount++;
$this->key= self::$keycount;
}
return$this->key;
}
}
下面将要讲到的类都将继承这个类,并且他和OperatorExpression(操作符表达式抽象类)是一个组合的关系,也就是说OperatorExpression在初始化时可以包含所有继承了Expression的子类(这也是本书一直在强调的要面向接口编程,这个Expression就是个接口,利用这个接口可以实现多态,不知道自己装B说的对不对,哈! 具体可以在看看原书的类图)
三、LiteralExpression 文字表达式类,作用就是将一个字符串保存到InterpreterContext这个小容器里,保存成一个索引数组,例如保存开头那二句自创代码中的 4 或者 four
代码如下:
classLiteralExpressionextendsExpression{
private$value;
function__construct ($value){ //初始化时传入要保存的值
$this->value=$value;
}
functioninterpret(InterpreterContext$context){ //调用InterpreterContext类的replace()将$value保存到InterpreterContext这个小容器里
$context->replace($this,$this->value);
}
}
四、VariableExpression 变量表达式类,和上面类的作用是一样的只不过数据将被保存成关联数组,关联数组中的健是变量名,值呢就是变量的值,例如开头二句中的变量"input" 和值 "4",
代码如下:
classVariableExpressionextendsExpression{
private$name; //变量名
private$val; //变量值
function__construct ($name,$val=null){
$this->name =$name;
$this->val =$val;
}
functioninterpret(InterpreterContext$context){
if(!is_null($this->val)){
$context->replace($this,$this->val);
$this->val = null;
}
}
functionsetValue($value){ //用于设置变量的值
$this->val =$value;
}
functiongetKey(){ //这个复写了父类的getKey()方法,在小容器InterpreterContext的lookup()方法调用这个类的实例的getKey()方法时 它将返回一个字符串(即变量名)而不是数字索引
return$this->name;
}
}
五、OperatorExpression 操作符表达式抽象基类,此类继承且组合了Expression抽象基类,实现的interpret()方法主要保存表达式的计算结果
代码如下:
abstractclassOperatorExpressionextendsExpression{
protected$l_op; //表达式左边的值
protected$r_op; //表达式右边的值
function__construct (Expression$l_op,Expression$r_op){ //初始化时可组合继承了Expression类的子类实例
$this->l_op =$l_op;
$this->r_op =$r_op;
}
functioninterpret(InterpreterContext$context){ //主要用于保存表达试的结果(保存到InterpreterContext 类的实例中)
$this->l_op->interpret($context); //将Expression子类实例的值或计算结果保存到InterpreterContext 类的实例中
$this->r_op->interpret($context);
$result_l=$context->lookup($this->l_op); //获取上一步的值或计算结果
$result_r=$context->lookup($this->r_op);
$this->doInterpret($context,$result_l,$result_r); //具体的比较运算由继承的子类来实现
}
protectedabstractfunctiondoInterpret(InterpreterContext$context,$result_l,$result_r);
}
六、EqualsExpression、BooleanOrExpression、BooleanAndExpression,分别为继承了OperatorExpression 抽象基类的相等表达式、或表达式、与表达式只有一个方法doInterpret()内部调用了InterpreterContext类的replace()方法将表达式的计算结果保存到InterpreterContext类的实例中
代码如下:
//相等表达式
classEqualsExpressionextendsOperatorExpression {
protectedfunctiondoInterpret(InterpreterContext$context,$result_l,$result_r){
$context->replace($this,$result_l==$result_r);
}
}
//或表达式
classBooleanOrExpressionextendsOperatorExpression{
protectedfunctiondoInterpret(InterpreterContext$context,$result_l,$result_r){
$context->replace($this,$result_l||$result_r);
}
}
//与表达式
classBooleanAndExpressionextendsOperatorExpression{
protectedfunctiondoInterpret(InterpreterContext$context,$result_l,$result_r){
$context->replace($this,$result_l&&$result_r);
}
}
到此为止此模式相关的类就介绍完毕,上述代码都是进过测试的,可直接复制粘贴运行来查看结果,现在我们就来看看客户端代码:
客户端代码一:
$context=newInterpreterContext();
$statement=newBooleanOrExpression ( //可尝试将此操作符表达式换成BooleanAndExpression 运行一下 看看执行结果
//可尝试将LiteralExpression中实例化的参数改成其他值看看运算结果,或者直接将EqualsExpression对象换成BooleanOrExpression 或BooleanAndExpression
newEqualsExpression(newLiteralExpression('four'),newLiteralExpression('four')),
newEqualsExpression(newLiteralExpression('b'),newLiteralExpression(Ɗ'))
);
$statement->interpret($context);
if($context->lookup($statement)){
echo'条件成立'
}else{
echo'条件不成立'
}
客户端代码二:
$context=newInterpreterContext();
$statement=newBooleanOrExpression(
newBooleanAndExpression(
newEqualsExpression(newLiteralExpression(Ɗ'),newLiteralExpression(Ɗ')),
newEqualsExpression(newLiteralExpression(Ɗ'),newLiteralExpression(Ɗ'))
),
newEqualsExpression(newLiteralExpression('b'),newLiteralExpression(Ɗ'))
);
$statement->interpret($context);
if($context->lookup($statement)){
echo'条件成立'
}else{
echo'条件不成立'
}
客户端代码三:
这是原文的客户端代码实例和上述客户端代码的区别在于使用了变量表达式VariableExpression
$context=newInterpreterContext();
$input=newVariableExpression('input'); //这里定义了一个变量input 但并未赋值
$statement=newBooleanOrExpression(
newEqualsExpression($input,newLiteralExpression('four')), //这里变量表达式和文字表达式的值将进行一个是否相等的比较
newEqualsExpression($input,newLiteralExpression(Ɗ'))
);
foreach(array("four","4","52")as$val){
$input->setValue($val); //对input这个变量赋值
print"变量input的值为:$val:
";
$statement->interpret($context); //进行比较并将比较结果存入InterpreterContext对象实例
if($context->lookup($statement)){ //获取比较的结果
print"条件成立
";
}else{
print"条件不成立
";
}
}
上述代码经过测试都可以正常运行,有需要的朋友可以复制下来,运行一下看看结果。
以上这篇PHP面向对象的解释器模式就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持本网站。
下面小编就为大家带来一篇PHP实现电商订单自动确认收货redis队列。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧一、场景
之前做的电商平台,用户在收到货之后,大部分都不会主动的点击确认收货,导致给商家结款的时候,商家各种投诉,于是就根据需求,要做一个订单在发货之后的x天自动确认收货。所谓的订单自动确认收货,就是在在特定的时间,执行一条update语句,改变订单的状态。
二、思路
最笨重的做法,通过linux后台定时任务,查询符合条件的订单,然后update。最理想情况下,如果每分钟都有需要update的订单,这种方式也还行。奈何平台太小,以及卖家发货时间大部分也是密集的,不会分散在24小时的每分钟。那么,定时任务的话,查询过多,不适合。这里可以先把将要自动确认收货的订单信息存储到其他介质上,比如redis,memcache,rabbitmq,然后执行的脚本从前面的介质获取到订单信息来判断,这里可以大大的减少数据库的查询压力。
redis队列的生产者
对此,我们选择每天在凌晨两点的时候,通过linux的定时任务把即将要确认收货的订单信息查询出来,然后存储在redis上,redis上我们选择的队列,队列处理的特点就是先进先出,前面的数据在查询订单时,通过发货时间排序,所以最先出队列的肯定是距离规定的自动收货时间最近的订单。代码如下
$successCount=0;
$failCount=0;
$screen_time= 3600*24*9;//设置筛选天数
$data=array();
$now_time= time();
//查询符合要求的数据
$sql="select id,send_timeasdeliver_time from `order` where is_send=1andis_del=0andis_cancel=0andis_token=0andsend_time>0andsend_time + {$screen_time} <$now_time
order by send_time asc";
$res=$con->query($sql);
//当队列还有数据时将数据记录并清除
while($redis->LLEN('auto_recevice_order')){
$txt='执行时间:'.date('Y-m-d H:i:s').',信息:'.$redis->RPOP('auto_recevice_order');
file_put_contents('./autoToken/fail_log.txt',$txt."\r\n".PHP_EOL,FILE_APPEND);
$failCount++;
}
//重新填充数据进队列
while($row=$res->fetch_assoc()) {
$successCount++;
$redis->LPUSH('auto_recevice_order',json_encode($row1));
}
$con->close();
$success=date('Y-m-d H:i:s').':[推送成功]:本次成功推送数据:'.$successCount.'条;记录上次处理失败数据:'.$failCount."条\r\n";
file_put_contents('./success_log.txt',$success."\r\n".PHP_EOL,FILE_APPEND);
redis队列的消费者
队列的消费者没有通过linux的定时任务去做,用linux的screen+php cli模式执行php脚本,消费者只需要不断的从队列中读取订单信息,然后判断订单信息中的发货时间,如果达到自动收货的要求,就执行update语句。同时如果没有达到收货的时间,而且与收货时间间距比较大的时候,可以让php脚本休眠sleep一定的时间数,这个时间数自己调节设计,获取出来的未达到时间要求的订单,需要重新推送到redis队列中去,而且还是队列的顶端。以便下次获取。代码如下:
$set_time= 3600*24*10;//设置几天后自动收货
while(true){
if($i%30==0){
usleep(10);//防止while 循环使CPU使用率过高
}
if($redis->LLEN('auto_recevice_order')){
$data= json_decode($redis->RPOP('auto_recevice_order'));
$id= (int)$data->id;//将数据转化为整形
$deliver_time= (int)$data->deliver_time;//将数据转化为整形
$res1=$res2=false;
$now_time= time();
if(($deliver_time+$set_time)<$now_time){
$sql1="update `order` set `is_token`=Ƈ',`token_time` = $now_time where id=$id and is_send=1 and is_del=0 and is_cancel=0 and is_token=0 and send_time + {$set_time} < $now_time";
$res1=$con->query($sql1);//更新数据
$rows= mysqli_affected_rows($con);
if($rows){
$ip=$this->getIp();
$sql2="insert into `order_log`(`order_id`,`log_msg`,`log_ip`,`log_role`,`log_user`,`log_order_state`,`log_time`) VALUES($id,'系统自动收货','$ip','系统','服务器','收货',$now_time)";//写入订单日志
$res2=$con->query($sql2);//添加日志数据
}
}
if($res1==false){//将没达到条件的数据重新插入队列中
$redis->RPUSH('auto_recevice_order',json_encode(array('id'=>$id,'deliver_time'=>$deliver_time)));
}
}
$i++;
}
这里执行php脚本,需要用到linux的screen或者supervisor、nohup守护进程。具体用法可自行百度.同样脚本里面最好有必须的日志记录。
三、思考
随着业务的增长,在队列中同一秒内,存在的多个需要处理的订单,而一次只能从队列中取出一个相关订单信息的时候,可以采用一个生产者多个消费者的模式,这种情况下,可以用到锁机制,保证一条消息只能到达一个消费者。当redis数据达到一定的量之后,也可以适当的调整生产者的执行频率和对应的条件。
以上这篇PHP实现电商订单自动确认收货redis队列就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持本网站。
相关文章
- 最近想自学PHP ,做了个验证码,但不知道怎么搞的,总出现一个如下图的小红叉,但验证码就是显示不出来,原因如下 未修改之前,出现如下错误; (1)修改步骤如下,原因如下,原因是apache权限没开, (2)点击打开php.int., 搜索extension=ph...2013-10-04
Django def clean()函数对表单中的数据进行验证操作
这篇文章主要介绍了Django def clean()函数对表单中的数据进行验证操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-07-09- 这篇文章主要为大家详细介绍了JS实现随机生成验证码,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2021-09-06
jQuery Real Person验证码插件防止表单自动提交
本文介绍的jQuery插件有点特殊,防自动提交表单的验证工具,就是我们经常用到的验证码工具,先给大家看看效果。效果图如下: 使用说明 需要使用jQuery库文件和Real Person库文件 同时需要自定义验证码显示的CSS样式 使用实例...2015-11-08- 验证码类文件 CreateImg.class.php <?php class ValidationCode { private $width,$height,$codenum; public $checkcode; //产生的验证码 private $checkimage; //验证码图片 private $disturbColor = ''; /...2015-11-08
- 这篇文章主要介绍了selenium 反爬虫之跳过淘宝滑块验证功能,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下...2020-08-27
- 这篇文章主要介绍了el-table树形表格表单验证(列表生成序号),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2020-06-01
- 这篇文章主要介绍了Nest.js 授权验证的方法示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-02-22
- 这篇文章主要介绍了基于JavaScript实现验证码功能的相关资料...2017-04-03
- 这篇文章主要为大家详细介绍了vue实现表单验证小功能,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2021-09-29
ThinkPHP使用心得分享-ThinkPHP + Ajax 实现2级联动下拉菜单
首先是数据库的设计。分类表叫cate.我做的是分类数据的二级联动,数据需要的字段有:id,name(中文名),pid(父id). 父id的设置: 若数据没有上一级,则父id为0,若有上级,则父id为上一级的id。数据库有内容后,就可以开始写代码,进...2014-05-31- 通过jquery.cookie.js插件可以快速实现“点击获取验证码后60秒内禁止重新获取(防刷新)”的功能效果图:先到官网(http://plugins.jquery.com/cookie/)下载cookie插件,放到相应文件夹,代码如下:复制代码 代码如下: <!DOCTYPE ht...2015-03-15
- 下面小编就为大家带来一篇单击按钮发送验证码,出现倒计时的简单实例。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧 代码...2017-07-06
- 验证码是一个现在WEB2.0中常见的一个功能了,像注册、登录又或者是留言页面,都需要注册码来验证当前操作者的合法性,我们会看到有些网站没有验证码,但那是更高级的验证了,...2016-11-25
- 这篇文章主要介绍了基于Pytorch版yolov5的滑块验证码破解思路详解,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下...2021-02-25
- 这篇文章主要为大家详细介绍了js canvas实现滑块验证,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2021-03-14
Bootstrap中文本框的宽度变窄并且加入一副验证码图片的实现方法
这篇文章主要介绍了Bootstrap中文本框的宽度变窄并且加入一副验证码图片的实现方法的相关资料,非常不错,具有参考借鉴价值,需要的朋友可以参考下...2016-06-24- 工信部的ICP备案网站登录时验证码一直输入不正确怎么回事,为了防止一些机器采集人工信部对于查询验证做得识别度极低,所以许多的朋友都会发现输入验证码一直有问题了,那...2016-10-10
- 这篇文章主要给大家介绍了关于利用Python验证的50个常见正则表达式的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-03-11
vue element table中自定义一些input的验证操作
这篇文章主要介绍了vue element table中自定义一些input的验证操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-07-18