原文:https://segmentfault.com/a/1190000000468220

发现,在调用session_start()的时候, session_start() 里面应该是有调用类似 setcookie()函数往客户端写cookie 的。

----------------------------------------------------------------

session 会话机制:

1)如果是基于cookie的会话机制,在调用session_start()之前,是不能够有任何实际的输出的,即使是空格或者是空行!

因为session_start()函数调用的时候,其实是通过setCookie()函数向cookie中设置了
PHPSESSID这个key,对应的value是一个随机的、唯一的32位字符串! 而setCookie前面是不可以有任何实际的输出的!

注意:这里的PHPSESSID名字是在php.ini文件中进行的配置!配置如下图所示:

2)这里我们需要明白session_start()函数的作用究竟是什么:

①如果session机制是基于cookie的,那么当脚本第一次运行的时候
A、 在客户端上session_start()函数会通过setCookie()函数向Cookie中保留一个key,默认情况下Key的名字是PHPSESSID,对应的值是一个32位的、唯一的、随机的字符串!
B、 在服务器端,会产生一个以PHPSESSID的value值为名字的文件!其中保留的是session中的数据!同时,在脚本中创建$_SESSION超全局数组,并将session文件的数据反序列化,添加到$_SESSION数组中!

②当脚本第二次,及以后运行的时候
A、 浏览器端会自动携带COOKIE中的PHPSESSID对应的value值,将数据送至服务器端!
B、在服务器端,一旦开启sessioin_start()的时候,会根据客户端提供的sessionid去寻找对应的session文件,将session中的变量读取出来!在脚本中创建$_SESSION超全局数组,将数据定义到$_SESSION数组中!

注意:$_SESION数组数组只有在调用session_start()函数之后,更确切确切的说,是开启session机制之后]才会存在!之所以这样说,是由于session_start()时,会先得到session_id,通过session_id找到对应的文件内容,然后进行反序列化!如果,我们接着使用session_id()函数来滞空session_id的话,我们就找不到session中的内容了!
同时,$_SESSION是超全局定义数组,他和常量一样,并没有作用域的概念!几乎在哪里都可以使用!


接下来,我们来看一点代码!当我们第一次运行这个脚本,脚本代码如下 脚本A中的代码:

 <?php
session_start();
$_SESSION['name']='maweibin';
?>

服务器端保存session文件夹中的的文件入下:

代码中的保值至如下:

在客户端保存了一个cookie文件,内容如下:

通过观察我们可以发现:PHPSESSIONID对应的value值和服务器端session文件的文件名是一致的!

第二次及其以后运行这个脚本的时候,在服务器端,并没有什么变化

在客户端,发起http请求的时候,我们可以看到:

请求的时候会字段的将cookie中的PHPSESSID带到服务器端!服务器端,会通过提供的sessionid值,将session文件中的数据读取出来!

3)同cookie不同的是,session中的数据不仅可以存放字符串,还可以存放数组和对象!

<?php
session_start();
$name = array('name'=>'jay', 'age'=>'23', 'addr'=>'吉林省');
$_SESSION['voice'] = $name;
?>
<?php
session_start();
var_dump($_SESSION['voice']);
?>

运行的结果如下:

array(3) { ["name"]=> string(3) "jay" ["age"]=> string(2) "23" ["addr"]=> string(9) "吉林省" }

注意:这里我们有必要讨论一下,为什么cookie中不能存放数组之类的变量,而只能够存放字符串!我们来看几段代码:

<?php
session_start();
$name = array('name'=>'jay', 'age'=>'23', 'addr'=>'吉林省');
$_SESSION['voice'] = $name;
?>
在session文件中存储的数据如下:
name|s:8:"maweibin";voice|a:3:{s:4:"name";s:3:"jay";s:3:"age";s:2:"23";s:4:"addr";s:9:"吉林省";} <?php
$expires = time()+3600;
$name = array('name'=>'jay', 'age'=>'23', 'addr'=>'吉林省');
setcookie('name',$name, $expires, '/talkphp/secondtalk/', 'php.test.com');
?>
此时,我们调用一下这段脚本:
<?php
$name = $_COOKIE['name'];
var_dump($name);
得到的结果是null,由此可见这样的写法是不支持的!
?> 我们再看一段代码:
<?php
$expires = time()+3600;
setcookie('person["name"]','liangbo' , $expires, '/talkphp/secondtalk/', 'php.test.com');
setcookie('person["age"]','23' , $expires, '/talkphp/secondtalk/','php.test.com');
?>
运行结果如下:
array(2) { [""name""]=> string(7) "liangbo" [""age""]=> string(2) "23" }

虽然也取到了数据,但是,cookie中的数据却和session中的数据并不相同!但是,我们来看一下cookie文件中存储的数据
person["name"]
liangbo
php.test.com/talkphp/secondtalk/
0
976582400
30365600
3634030379
30365591


person["age"]
23
php.test.com/talkphp/secondtalk/
0
976582400
30365600
3634060380
30365591

由此可见,cookie之所以不能够保存数组或者是对象等变量,是因为cookie本身并没有序列化,和反序列化这一步!这也提示我们,如果,我们手动将将变量进行了序列化和反序列化,就可以用cookie来存储变量了!

4)删除session数组需要注意的地方:

我们可以使用unset()方法干掉$_SESSION[‘key’] ,这样可以单独的干掉一个值,此时$_SESSION数组依然存在!
但是,如果我们需要清空session中的全部数据的时候,是不能够直接unset($_SESSION)。这样在当前脚本周期之内,超全局定义数组$_SESSION 就不存在了!我们也就没有办法操作session中的数据了!

我们来看一段代码:
<?php
var_dump($_SESSION);
session_start();
var_dump($_SESSION);
unset($_SESSION);
session_start();
var_dump($_SESSION);
?>
运行如下:
NULL array(0) { } NULL

从上面的代码中,我们至少可以得出两个结论:
① 开启session机制前,$_SESSION数组是不存在的!
② $_SESSION 数组在脚本周期内,一旦被干掉,就不会再产生!即使重新开始session机制之后,该数组也并没有出现!

因此,我们想要清空$_SESSION 中的数据的话,就需要使用$_SESSION = array()的形式,这样能够在脚本周期之内,保证$_SESSION数组的存在!

5)cookie中的值保存改变的问题!

有的时候,我们需要对cookie中的值进行重写!如果,后面并没有跟路径名、域名、过期时间的话,简单的进行重写就可以了!
但是,如果cookie中保存了这些参数,而我们重新设定值的时候,并没有设置相关参数,这些参数还会被完整保留下来么?还是没有了?看代码:

<?php
//文件路径: /talkphp.php
$expires = time()+3600;
setcookie('name','liangbo' , $expires);
运行结果如下图
?>

<?php
//文件路径: /secondtalk/se
setcookie('name','maweibin');
?>

<?php
$name = $_COOKIE['name'];
echo $name;
?>

从上面的代码中,我们可以得出以下结论:

①原有cookie中保存的值,是否被后来的cookie值是否被覆盖取决于,两个参数:
(1)path路径
如果在同一个域名,但是不同的路径下,设置了同名的cookie,那么该域名下回保存两个cookie,依据路径的不同,选择使用哪一个cookie(因为cookie中的key相同,那么只能依靠path路径进行区分!)
(2)domain域名。很明显,不同域名下的cookie肯定是两个,同名的话,也不冲突

②如果在同一个域名、同一个路径下,设置了同名的cookie,那么后面的cookie值会覆盖前面的值!
path路径、过期时间、所在域名都以最后一个cookie的为准!如果,该cookie并没有设置这些参数,那么cookie会话机制,会进行自动的填充!
path路径,以当前的运行脚本路径为准! 过期时间,就是一个会话周期,浏览器关闭后,就消失! domain域名,就是url中的域名!

6)cookie携带的问题

经过试验证明,凡是从浏览器端发起的请求,在访问同一个域名的时候,cookie是会被自动携带到服务器端的,无论是通过url访问、还是通过ajax调用访问cookie都是会被自动携带的!
注意:只有是从客户端发起的请求,cookie才会被自动携带!

【转】php中的会话机制(2)的更多相关文章

  1. WCF初探-26:WCF中的会话

    理解WCF中的会话机制 在WCF应用程序中,会话将一组消息相互关联,从而形成对话.会话”是在两个终结点之间发送的所有消息的一种相互关系.当某个服务协定指定它需要会话时,该协定会指定所有调用(即,支持调 ...

  2. JavaWeb中Cookie会话管理,理解Http无状态处理机制

    注:图片如果损坏,点击文章链接:https://www.toutiao.com/i6512995108961387015/ 1.<Servlet简单实现开发部署过程> 2.<Serv ...

  3. JavaWeb中Session会话管理,理解Http无状态处理机制

    注:图片如果损坏,点击文章链接:https://www.toutiao.com/i6512955067434271246/ 1.<Servlet简单实现开发部署过程> 2.<Serv ...

  4. 重温WCF之WCF中可靠性会话(十四)

    1.WCF中可靠性会话在绑定层保证消息只会被传输一次,并且保证消息之间的顺序.当使用TCP(Transmission Control Protocol,传输控制协议)通信时,协议本身保证了可靠性.然而 ...

  5. SQL Server 内存中OLTP内部机制概述(二)

    ----------------------------我是分割线------------------------------- 本文翻译自微软白皮书<SQL Server In-Memory ...

  6. note.js之 Mongodb在Nodejs上的配置及session会话机制的实现

    上篇我们使用nodejs实现了一个express4的网站构建配置,但一个有面的网站怎么可以缺少一个数据库呢.现在较为流行的就是使用MONGODB来作为nodejs网站引用的数据库,可能它与nodejs ...

  7. Servlet基础知识(三)—— 会话机制Session,Session和Cookie的异同

    Servlet会话机制: Http是一种无状态协议,它是无记忆的.也就是说,服务器不会保存用户的任何信息,当同一用户再次去访问时,服务器是不认识你的,它还是会建立新的连接. 但有时候我们需要服务器保留 ...

  8. Java Web(三) 会话机制,Cookie和Session详解

    很大一部分应该知道什么是会话机制,也能说的出几句,我也大概了解一点,但是学了之后几天不用,立马忘的一干二净,原因可能是没能好好理解这两种会话机制,所以会一直遗忘,一直重新回过头来学习它,今天好好把他总 ...

  9. Cookie&Seesion会话 共享数据 工作流程 持久化 Servlet三个作用域 会话机制

    Day37 Cookie&Seesion会话 1.1.1 什么是cookie 当用户通过浏览器访问Web服务器时,服务器会给客户端发送一些信息,这些信息都保存在Cookie中.这样,当该浏览器 ...

随机推荐

  1. AC日记——【模板】分块/带修改莫队(数颜色) 洛谷 P1903

    [模板]分块/带修改莫队(数颜色) 思路: 带修改莫队: (伏地膜xxy): 代码: #include <bits/stdc++.h> using namespace std; #defi ...

  2. ubuntu wine 使用

    运行程序 wine xxx.exe 图形界面程序(普通程序):直接使用 wine 命令行的DOS程序:wineconsole 代替 wine.这才是正常的运行方式.不使用wineconsole运行命令 ...

  3. 洛谷P1392 取数 [堆]

    题目传送门 取数 题目描述 在一个n行m列的数阵中,你须在每一行取一个数(共n个数),并将它们相加得到一个和.对于给定的数阵,请你输出和前k小的取数方法. 输入输出格式 输入格式: 第一行,三个数n, ...

  4. 洛谷P1940买蛋糕

    题目传送门 题意:给定你一个数n,要求用最小个数的整数组成小于等于n的所有整数,并求出方案数. 很明显,擅长二进制的大犇们肯定一眼就看得出方案数是log2(n)+1,然而我并不擅长,但是推了一小会儿也 ...

  5. AndroidManifest.xml文件详解(uses-permission)

    语法(SYNTAX): <uses-permissionandroid:name="string"/> 被包含于(CONTAINED IN): <manifest ...

  6. 拜拜了,GOPATH君!新版本Golang的包管理入门教程

    Go 1.11和1.12实现了对包管理的初步支持,Go的新依赖管理系统使依赖版本信息明确且易于管理.Using Go Modules - The Go Blog 新的包管理模式有什么不同? 作为Go语 ...

  7. 洛谷——P1743 矩阵 III

    P1743 矩阵 III 题目背景 usqwedf 改编系列题. 题目描述 给定一个n*m的矩阵,问从左上角走到右下角有多少条路径. 输入输出格式 输入格式: 一行两个正整数 n,m 输出格式: 路径 ...

  8. Linux命令之whereis

    whereis [选项] [文件] (1).选项 -b 只搜索二进制文件 -B <目录> 只在路径下查找二进制文件 -m 只搜索man手册 -M <目录> 只在路径下查找man ...

  9. Flask实战第61天:帖子板块过滤显示

    先在显示的帖子是所有版块的帖子,这节我们来完成点击某个版块,则显示此版块的帖子 要完成这个功能,我们需要在前端传递板块的id到后台, 编辑front_index.html 编辑首页视图 编辑板块选中样 ...

  10. Xamarin.Forms教程下载安装Windows版的Xamarin开发工具

    Xamarin.Forms教程下载安装Windows版的Xamarin开发工具 下载安装Windows版的Xamarin开发工具 本节将讲解如何下载并安装Windows版的Xamarin开发工具. 下 ...