搜尋
首頁後端開發php教程用PHP实现多服务器共享SESSION数据的方法_PHP教程

PHP 实现多服务器共享 SESSION 数据

稍大一些的网站,通常都会有好几个服务器,每个服务器运行着不同功能的模块,使用不同的二级域名,而一个整体性强的网站,用户系统是统一的,即一套用户名、密码在整个网站的各个模块中都是可以登录使用的。各个服务器共享用户数据是比较容易实现的,只需要在后端放个数据库服务器,各个服务器通过统一接口对用户数据进行访问即可。但还存在一个问题,就是用户在这个服务器登录之后,进入另一个服务器的别的模块时,仍然需要重新登录,这就是一次登录,全部通行的问题,映射到技术上,其实就是各个服务器之间如何实现共享 SESSION 数据的问题。

二、PHP SESSION 的工作原理

在解决问题之前,先来了解一下 PHP SESSION 的工作原理。在客户端(如浏览器)登录网站时,被访问的 PHP 页面可以使用 session_start() 打开 SESSION,这样就会产生客户端的唯一标识 SESSION ID(此 ID 可通过函数 session_id() 获取/设置)。SESSION ID 可以通过两种方式保留在客户端,使得请求不同的页面时,PHP 程序可以获知客户端的 SESSION ID;一种是将 SESSION ID 自动加入到 GET 的 URL 中,或者 POST 的表单中,默认情况下,变量名为 PHPSESSID;另一种是通过 COOKIE,将 SESSION ID 保存在 COOKIE 中,默认情况下,这个 COOKIE 的名字为 PHPSESSID。这里我们主要以 COOKIE 方式进行说明,因为应用比较广泛。

那么 SESSION 的数据保存在哪里呢?当然是在服务器端,但不是保存在内存中,而是保存在文件或数据库中。默认情况下,php.ini 中设置的 SESSION 保存方式是 files(session.save_handler = files),即使用读写文件的方式保存 SESSION 数据,而 SESSION 文件保存的目录由 session.save_path 指定,文件名以 sess_ 为前缀,后跟 SESSION ID,如:sess_c72665af28a8b14c0fe11afe3b59b51b。文件中的数据即是序列化之后的 SESSION 数据了。如果访问量大,可能产生的 SESSION 文件会比较多,这时可以设置分级目录进行 SESSION 文件的保存,效率会提高很多,设置方法为:session.save_path="N;/save_path",N 为分级的级数,save_path 为开始目录。当写入 SESSION 数据的时候,PHP 会获取到客户端的 SESSION_ID,然后根据这个 SESSION ID 到指定的 SESSION 文件保存目录中找到相应的 SESSION 文件,不存在则创建之,最后将数据序列化之后写入文件。读取 SESSION 数据是也是类似的操作流程,对读出来的数据需要进行解序列化,生成相应的 SESSION 变量。

三、多服务器共享 SESSION 的主要障碍及解决办法

通过了解 SESSION 的工作原理,我们可以发现,在默认情况下,各个服务器会各自分别对同一个客户端产生 SESSION ID,如对于同一个用户浏览器,A 服务器产生的 SESSION ID 是 30de1e9de3192ba6ce2992d27a1b6a0a,而 B 服务器生成的则是 c72665af28a8b14c0fe11afe3b59b51b。另外,PHP 的 SESSION 数据都是分别保存在本服务器的文件系统中。如下图所示:

 

确定了问题所在之后,就可以着手进行解决了。想要共享 SESSION 数据,那就必须实现两个目标:一个是各个服务器对同一个客户端产生的 SESSION ID 必须相同,并且可通过同一个 COOKIE 进行传递,也就是说各个服务器必须可以读取同一个名为 PHPSESSID 的 COOKIE;另一个是 SESSION 数据的存储方式/位置必须保证各个服务器都能够访问到。简单地说就是多服务器共享客户端的 SESSION ID,同时还必须共享服务器端的 SESSION 数据。

第一个目标的实现其实很简单,只需要对 COOKIE 的域(domain)进行特殊地设置即可,默认情况下,COOKIE 的域是当前服务器的域名/IP 地址,而域不同的话,各个服务器所设置的 COOKIE 是不能相互访问的,如 www.aaa.com 的服务器是不能读写 www.bbb.com 服务器设置的 COOKIE 的。

这里我们所说的同一网站的服务器有其特殊性,那就是他们同属于同一个一级域,如:aaa.infor96.com 和 www.infor96.com 都属于域 .infor96.com,那么我们就可以设置 COOKIE 的域为 .infor96.com,这样 aaa.infor96.com、www.infor96.com 等等都可以访问此 COOKIE。PHP 代码中的设置方法如下:

<CODE><SPAN style="COLOR: rgb(0,0,0)">
<SPAN style="COLOR: rgb(0,0,187)"><?php
ini_set</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(221,0,0)">'session.cookie_domain'</SPAN><SPAN style="COLOR: rgb(0,119,0)">, </SPAN><SPAN style="COLOR: rgb(221,0,0)">'.infor96.com'</SPAN><SPAN style="COLOR: rgb(0,119,0)">);
</SPAN><SPAN style="COLOR: rgb(0,0,187)">?></SPAN>
</SPAN>
</CODE>

这样各个服务器共享同一客户端 SESSION ID 的目的就达到了。

第二个目标的实现可以使用文件共享方式,如 NFS 方式,但设置、操作上有些复杂。我们可以参考先前所说的统一用户系统的方式,即使用数据库来保存 SESSION 数据,这样各个服务器就可以方便地访问同一个数据源,获取相同的 SESSION 数据了。

解决办法如下图所示:

 

四、代码实现

首先创建数据表,MySQL 的 SQL 语句如下:

  CREATE TABLE `sess` (
   `sesskey` varchar(32) NOT NULL default '',
   `expiry` bigint(20) NOT NULL default '0',
   `data` longtext NOT NULL,
   PRIMARY KEY (`sesskey`),
   KEY `expiry` (`expiry`)
  ) TYPE=MyISAM

sesskey 为 SESSION ID,expiry 为 SESSION 过期时间,data 用于保存 SESSION 数据。

默认情况下 SESSION 数据是以文件方式保存,想要使用数据库方式保存,就必须重新定义 SESSION 各个操作的处理函数。PHP 提供了session_set_save_handle() 函数,可以用此函数自定义 SESSION 的处理过程,当然首先要先将 session.save_handler 改成 user,可在 PHP 中进行设置:

<CODE><SPAN style="COLOR: rgb(0,0,0)">
<SPAN style="COLOR: rgb(0,0,187)"><?php 
session_module_name</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(221,0,0)">'user'</SPAN><SPAN style="COLOR: rgb(0,119,0)">);
</SPAN><SPAN style="COLOR: rgb(0,0,187)">?></SPAN>
</SPAN>
</CODE>

接下来着重讲一下 session_set_save_handle() 函数,此函数有六个参数:

session_set_save_handler ( string open, string close, string read, string write, string destroy, string gc )

各个参数为各项操作的函数名,这些操作依次是:打开、关闭、读取、写入、销毁、垃圾回收。PHP 手册中有详细的例子,在这里我们使用 OO 的方式来实现这些操作,详细代码如下:

<CODE><SPAN style="COLOR: rgb(0,0,0)">
<SPAN style="COLOR: rgb(0,0,187)"><?php
define</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(221,0,0)">'MY_SESS_TIME'</SPAN><SPAN style="COLOR: rgb(0,119,0)">, </SPAN><SPAN style="COLOR: rgb(0,0,187)">3600</SPAN><SPAN style="COLOR: rgb(0,119,0)">);   </SPAN><SPAN style="COLOR: rgb(255,128,0)">//SESSION 生存时长
//类定义
</SPAN><SPAN style="COLOR: rgb(0,119,0)">class </SPAN><SPAN style="COLOR: rgb(0,0,187)">My_Sess
</SPAN><SPAN style="COLOR: rgb(0,119,0)">{
    function </SPAN><SPAN style="COLOR: rgb(0,0,187)">init</SPAN><SPAN style="COLOR: rgb(0,119,0)">()
    {
        </SPAN><SPAN style="COLOR: rgb(0,0,187)">$domain </SPAN><SPAN style="COLOR: rgb(0,119,0)">= </SPAN><SPAN style="COLOR: rgb(221,0,0)">'.infor96.com'</SPAN><SPAN style="COLOR: rgb(0,119,0)">;
        </SPAN><SPAN style="COLOR: rgb(255,128,0)">//不使用 GET/POST 变量方式
        </SPAN><SPAN style="COLOR: rgb(0,0,187)">ini_set</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(221,0,0)">'session.use_trans_sid'</SPAN><SPAN style="COLOR: rgb(0,119,0)">,    </SPAN><SPAN style="COLOR: rgb(0,0,187)">0</SPAN><SPAN style="COLOR: rgb(0,119,0)">);
        </SPAN><SPAN style="COLOR: rgb(255,128,0)">//设置垃圾回收最大生存时间
        </SPAN><SPAN style="COLOR: rgb(0,0,187)">ini_set</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(221,0,0)">'session.gc_maxlifetime'</SPAN><SPAN style="COLOR: rgb(0,119,0)">,   </SPAN><SPAN style="COLOR: rgb(0,0,187)">MY_SESS_TIME</SPAN><SPAN style="COLOR: rgb(0,119,0)">);

        </SPAN><SPAN style="COLOR: rgb(255,128,0)">//使用 COOKIE 保存 SESSION ID 的方式
        </SPAN><SPAN style="COLOR: rgb(0,0,187)">ini_set</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(221,0,0)">'session.use_cookies'</SPAN><SPAN style="COLOR: rgb(0,119,0)">,      </SPAN><SPAN style="COLOR: rgb(0,0,187)">1</SPAN><SPAN style="COLOR: rgb(0,119,0)">);
        </SPAN><SPAN style="COLOR: rgb(0,0,187)">ini_set</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(221,0,0)">'session.cookie_path'</SPAN><SPAN style="COLOR: rgb(0,119,0)">,      </SPAN><SPAN style="COLOR: rgb(221,0,0)">'/'</SPAN><SPAN style="COLOR: rgb(0,119,0)">);
        </SPAN><SPAN style="COLOR: rgb(255,128,0)">//多主机共享保存 SESSION ID 的 COOKIE
        </SPAN><SPAN style="COLOR: rgb(0,0,187)">ini_set</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(221,0,0)">'session.cookie_domain'</SPAN><SPAN style="COLOR: rgb(0,119,0)">,    </SPAN><SPAN style="COLOR: rgb(0,0,187)">$domain</SPAN><SPAN style="COLOR: rgb(0,119,0)">);

        </SPAN><SPAN style="COLOR: rgb(255,128,0)">//将 session.save_handler 设置为 user,而不是默认的 files
        </SPAN><SPAN style="COLOR: rgb(0,0,187)">session_module_name</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(221,0,0)">'user'</SPAN><SPAN style="COLOR: rgb(0,119,0)">);
        </SPAN><SPAN style="COLOR: rgb(255,128,0)">//定义 SESSION 各项操作所对应的方法名:
        </SPAN><SPAN style="COLOR: rgb(0,0,187)">session_set_save_handler</SPAN><SPAN style="COLOR: rgb(0,119,0)">(
            array(</SPAN><SPAN style="COLOR: rgb(221,0,0)">'My_Sess'</SPAN><SPAN style="COLOR: rgb(0,119,0)">, </SPAN><SPAN style="COLOR: rgb(221,0,0)">'open'</SPAN><SPAN style="COLOR: rgb(0,119,0)">),   </SPAN><SPAN style="COLOR: rgb(255,128,0)">//对应于静态方法 My_Sess::open(),下同。
            </SPAN><SPAN style="COLOR: rgb(0,119,0)">array(</SPAN><SPAN style="COLOR: rgb(221,0,0)">'My_Sess'</SPAN><SPAN style="COLOR: rgb(0,119,0)">, </SPAN><SPAN style="COLOR: rgb(221,0,0)">'close'</SPAN><SPAN style="COLOR: rgb(0,119,0)">),
            array(</SPAN><SPAN style="COLOR: rgb(221,0,0)">'My_Sess'</SPAN><SPAN style="COLOR: rgb(0,119,0)">, </SPAN><SPAN style="COLOR: rgb(221,0,0)">'read'</SPAN><SPAN style="COLOR: rgb(0,119,0)">),
            array(</SPAN><SPAN style="COLOR: rgb(221,0,0)">'My_Sess'</SPAN><SPAN style="COLOR: rgb(0,119,0)">, </SPAN><SPAN style="COLOR: rgb(221,0,0)">'write'</SPAN><SPAN style="COLOR: rgb(0,119,0)">),
            array(</SPAN><SPAN style="COLOR: rgb(221,0,0)">'My_Sess'</SPAN><SPAN style="COLOR: rgb(0,119,0)">, </SPAN><SPAN style="COLOR: rgb(221,0,0)">'destroy'</SPAN><SPAN style="COLOR: rgb(0,119,0)">),
            array(</SPAN><SPAN style="COLOR: rgb(221,0,0)">'My_Sess'</SPAN><SPAN style="COLOR: rgb(0,119,0)">, </SPAN><SPAN style="COLOR: rgb(221,0,0)">'gc'</SPAN><SPAN style="COLOR: rgb(0,119,0)">)
        );
    }   </SPAN><SPAN style="COLOR: rgb(255,128,0)">//end function

    </SPAN><SPAN style="COLOR: rgb(0,119,0)">function </SPAN><SPAN style="COLOR: rgb(0,0,187)">open</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(0,0,187)">$save_path</SPAN><SPAN style="COLOR: rgb(0,119,0)">, </SPAN><SPAN style="COLOR: rgb(0,0,187)">$session_name</SPAN><SPAN style="COLOR: rgb(0,119,0)">) {
        return </SPAN><SPAN style="COLOR: rgb(0,0,187)">true</SPAN><SPAN style="COLOR: rgb(0,119,0)">;
    }   </SPAN><SPAN style="COLOR: rgb(255,128,0)">//end function

    </SPAN><SPAN style="COLOR: rgb(0,119,0)">function </SPAN><SPAN style="COLOR: rgb(0,0,187)">close</SPAN><SPAN style="COLOR: rgb(0,119,0)">() {
        global </SPAN><SPAN style="COLOR: rgb(0,0,187)">$MY_SESS_CONN</SPAN><SPAN style="COLOR: rgb(0,119,0)">;

        if (</SPAN><SPAN style="COLOR: rgb(0,0,187)">$MY_SESS_CONN</SPAN><SPAN style="COLOR: rgb(0,119,0)">) {    </SPAN><SPAN style="COLOR: rgb(255,128,0)">//关闭数据库连接
            </SPAN><SPAN style="COLOR: rgb(0,0,187)">$MY_SESS_CONN</SPAN><SPAN style="COLOR: rgb(0,119,0)">-></SPAN><SPAN style="COLOR: rgb(0,0,187)">Close</SPAN><SPAN style="COLOR: rgb(0,119,0)">();
        }
        return </SPAN><SPAN style="COLOR: rgb(0,0,187)">true</SPAN><SPAN style="COLOR: rgb(0,119,0)">;
    }   </SPAN><SPAN style="COLOR: rgb(255,128,0)">//end function

    </SPAN><SPAN style="COLOR: rgb(0,119,0)">function </SPAN><SPAN style="COLOR: rgb(0,0,187)">read</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(0,0,187)">$sesskey</SPAN><SPAN style="COLOR: rgb(0,119,0)">) {
        global </SPAN><SPAN style="COLOR: rgb(0,0,187)">$MY_SESS_CONN</SPAN><SPAN style="COLOR: rgb(0,119,0)">;

        </SPAN><SPAN style="COLOR: rgb(0,0,187)">$sql </SPAN><SPAN style="COLOR: rgb(0,119,0)">= </SPAN><SPAN style="COLOR: rgb(221,0,0)">'SELECT data FROM sess WHERE sesskey=' </SPAN><SPAN style="COLOR: rgb(0,119,0)">. </SPAN><SPAN style="COLOR: rgb(0,0,187)">$MY_SESS_CONN</SPAN><SPAN style="COLOR: rgb(0,119,0)">-></SPAN><SPAN style="COLOR: rgb(0,0,187)">qstr</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(0,0,187)">$sesskey</SPAN><SPAN style="COLOR: rgb(0,119,0)">) . </SPAN><SPAN style="COLOR: rgb(221,0,0)">' AND expiry>=' </SPAN><SPAN style="COLOR: rgb(0,119,0)">. </SPAN><SPAN style="COLOR: rgb(0,0,187)">time</SPAN><SPAN style="COLOR: rgb(0,119,0)">();
        </SPAN><SPAN style="COLOR: rgb(0,0,187)">$rs </SPAN><SPAN style="COLOR: rgb(0,119,0)">=& </SPAN><SPAN style="COLOR: rgb(0,0,187)">$MY_SESS_CONN</SPAN><SPAN style="COLOR: rgb(0,119,0)">-></SPAN><SPAN style="COLOR: rgb(0,0,187)">Execute</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(0,0,187)">$sql</SPAN><SPAN style="COLOR: rgb(0,119,0)">);
        if (</SPAN><SPAN style="COLOR: rgb(0,0,187)">$rs</SPAN><SPAN style="COLOR: rgb(0,119,0)">) {
            if (</SPAN><SPAN style="COLOR: rgb(0,0,187)">$rs</SPAN><SPAN style="COLOR: rgb(0,119,0)">-></SPAN><SPAN style="COLOR: rgb(0,0,187)">EOF</SPAN><SPAN style="COLOR: rgb(0,119,0)">) {
                return </SPAN><SPAN style="COLOR: rgb(221,0,0)">''</SPAN><SPAN style="COLOR: rgb(0,119,0)">;
            } else {    </SPAN><SPAN style="COLOR: rgb(255,128,0)">//读取到对应于 SESSION ID 的 SESSION 数据
                </SPAN><SPAN style="COLOR: rgb(0,0,187)">$v </SPAN><SPAN style="COLOR: rgb(0,119,0)">= </SPAN><SPAN style="COLOR: rgb(0,0,187)">$rs</SPAN><SPAN style="COLOR: rgb(0,119,0)">-></SPAN><SPAN style="COLOR: rgb(0,0,187)">fields</SPAN><SPAN style="COLOR: rgb(0,119,0)">[</SPAN><SPAN style="COLOR: rgb(0,0,187)">0</SPAN><SPAN style="COLOR: rgb(0,119,0)">];
                </SPAN><SPAN style="COLOR: rgb(0,0,187)">$rs</SPAN><SPAN style="COLOR: rgb(0,119,0)">-></SPAN><SPAN style="COLOR: rgb(0,0,187)">Close</SPAN><SPAN style="COLOR: rgb(0,119,0)">();
                return </SPAN><SPAN style="COLOR: rgb(0,0,187)">$v</SPAN><SPAN style="COLOR: rgb(0,119,0)">;
            }   </SPAN><SPAN style="COLOR: rgb(255,128,0)">//end if
        </SPAN><SPAN style="COLOR: rgb(0,119,0)">}   </SPAN><SPAN style="COLOR: rgb(255,128,0)">//end if
        </SPAN><SPAN style="COLOR: rgb(0,119,0)">return </SPAN><SPAN style="COLOR: rgb(221,0,0)">''</SPAN><SPAN style="COLOR: rgb(0,119,0)">;
    }   </SPAN><SPAN style="COLOR: rgb(255,128,0)">//end function

    </SPAN><SPAN style="COLOR: rgb(0,119,0)">function </SPAN><SPAN style="COLOR: rgb(0,0,187)">write</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(0,0,187)">$sesskey</SPAN><SPAN style="COLOR: rgb(0,119,0)">, </SPAN><SPAN style="COLOR: rgb(0,0,187)">$data</SPAN><SPAN style="COLOR: rgb(0,119,0)">) {
        global </SPAN><SPAN style="COLOR: rgb(0,0,187)">$MY_SESS_CONN</SPAN><SPAN style="COLOR: rgb(0,119,0)">;
        
        </SPAN><SPAN style="COLOR: rgb(0,0,187)">$qkey </SPAN><SPAN style="COLOR: rgb(0,119,0)">= </SPAN><SPAN style="COLOR: rgb(0,0,187)">$MY_SESS_CONN</SPAN><SPAN style="COLOR: rgb(0,119,0)">-></SPAN><SPAN style="COLOR: rgb(0,0,187)">qstr</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(0,0,187)">$sesskey</SPAN><SPAN style="COLOR: rgb(0,119,0)">);
        </SPAN><SPAN style="COLOR: rgb(0,0,187)">$expiry </SPAN><SPAN style="COLOR: rgb(0,119,0)">= </SPAN><SPAN style="COLOR: rgb(0,0,187)">time</SPAN><SPAN style="COLOR: rgb(0,119,0)">() + </SPAN><SPAN style="COLOR: rgb(0,0,187)">My_SESS_TIME</SPAN><SPAN style="COLOR: rgb(0,119,0)">;    </SPAN><SPAN style="COLOR: rgb(255,128,0)">//设置过期时间
        
        //写入 SESSION
        </SPAN><SPAN style="COLOR: rgb(0,0,187)">$arr </SPAN><SPAN style="COLOR: rgb(0,119,0)">= array(
            </SPAN><SPAN style="COLOR: rgb(221,0,0)">'sesskey' </SPAN><SPAN style="COLOR: rgb(0,119,0)">=> </SPAN><SPAN style="COLOR: rgb(0,0,187)">$qkey</SPAN><SPAN style="COLOR: rgb(0,119,0)">,
            </SPAN><SPAN style="COLOR: rgb(221,0,0)">'expiry'  </SPAN><SPAN style="COLOR: rgb(0,119,0)">=> </SPAN><SPAN style="COLOR: rgb(0,0,187)">$expiry</SPAN><SPAN style="COLOR: rgb(0,119,0)">,
            </SPAN><SPAN style="COLOR: rgb(221,0,0)">'data'    </SPAN><SPAN style="COLOR: rgb(0,119,0)">=> </SPAN><SPAN style="COLOR: rgb(0,0,187)">$data</SPAN><SPAN style="COLOR: rgb(0,119,0)">);
        </SPAN><SPAN style="COLOR: rgb(0,0,187)">$MY_SESS_CONN</SPAN><SPAN style="COLOR: rgb(0,119,0)">-></SPAN><SPAN style="COLOR: rgb(0,0,187)">Replace</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(221,0,0)">'sess'</SPAN><SPAN style="COLOR: rgb(0,119,0)">, </SPAN><SPAN style="COLOR: rgb(0,0,187)">$arr</SPAN><SPAN style="COLOR: rgb(0,119,0)">, </SPAN><SPAN style="COLOR: rgb(221,0,0)">'sesskey'</SPAN><SPAN style="COLOR: rgb(0,119,0)">, </SPAN><SPAN style="COLOR: rgb(0,0,187)">$autoQuote </SPAN><SPAN style="COLOR: rgb(0,119,0)">= </SPAN><SPAN style="COLOR: rgb(0,0,187)">true</SPAN><SPAN style="COLOR: rgb(0,119,0)">);
        return </SPAN><SPAN style="COLOR: rgb(0,0,187)">true</SPAN><SPAN style="COLOR: rgb(0,119,0)">;
    }   </SPAN><SPAN style="COLOR: rgb(255,128,0)">//end function

    </SPAN><SPAN style="COLOR: rgb(0,119,0)">function </SPAN><SPAN style="COLOR: rgb(0,0,187)">destroy</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(0,0,187)">$sesskey</SPAN><SPAN style="COLOR: rgb(0,119,0)">) {
        global </SPAN><SPAN style="COLOR: rgb(0,0,187)">$MY_SESS_CONN</SPAN><SPAN style="COLOR: rgb(0,119,0)">;

        </SPAN><SPAN style="COLOR: rgb(0,0,187)">$sql </SPAN><SPAN style="COLOR: rgb(0,119,0)">= </SPAN><SPAN style="COLOR: rgb(221,0,0)">'DELETE FROM sess WHERE sesskey=' </SPAN><SPAN style="COLOR: rgb(0,119,0)">. </SPAN><SPAN style="COLOR: rgb(0,0,187)">$MY_SESS_CONN</SPAN><SPAN style="COLOR: rgb(0,119,0)">-></SPAN><SPAN style="COLOR: rgb(0,0,187)">qstr</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(0,0,187)">$sesskey</SPAN><SPAN style="COLOR: rgb(0,119,0)">);
        </SPAN><SPAN style="COLOR: rgb(0,0,187)">$rs </SPAN><SPAN style="COLOR: rgb(0,119,0)">=& </SPAN><SPAN style="COLOR: rgb(0,0,187)">$MY_SESS_CONN</SPAN><SPAN style="COLOR: rgb(0,119,0)">-></SPAN><SPAN style="COLOR: rgb(0,0,187)">Execute</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(0,0,187)">$sql</SPAN><SPAN style="COLOR: rgb(0,119,0)">);
        return </SPAN><SPAN style="COLOR: rgb(0,0,187)">true</SPAN><SPAN style="COLOR: rgb(0,119,0)">;
    }   </SPAN><SPAN style="COLOR: rgb(255,128,0)">//end function

    </SPAN><SPAN style="COLOR: rgb(0,119,0)">function </SPAN><SPAN style="COLOR: rgb(0,0,187)">gc</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(0,0,187)">$maxlifetime </SPAN><SPAN style="COLOR: rgb(0,119,0)">= </SPAN><SPAN style="COLOR: rgb(0,0,187)">null</SPAN><SPAN style="COLOR: rgb(0,119,0)">) {
        global </SPAN><SPAN style="COLOR: rgb(0,0,187)">$MY_SESS_CONN</SPAN><SPAN style="COLOR: rgb(0,119,0)">;

        </SPAN><SPAN style="COLOR: rgb(0,0,187)">$sql </SPAN><SPAN style="COLOR: rgb(0,119,0)">= </SPAN><SPAN style="COLOR: rgb(221,0,0)">'DELETE FROM sess WHERE expiry<' </SPAN><SPAN style="COLOR: rgb(0,119,0)">. </SPAN><SPAN style="COLOR: rgb(0,0,187)">time</SPAN><SPAN style="COLOR: rgb(0,119,0)">();
        </SPAN><SPAN style="COLOR: rgb(0,0,187)">$MY_SESS_CONN</SPAN><SPAN style="COLOR: rgb(0,119,0)">-></SPAN><SPAN style="COLOR: rgb(0,0,187)">Execute</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(0,0,187)">$sql</SPAN><SPAN style="COLOR: rgb(0,119,0)">);
        </SPAN><SPAN style="COLOR: rgb(255,128,0)">//由于经常性的对表 sess 做删除操作,容易产生碎片,
        //所以在垃圾回收中对该表进行优化操作。
        </SPAN><SPAN style="COLOR: rgb(0,0,187)">$sql </SPAN><SPAN style="COLOR: rgb(0,119,0)">= </SPAN><SPAN style="COLOR: rgb(221,0,0)">'OPTIMIZE TABLE sess'</SPAN><SPAN style="COLOR: rgb(0,119,0)">;
        </SPAN><SPAN style="COLOR: rgb(0,0,187)">$MY_SESS_CONN</SPAN><SPAN style="COLOR: rgb(0,119,0)">-></SPAN><SPAN style="COLOR: rgb(0,0,187)">Execute</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(0,0,187)">$sql</SPAN><SPAN style="COLOR: rgb(0,119,0)">);
        return </SPAN><SPAN style="COLOR: rgb(0,0,187)">true</SPAN><SPAN style="COLOR: rgb(0,119,0)">;
    }   </SPAN><SPAN style="COLOR: rgb(255,128,0)">//end function
</SPAN><SPAN style="COLOR: rgb(0,119,0)">}   </SPAN><SPAN style="COLOR: rgb(255,128,0)">///:~

//使用 ADOdb 作为数据库抽象层。
</SPAN><SPAN style="COLOR: rgb(0,119,0)">require_once(</SPAN><SPAN style="COLOR: rgb(221,0,0)">'adodb/adodb.inc.php'</SPAN><SPAN style="COLOR: rgb(0,119,0)">);
</SPAN><SPAN style="COLOR: rgb(255,128,0)">//数据库配置项,可放入配置文件中(如:config.inc.php)。
</SPAN><SPAN style="COLOR: rgb(0,0,187)">$db_type </SPAN><SPAN style="COLOR: rgb(0,119,0)">= </SPAN><SPAN style="COLOR: rgb(221,0,0)">'mysql'</SPAN><SPAN style="COLOR: rgb(0,119,0)">;
</SPAN><SPAN style="COLOR: rgb(0,0,187)">$db_host </SPAN><SPAN style="COLOR: rgb(0,119,0)">= </SPAN><SPAN style="COLOR: rgb(221,0,0)">'192.168.212.1'</SPAN><SPAN style="COLOR: rgb(0,119,0)">;
</SPAN><SPAN style="COLOR: rgb(0,0,187)">$db_user </SPAN><SPAN style="COLOR: rgb(0,119,0)">= </SPAN><SPAN style="COLOR: rgb(221,0,0)">'sess_user'</SPAN><SPAN style="COLOR: rgb(0,119,0)">;
</SPAN><SPAN style="COLOR: rgb(0,0,187)">$db_pass </SPAN><SPAN style="COLOR: rgb(0,119,0)">= </SPAN><SPAN style="COLOR: rgb(221,0,0)">'sess_pass'</SPAN><SPAN style="COLOR: rgb(0,119,0)">;
</SPAN><SPAN style="COLOR: rgb(0,0,187)">$db_name </SPAN><SPAN style="COLOR: rgb(0,119,0)">= </SPAN><SPAN style="COLOR: rgb(221,0,0)">'sess_db'</SPAN><SPAN style="COLOR: rgb(0,119,0)">;
</SPAN><SPAN style="COLOR: rgb(255,128,0)">//创建数据库连接,这是一个全局变量。
</SPAN><SPAN style="COLOR: rgb(0,0,187)">$GLOBALS</SPAN><SPAN style="COLOR: rgb(0,119,0)">[</SPAN><SPAN style="COLOR: rgb(221,0,0)">'MY_SESS_CONN'</SPAN><SPAN style="COLOR: rgb(0,119,0)">] =& </SPAN><SPAN style="COLOR: rgb(0,0,187)">ADONewConnection</SPAN><SPAN style="COLOR: rgb(0,119,0)">(</SPAN><SPAN style="COLOR: rgb(0,0,187)">$db_type</SPAN><SPAN style="COLOR: rgb(0,119,0)">);
</SPAN><SPAN style="COLOR: rgb(0,0,187)">$GLOBALS</SPAN><SPAN style="COLOR: rgb(0,119,0)">[</SPAN><SPAN style="COLOR: rgb(221,0,0)">'MY_SESS_CONN'</SPAN><SPAN style="COLOR: rgb(0,119,0)">]-></SPAN><SPAN style="COLOR: rgb(0,0,187)">Connect</SPAN><SPAN style="COLOR: rgb(0,119,0)">( </SPAN><SPAN style="COLOR: rgb(0,0,187)">$db_host</SPAN><SPAN style="COLOR: rgb(0,119,0)">, </SPAN><SPAN style="COLOR: rgb(0,0,187)">$db_user</SPAN><SPAN style="COLOR: rgb(0,119,0)">, </SPAN><SPAN style="COLOR: rgb(0,0,187)">$db_pass</SPAN><SPAN style="COLOR: rgb(0,119,0)">, </SPAN><SPAN style="COLOR: rgb(0,0,187)">$db_name</SPAN><SPAN style="COLOR: rgb(0,119,0)">);
</SPAN><SPAN style="COLOR: rgb(255,128,0)">//初始化 SESSION 设置,必须在 session_start() 之前运行!!
</SPAN><SPAN style="COLOR: rgb(0,0,187)">My_Sess</SPAN><SPAN style="COLOR: rgb(0,119,0)">::</SPAN><SPAN style="COLOR: rgb(0,0,187)">init</SPAN><SPAN style="COLOR: rgb(0,119,0)">();
</SPAN><SPAN style="COLOR: rgb(0,0,187)">?></SPAN>
</SPAN>
</CODE>

五、遗留问题

如果网站的访问量很大的话,SESSION 的读写会频繁地对数据库进行操作,这样效率就会明显降低。考虑到 SESSION 数据一般不会很大,可以尝试用 C/Java 写个多线程的程序,用 HASH 表保存 SESSION 数据,并通过 socket 通信进行数据读写,这样 SESSION 就保存在内存中,读写速度应该会快很多。另外还可以通过负载均衡来分担服务器负载。不过这些都只是我自己的一些想法和假设,并没有实践过

www.bkjia.comtruehttp://www.bkjia.com/PHPjc/317992.htmlTechArticlePHP 实现多服务器共享 SESSION 数据 一、问题起源 稍大一些的网站,通常都会有好几个服务器,每个服务器运行着不同功能的模块,使用不同...
陳述
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
PHP的當前狀態:查看網絡開發趨勢PHP的當前狀態:查看網絡開發趨勢Apr 13, 2025 am 12:20 AM

PHP在現代Web開發中仍然重要,尤其在內容管理和電子商務平台。 1)PHP擁有豐富的生態系統和強大框架支持,如Laravel和Symfony。 2)性能優化可通過OPcache和Nginx實現。 3)PHP8.0引入JIT編譯器,提升性能。 4)雲原生應用通過Docker和Kubernetes部署,提高靈活性和可擴展性。

PHP與其他語言:比較PHP與其他語言:比較Apr 13, 2025 am 12:19 AM

PHP適合web開發,特別是在快速開發和處理動態內容方面表現出色,但不擅長數據科學和企業級應用。與Python相比,PHP在web開發中更具優勢,但在數據科學領域不如Python;與Java相比,PHP在企業級應用中表現較差,但在web開發中更靈活;與JavaScript相比,PHP在後端開發中更簡潔,但在前端開發中不如JavaScript。

PHP與Python:核心功能PHP與Python:核心功能Apr 13, 2025 am 12:16 AM

PHP和Python各有優勢,適合不同場景。 1.PHP適用於web開發,提供內置web服務器和豐富函數庫。 2.Python適合數據科學和機器學習,語法簡潔且有強大標準庫。選擇時應根據項目需求決定。

PHP:網絡開發的關鍵語言PHP:網絡開發的關鍵語言Apr 13, 2025 am 12:08 AM

PHP是一種廣泛應用於服務器端的腳本語言,特別適合web開發。 1.PHP可以嵌入HTML,處理HTTP請求和響應,支持多種數據庫。 2.PHP用於生成動態網頁內容,處理表單數據,訪問數據庫等,具有強大的社區支持和開源資源。 3.PHP是解釋型語言,執行過程包括詞法分析、語法分析、編譯和執行。 4.PHP可以與MySQL結合用於用戶註冊系統等高級應用。 5.調試PHP時,可使用error_reporting()和var_dump()等函數。 6.優化PHP代碼可通過緩存機制、優化數據庫查詢和使用內置函數。 7

PHP:許多網站的基礎PHP:許多網站的基礎Apr 13, 2025 am 12:07 AM

PHP成為許多網站首選技術棧的原因包括其易用性、強大社區支持和廣泛應用。 1)易於學習和使用,適合初學者。 2)擁有龐大的開發者社區,資源豐富。 3)廣泛應用於WordPress、Drupal等平台。 4)與Web服務器緊密集成,簡化開發部署。

超越炒作:評估當今PHP的角色超越炒作:評估當今PHP的角色Apr 12, 2025 am 12:17 AM

PHP在現代編程中仍然是一個強大且廣泛使用的工具,尤其在web開發領域。 1)PHP易用且與數據庫集成無縫,是許多開發者的首選。 2)它支持動態內容生成和麵向對象編程,適合快速創建和維護網站。 3)PHP的性能可以通過緩存和優化數據庫查詢來提升,其廣泛的社區和豐富生態系統使其在當今技術棧中仍具重要地位。

PHP中的弱參考是什麼?什麼時候有用?PHP中的弱參考是什麼?什麼時候有用?Apr 12, 2025 am 12:13 AM

在PHP中,弱引用是通過WeakReference類實現的,不會阻止垃圾回收器回收對象。弱引用適用於緩存系統和事件監聽器等場景,需注意其不能保證對象存活,且垃圾回收可能延遲。

解釋PHP中的__ Invoke Magic方法。解釋PHP中的__ Invoke Magic方法。Apr 12, 2025 am 12:07 AM

\_\_invoke方法允許對象像函數一樣被調用。 1.定義\_\_invoke方法使對象可被調用。 2.使用$obj(...)語法時,PHP會執行\_\_invoke方法。 3.適用於日誌記錄和計算器等場景,提高代碼靈活性和可讀性。

See all articles

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

AI Hentai Generator

AI Hentai Generator

免費產生 AI 無盡。

熱門文章

R.E.P.O.能量晶體解釋及其做什麼(黃色晶體)
3 週前By尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.最佳圖形設置
3 週前By尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.如果您聽不到任何人,如何修復音頻
3 週前By尊渡假赌尊渡假赌尊渡假赌
WWE 2K25:如何解鎖Myrise中的所有內容
4 週前By尊渡假赌尊渡假赌尊渡假赌

熱工具

mPDF

mPDF

mPDF是一個PHP庫,可以從UTF-8編碼的HTML產生PDF檔案。原作者Ian Back編寫mPDF以從他的網站上「即時」輸出PDF文件,並處理不同的語言。與原始腳本如HTML2FPDF相比,它的速度較慢,並且在使用Unicode字體時產生的檔案較大,但支援CSS樣式等,並進行了大量增強。支援幾乎所有語言,包括RTL(阿拉伯語和希伯來語)和CJK(中日韓)。支援嵌套的區塊級元素(如P、DIV),

SecLists

SecLists

SecLists是最終安全測試人員的伙伴。它是一個包含各種類型清單的集合,這些清單在安全評估過程中經常使用,而且都在一個地方。 SecLists透過方便地提供安全測試人員可能需要的所有列表,幫助提高安全測試的效率和生產力。清單類型包括使用者名稱、密碼、URL、模糊測試有效載荷、敏感資料模式、Web shell等等。測試人員只需將此儲存庫拉到新的測試機上,他就可以存取所需的每種類型的清單。

EditPlus 中文破解版

EditPlus 中文破解版

體積小,語法高亮,不支援程式碼提示功能

SublimeText3 Linux新版

SublimeText3 Linux新版

SublimeText3 Linux最新版

Dreamweaver Mac版

Dreamweaver Mac版

視覺化網頁開發工具