php官方已经提供了Iterator(迭代器)接口,通过网上资料的学习,目前看适合超大集合或者数组提取使用。学习一个函数的实现对比内存占用差距.
使用迭代器和普通循环实现range()函数。
(1).普通循环实现range()函数。
function newrange($low, $hign, $step = 1) { $ret = []; for ($i = 0; $i < $hign; $i += $step) { $ret[] = $i; } return $ret; } //统计内存占用 $oldMemory = memory_get_usage(); $result = newrange(0, 500000); $newMemory = memory_get_usage(); $useMemory = $newMemory - $oldMemory; echo 'used:' . $useMemory . 'byte';
used:14680128byte
(1).迭代器实现range()函数。
class newrange implements Iterator { protected $low; protected $high; protected $step; protected $current; public function __construct($low, $high, $step = 1) { $this->low = $low; $this->high = $high; $this->step = $step; } //返回到迭代器的第一个元素 public function rewind() { var_dump('rewind'); $this->current = $this->low; } //向前移动到下一个元素 public function next() { var_dump('next'); $this->current += $this->step; } //返回当前元素 public function current() { var_dump('current'); return $this->current; } //返回当前元素的键 public function key() { var_dump('key'); return $this->current + 1; } //检查当前位置是否有效 public function valid() { var_dump('valid'); return $this->current <= $this->high; } } //统计内存占用 $oldMemory = memory_get_usage(); $result = new newrange(0, 9, 1); /* foreach ($result as $key => $value) { } */ $newMemory = memory_get_usage(); $useMemory = $newMemory - $oldMemory; echo 'used:' . $useMemory . 'byte';
used:96byte
再来使用php自带的pdo实现的迭代器测试大数据占用内存.
(1).数据是我们公司正式服务器拉取下拉的,保证数据不是随便生成的,之前在迁移数据使用mysqli提取单表53w数组内存超过限制(每条记录字段50个)。
(2).本次改用pdo的query方法,因为query内部用php迭代器实现
//起始内存 $oldMemory = memory_get_usage(); $host = '127.0.0.1:9944'; $datebase = 'dev_dnet1'; $username = 'dev_dnet2'; $password = 'dev_dnet3'; $conn = "mysql:host=$host;dbname=$datebase"; $db = new PDO($conn, $username, $password); $sql = 'select * from prd_product'; $result = $db->query($sql); //结束内存 $newMemory = memory_get_usage(); $useMemory = $newMemory - $oldMemory; echo 'used:' . $useMemory . 'byte';
used:1563928408byte 1491M
总结:
(1).迭代器仅仅只是产生1个对象的内存,因此在使用超大数组的时候可以选择迭代器,只不过使用起来麻烦点,但是至少比使用数组性能好多了。
(2).如果想要看到内部执行过程只需要使用foreach直接循环结果集,php内部帮我们执行了对应方法来实现迭代
(3).PHP数组的内存利用率只有 1/10,但是我们可以用迭代器啊,世界上最好的编程语言
<?php $base64_body = substr(strstr($_POST[base64],','),1); $data= base64_decode($base64_body); file_put_contents($_SERVER[&q...
使用php函数array_multisort()即可实现和SQL一样的order by排序. 例如我们需要对会员表按照主键降序排列,年龄升序排列://会员表数据 $list = []; $list[] = ['mid' =>&n...
逛公众号文章看到文章"php实现事件监听与触发的方法,你用过吗?",我就好奇了,php又不是asp.net的webform,哪里来的服务端事件监听。于是学习了一波。先看下监听类:class Event { /** &nbs...
php arrayaccess 官方的说法是让你能以数组的形式访问对象,对于这种php内置接口一直不太明白有什么用,坚持多看文章,终于理解,特来分享,思路不同,更易于理解。(1).创建一个学生类,并且实现arrayaccess 接口。<?php class studen...
ThinkPHP中有一个debug调试功能,能输出报错文件的信息,并能看到这个函数被哪些函数调用,从框架的启动开始记录,特别方便调试。于是研究了下它的底层给予了实现。<?php //--框架核心--Start //框架内置错误处理 function errDealWith($er...
重构框架的时候想要考虑支持下cli模式,于是参考了thinkphp的底层。/** * 获取应用根目录 * @return string */ public static function getRootP...