分类: PHP

  • PhpStorm配置Xdebug调试

    xdebug 安装

    打开 https://xdebug.org/wizard.php 复制phpinfo的信息到文本框,会返回安装指导:

    Tailored Installation Instructions
    
    Summary
    
    Xdebug installed: no
    Server API: FPM/FastCGI
    Windows: no
    Zend Server: no
    PHP Version: 5.6.9
    Zend API nr: 220131226
    PHP API nr: 20131226
    Debug Build: no
    Thread Safe Build: no
    Configuration File Path: /usr/local/php/etc
    Configuration File: /usr/local/php/etc/php.ini
    Extensions directory: /usr/local/php/lib/php/extensions/no-debug-non-zts-20131226
    Instructions
    
    Download xdebug-2.5.1.tgz
    Unpack the downloaded file with tar -xvzf xdebug-2.5.1.tgz
    Run: cd xdebug-2.5.1
    Run: phpize (See the FAQ if you don't have phpize.
    
    As part of its output it should show:
    
    Configuring for:
    ...
    Zend Module Api No:      20131226
    Zend Extension Api No:   220131226
    If it does not, you are using the wrong phpize. Please follow this FAQ entry and skip the next step.
    
    Run: ./configure
    Run: make
    Run: cp modules/xdebug.so /usr/local/php/lib/php/extensions/no-debug-non-zts-20131226
    Edit /usr/local/php/etc/php.ini and add the line
    zend_extension = /usr/local/php/lib/php/extensions/no-debug-non-zts-20131226/xdebug.so
    Restart the webserver
    If you like Xdebug, and thinks it saves you time and money, please have a look at the donation page.
    

    xdebug 配置

    编辑php.ini文件,加入

    [Xdebug]  
    ;指定Xdebug扩展文件的绝对路径  
    zend_extension = /usr/local/php/lib/php/extensions/no-debug-non-zts-20131226/xdebug.so
    ;允许远程IDE调试
    xdebug.remote_enable        = true
    ;通知 PHP 开启调试的标识
    xdebug.idekey = PHPSTORM
    ;远程主机
    xdebug.remote_host          = 192.168.xxx.xxx
    ;xdebug.remote_port         = 9000 ;默认端口 9000
    

    配置参数选项附录

    配置参数选项 参数值类型与默认值    参数选项描述
    xdebug.auto_trace   boolean类型,默认值=0 是否在脚本运行之前自动调用相关追踪函数。
    xdebug.cli_color    integer类型,默认值=0 该参数自2.2版本开始引入。如果值=1,当处于CLI模式或连接虚拟控制台时,Xdebug将高亮显示var_dumps()和堆栈输出,;在Windows中,这需要安装ANSICON工具。如果值=2,不管是否处于CLI模式或连接虚拟控制台,Xdebug都会高亮显示var_dumps()或堆栈输出;这种情况下,你可能会看到转义后的代码。
    xdebug.collect_assignments  boolean类型,默认值=0 该参数自2.1版本开始引入。用于控制是否为函数跟踪添加变量赋值功能。
    xdebug.collect_includes boolean类型,默认值=1 控制是否在跟踪文件中写入include()、include_once()、require()、require_once()等函数中用到的文件名。
    xdebug.collect_params   integer类型,默认值=0 
    控制在调用函数时,是否收集传递给函数的参数信息。如果参数值过大,这可能会占用大量的内存;不过,在Xdebug 2中不会出现该问题,因为Xdebug 2将相关数据写入磁盘中,而不是占用内存。
    
    如果值=0,则不显示任何信息。
    如果值=1,只显示类型和大小信息,例如:string(6)、array(8)。
    如果值=2,将显示类型和大小,以及全部信息的工具提示。
    如果值=3,将显示变量的全部内容。
    如果值=4,将显示变量的全部内容和变量名。
    
    xdebug.collect_return   boolean类型,默认值=0 控制是否在追踪文件中写入函数调用的返回值。
    xdebug.collect_vars boolean类型,默认值=0 控制是否收集指定作用域中的变量信息。由于需要反向工程PHP的操作码数组,因此Xdebug的分析速度可能比较慢。
    xdebug.coverage_enable  boolean类型,默认值=1 该参数自2.2版本开始引入。控制是否允许通过设置内部结构来启用代码覆盖率功能。
    xdebug.default_enable   boolean类型,默认值=1 当发生异常或错误时,是否默认显示堆栈信息。
    xdebug.dump.*   string类型,默认值=Empty  这里的*可以是COOKIE, FILES, GET, POST, REQUEST, SERVER, SESSION中的任意一个。用于指定发生错误时是否显示超全局变量数组中的索引变量信息。比如,你想要显示请求的IP地址和请求方式,可以设置为
    xdebug.dump.SERVER=REMOTE_ADD,REQUEST_METHOD
    多个索引变量用英文逗号隔开,如果要输出其中的所有变量,可以直接用*,例如:
    xdebug.dump.GET=*
    xdebug.dump_globals boolean类型,默认值=1 控制是否显示通过xdebug.dump.*定义的所有超全局变量的信息。
    xdebug.dump_once    boolean类型,默认值=1 如果出现多个错误,控制超全局变量信息是在所有错误中显示,还是只在第一个错误中显示。
    xdebug.dump_undefined   boolean类型,默认值=1 控制是否显示超全局变量中未定义的值。
    xdebug.extended_info    integer类型,默认值=1 是否强制进入PHP解析器的"extended_info"模式,这将允许Xdebug以远程调试器对文件或行添加断点。开启此模式将拖慢脚本的允许速度,该参数只能在php.ini中设置。
    xdebug.file_link_format string类型,默认值=,  自2.2版本开始引入。用于指定堆栈信息中用到的文件名称的链接样式,这允许IDE通过设置链接协议,直接点击堆栈信息中的文件名称,即可快速打开指定的文件。例如:ZendStudio://%f@%l(%f表示文件路径,%f表示行号)。
    xdebug.force_display_errors integer类型,默认值=0 自2.3版本开始引入。是否强制显示错误信息。
    xdebug.force_error_reporting    integer类型,默认值=0 自2.3版本开始引入。是否强制显示所有错误级别的信息。
    xdebug.halt_level   integer类型,默认值=0 自2.3版本开始引入。指定出现那些错误级别的错误时,中止程序运行。例如:xdebug.halt_level=E_WARNING|E_NOTICE|E_USER_WARNING|E_USER_NOTICE(也仅支持上述4种错误级别)。
    xdebug.idekey   string类型,默认值=*complex*  指定传递给DBGp调试器处理程序的IDE Key。
    xdebug.manual_url   string类型,默认值=http://www.php.net 仅2.2.1以下版本可用,用于指定从函数堆栈和错误信息链接到的帮助手册的基本URL。
    xdebug.max_nesting_level    integer类型,默认值=100   指定递归的嵌套层级数。
    xdebug.overload_var_dump    boolean类型,默认值=1 自2.2版本开始引入,当php.ini中的html_error设为1时,Xdebug是否默认使用自身的改进版本来重载var_dump()。
    xdebug.profiler_append  integer类型,默认值=0 当多个请求映射到相同文件时,指定是覆盖之前的调试信息文件还是追加内容到该文件中。
    xdebug.profiler_enable  integer类型,默认值=0 指定是否启用Xdebug的性能分析,并创建性能信息文件。
    xdebug.profiler_output_dir  string类型,默认值=/tmp   指定性能分析信息文件的输出目录
    xdebug.profiler_output_name string类型,默认值=cachegrind.out.%p  指定性能分析信息文件的名称
    xdebug.remote_enable    boolean类型,默认值=0 是否开启远程调试
    xdebug.remote_handler   string类型,默认值=dbgp   指定远程调试的处理协议
    xdebug.remote_host  string类型,默认值=localhost  指定远程调试的主机名
    xdebug.remote_log   string类型,默认值=   指定远程调试的日志文件名
    xdebug.remote_mode  string类型,默认值=req    可以设为req或jit,req表示脚本一开始运行就连接远程客户端,jit表示脚本出错时才连接远程客户端。
    xdebug.remote_port  integer类型,默认值=9000  指定远程调试的端口号
    xdebug.trace_options    integer类型,默认值=0 指定对于之后的请求,追踪文件是追加内容还是覆盖之前内容。
    xdebug.trace_output_dir string类型,默认值=/tmp   指定追踪文件的存放目录
    xdebug.trace_output_name    string类型,默认值=trace.%c   指定追踪文件的名称
    

    PhpStorm 配置

    先配置一个 server(一定要配置目录映射),打开
    菜单栏->File->Settings->Languages & Frameworks->PHP->Server
    开启一下 9000 端口 的监听,打开
    菜单栏->Run->Start Listing for PHP Debug Connections

    调试

    chrome 可以安装一个插件xdebug helper,需要调试的页面打开debug,会设置一个cookie来自动加调试标识。
    在代码中打断点,访问浏览器,phpstorm 就会跳出一个调试栏,显示调试信息。

  • laravel 订单通知队列

    需求

    订单完成支付后通知服务器已到账,通知失败则重试,最多3次,第二次5秒后,第三次10秒后

    实现方案一

    生成任务类

    php artisan make:job PaymentNotify
    

    命令将会在app/Jobs目录下生成一个新的类,编辑:

    < ?php
    
    namespace App\Jobs;
    
    use App\Exceptions\SignException;
    use Illuminate\Bus\Queueable;
    use Illuminate\Queue\SerializesModels;
    use Illuminate\Queue\InteractsWithQueue;
    use Illuminate\Contracts\Queue\ShouldQueue;
    
    class PaymentNotify implements ShouldQueue
    {
        use InteractsWithQueue, Queueable, SerializesModels;
    
        protected $data;
    
        /**
         * Create a new job instance.
         *
         * @param $data
         */
    
        public function __construct($data)
        {
            $this->data = $data;
        }
    
        /**
         * Execute the job.
         *
         * @throws \Exception
         */
        public function handle()
        {
            if ($this->attempts() == 1) {
                //处理订单通知
                info('notify'.$this->attempts());
                if(true){//测试强制通知失败
                    $this->release(5);//手动释放任务回队列,带延时执行时间
                }
            }
    
            if ($this->attempts() == 2) {
                //处理订单通知
                info('notify'.$this->attempts());
                if(true){
                    $this->release(10);
                }
            }
    
            if ($this->attempts() == 3) {
                //处理订单通知
                info('notify'.$this->attempts());
                if(true){
                    throw new SignException('fails');//抛出异常,任务失败,自动入库
                }
    
            }
        }
    
    }
    

    添加测试路由并访问:

    Route::get('/pay/notify', function(){
        dispatch((new \App\Jobs\PaymentNotify(['order'=>time().mt_rand(1000,9999)]))->onQueue('PaymentNotify'));
        return 'ok';
    });
    

    监听队列

    php artisan queue:work  --queue=PaymentNotify --tries=3  --sleep=0
    

    可查看使用说明

    $php artisan queue:work --help
    Usage:
      queue:work [options] [--] [<connection>]
    
    Arguments:
      connection               The name of connection
    
    Options:
          --queue[=QUEUE]      The queue to listen on
          --daemon             Run the worker in daemon mode (Deprecated)
          --once               Only process the next job on the queue
          --delay[=DELAY]      Amount of time to delay failed jobs [default: "0"]
          --force              Force the worker to run even in maintenance mode
          --memory[=MEMORY]    The memory limit in megabytes [default: "128"]
          --sleep[=SLEEP]      Number of seconds to sleep when no job is available [default: "3"]
          --timeout[=TIMEOUT]  The number of seconds a child process can run [default: "60"]
          --tries[=TRIES]      Number of times to attempt a job before logging it failed [default: "0"]
      -h, --help               Display this help message
      -q, --quiet              Do not output any message
      -V, --version            Display this application version
          --ansi               Force ANSI output
          --no-ansi            Disable ANSI output
      -n, --no-interaction     Do not ask any interactive question
          --env[=ENV]          The environment the command should run under
      -v|vv|vvv, --verbose     Increase the verbosity of messages: 1 for normal output, 2 for more verbose output and 3 for debug
    

    命令意思是:使用默认队列连接的PaymentNotify队列,失败最多尝试执行3次,轮询新任务之前的等待时间为0(即有任务随时检测执行,不会在无任务时默认等待3秒)

    任务执行情况

    [2017-03-07 16:22:44] local.INFO: notify1  
    [2017-03-07 16:22:49] local.INFO: notify2  
    [2017-03-07 16:22:59] local.INFO: notify3
    

    符合预期

    方案二

    另一种思路是订单完成后直接生成3个队列,

    $order = time().mt_rand(1000,9999);
    
    dispatch((new \App\Jobs\PaymentNotify(['order'=>$order]))->onQueue('PaymentNotify'));
    
    dispatch((new \App\Jobs\PaymentNotify(['order'=>$order]))->onQueue('PaymentNotify')->delay(5));
    
    dispatch((new \App\Jobs\PaymentNotify(['order'=>$order]))->onQueue('PaymentNotify')->delay(10));
    

    后两个队列带延时执行参数,设置最多执行次数为1,执行队列时检测已经通知成功则跳过执行

    总结

    方案一:在处理队列时需判断重试次数,分类处理,业务需求变化需改动多;
    方案二:在redis中相同队列存在多条,业务需求变化处理逻辑相对改动少;
    各有利弊。

  • laravel自定义用户认证

    添加自定义的 Guard

    需要通过Auth门面的extend方法定义自己的认证guard,在App\Providers\AuthServiceProvider的boot方法中实现:

    public function boot()
    {
        $this->registerPolicies();
    
        Auth::extend('XXX', function($app, $name, array $config) {
            // 返回 Illuminate\Contracts\Auth\Guard 实例
            $guard = new XXXGuard($name,Auth::createUserProvider($config['provider']),$this->app['session.store']);
            //事件
            if (method_exists($guard, 'setDispatcher')) {
                $guard->setDispatcher($this->app['events']);
            }
            //请求
            if (method_exists($guard, 'setRequest')) {
                $guard->setRequest($this->app->refresh('request', $guard, 'setRequest'));
            }
            return $guard;
        });
    }
    

    (更多…)

  • 抽奖概率算法

    $reward = array(
    0=>array(‘id’=>0,’area’=>35,’prize’=>’金币5000’),
    1=>array(‘id’=>1,’area’=>30,’prize’=>’蓝钻1天’),
    2=>array(‘id’=>2,’area’=>15,’prize’=>’金币20000’),
    3=>array(‘id’=>3,’area’=>8,’prize’=>’蓝钻3天’),
    4=>array(‘id’=>4,’area’=>6,’prize’=>’金币30000’),
    5=>array(‘id’=>5,’area’=>5,’prize’=>’紫钻2天’),
    6=>array(‘id’=>6,’area’=>1,’prize’=>’获取抽奖机会一次’)
    );
    //area为中奖概率
    $r = mt_rand(1, 100); //100为area之和
    $b = 0;
    foreach ($reward as $k => $v) {
    $b += $v[‘area’];
    if ($r <= $b) {
    $award = $v;
    break;
    }
    }
  • mcrypt_encrypt() Size of key is too large for this algorithm in …

    function encrypt($str, $key) {

    $block = mcrypt_get_block_size(‘des’, ‘ecb’);

    $pad = $block – (strlen($str) % $block);

    $str .= str_repeat(chr($pad), $pad);

    $tt = mcrypt_encrypt(MCRYPT_DES,$key, $str,MCRYPT_MODE_ECB);

    return base64_encode($tt);

    }

    function decrypt($str, $key) {

    $str = base64_decode($str);

    $str = mcrypt_decrypt(MCRYPT_DES, $key, $str, MCRYPT_MODE_ECB); 

    $block = mcrypt_get_block_size(‘des’, ‘ecb’);

    $pad = ord($str[($len = strlen($str)) – 1]);

    return substr($str, 0, strlen($str) – $pad);

    }
    在使用以上DES加密解密过程中报了错,经查原来是PHP版本中$key的长度不能超过8位,之前的KEY随机生成了14位。

  • 在 Ubuntu 中安装 Laravel

    安装 Composer

    curl -sS https://getcomposer.org/installer | php 

    下载完成之后将它放在系统 PATH 中,供全局调用。

    mv composer.phar /usr/local/bin/composer

    安装 Larave

    composer create-project laravel/laravel /home/wwwroot/laravel/

    执行下面的命令来授予 storage 文件夹的读写操作权限

    chgrp -R www /home/wwwroot/laravel/

    chmod -R 775 /home/wwwroot/laravel/app/storage

    建立网站域名指向/home/wwwroot/laravel/public(略)

    输入域名若提示:

    open_basedir restriction in effect. File() is not within the allowed path(s):XXX

    open_basedir可将用户访问文件的活动范围限制在指定的区域,通常是当前网站根目录的路径。

    在Apache的httpd.conf中VirtualHost的相应设置方法: 

    php_admin_value open_basedir “/home/wwwroot/laravel:/home/wwwroot/laravel/public:/tmp/:/var/tmp/:/proc/“

    重启apache解决

  • Laravel requires the Mcrypt PHP extension.

    I am trying to use the migrate function in Laravel 4 on OSX, however I am getting the error

    ​Laravel requires the Mcrypt PHP extension.

    What is wrong, and how can I fix it?

    (更多…)

  • mac下安装oci8扩展来支持php连接oracle

    下载oci8扩展包:
    http://pecl.php.net/package/oci8

    cd到解压目录执行phpize,发现错误,

    错误提示:
    Cannot find autoconf. Please check your autoconf installation and the
    $PHP_AUTOCONF environment variable. Then, rerun this script.

    解决方法:
    curl -OL http://ftpmirror.gnu.org/autoconf/autoconf-latest.tar.gztar xzf autoconf-latest.tar.gz
    cd autoconf-*./configure –prefix=/usr/local
    make
    sudo make install

    (更多…)

  • 通过传递session_id解决uploadify上传PHP中丢失SESSION值的问题

     

    由于uploadify通过flash提交上传的文件,会导致在接收文件的PHP端出现获取SESSION值为空的情况,如果你要在PHP端验证用户的登录状态或验证用户输入的验证码是否正确你就需要解决SESSION取值的问题,下面给出uploadify上传中获取SESSION的方法:

    (更多…)

  • PHP中一个导出到excel的函数

    /**

    * 导出数据为excel表格

    *@param $data    一个二维数组,结构如同从数据库查出来的数组

    *@param $title   excel的第一行标题,一个数组,如果为空则没有标题

    *@param $filename 下载的文件名

    *@examlpe

    $stu = M (‘User’);

    $arr = $stu -> select();

    exportexcel($arr,array(‘id’,’账户’,’密码’,’昵称’),’文件名!’);

    */

    function exportexcel($data=array(),$title=array(),$filename=’report’){

    header(“Content-type:application/octet-stream”);

    header(“Accept-Ranges:bytes”);

    header(“Content-type:application/vnd.ms-excel”);

    header(“Content-Disposition:attachment;filename=”.$filename.”.xls”);

    header(“Pragma: no-cache”);

    header(“Expires: 0”);

    //导出xls 开始

    if (!empty($title)){

    foreach ($title as $k => $v) {

    $title[$k]=iconv(“UTF-8”, “GB2312”,$v);

    }

    $title= implode(“\t”, $title);

    echo “$title\n”;

    }

    if (!empty($data)){

    foreach($data as $key=>$val){

    foreach ($val as $ck => $cv) {

    $data[$key][$ck]=iconv(“UTF-8”, “GB2312”, $cv);

    }

    $data[$key]=implode(“\t”, $data[$key]);

    }

    echo implode(“\n”,$data);

    }

    }