Nginx系列二:负载均衡与反向代理

      

一、负载均衡

 

       1、什么是负载均衡

随着请求数的快速增长,单服务器已经无法承担大量用户的并发访问,这个时候,就需要建立服务器集群,来让多台服务器协同工作,提高整体项目的吞吐量和QPS。假设一台设备资源占有率已经饱和,而另一台服务器缺只有零星几个请求处理,这显然是不合理的。我们期望多台服务器需要平均承担客户端发来的请求,使每台机器都充分利用。这时,我们就需要用到一种技术,叫做负载均衡。

负载均衡是多台服务器组合为一个集群,其中每台机器可以单独运行,服务器之间地位相同,通过负载均衡技术,将客户端发来的请求平均分配到每台服务器中,使得项目的负载能够均衡的分布,每台服务器的资源使用量也基本相同。直白的说,通过负载均衡技术,使得项目巨额访问量从单台服务器均匀的分摊到每台服务器上。这种思想,也是高性能网站结构的核心思想:分!

 

2、负载均衡之DNS轮询

我们对同一个域名添加多个A记录解析。DNS服务器会将请求随机分配到其中一条记录中。DNS轮询是否支持需要询问域名注册商。比如,我对博客www.lanecn.com添加了一个二级域名nginx.lanecn.com。在万网的操作后台,对这个域名添加了4A记录解析,分别为10.10.10.110.10.10.210.10.10.310.10.10.4。如果有请求来访问nginx.lanecn.com,域名会随机被解析到10.10.10.1 – 10.10.10.4中的其中一台。这就是用DNS轮询的方式来做负载均衡。

无标题.png

 

优点:DNS轮询的成本非常低。

缺点:1)分配不均匀。首先本地浏览器缓存问题会使下次请求根本不会去查询域名解析的IP,其次最简单的随机分配算法,不能根据服务器的性能差异来为某台高配机器多分,低配机器少分。这使得低配机器已经高负荷运行了,而高配机器只有10%的资源消耗。2)不可靠。DNS轮询没有心跳检测,假设其中一台服务器宕机,域名供应商是不可知的。其次DNS生效时间过长,而且各级都有缓存。

应用:在可靠性要求不高的服务器集群可以使用DNS轮询

 

3、负载均衡之四\七层负载均衡设备

       在可靠性要求高的应用场景,硬件层面通常采用四\七层负载均衡设备。鼎鼎大名的就是F5负载均衡设备。

       在网络中,有一个叫做开放系统互联模型,也就是OSI七层模型。一到四层和数据传输相关,五到七层是和应用程序数据相关,数据一层层的传输。从第一层到第七层名称依次为:物理层,数据链路层,网络层,传输层,会话层,表示层,应用层。其中,我们熟知的IP协议位于第三层网络层,TCP协议位于第四层传输层,HTTP/FTP等协议位于最高的一层第七层应用层。

       常见的负载均衡交换机有F5 BIG-IPCisco CSS等,价格高昂。这些设备通常应用在第四层或者第七层。因此,被成为四\七层负载均衡设备。

 

4、负载均衡之软件层面

1)应用在第四层的负载均衡软件的代表是Linux Virtual Server,简称为LVSLVS根据IP和内容进行请求奋发。并且可以自动识别并屏蔽故障服务器,使得服务器集群更加高可用。

2)、应用在第七层的负载均衡软件的代表是基于HTTP反向代理的NginxHA Proxy等。Nginx的反向代理支持虚拟主机,可配置。能够按照轮询、IPURL等方式进行负载均衡,和LVS一样,也支持故障检查。

 

二、反向代理

       1、什么是反向代理

我们先来看看什么叫代理:客户端使用代理服务器,来访问一些在国家政策不允许访问的国外网站。基本流程是:客户端发送请求给代理服务器去,代理服务器去访问目标网站,然后代理服务器将目标网站返回的结果发送给客户端。

现在,我们来看看什么叫反向代理:网站内部使用代理服务器,来中转客户端发来的请求。基本流程是:客户端发送请求给目标网站,目标网站先经过反向代理服务器,然后将请求转发给内部网络的其他服务器来处理,最后将数据返回。

通俗的讲:代理,就是客户端使用的,对目标网站来说,代理服务器就是一个客户。而反向代理,就是目标网站使用的,对于用户来讲,反向代理服务器就是一个网站服务器。

定义:反向代理就是客户端发送请求给代理服务器,代理服务器再转发给内部的Web服务器集群,然后将Web服务器集群返回的数据发送给客户端。在客户端看来,代理服务器的表现就是一个真正的Web服务器。

反向代理服务器并不能处理静态网页或CGI程序,也不保存任何数据。网站数据都是保存在内部网络的Web服务器中。因此,如果暴露在公网的反向代理服务器被攻击,也不会使网站数据遭到破坏,这就增强了Web服务器的安全性。

 

三、Nginx的负载均衡和反向代理

       Nginx的反向代理通过设置Upstream指令。Upstream指令包含一组目标服务器IP、权重等信息。在Nginx的配置文件中server{}模块中的proxy_passfastcgi_pass指令来使用Upstream定义的一组服务器集群。

       值得一提的是,在PHP中,我们可以通过超全局变量$_SERVER[“REMOTE_ADDR”]来获取客户端的IP地址。但是,在反向代理环境中,我们的PHP程序获取到的时候反向代理服务器IP,而非真实客户端的IP。这时,通过在配置文件中添加proxy_set_header X-Forwarded-For $remote_addr; 可以使代理服务器收到的客户端IP传递给后端Web服务器。

       我们来看看如何配置Nginx的反向代理配置:

 

user www www;

worker_processes 4;

error_log /usr/local/nginx/logs/nginx_error.log crit;

pid  /usr/local/nginx/logs/nginx.pid;

worker_rlimit_nofile 51200;

events

{

    use epoll;

    worker_connections 51200;

}

 

http

{

    include mime.types;

    default_type application/octet-stream;

    charset UTF-8;

    server_names_hash_bucket_size 128;

    client_header_buffer_size 32k;

    large_client_header_buffers 4 32k;

 

    sendfile on;

    tcp_nopush on;

    keepalive_timeout 60;

    tcp_nodelay on;

 

    #允许客户端请求的最大字节

    client_max_body_size 50m;

    #缓冲区最大字节

    client_body_buffer_size 256k;

    #代理服务器链接后端服务器的超时时间

    proxy_connect_timeout 30;

    #代理服务器等待后端服务器响应的超时时间

    proxy_read_timeout 60;

    #后端服务器返回数据给代理服务器的最大传输时间

    proxy_send_timeout 30;

    #代理服务器缓冲区大小,客户端的头信息会保存在这里

    proxy_buffer_size 64k;

    #代理服务器有几个缓冲区,最大是多大

    proxy_buffers 4 64k;

    #代理服务器烦方式可以申请更大的缓冲区,Nginx官方推荐为*2即可

    proxy_busy_buffers_size 128k;

    #代理服务器临时文件大小

    proxy_temp_file_write_size 256k;

 

    #设置服务器集群池,每台服务器的权重等信息,www_server_pool是这一组服务器的名字,可以自行修改.后续将配置www_server_pool服务器集群用来访问www.lanecn.com

    upstream www_server_pool

    {

        server 192.168.1.100:80 weight=1 max_fails=2 fail_timeout=30s;

        server 192.168.1.101:80 weight=2 max_fails=2 fail_timeout=30s;

        server 192.168.1.102:80 weight=1 max_fails=2 fail_timeout=30s;

    }

 

    #设置服务器集群池,每台服务器的权重等信息,lanewechat_server_pool是这一组服务器的名字,可以自行修改.后续将配置lanewechat_server_pool服务器集群用来访问lanewechat.lanecn.com

    upstream lanewechat_server_pool

    {

        server 192.168.1.110:80 weight=3 max_fails=2 fail_timeout=30s;

        server 192.168.1.111:80 weight=1 max_fails=2 fail_timeout=30s;

        server 192.168.1.112:80 weight=2 max_fails=2 fail_timeout=30s;

    }

 

    #设置第一个虚拟主机,域名为博客主站www.lanecn.com.使用www_server_pool这一组服务器集群.

    server

    {

        listen 80;

        server_name www.lanecn.com;

        access_log /usr/local/nginx/logs/www.lanecn.com_access.log;

 

        location /

        {

            #如果www_server_pool这个服务器集群中的某台服务器返回超时或者502等错误,则自动转发到集群中的其他服务器.

            proxy_next_upstream error timeout invalid_header http_500 http_502 http_503;

            proxy_pass http://www.lanecn.com;

            proxy_max_temp_file_size 128m;

            proxy_set_header X-Forwarded-For $remote_addr;

            proxy_set_header Host www.lanecn.com;

        }

    }

 

    #设置第二个虚拟主机,域名为LaneWeChat项目站lanewechat.lanecn.com.使用lanewechat_server_pool这一组服务器集群.

    server

    {

        listen 80;

        server_name lanewechat.lanecn.com;

        access_log /usr/local/nginx/logs/lanewechat.lanecn.com_access.log;

 

        location /

        {

            #如果lanewechat_server_pool这个服务器集群中的某台服务器返回超时或者502等错误,则自动转发到集群中的其他服务器.

            proxy_next_upstream error timeout invalid_header http_500 http_502 http_503;

            proxy_pass http://lanewechat.lanecn.com;

            proxy_max_temp_file_size 128m;

            proxy_set_header X-Forwarded-For $remote_addr;

            proxy_set_header Host lanewechat.lanecn.com;

        }

    }

}

Nginx系列一:信号与配置

一、Nginx与信号

Nginx支持平滑重启,相比于Apache,修改了配置文件后可以不需要先停止程序,再重新启动。

1、启动

nginx –c nginx.conf

其中,-c nginx.conf可以省略不写。如果省略,则默认加载安装目录下的conf子目录中的nginx.conf

2、停止

停止的方式有很多种,kill时传入不同的信号来结束或者平滑重启。Nginx的进程号记录在Pid文件中,Pid文件的位置可以在conf/nginx.conf中找到。如下图:

无标题.png

当然,也可以根据

ps –ef | grep nginx

来查找Nginx的进程号。我们可以通过kill命令来结束Nginx

       从容停止Nginx

kill – QUIT Nginx进程ID

kill – QUIT /usr/local/nginx/logs/nginx.pid

       快速停止Nginx

kill – TERM Nginx进程ID

kill – TERM /usr/local/nginx/logs/nginx.pid

             

kill – INT Nginx进程ID

kill – INT /usr/local/nginx/logs/nginx.pid

       强制停止Nginx

kill –9 Nginx进程ID

kill -9 /usr/local/nginx/logs/nginx.pid

pkill -9 nginx

 

       3、重启

       如果修改了Nginx的配置文件,想要重启Nginx。同样可以使用kill命令来传递信号。不过,在此之前,我强烈建议先检查并测试配置文件是否正确。

       测试配置文件:nginx –t –c conf/nginx.conf

       若提示unknow directive *** in conf/nginx.conf:55.  Configuration file conf/nginx.conf test failed,则证明在第55行的***是非法的,需要修改。

       若提示the configuration file conf/nginx.conf syntax is ok.  Configuration file conf/nginx.conf test is successful,则证明配置文件测试通过,可以重启Nginx了。

平滑重启Nginx

kill –HUP Nginx进程ID

kill – HUP /usr/local/nginx/logs/nginx.pid

       Nginx收到HUP信号的时候,首先会尝试解析配置文件,如果成功,则应用新的配置文件并完成重启。

       4Nginx升级

       Nginx可以平滑升级,在我们重新便利Nginx、添加或删除服务器模块等操作后,通过kill命令并传递USR2信号进行升级。在此之前,请备份旧的可执行文件。

       1)、新的版本仍旧会安装在旧目录当中。

       2)、旧Pid文件被重命名为.oldbin

       3)、执行新版本的Nginx,启动主进程和子进程。

       4)、此时新旧版本同时在运行,需要使用kill命令并发送WINCH信号给旧的进程ID,是它从容关闭。

       5Nginx的信号

              1)、TERMINT 快速关闭

              2)、QUIT从容关闭

              3)、HUP平滑重启,重新加载配置文件

              4)、USR1 重新打开日志文件

              5)、USR2 平滑升级可执行程序

              6)、WINCH 从容关闭工作进程

 

二、基本配置

       1、主配置文件

主配置文件默认位于Nginx安装目录下的conf/nginx.conf。下面我们将逐行解读Nginx 的主配置文件。

 

#使用的用户和用户组

user  www www;

 

#子进程个数,一般等于CPU的总核心数,4CPU则为4

worker_processes  4 ;

 

#一个nginx进程打开的最多文件描述符数目,理论值应该是最多打开文件数(ulimit -n)与nginx进程数相除,但是nginx分配请求并不是那么均匀,所以最好与ulimit -n 的值保持一致。现在在linux 2.6内核下开启文件打开数为65535worker_rlimit_nofile就相应应该填写65535

worker_rlimit_nofile 65535;

 

#错误日志的路径,可选级别为debug\info\notice\warn\error\crit

#error_log  logs/error.log;

#error_log  logs/error.log  notice;

error_log  logs/error.log  crit;

 

#pid文件的路径,该文件中记录了当前正在运行的Nginx的主进程ID

pid        logs/nginx.pid;

 

#事件相关

events {

    #使用的网络IO模型,Linux推荐epoll, FreeBSD推荐kqueue. Apacheselect是非常低效的

use epoll;

 

    #允许的连接数

    worker_connections  10240;

}

 

#http相关

http {

    #文件扩展名与文件类型映射表

include       mime.types;

 

    #默认文件类型

default_type  application/octet-stream;

 

    #日志格式 ip - 用户 [时间] 请求 状态 发送的字节

    log_format    main '$remote_addr - $remote_user [$time_iso8601] "$request" '

                        '$status $body_bytes_sent ';

 

    #接受请求的日志路径

access_log  logs/$server_name.log main;

 

    #客户端所发请求的最大值

client_max_body_size 128m;

 

    #开启目录列表访问,合适下载服务器,默认关闭

autoindex off;

 

    #开启高效文件传输模式,是否调用sendfile函数来输出文件,通常为on,如果用来下载等磁盘IO重负载应用,可设置为off,以平衡磁盘与网络I/O处理速度,降低系统的负载。注意:如果图片显示不正常把这个改成off

sendfile        on;

 

    #防止网络阻塞

tcp_nopush     on;

 

    #长连接超时时间,单位是秒

keepalive_timeout  65;

 

    #是否开启Gzip压缩

gzip  on;

 

    #引入其他的配置文件.我配置了多个虚拟主机,每个应用一个配置文件,所以这里需要引入多个配置文件

include vhost/vhost-*.conf;

 

    #服务相关

    server {

       #监听80端口,

listen 80 default backlog=20480;

 

       #域名,多个用空格隔开

      server_name  localhost;

      location /nginx_status {

            allow all;

            stub_status on;

            access_log off;

      }

    }

}

 

2、配置虚拟主机

      利用虚拟主机技术,可以不用为每个应用单独的提供一组Nginx进程。在同一台服务器,同一组Nginx进程,可以运行多个网站。

我在Nginx的安装目录下的conf子目录下新建了一个目录为vhost(在这个目录下放置所有的虚拟主机配置文件),在vhost目录中新建了一个文件,名为vhost-blog.conf。在Nginx的主配置文件conf/nginx.conf中的http模块下添加了一行include vhost/vhost-*.conf; 那么在Nginx启动时加载conf/nginx.conf,同时也会加载conf/vhost/目录下的所有配置文件。

下面,我们来看看blog这个虚拟主机的配置文件vhost-blog.conf吧:

 

#server模块,每个server都是一个虚拟主机,建议一个配置文件只放一个

server {

    #监听80端口

     listen       80;

     #域名,多个用空格分割

     server_name  www.lanecn.com;

     #location模块

     location / {

         #代码根目录

            root   /var/www/blog;

            #首页文件

            index  index.html index.php;

            #REWRITE规则

            if (!-e $request_filename) {

               rewrite  ^(.*)$  /index.php?s=$1  last;

               break;

            }

     }

 

    #解析PHP,使用FastCGI的方式.

    location ~ .php

    {

        set $path_info "";

        set $real_script_name $fastcgi_script_name;

    if ($fastcgi_script_name ~ "^(.+?\.php)(/.+)$") {

                    set $real_script_name $1;

                    set $path_info $2;

        }

 

        fastcgi_param SCRIPT_FILENAME $document_root$real_script_name;

        fastcgi_param SCRIPT_NAME $real_script_name;

        fastcgi_param PATH_INFO $path_info;

 

        root /var/www/blog;

        fastcgi_pass   127.0.0.1:9000;

        fastcgi_index  index.php;

        #FastCGI的配置文件.默认和nginx同目录(conf/目录下)

        include        fastcgi.conf;

    }

}

 

三、Nginx压缩输出

       GzipNginx压缩输出的模块。GzipGun ZIP,使用压缩技术,经过Gzip的压缩可以使得页面只有原来30%甚至更小。减少传输带宽。在服务器,Nginx调用Gzip模块进行压缩,然后消耗网络资源,将压缩数据发送给客户端的浏览器。客户端的浏览器进行解压,最终显示出来。

       NginxGzip配置在配置文件的http{}中。示例如下:

http{

#gzip模块设置

#开启gzip压缩输出

gzip on;

#最小压缩文件大小

gzip_min_length 1k;

#压缩缓冲区

gzip_buffers 4 16k;

#压缩版本(默认1.1,前端如果是squid2.5请使用1.0

gzip_http_version 1.0;

#压缩等级

gzip_comp_level 2;

#压缩类型,默认就已经包含text/html,所以下面就不用再写

gzip_types text/plain application/x-javascript text/css application/xml;

gzip_vary on;

}

 


四、Nginx缓存设置

       设置Nginx缓存,可以让Nginx告诉浏览器,本次响应的内容请保存起来,下次直接给用户看,别再来烦我了。

       缓存的方式有效的节省了服务器请求次数、带宽消耗。

       浏览器缓存可以通过expires指令输出Header头来实现。

       语法:expires [time|epoch|max|off]

       默认值:expires off

       在配置文件中,可以卸载http{}server{}location{}中。

       示例:

#图片缓存时间设置,缓存时间为30

location ~ .*\.(gif|jpg|jpeg|png|bmp|swf)$

{

      expires 30d;

}

#JSCSS缓存时间设置,缓存时间为1小时

location ~ .*\.(js|css)?$

{

      expires 1h;

}

PHP7

2015.12.3发生了两件大事,PHP7问世了,Swift开源了。

最好的语言发布了新的版本,一个划时代的大版本:PHP7

PHP7修复了大量BUG,新增了功能和语法糖。这些改动涉及到了核心包、GD库、PDOZIPZLIB等熟悉和不熟悉的核心功能与扩展包。

PHP7移除了已经被废弃的函数,如mysql_系列函数在PHP5.5被废弃,在PHP7被删除。

PHP7的性能高于HHVM。并且是PHP5.6的两倍。

http://php.net/archive/2015.php#id2015-12-03-1

2015123

PHP开发团队宣布PHP 7.0.0即将上市。本次发布标志着新的重要的PHP 7系列的开始。

 

PHP 7.0.0附带了一个新版本的Zend引擎中,无数的改进和新功能,如

性能改善:PHP 7高达两倍快的PHP 5.6

显著减少内存使用

抽象语法树

一致的64位支持

改进的异常层次结构

许多转化为异常致命错误

安全随机数发生器

删除旧的和不支持的SAPIs和扩展

空合并运算符(?)

返回和标量类型声明

匿名类

零成本断言

 

这是下一个主要版本的PHP。它的发布是近两年的发展征程的结果。这是核心团队的一个非常特殊的成就。而且,它是许多活跃的社区成员难以置信努力的结果。事实上,这是一个新的PHP一代的崛起与巨大潜力。

恭喜大家,这是一个壮观的PHP的世界!

感谢感谢所有的贡献者和支持者!

 

根据更新日志,我整理了一下涉及到的类库:CoreCLI_serverCOMCurlDateDBADOMEXIFFileinfoFilterFPMFTPGDGMPhashIMAPIntlJSONLDAPLiteSpeedlibxmlMcryptMysqliOCI8ODBCOpcacheOpenSSLPcntlPCREPDOPDO_DBlibPDO_mysqlPDO_OCIPDO_pgsqlPharPhpdbgReflectionSessionOAPSPLSQLite3tandardStreamsTokenizerXMLReaderXMLRPCXSLZlibZip

一、PHP7的前世今生

            以下摘自并修改与鸟哥微信

PHP7开始于2014年春节,因为基于PHP-5.5Opcache JIT因为无法得到期望而搁置了,并且让鸟哥等人认识到, 基础部分还不够好, 并不能很好的支持JIT, 所以开始了重构项目,希望通过得到30%以上的提升。随后发现性能提升比我们想象的还要大,于是定名为PHP NG项目。

经过发起投票, 绝大部分人都支持了PHP NG项目, 并决定以PHP NG为基础, 开发新版的PHP。社区曾开发过PHP6,后来PHP6的特性在PHP5.55.6等版本都逐渐实现,所以PHP6被搁置。经过社区投票,新项目命名为PHP7

在这近两年的时间里,各种新特性的加入, 性能的持续提升,很多以前不合理的地方改进等等, 都加入到了PHP7, PHP7越来越丰满. 从最底层的ZVAL的改变, 到标量类型提示, 从最初的30%的性能提升, 到现在超过100%的性能飞跃, 每一处变化都让人值得期待. 然后经过几次不情愿的跳票, 终于, 到今天, 这一切都将呈现于你面前。

二、安装

安装:我们编译了核心包以及PDOGDmysqliZip

>$ ./configure --prefix=/usr/local/php7 --enable-fpm --with-zlib --enable-mbstring --with-openssl --with-mysqli --with-mysql-sock --with-gd --enable-gd-native-ttf  --enable-pdo --with-pdo-mysql --with-gettext --with-curl --with-pdo-mysql --enable-sockets --enable-bcmath --enable-xml --with-bz2 --enable-zip -enable-pcntl

>$ make

>$ sudo make install

屏幕快照 2015-12-04 上午10.22.33.png

 

三、测试

            测试版本:

旧版PHP 5.5.29,新版 PHP 7.0.0


屏幕快照 2015-12-04 上午10.45.01.png屏幕快照 2015-12-04 上午10.22.33.png

 

测试机配置如下:

屏幕快照 2015-12-04 上午10.38.15.png

 

1、测试用例一:

生成五十万个数组,并查询五十万次key是否存在

<?php

    $a = array();

    for($i=0;$i<500000;$i++){

        $a[$i] = $i;

    }

    foreach($a as $i)

    {

        array_key_exists($i, $a);

}

?>

测试结果如下:


  time php test.php

php test.php 

0.60s user

0.05s system

98% cpu

0.667 total

  time /usr/local/php7/bin/php test.php

/usr/local/php7/bin/php test.php

0.05s user

0.02s system

92% cpu

0.073 total

 

PHP7速度是PHP5.59

2、测试用例二:

生成五十万个数组,并查询五十万次value是否存在

<?php

    $a = array();

    for($i=0;$i<10000;$i++){

        $a[$i] = $i;

    }

    foreach($a as $i)

    {

        array_search($i, $a);

}

?>

  time php test.php

php test.php

0.79s user

0.01s system

99% cpu

0.809 total

  time /usr/local/php7/bin/php test.php

/usr/local/php7/bin/php test.php 

0.08s user

0.01s system

97% cpu

0.091 total

 

PHP7速度是PHP5.58.7

3、测试用例三:

示例与结果摘自鸟哥博客。以Wordpress为基础,测试PHP7HHVM3.2。用Apacheab测试工具。100个并发, 10000个请求。测试前都会用100个请求预热。

PHP7结果如下:

Concurrency Level:      100

Time taken for tests:   38.726 seconds

Complete requests:      10000

Failed requests:        0

Write errors:           0

Total transferred:      89290000 bytes

HTML transferred:       86900000 bytes

Requests per second:    258.22 [#/sec] (mean)

Time per request:       387.260 [ms] (mean)

Time per request:       3.873 [ms] (mean, across all concurrent requests)

Transfer rate:          2251.64 [Kbytes/sec] received

HHVM-3.2

 

HHVM结果如下:

Document Path:          /wordpress/

Document Length:        8690 bytes

Concurrency Level:      100

Time taken for tests:   43.296 seconds

Complete requests:      10000

Failed requests:        0

Write errors:           0

Total transferred:      89260000 bytes

HTML transferred:       86900000 bytes

Requests per second:    230.97 [#/sec] (mean)

Time per request:       432.957 [ms] (mean)

Time per request:       4.330 [ms] (mean, across all concurrent requests)

Transfer rate:          2013.31 [Kbytes/sec] received

 

PHP7 – 258.22 QPS HHVM – 230.97 QPS

四、新特性

1、标量类型声明

有两种模式: 强制 (默认) 严格模式。 现在可以使用下列类型参数(无论用强制模式还是严格模式): 字符串(string), 整数 (int), 浮点数 (float), 以及布尔值 (bool)。它们扩充了PHP5中引入的其他类型:类名,接口,数组和 回调类型。在旧版中,函数的参数申明只能是(Array $arr)(CLassName $obj)等,基本类型比如IntString等是不能够被申明的

<?php

function check(int $bool){

    var_dump($bool);

}

check(1);

check(true);

?>

若无强制类型转换,会输入int(1)bool(true)。转换后会输出bool(true) bool(true)

2、返回值类型声明

PHP 7 增加了对返回类型声明的支持。返回类型声明指明了函数返回值的类型。可用的类型与参数声明中可用的类型相同。

<?php

function arraysSum(array ...$arrays): array

{

    return array_map(function(array $array): int {

        return array_sum($array);

    }, $arrays);

}

print_r(arraysSum([1,2,3], [4,5,6], [7,8,9]));

以上例程会输出:

Array

(

    [0] => 6

    [1] => 15

    [2] => 24

)

3null合并运算符

项目中存在大量同时使用三元表达式和 isset()的情况,新增了null合并运算符 (??) 这个语法糖。如果变量存在且值不为NULL 它就会返回自身的值,否则返回它的第二个操作数。

旧版:isset($_GET[‘id']) ? $_GET[id] : err;

新版:$_GET['id'] ?? 'err';

4、太空船操作符(组合比较符)

            太空船操作符用于比较两个表达式。当$a小于、等于或大于$b时它分别返回-101 比较的原则是沿用 PHP 的常规比较规则进行的。

<?php

// Integers

echo 1 <=> 1; // 0

echo 1 <=> 2; // -1

echo 2 <=> 1; // 1

// Floats

echo 1.5 <=> 1.5; // 0

echo 1.5 <=> 2.5; // -1

echo 2.5 <=> 1.5; // 1

// Strings

echo "a" <=> "a"; // 0

echo "a" <=> "b"; // -1

echo "b" <=> "a"; // 1

?>

5、通过define()定义常量数组

<?php

define('ANIMALS', ['dog', 'cat', 'bird']);

echo ANIMALS[1]; // outputs "cat"

?>

6、匿名类

现在支持通过new class 来实例化一个匿名类,这可以用来替代一些“用后即焚”的完整类定义。

<?php

interface Logger {

    public function log(string $msg);

}

class Application {

    private $logger;

    public function getLogger(): Logger {

        return $this->logger;

    }

    public function setLogger(Logger $logger) {

        $this->logger = $logger;

    }

}

$app = new Application;

$app->setLogger(new class implements Logger {

    public function log(string $msg) {

        echo $msg;

    }

});

var_dump($app->getLogger());

7Unicode codepoint 转译语法

这接受一个以16进制形式的 Unicode codepoint,并打印出一个双引号或heredoc包围的 UTF-8 编码格式的字符串。 可以接受任何有效的 codepoint,并且开头的 0 是可以省略的。

<?php

 echo “\u{9876}”

?>

            旧版输出:\u{9876}

            新版输入:顶

8Closure::call()

Closure::call() 现在有着更好的性能,简短干练的暂时绑定一个方法到对象上闭包并调用它。

<?php

class Test{public $name = "lixuan";}

 

//PHP7PHP5.6都可以

$getNameFunc = function(){return $this->name;};

$name = $getNameFunc->bindTo(new Test, 'Test');

echo $name();

//PHP7可以,PHP5.6报错

$getX = function() {return $this->name;};

echo $getX->call(new Test);

9、为unserialize()提供过滤

这个特性旨在提供更安全的方式解包不可靠的数据。它通过白名单的方式来防止潜在的代码注入。

<?php

//将所有对象分为__PHP_Incomplete_Class对象

$data = unserialize($foo, ["allowed_classes" => false]);

//将所有对象分为__PHP_Incomplete_Class 对象 除了ClassName1ClassName2

$data = unserialize($foo, ["allowed_classes" => ["ClassName1", "ClassName2"]);

//默认行为,和 unserialize($foo)相同

$data = unserialize($foo, ["allowed_classes" => true]);

10IntlChar

            新增加的 IntlChar 类旨在暴露出更多的 ICU 功能。这个类自身定义了许多静态方法用于操作多字符集的 unicode 字符。IntlPecl扩展,使用前需要编译进PHP中,也可apt-get/yum/port install php5-intl

<?php

printf('%x', IntlChar::CODEPOINT_MAX);

echo IntlChar::charName('@');

var_dump(IntlChar::ispunct('!'));

?>

以上例程会输出:

10ffff

COMMERCIAL AT

bool(true)

11、预期

预期是向后兼用并增强之前的 assert() 的方法。 它使得在生产环境中启用断言为零成本,并且提供当断言失败时抛出特定异常的能力。 老版本的API出于兼容目的将继续被维护,assert()现在是一个语言结构,它允许第一个参数是一个表达式,而不仅仅是一个待计算的 string或一个待测试的boolean

<?php

ini_set('assert.exception', 1);

class CustomError extends AssertionError {}

assert(false, new CustomError('Some error message'));

?>

以上例程会输出:

Fatal error: Uncaught CustomError: Some error message

12Group use declarations

            从同一 namespace 导入的类、函数和常量现在可以通过单个 use 语句 一次性导入了。

<?php

//PHP7之前

use some\namespace\ClassA;

use some\namespace\ClassB;

use some\namespace\ClassC as C;

use function some\namespace\fn_a;

use function some\namespace\fn_b;

use function some\namespace\fn_c;

use const some\namespace\ConstA;

use const some\namespace\ConstB;

use const some\namespace\ConstC;

// PHP7之后

use some\namespace\{ClassA, ClassB, ClassC as C};

use function some\namespace\{fn_a, fn_b, fn_c};

use const some\namespace\{ConstA, ConstB, ConstC};

?>

13intdiv()

            接收两个参数作为被除数和除数,返回他们相除结果的整数部分。

<?php

var_dump(intdiv(7, 2));

?>

输出int(3)

14CSPRNG

新增两个函数: random_bytes() and random_int().可以加密的生产被保护的整数和字符串。我这蹩脚的翻译,总之随机数变得安全了。

random_bytes — 加密生存被保护的伪随机字符串

random_int —加密生存被保护的伪随机整数

15preg_replace_callback_array()

            新增了一个函数preg_replace_callback_array(),使用该函数可以使得在使用preg_replace_callback()函数时代码变得更加优雅。在PHP7之前,回调函数会调用每一个正则表达式,回调函数在部分分支上是被污染了。

16Session options

            现在,session_start()函数可以接收一个数组作为参数,可以覆盖php.inisession的配置项。

            比如,把cache_limiter设置为私有的,同时在阅读完session后立即关闭。

<?php

session_start([

    'cache_limiter' => 'private',

    'read_and_close' => true,

]);

?>

17、生成器的返回值

            PHP5.5引入生成器的概念。生成器函数每执行一次就得到一个yield标识的值。在PHP7中,当生成器迭代完成后,可以获取该生成器函数的返回值。通过Generator::getReturn()得到。

<?php

function generator() {

    yield 1;

    yield 2;

    yield 3;

    return "a";

}

$generatorClass = ("generator")();

foreach ($generatorClass as $val) {

    echo $val.” “;

}

echo $generatorClass->getReturn();

?>

输出为:1 2 3 a

18、生成器中引入其他生成器

            在生成器中可以引入另一个或几个生成器,只需要写yield from functionName1

<?php

function generator1(){

    yield 1;

    yield 2;

    yield from generator2();

    yield from generator3();

}

function generator2(){

    yield 3;

    yield 4;

}

function generator3(){

    yield 5;

    yield 6;

}

foreach (generator1() as $val){

    echo $val, " ";

}

?>

输出:1 2 3 4 5 6

五、不兼容性

1foreach不再改变内部数组指针

PHP7之前,当数组通过 foreach 迭代时,数组指针会移动。现在开始,不再如此,见下面代码。

<?php

$array = [0, 1, 2];

foreach ($array as &$val) {

    var_dump(current($array));

}

?>

PHP5输出:

int(1)

int(2)

bool(false)

PHP7输出:

int(0)

int(0)

int(0)

2foreach通过引用遍历时,有更好的迭代特性

当使用引用遍历数组时,现在 foreach 在迭代中能更好的跟踪变化。例如,在迭代中添加一个迭代值到数组中,参考下面的代码:

<?php

$array = [0];

foreach ($array as &$val) {

    var_dump($val);

    $array[1] = 1;

}

?>

PHP5输出:

int(0)

PHP7输出:

int(0)

int(1)

3、十六进制字符串不再被认为是数字

    含十六进制字符串不再被认为是数字

<?php

var_dump("0x123" == "291");

var_dump(is_numeric("0x123"));

var_dump("0xe" + "0x1");

var_dump(substr("foo", "0x1"));

?>

PHP5输出:

bool(true)

bool(true)

int(15)

string(2) "oo"

PHP7输出:

bool(false)

bool(false)

int(0)

Notice: A non well formed numeric value encountered in /tmp/test.php on line 5

string(3) "foo"

4PHP7中被移除的函数

被移除的函数列表如下:

call_user_func() call_user_func_array()PHP 4.1.0开始被废弃。

已废弃的 mcrypt_generic_end() 函数已被移除,请使用mcrypt_generic_deinit()代替。

已废弃的 mcrypt_ecb(), mcrypt_cbc(), mcrypt_cfb() mcrypt_ofb() 函数已被移除。

set_magic_quotes_runtime(), 和它的别名 magic_quotes_runtime()已被移除. 它们在PHP 5.3.0中已经被废弃,并且 in PHP 5.4.0也由于魔术引号的废弃而失去功能。

已废弃的 set_socket_blocking() 函数已被移除,请使用stream_set_blocking()代替。

dl() PHP-FPM 不再可用,在 CLI embed SAPIs 中仍可用。

GD库中下列函数被移除:imagepsbbox()imagepsencodefont()imagepsextendfont()imagepsfreefont()imagepsloadfont()imagepsslantfont()imagepstext()

在配置文件php.ini中,always_populate_raw_post_dataasp_tagsxsl.security_prefs被移除了。

5new 操作符创建的对象不能以引用方式赋值给变量

new 操作符创建的对象不能以引用方式赋值给变量

<?php

class C {}

$c =& new C;

?>

PHP5输出:

Deprecated: Assigning the return value of new by reference is deprecated in /tmp/test.php on line 3

PHP7输出:

Parse error: syntax error, unexpected 'new' (T_NEW) in /tmp/test.php on line 3

6、移除了 ASP script PHP 标签

使用类似 ASP 的标签,以及 script 标签来区分 PHP 代码的方式被移除。 受到影响的标签有:<% %><%= %><script language="php"> </script>

7、从不匹配的上下文发起调用

            在不匹配的上下文中以静态方式调用非静态方法, PHP 5.6 中已经废弃, 但是在 PHP 7.0 中, 会导致被调用方法中未定义 $this 变量,以及此行为已经废弃的警告。

<?php

class A {

    public function test() { var_dump($this); }

}

// 注意:并没有从类 A 继承

class B {

    public function callNonStaticMethodOfA() { A::test(); }

}

(new B)->callNonStaticMethodOfA();

?>

PHP5输出:

Deprecated: Non-static method A::test() should not be called statically, assuming $this from incompatible context in /tmp/test.php on line 8

object(B)#1 (0) {

}

PHP7输出:

Deprecated: Non-static method A::test() should not be called statically in /tmp/test.php on line 8

Notice: Undefined variable: this in /tmp/test.php on line 3

NULL

8、在数值溢出的时候,内部函数将会失败

将浮点数转换为整数的时候,如果浮点数值太大,导致无法以整数表达的情况下, 在之前的版本中,内部函数会直接将整数截断,并不会引发错误。 PHP 7.0 中,如果发生这种情况,会引发 E_WARNING 错误,并且返回 NULL

9JSON 扩展已经被 JSOND 取代

JSON 扩展已经被 JSOND 扩展取代。 对于数值的处理,有以下两点需要注意的: 第一,数值不能以点号(.)结束 (例如,数值 34. 必须写作 34.0 34)。 第二,如果使用科学计数法表示数值,e 前面必须不是点号(. (例如,3.e3 必须写作 3.0e3 3e3)。

10INI 文件中 # 注释格式被移除

在配置文件INI文件中,不再支持以 # 开始的注释行, 请使用 ;(分号)来表示注释。 此变更适用于 php.ini 以及用 parse_ini_file() parse_ini_string() 函数来处理的文件。

11$HTTP_RAW_POST_DATA 被移除

不再提供 $HTTP_RAW_POST_DATA 变量。 请使用 php://input 作为替代。

12yield 变更为右联接运算符

在使用 yield 关键字的时候,不再需要括号, 并且它变更为右联接操作符,其运算符优先级介于 print => 之间。 这可能导致现有代码的行为发生改变。可以通过使用括号来消除歧义。

<?php

echo yield -1;

// 在之前版本中会被解释为:

echo (yield) - 1;

// 现在,它将被解释为:

echo yield (-1);

 

yield $foo or die;

// 在之前版本中会被解释为:

yield ($foo or die);

// 现在,它将被解释为:

(yield $foo) or die;

?>

 

PHP官方网站文档:http://php.net/manual/zh/migration70.php

可以浏览PHP5.6PHP7时,新特性、新增函数、已经被移除的函数、不兼容性、新增的类和接口等内容。


    最近在学习Golang,看资料过程中收集了一部分Go入门资源。供大家浏览,也当自己收藏。包括Go语言的安装和配置,Go入门,语法词法。还有一些函数库,包括net/http,time,buffer等。持续更新


1、安装和入门,基础语法:https://blog.linguofeng.com/pages/language/go.html


2、Go指针详解:http://my.oschina.net/u/943306/blog/131269


3、time包:http://my.oschina.net/u/943306/blog/149395


4、bytes/buffer包:http://my.oschina.net/u/943306/blog/127981


5、net/http包:http://my.oschina.net/u/943306/blog/151293


6、Go Web编程(开源书籍,有PDF和EPUB和Gitbook):https://github.com/astaxie/build-web-application-with-golang





   想了解PHP-MVC框架的原理和实现方式是什么?在这里,我们用PHP开发一个自己的MVC框架。小巧精悍。深入理解市面上PHP主流的MVC框架原理。唯一入口文件该干什么,如何自动载入。本章代码基于PHP5.3+,采用命名空间。

    PHP框架是什么?http://baike.baidu.com/link?url=3IBWTY1jzqQiHAfVWRn6BzXvkUxkWTDffhe7_J4o7Eqs2Hxp06PMnAXsK82vGdnQEfEYuueszQ3_EFdxMUOvEa

    MVC是什么?http://baike.baidu.com/view/5432454.htm?fromtitle=mvc&fromid=85990&type=syn

    源码地址:https://github.com/lixuancn/LaneSmartFW

    开源协议:Do What The Fuck You Want To Public License


一、起名:

    先给我们的PHP-MVC框架起个名字,叫宇宙无敌框架UniverseInvincibleFrameWork


二、实现功能

    1、MVC分层

    2、唯一入口

    3、关键常量可配置

    4、自动载入函数

    5、路由分发

    6、数据库工厂

    7、多数据支持

    8、多项目支持


三、详细分解如何PHP-MVC框架

    1、MVC分层

        1)、目录结构

框架根目录



项目一的目录



配置文件目录



框架核心文件目录



框架核心文件之数据库目录



        2)、目录简介

            (1)、Home、Admin是项目名,可以无限扩展

            (2)、Config是配置文件所在目录,UniverseInvincibleFrameWork是框架核心文件所在目录

            (3)、Index.php是唯一入口文件

            (4)、Home目录下就是标准的Controller、Model、View,另外新增了Service

            (5)、UniverseInvincibleFrameWork目录下是核心框架入口类、自动载入类、路由类已经数据库文件所在的DB目录

            (6)、DB目录是数据库相关操作。比如数据库工厂类,接口规范类,CURD操作等。


    2、唯一入口

        1)、采用单一入口模式进行项目部署和访问,无论完成什么功能,一个项目都有一个统一的入口。

        2)、只需要引入框架核心文件App.php,然后执行该类的方法

<?php

/**

 * 宇宙无敌框架UniverseInvincibleFrameWork

 * 唯一入口

 * Created by lixuan-it@360.cn

 * User: lane

 * Date: 15/8/27

 * Time: 下午3:17

 * E-mail: lixuan868686@163.com

 * WebSite: http://www.lanecn.com

 */

//引入框架核心文件

require_once 'UniverseInvincibleFrameWork/App.php';

//初始化框架

$obj = new UniverseInvincibleFrameWork\App();

$obj->init();

        3)、框架核心文件源码:

<?php

namespace UniverseInvincibleFrameWork;



class App{

    public function init(){

        //设置头 - utf-8

        $this->_setHeader();

        //载入系统配置文件

        $this->_loadSysFile();

        //自动载入函数

        $this->_setAutoload();

        //设置路由

        $this->_setRoute();

    }


    /**

     * 载入系统配置文件

     */

    private function _loadSysFile(){

        require_once dirname(__FILE__).'/Function.php';

        //1、 require_once dirname(__FILE__).'/../config/config.php';

        //2、$GLOBALS['config'] = config.php的所有内容

        $GLOBALS['config'] = require_once dirname(__FILE__).'/../config/config.php';

    }


    /**

     * 头

     */

    private function _setHeader(){

        header('Content-type: text/html; charset=UTF-8');

    }


    /**

     * 自动载入函数

     */

    private function _setAutoload(){

        //自动载入函数

        require_once dirname(__FILE__).'/../UniverseInvincibleFrameWork/Autoload.php';

        $autoload = new Autoload();

        $autoload->register();

    }


    /**

     * 设置路由

     */

    private function _setRoute(){

        $routeObj = new Route();

        $routeObj->parse();

    }

}

    3、关键常量可配置

        1)、谁也不会傻呼呼的到把数据库链接信息等配置信息写死到代码里,那么就必须有一个配置文件。它定义系统常量,包括但不限于项目名称、数据库账号密码,默认应用名称/控制器/方法名等

        2)、配置文件还有个好处,定义生长环境、测试环境、开发环境等不同的参数,可以根据来访域名、所在机器IP等信息来使自动选择加载不同的系统和数据库配置。

<?php

/**

 * Created by lixuan-it@360.cn

 * User: lane

 * Date: 15/8/27

 * Time: 下午3:28

 * E-mail: lixuan868686@163.com

 * WebSite: http://www.lanecn.com

 */

return array(

    //默认加载的项目

    'DEFAULT_APP_NAME' => 'Home',

    //默认加载的控制器

    'DEFAULT_CONTROLLER' => 'Index',

    //默认加载的方法

    'DEFAULT_METHOD' => 'index',

 //默认数据库配置

    'DB_CONFIG' => array(

        'DB_TYPE' => 'mysql',

        'DB_HOST' => 'localhost',

        'DB_PORT' => '3306',

        'DB_USERNAME' => 'root',

        'DB_PASSWORD' =>’’,

        'DB_NAME' => db1,

    ),

    //默认数据二配置

    'DB_CONFIG2' => array(

        'DB_TYPE' => 'mysql',

        'DB_HOST' => 'localhost',

        'DB_PORT' => '3306',

        'DB_USERNAME' => 'root',

        'DB_PASSWORD' =>’’,

        'DB_NAME' => 'db2',

    ),

);

     在Function.php增加一个函数,用来读取配置文件。

<?php

/**

 * Created by lixuan-it@360.cn

 * User: lane

 * Date: 15/8/27

 * Time: 下午4:17

 * E-mail: lixuan868686@163.com

 * WebSite: http://www.lanecn.com

 */

function getConfig($name){

    return $GLOBALS['config'][$name] ? : '';

}


    4、自动载入函数

        1)、不用自动载入函数,难道要在代码里不断的去include其他的文件吗?

        2)、我们用spl_autoload_register()。从PHP5.1.2引入。摒弃了__autoload()。它的优势是一个项目可以有多个spl_autoload_register()函数。使得项目框架、各种插件(LaneWeChat、PHPMailer、PHPEXCEL等)不会相互冲突

<?php

namespace UniverseInvincibleFrameWork;

/**

 * 自动载入

 * Created by lixuan-it@360.cn

 * User: lane

 * Date: 15/8/27

 * Time: 下午3:28

 * E-mail: lixuan868686@163.com

 * WebSite: http://www.lanecn.com

 */

class Autoload{

    public function register(){

        spl_autoload_register(array($this, 'autoload'));

    }


    public function autoload($className){

        $pathArr = explode('\\', $className);

        $filename = array_pop($pathArr);

        $dir = implode(DIRECTORY_SEPARATOR, $pathArr);

        $filename = $dir . '/' . $filename . '.php';

        if(file_exists($filename)){

            require_once $filename;

        }else{

            exit('Error:'.$className.' loading Failed');

        }

    }

}


    5、路由分发

        1)、我们的URL规则:http://www.lanecn.com/index.php/项目名/类名/方法名

        2)、所有的URL,都会去执行index.php。然后路由的作用的是根据不同的URL,来执行不同的Controller。

<?php

namespace UniverseInvincibleFrameWork;

/**

 * 路由

 * Created by lixuan-it@360.cn

 * User: lane

 * Date: 15/8/27

 * Time: 下午3:29

 * E-mail: lixuan868686@163.com

 * WebSite: http://www.lanecn.com

 */

class Route{

    /**

     * 分析URL

     */

    public function parse(){

        $pathInfo = !empty($_SERVER['PATH_INFO']) ? explode('/', $_SERVER['PATH_INFO']) : array();

        $appName = !empty($pathInfo[1]) ? $pathInfo[1] : getConfig('DEFAULT_APP_NAME');

        $className = !empty($pathInfo[2]) ? $pathInfo[2] : getConfig('DEFAULT_CONTROLLER');

        $methodName = !empty($pathInfo[3]) ? $pathInfo[3] : getConfig('DEFAULT_METHOD');

        $c = $appName . '\Controller\\' . $className.'Controller';

        $obj = new $c();

        $obj->$methodName();

    }

}


    6、数据库工厂

        1)、大型项目中,我们会用到Mysql、Redis等多种数据库。甚至前期是ACCESS,后期是Mysql/Oracle/SQL SERVER,在切换数据库的过程中,只需要修改一个常量而不需要修改代码。

        2)、根据配置文件中定义的数据库类型,我们选在加载不同的数据库类

<?php

namespace UniverseInvincibleFrameWork\DB;

/**

 * 数据工厂

 * Created by lixuan-it@360.cn

 * User: lane

 * Date: 15/8/27

 * Time: 下午3:29

 * E-mail: lixuan868686@163.com

 * WebSite: http://www.lanecn.com

 */

class Db{

    public static function factor($dbConfigKey='DB_CONFIG'){

        //根据参数选择加载不同的数据库配置

        $dbType = strtolower(getConfig($dbConfigKey)['DB_TYPE']);

        switch($dbType){

            case 'mysql':

                $className = 'Mysql';

                break;

            default:

                exit('Error:Database Type');

        }

        $className = 'UniverseInvincibleFrameWork\DB\\'.$className;

        return new $className($dbConfigKey);

    }

}

        3)、项目中的Model文件,继承Model类。该类定义了常用的数据库操作,是所有数据库的抽象类。如增删改查和自定义SQL等。该类使用数据工厂中返回的示例,来操作具体的数据库类。

<?php

namespace UniverseInvincibleFrameWork\DB;

/**

 * 基础Model类,所有的Model文件均继承本类

 * Created by lixuan-it@360.cn

 * User: lane

 * Date: 15/8/27

 * Time: 下午6:35

 * E-mail: lixuan868686@163.com

 * WebSite: http://www.lanecn.com

 */

class Model implements DbInterface {


    protected $dbConfigKey = null;


    private $_db = null;


    private function _getInstance(){

        if(is_null($this->_db)){

            if(is_null($this->dbConfigKey)){

                $this->_db = DB::factor();

            }else{

                $this->_db = DB::factor($this->dbConfigKey);

            }

        }

        return $this->_db;

    }


    public function close(){

        $this->_getInstance()->close();

    }



    public function query($sql){

        return $this->_getInstance()->query($sql);

    }


    public function fetchAssoc($resource){

        return $this->_getInstance()->fetchAssoc($resource);

    }


    public function select($sql){

        return $this->_getInstance()->select($sql);

    }

}

        4)、最后该编码数据库实例类了。已Mysql为例

<?php

namespace UniverseInvincibleFrameWork\DB;


/**

 * Created by lixuan-it@360.cn

 * User: lane

 * Date: 15/8/27

 * Time: 下午3:29

 * E-mail: lixuan868686@163.com

 * WebSite: http://www.lanecn.com

 */

class Mysql implements DbInterface{


    private $_conn = null;


    public function __construct($dbConfigKey='DB_CONFIG'){

        if(is_null($this->_conn)){

            $this->_connect($dbConfigKey);

        }

    }


    private function _connect($dbConfigKey='DB_CONFIG'){

        $dbConfig = getConfig($dbConfigKey);

        $this->_conn = mysqli_connect($dbConfig['DB_HOST'], $dbConfig['DB_USERNAME'], $dbConfig['DB_PASSWORD'], $dbConfig['DB_NAME'], $dbConfig['DB_PORT']);

    }


    public function close(){

        mysqli_close($this->_getInstance());

    }


    public function query($sql){

        $result = mysqli_query($this->_conn, $sql);

        return $result;

    }


    public function fetchAssoc($resource){

        $rowList = array();

        while($row = mysqli_fetch_assoc($resource)){

            $rowList[] = $row;

        }

        return $rowList;

    }


    public function select($sql){

        $result = $this->query($sql);

        $rowList = $this->fetchAssoc($result);

        return $rowList;

    }

}

        5)、补充一点。我们有一个接口类,为了约定各个数据库的实例类的规范,他们都要实现几个关键方法。

<?php

namespace UniverseInvincibleFrameWork\DB;

/**

 * 数据库实例类的接口

 * Created by lixuan-it@360.cn

 * User: lane

 * Date: 15/8/27

 * Time: 下午5:57

 * E-mail: lixuan868686@163.com

 * WebSite: http://www.lanecn.com

 */

Interface DbInterface{


    public function close();


    public function query($sql);


    public function fetchAssoc($resource);


    public function select($sql);

}


    7、多数据支持

        1、项目中,常常会遇到既需要数据库A,又需要数据库B。那就需要多数据库支持。

        2、在数据库A中获得用户ID列表,在数据库B中根据用户ID列表获得用户详细信息


    8、多项目支持

        1、框架实现了多项目支持。比如前台、后台、项目三、项目四



四、测试

    1、我们在项目Home中,进行测试。编写两个Model文件。第一个是Home/Model/IndexModel.php 来查询默认数据库的Mysql 版本。第二个是Home/ Model / TestModel.php。来查询数据库2的所有表的名字。

<?php

namespace Home\Model;

use UniverseInvincibleFrameWork\DB\Model;


/**

 * Created by lixuan-it@360.cn

 * User: lane

 * Date: 15/8/27

 * Time: 下午4:36

 * E-mail: lixuan868686@163.com

 * WebSite: http://www.lanecn.com

 */

class IndexModel extends Model{

    public function getVersion(){

        $sql = 'SELECT VERSION() as `version`';

        $result = $this->query($sql);

        $result = $this->fetchAssoc($result);

        return $result;

    }

}


<?php

namespace Home\Model;

use UniverseInvincibleFrameWork\DB\Model;


/**

 * Created by lixuan-it@360.cn

 * User: lane

 * Date: 15/8/27

 * Time: 下午4:36

 * E-mail: lixuan868686@163.com

 * WebSite: http://www.lanecn.com

 */

class TestModel extends Model{

    public function __construct(){

        $this->dbConfigKey = 'DB_CONFIG2';

    }


    public function getTables(){

        $sql = 'show tables';

        $result = $this->query($sql);

        $result = $this->fetchAssoc($result);

        return $result;

    }

}


    2、写一个Server文件。Home/Server/IndexServer.php来调用刚才写的两个Model文件并返回

<?php

namespace Home\Service;

/**

 * Created by lixuan-it@360.cn

 * User: lane

 * Date: 15/8/27

 * Time: 下午4:35

 * E-mail: lixuan868686@163.com

 * WebSite: http://www.lanecn.com

 */

class IndexService{

    public static function getVersion(){

        $ret = array();

        $ret['php_version'] = PHP_VERSION;

        $model = new \Home\Model\IndexModel();;

        $ret['mysql_version'] = $model->getVersion()[0]['version'];

        return $ret;

    }


    public static function getTables(){

        $model = new \Home\Model\TestModel();;

        $ret = $model->getTables();

        return $ret;

    }

}


    3、Home/Controller/IndexController.php编写4个测试示例。

<?php

namespace Home\Controller;

/**

 * Created by lixuan-it@360.cn

 * User: lane

 * Date: 15/8/27

 * Time: 下午4:31

 * E-mail: lixuan868686@163.com

 * WebSite: http://www.lanecn.com

 */

class IndexController{

    public function index(){

        echo 'hello world';

    }


    public function test(){

        echo 'hello 360';

    }


    public function getVersion(){

        $versionList = \Home\Service\IndexService::getVersion();

        echo 'PHP版本:' . $versionList['php_version'].'。Mysql版本:' . $versionList['mysql_version'];

    }


    public function getTables(){

        $tables = \Home\Service\IndexService::getTables();

        var_dump($tables);

    }

}


    4、结果:

        1)、浏览器运行http://framework/index.php/Home/Index/index。正常输出:“hello world”

        2)、浏览器运行http://framework/index.php/Home/Index/test。正常输出:“hello 360”

        3)、浏览器运行http://framework/index.php/Home/Index/getVersion。正常输出:“PHP版本:5.5.27。Mysql版本:5.6.17”

        4)、浏览器运行http://framework/index.php/Home/Index/getTables。正常输出数据库2的所有表名