首页
关于
Search
1
给你10个市场数据调研报告的免费下载网站!以后竞品数据就从这里找!
138 阅读
2
php接口优化 使用curl_multi_init批量请求
130 阅读
3
2024年备考系统架构设计师
102 阅读
4
《从菜鸟到大师之路 ElasticSearch 篇》
101 阅读
5
PHP 文件I/O
89 阅读
php
thinkphp
laravel
工具
开源
mysql
数据结构
总结
思维逻辑
令人感动的创富故事
读书笔记
前端
vue
js
css
书籍
开源之旅
架构
消息队列
docker
教程
代码片段
redis
服务器
nginx
linux
科普
java
c
ElasticSearch
测试
php进阶
php基础
登录
Search
标签搜索
php函数
php语法
性能优化
安全
错误和异常处理
问题
vue
Composer
Session
缓存
框架
Swoole
api
并发
异步
正则表达式
php-fpm
mysql 索引
开发规范
协程
dafenqi
累计撰写
785
篇文章
累计收到
7
条评论
首页
栏目
php
thinkphp
laravel
工具
开源
mysql
数据结构
总结
思维逻辑
令人感动的创富故事
读书笔记
前端
vue
js
css
书籍
开源之旅
架构
消息队列
docker
教程
代码片段
副业
redis
服务器
nginx
linux
科普
java
c
ElasticSearch
测试
php进阶
php基础
页面
关于
搜索到
559
篇与
的结果
2023-08-09
PHP 闭包详解及实例代码
PHP 闭包详解及实例代码这篇文章主要介绍了PHP 闭包详解及实例代码的相关资料,需要的朋友可以参考下闭包和匿名函数在PHP5.3.0中引入的。闭包是指:创建时封装周围状态的函数。即使闭包所处的环境不存在了,闭包中封装的状态依然存在。理论上,闭包和匿名函数是不同的概念。但是PHP将其视作相同概念。实际上,闭包和匿名函数是伪装成函数的对象。他们是Closure类的实例。闭包和字符串、整数一样,是一等值类型。创建闭包<?php$clousre = function ($name) {return 'Hello ' . $name;};echo $closure('nesfo');我们之所以能调用$closure变量,是因为这个变量的值是一个闭包,而且闭包对象实现了__invoke()魔术方法。只要变量名后有(),PHP就会查找并调用__invoke()方法。通常会把PHP闭包当作函数的回调使用。array_map(), preg_replace_callback()方法都会用到回调函数,这是使用闭包的最佳时机!举个例子:<?php$numbersPlusOne = array_map(function ($number) {return $number + 1;}, [1, 2, 3]);print_r($numbersPlusOne);得到结果:[2, 3, 4]在闭包出现之前,只能单独创建具名函数,然后使用名称引用那个函数。这么做,代码执行会稍微慢点,而且把回调的实现和使用场景隔离了。<?phpfunction incrementNum ($number) {return $number + 1;}$numbersPlusOne = array_map('incrementNum', [1, 2, 3]);print_r($numbersPlusOne);附加状态匿名函数不止可以当回调使用,还可以为PHP附加并封装状态。PHP中,必须手动调用闭包对象的bindTo()方法或者使用use关键字,才能把状态附加到PHP闭包上。<?phpfunction enclosePerson ($name) {return function ($doCommand) use ($name) {return $name . ', ' . $doCommand;}}$clay = enclosePerson('Clay');echo $clay('get me sweet tea!');得到结果:"Clay, get me sweet tea!"PHP闭包是对象,每个闭包实例都可以使用$this关键字获取闭包的内部状态。闭包对象的默认状态没什么用,只有__invoke()方法和bindTo方法而已。我们可以使用bindTo()这个方法,将Closure对象的内部状态绑定到其它对象上。bindTo()方法的第二个参数:其作用是指定绑定闭包的那个对象所属的PHP类。因此,闭包可以访问绑定闭包的对象中受保护和私有的成员。PHP框架经常使用bindTo()方法把路由URL映射到匿名回调函数上。这么做可以在这个匿名函数中使用$this关键字引用重要的应用对象。使用bindTo()方法附加闭包状态<?phpclass App{protected $routes = [];protected $responseStatus = '200 OK';protected $responseContentType = 'text/html';protected $responseBody = 'Hello world';public function addRoute($routePath, $routeCallback){$this->routes[$routePath] = $routeCallback->bindTo($this, __CLASS__);}public function dispatch($currentPath){foreach($this->routes as $routePath => $callback){if ($routePath === $currentPath) {$callback();}}header('HTTP/1.1' . $this->responseStatus);header('Content-type: ' . $this->responseContentType);header('Content-length' . mb_strlen($this->responseBody));echo $this->responseBody;}}<?php$app = new App();$app->addRoute('/user/nesfo', function () {$this->responseContentType = 'application/json; charset=utf8';$this->responseBody = '{"name": "nesfo"}';});$app->dispatch('/user/nesfo');
2023年08月09日
13 阅读
0 评论
0 点赞
2023-08-09
php基于闭包实现函数的自调用(递归)实例分析
php基于闭包实现函数的自调用(递归)实例分析这篇文章主要介绍了php基于闭包实现函数的自调用,结合实例形式分析了php闭包实现递归的操作方法,需要的朋友可以参考下本文实例讲述了php基于闭包实现函数的自调用(递归)的方法。分享给大家供大家参考,具体如下:php的闭包可能不常用,但是在某些场合之下还是可以考虑用php的闭包来实现某些功能的,比如递归,这里讲一下用php的闭包实现递归//php闭包实现函数的自调用,也就是实现递归function closure($n,$counter,$max){ //匿名函数,这里函数的参数加&符号是,引址调用参数自己 $fn = function (&$n,&$counter,&$max=1) use(&$fn){//use参数传递的是函数闭包函数自身$n++; if($n < $max){//递归点,也就是递归的条件 $counter .=$n.'<br />'; //递归调用自己 $fn($n,$counter,$max); } return $counter;};//记得这里必须加;分号,不加分号php会报错,闭包函数 /* *这里函数closure的返回值就是调用闭包的匿名函数 *而闭包函数,引用closure函数传进来的参数 */ return $fn($n,$counter,$max);}echo (closure(0,'',10));这是一个简单的使用php闭包实现递归的函数,其实,假如我们对这个函数稍微修改一下,是可以实现很多功能的,比如说,无限极分类。关于里面的闭包函数,其实,我们也可以不用引用外面的closure参数,里面的闭包函数的参数,我们可以在closure的内部设置。我们也可以不用返回闭包函数的值,因为闭包函数是在closure函数里面,我们可以在closure里面设置一下变量传递给闭包函数,然后,用闭包递归获取我们想要的东西存储在数组中,再利用closure函数返回闭包函数获取的东西,所以,闭包函数也可以没有返回值的,关键是你要获取什么东西。总之,你只要利用这个闭包函数实现你要的东西,然后返回你需要的东西即可,不一定非要从闭包函数的返回值中获取。
2023年08月09日
13 阅读
0 评论
0 点赞
2023-08-09
php实现递归的三种基本方式
php实现递归的三种基本方式这篇文章主要介绍了php实现递归的三种基本方法,包括利用引用做参数,利用全局变量,利用静态变量来实现递归,并附上了相关示例,最后给大家一个演示,涉及php的递归操作技巧,需要的朋友可以参考下递归函数是我们常用到的一类函数,最基本的特点是函数自身调用自身,但必须在调用自身前有条件判断,否则无限无限调用下去。实现递归函数可以采取什么方式呢?本文列出了三种基本方式。理解其原来需要一定的基础知识水品,包括对全局变量,引用,静态变量的理解,也需对他们的作用范围有所理解。递归函数也是解决无限级分类的一个很好地技巧。如果对无限级分类感兴趣,请参照php利用递归函数实现无限级分类。我习惯套用通俗的话解释复杂的道理,您确实不明白请参见手册。 利用引用做参数先不管引用做不做参数,必须先明白引用到底是什么?引用不过是指两个不同名的变量指向同一块存储地址。本来每个变量有各自的存储地址,赋值删除各行其道。现在可好,两个变量共享一块存储地址。 $a=&$b; 。实际上指的是 $a 不管不顾自己原来的存储地址,非要和 $b 共享一室了。因而任何对存储地址数值的改变都会影响两个值。函数之间本来也是各行其是,即便是同名函数。递归函数是考虑将引用作为参数,成为一个桥梁,形成两个函数间的数据共享。虽然两个函数见貌似操作的是不同地址,但是实际上操作的是一块儿内存地址。function test($a=0,&$result=array()){$a++;if ($a<10) { $result[]=$a; test($a,$result);}echo $a;return $result;}上面的例子非常简答,以a<10作为判断条件,条件成立,则把a赋给result[];将result的引用传入函数,会将每一次递归产生的a添加到结果数组result。因而本例生成的$result数组是 Array ( [0] => 1 [1] => 2 [2] => 3 [3] => 4 [4] => 5 [5] => 6 [6] => 7 [7] => 8 [8] => 9 ) 。本例比较有意思的是echo a的值。相信很多人认为是12345678910吧,其实不然,是1098765432。为什么呢?因为函数还没执行echoa前就进行了下一次的函数递归。真正执行echo a是当a<10条件不满足的时候,echo a,返回result,对于上一层而言,执行完递归函数,开始执行本层的echo $a,依次类推。 利用全局变量利用全局变量完成递归函数,请确保你确实理解什么是全局变量。global在函数内申明变量不过是外部变量的同名引用。变量的作用范围仍然在本函数范围内。改变这些变量的值,外部同名变量的值自然也改变了。但一旦用了&,同名变量不再是同名引用。利用全局变量实现递归函数没必要理解到这么深的一层,还保持原有对全局变量的看法就可以顺理成章理解递归函数。function test($a=0,$result=array()){ global $result; $a++; if ($a<10) {$result[]=$a; test($a,$result);} return $result;} 利用静态变量我们常常在类中见到static,今天我们把它利用到递归函数中。请记住static的作用:仅在第一次调用函数的时候对变量进行初始化,并且保留变量值。举个栗子:function test(){static $count=0;echo $count;$count++;}test();test();test();test();test();请问这一段代码的执行结果是多少?是00000么?必然不是。是01234。首先第一次调用test(),static对 $count 进行初始化,其后每一次执行完都会保留 $count 的值,不再进行初始化,相当于直接忽略了 static $count=0; 这一句。因而将static应用到递归函数作用可想而知。在将需要作为递归函数间作为“桥梁"的变量利用static进行初始化,每一次递归都会保留"桥梁变量"的值。function test($a=0){ static $result=array(); $a++; if ($a<10) {$result[]=$a; test($a);} return $result;} 总结所谓递归函数,重点是如何处理函数调用自身是如何保证所需要的结果得以在函数间合理"传递",当然也有不需要函数之间传值得递归函数,例如:function test($a=0){ $a++; if ($a<10) {echo $a; test($a);}}面对这样的函数,我们就不必大伤脑筋了。顺便说一句,深入理解变量引用相关知识对解决这类问题大有裨益。最后给大家分享一个php实现递归与无限分类的方法,具体实现方法如下:<?phpecho "";$area = array(array('id'=>1,'area'=>'北京','pid'=>0),array('id'=>2,'area'=>'广西','pid'=>0),array('id'=>3,'area'=>'广东','pid'=>0),array('id'=>4,'area'=>'福建','pid'=>0),array('id'=>11,'area'=>'朝阳区','pid'=>1),array('id'=>12,'area'=>'海淀区','pid'=>1),array('id'=>21,'area'=>'南宁市','pid'=>2),array('id'=>45,'area'=>'福州市','pid'=>4),array('id'=>113,'area'=>'亚运村','pid'=>11),array('id'=>115,'area'=>'奥运村','pid'=>11),array('id'=>234,'area'=>'武鸣县','pid'=>21));function t($arr,$pid=0,$lev=0){static $list = array();foreach($arr as $v){if($v['pid']==$pid){echo str_repeat(" ",$lev).$v['area']."";//这里输出,是为了看效果$list[] = $v;t($arr,$v['id'],$lev+1);}}return $list;}$list = t($area);echo "";print_r($list);?>
2023年08月09日
6 阅读
0 评论
0 点赞
2023-08-09
php 生成器 yield 实现的递归
php 生成器 yield 实现的递归原始递归方法:<?php echo "<pre>"; $area = array( array('id'=>1,'area'=>'北京','pid'=>0), array('id'=>2,'area'=>'广西','pid'=>0), array('id'=>3,'area'=>'广东','pid'=>0), array('id'=>4,'area'=>'福建','pid'=>0), array('id'=>11,'area'=>'朝阳区','pid'=>1), array('id'=>12,'area'=>'海淀区','pid'=>1), array('id'=>21,'area'=>'南宁市','pid'=>2), array('id'=>45,'area'=>'福州市','pid'=>4), array('id'=>113,'area'=>'亚运村','pid'=>11), array('id'=>115,'area'=>'奥运村','pid'=>11), array('id'=>234,'area'=>'武鸣县','pid'=>21) ); function t($arr,$pid=0,$lev=0){ static $list = array(); foreach($arr as $v){ if($v['pid']==$pid){ echo str_repeat(" ",$lev).$v['area']."<br />"; //这里输出,是为了看效果 $list[] = $v; t($arr,$v['id'],$lev+1); } } return $list; } $list = t($area); echo "<hr >"; print_r($list); ?>用 yield 实现<?php echo "<pre>"; $area = array( array('id'=>1,'area'=>'北京','pid'=>0), array('id'=>2,'area'=>'广西','pid'=>0), array('id'=>3,'area'=>'广东','pid'=>0), array('id'=>4,'area'=>'福建','pid'=>0), array('id'=>11,'area'=>'朝阳区','pid'=>1), array('id'=>12,'area'=>'海淀区','pid'=>1), array('id'=>21,'area'=>'南宁市','pid'=>2), array('id'=>45,'area'=>'福州市','pid'=>4), array('id'=>113,'area'=>'亚运村','pid'=>11), array('id'=>115,'area'=>'奥运村','pid'=>11), array('id'=>234,'area'=>'武鸣县','pid'=>21) ); function t($arr,$pid=0,$lev=0){ static $list = array(); foreach($arr as $v){ if($v['pid']==$pid){ echo str_repeat(" ",$lev).$v['area']."<br />"; //这里输出,是为了看效果 $list[] = $v; foreach(t($arr,$v['id'],$lev+1) as $value){ yield $value; } } } yield $list; } foreach (t($area) as $key => $value) { $list=$value; } echo "<hr >"; print_r($list); ?>
2023年08月09日
26 阅读
0 评论
0 点赞
2023-08-09
浅析 PHP 中的异常与错误
浅析 PHP 中的异常与错误异常与错误异常是指程序运行中不符合预期情况以及与正常流程不同的状况。错误则属于自身问题,是一种非法语法或者环境问题导致的、让编译器无法通过检查设置无法运行的情况。由于php最开始是没有异常处理,后来为了进军企业级开发,模仿java等语言,推出了异常。导致php中遇到任何自身错误都会触发一个错误,而不是抛出一个异常(某些情况下,会同时抛出错误和异常)。PHP一旦遇到非正常代码,大多数情况下,都是直接抛出错误,而不是异常。php只有在你throw 一个异常后,才能用try...catch来捕获异常(一般情况下如此,也有部分异常可以自动捕获)。在php中通常会在以下场景中使用异常:对程序的悲观预测:如果认为自己的代码无法一一处理各种可预见的情况、不可预见的情况。程序的需要和对业务的关注 : 如果对数据的一致性要求很高时,可以用try...catch把异常造成的逻辑中断破坏将到最小,并且经过补救处理后,不影响业务逻辑的完整性。语言级别的健壮性要求 : 通过精确控制运行时的流程,在程序中断时,有预见的用try...catch缩小可能出错的范围,及时捕获异常并做出相应的补救。怎样看待php的异常历史原因导致php的异常处理是不足的,绝大多数情况下,无法自动抛出异常,必须使用if...else先进行判断,再手动抛出异常。手动抛出异常的意义不是很大,因为这意味着在代码里已经充分的预期到错误的出现。同时这种方式还会让你在复杂的逻辑判断和处理中晕头转向。导致失去异常真正的优点。那么有更好的异常抛出方法吗?有,那就是结合使用错误php中的错误错误就是会使脚本运行不正常的情况。在php中主要的错误等级如下:deprecated: 最低级别的错误,表示"不推荐, 不建议"。例如在php 5中使用了ereg系列的正则函数就会出现。这类错误一般由于使用了不推荐的、过时的函数或语法造成。不影响程序正常运行,但建议修正。notice: 一般指语法中存在不恰当的地方。如使用变量但是未定义就会报此错误。不影响程序正常流程。warning: 较高级别的错误,在语法中出现很不恰当的情况才会出现此错误,比如函数参数不匹配。会导致得不到预期的结果,需要修改代码。fetal error: 致命错误,直接导致程序终止运行。这类错误必须修改。prase error: 语法解析错误,上面几种都属于运行时错误,此错误在运行前就会抛出。在php中,总共有16错误级别,但是主要的就是上面几种。error.php$data = '2012-12-20';if (ereg("([0-9]{4})-([0-9]{1,2})-([0-9]{1,2})", $data, $regs)) {echo "$reg[3].$regs[2].$regs[1]";} else {echo "Invalid data format: $data";}$a = array('o' => 2, 4, 6, 8);echo $a[o];$result = array_sum($a, 3);echo func();echo '致命错误后,还会执行吗?';//echo '最高级别错误', $55;上面代码执行后,会有四个错误级别,如果你无法完全看到的话,你需要去修改你的ini配置文件中错误显示级别为 E_ALL自定义错误处理程序可以使用 set_error_handler() 函数来托管错误处理程序,可自行定制错误的处理流程。如果要取消托管的话,可以在同一个页面中使用restore_error_handler()来取消托管。如果想要自己抛出一个错误的话,可以使用trigger_error()函数。<?php//自定义错误处理程序function customError($errno, $errstr, $errfile, $errline){echo "<b>错误代码:</b>[{$errno}] {$errstr}", PHP_EOL; echo "错误所在代码行:{$errline} 文件{$errfile}", PHP_EOL; echo "PHP版本", PHP_VERSION, "(", PHP_OS, ")", PHP_EOL;}set_error_handler("customError", E_ALL | E_STRICT);$a = array('o' => 2, 4, 6, 8);echo $a[o];执行上面的代码,可以看到错误信息是由我们自定义的处理程序输出的,完全绕开了系统的处理程序。如果错误发生在自定义处理程序前,则不会调用我们自定义的错误处理程序,所以应当先定义错误处理程序。当然不是所有的错误级别都可以用set_error_handler来托管,如E_ERROR、E_PARSE、E_CODE_WARNING、E_COMPILE_ERROR、E_COMPILE_WARNING以及E_STRICT中的部分。这些错误信息会以原始的方式来显示或者不现实。PHP把许多异常看作是错误,所以这些"异常"同样可以使用set_error_handler来接管:function customError($errno, $errstr, $errfile, $errline){//自定义错误处理是,手动抛出异常 throw new Exception($errstr);}set_error_handler('customError', E_ALL | E_STRICT);try {$a = 5/0;} catch (Exception $e) {echo '错误信息:', $e->getMessage();}当然这种处理方式也有自己的优缺点:缺点: 必须依靠程序员自己来掌控对异常的处理,对于异常的高发区、敏感区,如果程序员处理不好,就会导致业务数据不一致的问题。优点: 可以获得程序运行时的上下文信息,以进行针对性的补救。fetal error这样的错误无法捕获,也无法在发生后恢复流程处理,但是可以使用register_shutdown_function()函数在程序终止或die时触发一个函数,给程序带来一个短暂的回光返照。在php4时,不支持析构函数,也常用于模拟实现析构函数。class Shutdown{public function stop() { if (error_get_laster()) { print_r(error_get_laster()); } die('Stop.'); }}register_shutdown_function(array(new Shutdown(), 'stop'));$a = new a(); //致命错误,导致失败echo '必须终止';Parse error级别的错误,除了修改ini文件,将错误信息写到日志中,什么也做不了。小结php中错误和异常是两个不同的概念,这种设计根本上导致了php的异常和错误与其它语言相异。java中,异常时错误唯一的报告方式。说到底,两者的区别就是对异常的认识不同产生的。php异常绝大部分是通过某种方式手动抛出,才能捕获到。是一种半自动化的异常处理机制。无论是错误还是异常,都可以使用handler接管系统已有的处理机制。
2023年08月09日
12 阅读
0 评论
0 点赞
1
...
62
63
64
...
112