zoukankan      html  css  js  c++  java
  • 关于PHP程序员技术职业生涯规划

    看到很多PHP程序员职业规划的文章,都是直接上来就提Linux、PHP、MySQL、Nginx、Redis、Memcache、jQuery这些,然后就直接上手搭环境、做项目,中级就是学习各种PHP框架和类库,高级阶段就是MySQL优化、PHP内核与扩展、架构设计这些了。

    这些文章都存在一个严重的缺陷,不重视基础。就好比练武功,只求速成,不修炼内功和心法,只练各种招式,这样能高到哪里去?我所见过的PHP大牛每一个都是具备非常扎实的基础,他们之所以能成为大牛,是因为基础足够好。基础不稳,面对技术复杂的系统,如同盲人摸象、管中窥豹,只得其门不得其法。而且如果基础不扎实,也没办法进入大公司。国外的Google、Facebook,国内的腾讯、阿里、百度、滴滴、京东、新浪等知名互联网企业,无论哪一家公司面试必然会考验应聘者的技术功底。无法进入一个拥有大规模并发请求的项目中得到历练,不坚持提升自己,那也只能在小公司混日子了。

    我最开始工作也是在2家小公司,后来加入腾讯阿里,主要原因还是我坚持学习基础知识,从而得倒了这个机会。有几个方面的基础知识,我建议每一位PHP程序员都应该好好学习一下。我推荐几本书给大家,包括深入理解计算机系统、现代操作系统、C程序设计语言、C语言数据结构和算法、Unix环境高级编程、TCP/IP网络通信详解。另外我建议大家学习一下面向对象方面知识,PHP这方面的书不太多,建议看Java面向对象编程、Java编程思想、J2EE这些书。PHP语言基础方面,建议认真地把PHP5权威编程这本书好好读完。另外不光要读,还要照着书中的讲解动手去编程实践。

    总之有一个好的基础,再去学LAMP、Redis、PHP框架、前端,这样取得的成就更大。这与年龄无关、与学历无关、与智力无关,与天赋也无关。只要肯努力学习,人人可以成为技术大牛。

     
    分类:PHP人生编程标签:

    PHP+Swoole的闭包写法

    2016年10月25日评论已被关闭
     

    JS程序员总是嘲笑PHP没有闭包,今天抽空写一篇文章来专门介绍一下PHP的闭包。从5.3版本开始PHP就增加了匿名函数支持,经过数个版本迭代到现在的PHP5.6、PHP7,PHP语言的闭包已经非常完善了。再结合Swoole提供的事件驱动支持,PHP的闭包功能非常强大而且很优雅。

    匿名函数

    匿名函数是闭包的核心,匿名函数在PHP里实际上是一个Closure类的对象(请注意是对象)。与普通的面向对象编程方式不同,匿名函数的代码是直接写在调用处的,不需要额外写一个类,编写方法的代码。这样的好处就是更直接。下面的示例是设置一个定时器,每2秒输出hello world。

    传统写法

    function timer () {
        echo "hello world";
    }
    SwooleTimer::tick(2000, 'timer');
    

    闭包写法

    SwooleTimer::tick(2000, function () {
        echo "hello world";
    });
    

    非闭包的传统写法,先要声明一个函数,再转入函数名称字符串。两段代码是分离的,不够直观。而闭包的写法把定时器的声明和定时器要执行的代码写在了一起,逻辑非常清晰直观。使用闭包语法可以很方便编写回调函数。在事件驱动编程、排序、array_walk等需要用户传入一段执行代码的场景中,闭包的写法非常优雅。

    闭包更强大的地方在于它可以直接在调用处引入外部变量。PHP中实现的方法就是use关键词。

    Use语法

    如果刚才的定时器需要传入一个变量,传统的写法只能通过全局变量来实现。与JS不同,PHP的变量引入是显式的,如果要引用外部变量必须使用use来声明。而JS是隐式的,匿名函数内部可以随意操作外部变量,无需声明。这样好处是少写了一点代码,缺点是存在风险和混乱。

    传统写法

    $str = "hello world";
    function timer () {
        global $str;
        echo $str;
    }
    SwooleTimer::tick(2000, 'timer');
    

    闭包写法

    $str = "hello world";
    SwooleTimer::tick(2000, function () use ($str) {
        echo $str;
    });
    

    闭包写法使用use直接引入了当前的$str变量,而不需要使用global全局变量。另外如果是在swoole的事件驱动编程模式,使用global就无法实现异步并发了,因为global全局变量只有1个,如果同时有多个客户端请求,每个请求要查询数据库,输出不同的内容,传统的编程方法就不太容易实现,需要使用全局变量数组,以客户端的ID为KEY保存各自的数据。

    传统写法

    $requestArray = array();
    $dbResultArray = array();
    
    function my_request($request, $response) {
        global $dbResultArray, $requestArray;
        $queryId = $db->query($sql, 'get_result');
        $requestArray[$request->fd] = array($request, $response);
        $dbResultArray[$queryId] = $request->fd;
    }
    
    function get_result($queryId, $queryResult) {
        global $dbResultArray, $requestArray;
        list($request, $response) = $requestArray[$dbResultArray[$queryId]];
        $response->end($queryResult);
    }
    
    $server->on('request', 'my_request');
    

    闭包写法

    $server->on('request', function ($request, $response) {
        $queryId = $db->query($sql, function ($queryId, $queryResult) use ($request, $response) {
            $response->end($queryResult);
        });
    });
    

    传统的写法非常复杂,需要反复多次从全局数组保存/提取数据。而闭包的写法非常简洁优雅,只用了几行代码就实现了同样的功能。闭包写法非常适合用来编写异步非阻塞回调模式的服务器程序。目前热门的编程语言中只有PHP和JS具备这种能力。

    闭包更多特性

    在类的方法中使用匿名函数,5.4以上的版本无需使用use引入$this,直接可以在匿名函数中使用$this来调用当前对象的方法。在swoole编程中,可以利用此特性减少$serv对象的use引入传递。

    class Server extends SwooleServer {
        function onReceive($serv, $fd, $reactorId, $data) {
            $db->query($sql, function ($queryId, $queryResult) use ($fd) {
                $this->send($fd, $queryResult);
            }
        }
    }
    

    另外如果希望在闭包函数中修改外部变量,可以在use时为变量增加&引用符号即可。注意对象类型不需要加&,因为在PHP中对象默认就是传引用而非传值。

     
    分类:Node.jsPHPSwoole扩展标签:

    基于swoole的task功能实现程序内的map-reduce

    2016年3月31日评论已被关闭
     

    Swoole扩展自带的Task进程功能非常强大,可以用来实现各种复杂的业务逻辑。本文主要介绍使用task/finish功能实现程序内的Map-Reduce并发任务处理。一个聊天服务经常会有群聊需求,我的群组和群组内成员,另外群组内成员需要按照积分排序,类似与这样的功能就可以使用Swoole简单实现。

    传统多线程方案

    创建2个全局变量Map,group_map以group_id为Key,存储成员set。user_map以uid为Key存储当前用户加入的所有group。

    多线程环境下实际上不能直接操作这2个Map,必须要加锁。当添加用户到一个组或者用户退出一个组时需要操作这2个map,必须要加锁。如果操作很频繁,实际上锁的碰撞是很严重的,这部分操作就会变成串行的。同时只有一个线程可以对map进行操作。锁的争抢也会带来大量线程切换浪费很多CPU资源。

    lock.lock();
    group_map[group_id].append([uid, score]);
    user_map[uid].append(group_id);
    group_map.sortByScore();
    lock.unlock();
    

    基于Swoole的Task功能

    基于Swoole的Task功能,可以将任务切片,然后hash投递到不同的Task进程,完成任务。排序功能可以直接使用PHP提供的SplHeap实现,时间复杂度为O(logn),如果要实现查询功能,如根据UID查询用户加入的所有群组,根据GroupId查询有哪些成员。可以先计算Hash找到对应Task进程,然后通过task/taskwait发送指令,直接读取进程的变量查找到信息。

    $serv->set(array("task_worker_num" => 24));
    
    $serv->task(array("cmd" => "user", "uid" => $uid, "gid" => $gid, "score" => $score), $gid % $task_worker_num);
    $serv->task(array("cmd" => "group", "uid" => $uid, "gid" => $gid), $uid % $task_worker_num);
    
    class MyMaxHeap extends SplHeap
    {
        public function compare($value1, $value2)
        {
            return ($value1['score'] - $value2['score']);
        }
    }
    
    function onTask($serv, $taskId, $srcWorkerId, $data) {
        static $userMap = array();
        static $groupMap = array();
        
        if ($data['cmd'] == 'group')
        {
            if (!isset($groupMap[$data['gid']]))
            {
                $groupMap[$data['gid']] = new MyMaxHeap();
            }
            $heap = $groupMap[$data['gid']];
            $heap->insert(array("uid" => $data['uid'], "score" => $data['score']));
        }
        elseif ($data['cmd'] == 'user')
        {
            $userMap[$data['uid']][] = $data['gid'];
        }
    }
    

    由于Task进程只有数组操作,所以是非阻塞的,只需要开启与CPU核数相同的进程数量即可。进程间无任何加锁争抢,性能非常好。Swoole的Task进程通信使用UnixSocket,是内核提供的全内存通信方式无任何IO,一写一读单进程可达100万/秒。虽然没有直接读变量的速度快,但性能也足够了。

  • 相关阅读:
    MySQL优化查询语句Explain
    Spring Kafka(二)操作Topic以及Kafka Tool 2的使用
    集群作业管理OpenPBS与OpenPBS Interface Library
    OpenPBS 脚本样本 非常值得参考
    mpirun 与 PBS 问题
    MPIRUN 与 PBS 脚本
    什么是IA架构服务器
    集群的管理pbs
    软件包 javax.servlet 不存在
    An introduction to PORTABLE BATCH SYSTEM
  • 原文地址:https://www.cnblogs.com/xiangshihua/p/13292520.html
Copyright © 2011-2022 走看看