@phper
2017-01-10T10:35:06.000000Z
字数 3908
阅读 4914
swoole
异步IO,分很多种,但凡是IO相关的都可以借助swoole来完成。
Swoole\Async::readFile最大可读取4M的文件,如果要读取超大文件,请使用Swoole\Async::read函数
读取文件一定要先存在,不然报错,我准备一个2.text,内容是123456.
//swoole_async_readfile最大可读取4M的文件//如果要读取超大文件,请使用swoole_async_read函数Swoole\Async::readFile("./2.text", function($fileName, $content){var_dump($fileName, $content);});echo "done" .PHP_EOL;
运行下输出:
$ php async_file.phpdonestring(8) "./2.text"123456
Swoole\Async::writeFile最大可写入4M, 文件不存在,会自动创建,写入会从头覆盖原先的内容
//文件不存在,会自动创建,最大可写入4M//会覆盖原先的内容,从头开始覆盖Swoole\Async::writeFile("./2.text", 'v', function($fileName){var_dump($fileName);});echo "done" .PHP_EOL;//再异步读一下,写入成功没?Swoole\Async::writeFile("./2.text", 'v', function($fileName){var_dump($fileName);});
运行下输出:
$ php async_file.phpdonestring(8) "./2.text"v23456
可以看到会从头写入,原先的1被v给取代了。暂时不知此从尾部写文件。
异步读大文件,分段读取,每次只读$trunk_size个字节,不会占用太多内存。
我准备了一个1.text的文件,大概有 35k 大小。
$trunk_size = 8192;$offset = 0Swoole\Async::read("./1.text", function($fileName, $content){var_dump($fileName, strlen($content));//默认是return true, 会分段,多次执行//如果 return false,则执行一次就退出.//return false;}, $trunk_size, $offset);echo "done" .PHP_EOL;
$size,读取数据的最大长度,默认为8K = 8192字节
$offset,偏移文件指针,默认为0,表示从文件头部开始读取。必须大于等于0且小于文件总长度
运行下输出:
$ php async_file.phpdonestring(8) "./1.text"int(8192)string(8) "./1.text"int(8192)string(8) "./1.text"int(8192)string(8) "./1.text"int(8192)string(8) "./1.text"int(6459)string(8) "./1.text"int(0)
会发现了执行了多次,一次一次的异步分段在读取。$size参数设定每一次读取最大的值。
Swoole\Async::write是分段读写的。不需要一次性将要写的内容放到内存里,所以只占用少量内存。
我准备了一个1.text的文件,大概有 35k 大小。写入到4.text中。
$content = file_get_contents('./1.text');Swoole\Async::write("./4.text", $content, -1, function($fileName){var_dump($fileName);});echo "done" .PHP_EOL;
运行下输出:
$ php async_file.phpdonestring(8) "./4.text"
当DNS查询完成时,自动回调指定的callback函数。
当DNS查询失败时,比如域名不存在,回调函数传入的$ip为空
Swoole\Async::dnsLookup("www.baidu.com", function($host, $ip){var_dump($host, $ip);});Swoole\Async::dnsLookup("www.esd4343.com", function($host, $ip){var_dump($host, $ip);});echo 'dns done';
执行下:
$ php async.phpdns donestring(13) "www.baidu.com"string(13) "14.215.177.37"string(15) "www.esd4343.com"string(0) ""
设置一个间隔时钟定时器,单位是微妙。与after定时器不同的是tick定时器会持续触发,直到调用Swoole\Timer::clear清除。而且可以设置多个相同时间间隔的定时器。
$tmer_id1 = Swoole\Timer::tick(1000, function($timer_id, $params = null){echo $timer_id. "timeout\n" . microtime(true);//Swoole\Timer::clear($timer_id);});//1var_dump($timer_id1);// truevar_dump(Swoole\Timer::exists($timer_id1));die;
一个一次性定时器,执行完成后就会销毁,回调函数没有任何参数
$tmer_id2 = Swoole\Timer::after(1000, function(){echo "timeout\n" . microtime(true);});//2var_dump($tmer_id2);// truevar_dump(Swoole\Timer::exists($tmer_id2));die;
使用定时器ID来删除定时器,不能用于清除其他进程的定时器,只作用于当前进程。
$tmer_id1 = Swoole\Timer::tick(1, function($timer_id, $params = null){echo $timer_id. "loop" . microtime(true) . PHP_EOL;global $a; $a++;//1万次后清除if ($a >= 10000) {Swoole\Timer::clear($timer_id);}});
异步mysql就是异步的去连接数据库。直接上代码。
<?php/*** Created by PhpStorm.* User: yangyi* Date: 2017/1/5* Time: 17:52*/$db = new Swoole\Mysql;//配置文件$server = array('host' => '127.0.0.1','user' => 'root','password' => '123456','database' => 'user',);//concent方法 2个参数,第一个是配置文件,第二个是一个回调函数。$db->connect($server, function ($db, $result) {//$result 连接是否成功,只有为true时才可以执行query查询//$result 为false,可以通过`connect_errno`和`connect_error`得到失败的错误码和错误信息if ($result === false) {var_dump($db->connect_errno, $db->connect_error);die;}$sql = 'select * from ap_room limit 1';//query方法就是执行sql的方法//第一个参数是sql, 第二个参数是回调函数$db->query($sql, function($db, $result) {/** 执行失败,$result为false* error属性获得错误信息,errno属性获得错误码*/if ($result === false) {var_dump($db->error, $db->errno);}/** 执行成功* SQL为非查询语句(inset,update,delete)$result为true* affected_rows 为影响的行数,insert_id 获得*/else if ($result === true ) {var_dump($db->affected_rows, $db->insert_id);}/** 执行成功,SQL为select语句,$result为结果数组*/else {var_dump($result);}//关闭mysql连接$db->close();});});echo 33;