PHP中的会话控制

了解HTTP(超文本传输协议)可以知道,它采用请求与响应的模式,最大的特点就是无连接无状态。

  • 无连接:每次连接仅处理一个客户端的请求,得到服务器响应后,连接就结束了
  • 无状态:每个请求都是独立的,服务器无法识别和区分它们的身份

这就造成了一个问题,在不同网页之间如何传递信息,会话控制的思想就是为了解决这个问题的,它的解决方案主要分为Cookie和Session。

一、Cookie

保存在客户端中,又分为内存cookie和硬盘cookie。

  • 内存cookie:由浏览器维护,保存在内存中,浏览器关闭之后就消失了,存在时间短暂
  • 硬盘cookie:保存在硬盘中,有一个过期时间,仅手动删除或过期才消失

Cookie的使用场景主要有记住登录,购物车等,在PHP中操作cookie主要通过setcookie和setrawcookie两个方法来设置。

setcookie(name,value,[expire],[path],[domain],[secure],[httponly])

  • name 必需。名称。
  • value 必需。值。
  • expire 可选。有效期。
  • path 可选。服务器有效路径(默认当前路径)。
  • domain 可选。作用域(默认本域)。
  • secure 可选。仅HTTPS可用(默认false)。
  • httponly可选。仅http可访问(默认false),防止JS修改,减少XSS攻击。
setcookie("abc","123",time()+3600)

读取cookie使用$_COOKIE,更新和删除均使用setcookie方法,注意保证path和domain与之前一致,删除设置过期即可,如time()-1

还可以通过header设置cookie:

header("Set-Cookie:abc=123; expires=".gmstrftime("%A,%d-%b-%Y %H:%M:%S GMT",time()+3600));

通过JS操作cookie的值,下面是一个封装的实例:

var Cookie={
  set:function(key,val,expire){
    // 判断是否设置了过期时间
    if(expire){
      var date=new Date();
      date.setTime(date.getTime()+expire*24*3600*1000);//格式化时间
      var expireStr="expires="+date.toGMTString()+';';
    }else{
      var expireStr='';
    }
    document.cookie=key+'='+escape(val)+';'+expireStr;
  },
  get:function(key){
    var getCookie=document.cookie.replace(/[ ]/g,'');//去空格
    var resArr=getCookie.split(';')
    var res;
    for(var i=0,l=resArr.length;i<l;i++){
      var arr=resArr.split('=');
      if(arr[0]===key){
        res=arr[1];break;
      }
    }
    return unescape(res);
  }
}

 cookie实现自动登录,主要通过生成令牌存储到本地,下次访问时获取令牌中的id,查询数据库得到用户名和密码,加上掩值重新生成令牌与之比对,如相同则直接登录。

<?php
define('GAVIN', true);//权限标示常量,申明后方可引入文件
include("../include/Gavin.class.php");

$username=$_POST['username'];
$password=md5($_POST['password']);
$autologin=$_POST['autologin'];

Gavin::connDb();
$res = Gavin::dbSelect("SELECT * FROM main WHERE name='$username' and password='$password'");
if(count($res)===1){
    if($autologin==="1"){
        setcookie("username",$username,strtotime('+7 days'));
        // 给用户一个登录凭证
        $salt='xiaoguge';//加密掩值
        $tokens=md5($username.$password.$salt).":".$res[0][0];//拼接令牌:加密(用户名+密码+掩值)拼接用户id
        setcookie("tokens",$tokens,strtotime('+7 days'));
    }else{setcookie("username",$username);}echo Gavin::createJson(200,"登录成功!");
}
else{echo Gavin::createJson(400,"登录失败,用户名或密码错误!");}
?>

cookie的缺点:不够安全数据容易被截取;每个域名下大小有限制,cookie中最大字节数为4K;cookie每次都附着在http请求头中。

为了解决cookie在存储上的这些缺陷,HTML5提出了本地存储方案localStorage和sessionStorage。

二、session

session的工作原理:

  1. 准备建立会话时,PHP首先查看请求的cookie中是否包含session_id,如果没有则创建一条session信息(一般以文件形式存在服务器上)。
  2. 服务器将新创建session信息的session_id发送给浏览器,一般浏览器将其存放在cookie中。
  3. 当浏览器再次访问服务器时,会携带这个session_id,凭借此到服务器session认领对应信息。
  4. 取消会话,可以删除服务器中session的信息。

在PHP中使用会话,必须先使用session_start()开启,再使用$_SESSION进行设置和读取

session_start();
$_SESSION['account']=$account;

删除session:

session_start();

//将session数据清空
$_SESSION=[];

//删除会话cookie
if(ini_get('session.use_cookie')){
  $params=session_get_cookie_params();
  setcookie(session_name(),'',time()-1,$params['path'],$params['domain'],$params['secure'],$params['httponly'])    
}

//销毁会话
session_destroy();

session默认是采用文件形式存储,当然也可也修改PHP的设置,将其存储到数据库中(加快查询速度)。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏吴伟祥

单点登录原理与简单实现 原

web应用采用browser/server架构,http作为通信协议。http是无状态协议,浏览器的每一次请求,服务器会独立处理,不与之前或之后的请求产生关联...

1075
来自专栏ppjun专栏

aapt异常

在编译的时候出现以上错误是--no-version-vectors,在异常中可以发现aapt报错,而我们知道aapt是将对应的资源生成索引就是R文件。下面我尝试...

762
来自专栏DeveWork

WordPress登陆不了后台的原因及解决方法(登陆界面不断返回)

应该说,这是一个常见的现象。这种现象具体为:在后台登陆界面输入账号密码后,点击进入却无任何反应及提示。更换浏览器也是一样。 出现如上问题,请检查一下,你的电脑是...

26611
来自专栏北京马哥教育

ulimit设置句柄数

这几天在做一个性能测试,写了一个模拟发送http的程序。模拟100并发的情况下,随机发http get的请求。放到服务器上运行一段时间抛出Too many op...

3505
来自专栏coder修行路

python爬虫从入门到放弃(二)之爬虫的原理

在上文中我们说了:爬虫就是请求网站并提取数据的自动化程序。其中请求,提取,自动化是爬虫的关键!下面我们分析爬虫的基本流程 爬虫的基本流程 发起请求 通过HTTP...

2119
来自专栏黑白安全

emlog敏感信息泄漏漏洞

漏洞复现:登录状态后访问http://xxx.xxx/admin/index.php?action=phpinfo

812
来自专栏java技术学习之道

单点登录原理与简单实现

1232
来自专栏雪胖纸的玩蛇日常

使用Navicat for MySQL把本地数据库上传到服务器

6945
来自专栏IT杂记

Windows下jps, jconsole无法查看本地java进程问题解决

先通过本地java代码运行: System.out.println(System.getProperties()); 查看属性java.io.tmpdir=C:...

2776
来自专栏linux系统运维

访问日志不记录静态文件,访问日志切割,静态元素过期时间

2106

扫码关注云+社区