PHP利用MySQL保存session的实现思路及示例代码,mysqlsession
实现环境:
PHP 5.4.24
MySQL 5.6.19
OS X 10.9.4/Apache 2.2.26
一、代码
CREATE TABLE `session` ( `skey` char(32) CHARACTER SET ascii NOT NULL, `data` text COLLATE utf8mb4_bin, `expire` int(11) NOT NULL, PRIMARY KEY (`skey`), KEY `index_session_expire` (`expire`) USING BTREE ) ENGINE=MyISAM DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_bin;
<?php /* * 连接数据库所需的DNS、用户名、密码等,一般情况不会在代码中进行更改, * 所以使用常量的形式,可以避免在函数中引用而需要global。 */ define('SESSION_DNS', 'mysql:host=localhost;dbname=db;charset=utf8mb4'); define('SESSION_USR', 'usr'); define('SESSION_PWD', 'pwd'); define('SESSION_MAXLIFETIME', get_cfg_var('session.gc_maxlifetime')); //创建PDO连接 //持久化连接可以提供更好的效率 function getConnection() { try { $conn = new PDO(SESSION_DNS, SESSION_USR, SESSION_PWD, array( PDO::ATTR_PERSISTENT => TRUE, PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION, PDO::ATTR_EMULATE_PREPARES => FALSE )); return $conn; } catch (Exception $ex) { } } //自定义的session的open函数 function sessionMysqlOpen($savePath, $sessionName) { return TRUE; } //自定义的session的close函数 function sessionMysqlClose() { return TRUE; } /* * 由于一般不会把用户提交的数据直接保存到session,所以普通情况不存在注入问题。 * 且处理session数据的SQL语句也不会多次使用。因此预处理功能的效益无法体现。 * 所以,实际工程中可以不必教条的使用预处理功能。 */ /* * sessionMysqlRead()函数中,首先通过SELECT count(*)来判断sessionID是否存在。 * 由于MySQL数据库提供SELECT对PDOStatement::rowCount()的支持, * 因此,实际的工程中可以直接使用rowCount()进行判断。 */ //自定义的session的read函数 //SQL语句中增加了“expire > time()”判断,用以避免读取过期的session。 function sessionMysqlRead($sessionId) { try { $dbh = getConnection(); $time = time(); $sql = 'SELECT count(*) AS `count` FROM session ' . 'WHERE skey = ? and expire > ?'; $stmt = $dbh->prepare($sql); $stmt->execute(array($sessionId, $time)); $data = $stmt->fetch(PDO::FETCH_ASSOC)['count']; if ($data = 0) { return ''; } $sql = 'SELECT `data` FROM `session` ' . 'WHERE `skey` = ? and `expire` > ?'; $stmt = $dbh->prepare($sql); $stmt->execute(array($sessionId, $time)); $data = $stmt->fetch(PDO::FETCH_ASSOC)['data']; return $data; } catch (Exception $e) { return ''; } } //自定义的session的write函数 //expire字段存储的数据为当前时间+session生命期,当这个值小于time()时表明session失效。 function sessionMysqlWrite($sessionId, $data) { try { $dbh = getConnection(); $expire = time() + SESSION_MAXLIFETIME; $sql = 'INSERT INTO `session` (`skey`, `data`, `expire`) ' . 'values (?, ?, ?) ' . 'ON DUPLICATE KEY UPDATE data = ?, expire = ?'; $stmt = $dbh->prepare($sql); $stmt->execute(array($sessionId, $data, $expire, $data, $expire)); } catch (Exception $e) { echo $e->getMessage(); } } //自定义的session的destroy函数 function sessionMysqlDestroy($sessionId) { try { $dbh = getConnection(); $sql = 'DELETE FROM `session` where skey = ?'; $stmt = $dbh->prepare($sql); $stmt->execute(array($sessionId)); return TRUE; } catch (Exception $e) { return FALSE; } } //自定义的session的gc函数 function sessionMysqlGc($lifetime) { try { $dbh = getConnection(); $sql = 'DELETE FROM `session` WHERE expire < ?'; $stmt = $dbh->prepare($sql); $stmt->execute(array(time())); $dbh = NULL; return TRUE; } catch (Exception $e) { return FALSE; } } //自定义的session的session id设置函数 /* * 由于在session_start()之前,SID和session_id()均无效, * 故使用$_GET[session_name()]和$_COOKIE[session_name()]进行检测。 * 如果此两者均为空,则表明session尚未建立,需要为新session设置session id。 * 通过MySQL数据库获取uuid作为session id可以更好的避免session id碰撞。 */ function sessionMysqlId() { if (filter_input(INPUT_GET, session_name()) == '' and filter_input(INPUT_COOKIE, session_name()) == '') { try { $dbh = getConnection(); $stmt = $dbh->query('SELECT uuid() AS uuid'); $data = $stmt->fetch(PDO::FETCH_ASSOC)['uuid']; $data = str_replace('-', '', $data); session_id($data); return TRUE; } catch (Exception $ex) { return FALSE; } } } //session启动函数,包括了session_start()及其之前的所有步骤。 function startSession() { session_set_save_handler( 'sessionMysqlOpen', 'sessionMysqlClose', 'sessionMysqlRead', 'sessionMysqlWrite', 'sessionMysqlDestroy', 'sessionMysqlGc'); register_shutdown_function('session_write_close'); sessionMysqlId(); session_start(); }
二、简介
使用MySQL保存session,需要保存三个关键性的数据:session id、session数据、session生命期。
考虑到session的使用方式,没必要使用InnoDB引擎,MyISAM引擎可以获得更好的性能。如果环境允许,可以尝试使用MEMORY引擎。
保存session数据的列,有需要的话,可以使用utf8或utf8mb4字符集;保存session id的列则没有必要,一般情况使用ascii字符集就可以了,可以节约存储成本。
保存session生命期的列,可以根据工程需要进行设计。比如datetime类型、timestamp类型、int类型。对于datetime、int类型可以保存session生成时间或过期时间。
如果有必要可以扩展session表的列并修改读、写函数以支持(维护)相关列来保存诸如用户名等信息。
当前版本,只要通过session_set_save_handler注册自定义的会话维护函数就可以,不需要在其之前使用session_module_name('user')函数。
当read函数获取数据并返回,PHP会自动对其进行反序列化,一般情况请不要对数据进行更改。
PHP传递给write函数的date参数是序列化之后的session数据,直接保存即可,一般情况请不要对数据进行更改。
按照本段代码的逻辑,PHP配置选项关于会话生命期的设置已经不再有效,这个值可以自行维护,不一定需要通过get_cfg_var获取。
sessionMysqlId()函数是为了避免大用户量、多台Web服务器情况下的碰撞,一般情况PHP自动生成的session id是可以满足用户要求的。
没了
三、需求
当用户量非常大,需要多台服务器提供应用的时候,使用MySQL存储会话相对使用会话文件具有一定的优越性。比如具有最小的存储开销,比如可以避免文件共享带来的复杂性,比如可以更好的避免发生碰撞,比如相比会话文件共享具有更好的性能。总体上来说,当访问量剧增的时候,如果使用数据库保存会话带来的问题是线性增长的,那么使用会话文件带来的问题几乎是爆炸性的。好吧,换一个更直白的说法吧:如果您的应用用户量不大,其实让PHP自己处理session就好了,没必要考虑MySQL。
这个问题,可以用到数组存储,最后输出成分隔的字符串,是多字符串的操作。
我可以发一份示例代码给你。
$gb_DBname="charles_friend";//数据库名称
$gb_DBuser="charles_friend";//数据库用户名称
$gb_DBpass="wxyzoui";//数据库密码
$gb_DBHOSTname="localhost";//主机的名称或是IP地址
$SESS_DBH="";
$SESS_LIFE=get_cfg_var("session.gc_maxlifetime");//得到session的最大有效期。
function sess_open($save_path,$session_name){
global $gb_DBHOSTname,$gb_DBname,$gb_DBuser,$gb_DBpass,$SESS_DBH;
if(!$SESS_DBH=mysql_pconnect($gb_DBHOSTname,$gb_DBuser,$gb_DBpass)){
echo "
die();
}
if(!mysql_select_db($gb_DBname,$SESS_DBH)){
echo "
die();
}
return true;
}
function sess_close(){
return true;
}
function sess_read($key){
global $SESS_DBH,$SESS_LIFE;
$qry="select value from db_session where sesskey = '$key' and expiry > ".time();
$qid=mysql_query($qry,$SESS_DBH);
if(list($value)=mysql_fetch_row($qid)){
return $value;
}
return false;
}
function sess_write($key,$val){
global $SESS_DBH,$SESS_LIFE;
$expiry=time()+$SESS_LIFE;
$value=$val;
$qry="insert into db_session values('$key',$expiry,'$value')";
$qid=mysql_query($qry,$SESS_DBH);
if(!$qid){
$qry="update db_session set expiry=$expiry, value='$value' where sesskey='$key' and expiry >".time();
$qid=mysql_query($qry,$SESS_DBH);
}
return $qid;
} ......余下全文>>

여전히 인기있는 것은 사용 편의성, 유연성 및 강력한 생태계입니다. 1) 사용 편의성과 간단한 구문은 초보자에게 첫 번째 선택입니다. 2) 웹 개발, HTTP 요청 및 데이터베이스와의 우수한 상호 작용과 밀접하게 통합되었습니다. 3) 거대한 생태계는 풍부한 도구와 라이브러리를 제공합니다. 4) 활성 커뮤니티와 오픈 소스 자연은 새로운 요구와 기술 동향에 맞게 조정됩니다.

PHP와 Python은 웹 개발, 데이터 처리 및 자동화 작업에 널리 사용되는 고급 프로그래밍 언어입니다. 1.PHP는 종종 동적 웹 사이트 및 컨텐츠 관리 시스템을 구축하는 데 사용되며 Python은 종종 웹 프레임 워크 및 데이터 과학을 구축하는 데 사용됩니다. 2.PHP는 Echo를 사용하여 콘텐츠를 출력하고 Python은 인쇄를 사용합니다. 3. 객체 지향 프로그래밍을 지원하지만 구문과 키워드는 다릅니다. 4. PHP는 약한 유형 변환을 지원하는 반면, 파이썬은 더 엄격합니다. 5. PHP 성능 최적화에는 Opcache 및 비동기 프로그래밍 사용이 포함되며 Python은 Cprofile 및 비동기 프로그래밍을 사용합니다.

PHP는 주로 절차 적 프로그래밍이지만 객체 지향 프로그래밍 (OOP)도 지원합니다. Python은 OOP, 기능 및 절차 프로그래밍을 포함한 다양한 패러다임을 지원합니다. PHP는 웹 개발에 적합하며 Python은 데이터 분석 및 기계 학습과 같은 다양한 응용 프로그램에 적합합니다.

PHP는 1994 년에 시작되었으며 Rasmuslerdorf에 의해 개발되었습니다. 원래 웹 사이트 방문자를 추적하는 데 사용되었으며 점차 서버 측 스크립팅 언어로 진화했으며 웹 개발에 널리 사용되었습니다. Python은 1980 년대 후반 Guidovan Rossum에 의해 개발되었으며 1991 년에 처음 출시되었습니다. 코드 가독성과 단순성을 강조하며 과학 컴퓨팅, 데이터 분석 및 기타 분야에 적합합니다.

PHP는 웹 개발 및 빠른 프로토 타이핑에 적합하며 Python은 데이터 과학 및 기계 학습에 적합합니다. 1.PHP는 간단한 구문과 함께 동적 웹 개발에 사용되며 빠른 개발에 적합합니다. 2. Python은 간결한 구문을 가지고 있으며 여러 분야에 적합하며 강력한 라이브러리 생태계가 있습니다.

PHP는 현대화 프로세스에서 많은 웹 사이트 및 응용 프로그램을 지원하고 프레임 워크를 통해 개발 요구에 적응하기 때문에 여전히 중요합니다. 1.PHP7은 성능을 향상시키고 새로운 기능을 소개합니다. 2. Laravel, Symfony 및 Codeigniter와 같은 현대 프레임 워크는 개발을 단순화하고 코드 품질을 향상시킵니다. 3. 성능 최적화 및 모범 사례는 응용 프로그램 효율성을 더욱 향상시킵니다.

phphassignificallyimpactedwebdevelopmentandextendsbeyondit

PHP 유형은 코드 품질과 가독성을 향상시키기위한 프롬프트입니다. 1) 스칼라 유형 팁 : PHP7.0이므로 int, float 등과 같은 기능 매개 변수에 기본 데이터 유형을 지정할 수 있습니다. 2) 반환 유형 프롬프트 : 기능 반환 값 유형의 일관성을 확인하십시오. 3) Union 유형 프롬프트 : PHP8.0이므로 기능 매개 변수 또는 반환 값에 여러 유형을 지정할 수 있습니다. 4) Nullable 유형 프롬프트 : NULL 값을 포함하고 널 값을 반환 할 수있는 기능을 포함 할 수 있습니다.


핫 AI 도구

Undresser.AI Undress
사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover
사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool
무료로 이미지를 벗다

Clothoff.io
AI 옷 제거제

AI Hentai Generator
AI Hentai를 무료로 생성하십시오.

인기 기사

뜨거운 도구

MinGW - Windows용 미니멀리스트 GNU
이 프로젝트는 osdn.net/projects/mingw로 마이그레이션되는 중입니다. 계속해서 그곳에서 우리를 팔로우할 수 있습니다. MinGW: GCC(GNU Compiler Collection)의 기본 Windows 포트로, 기본 Windows 애플리케이션을 구축하기 위한 무료 배포 가능 가져오기 라이브러리 및 헤더 파일로 C99 기능을 지원하는 MSVC 런타임에 대한 확장이 포함되어 있습니다. 모든 MinGW 소프트웨어는 64비트 Windows 플랫폼에서 실행될 수 있습니다.

SublimeText3 중국어 버전
중국어 버전, 사용하기 매우 쉽습니다.

에디트플러스 중국어 크랙 버전
작은 크기, 구문 강조, 코드 프롬프트 기능을 지원하지 않음

Atom Editor Mac 버전 다운로드
가장 인기 있는 오픈 소스 편집기

스튜디오 13.0.1 보내기
강력한 PHP 통합 개발 환경
