在标题中我加了一个大家都很熟悉的单词——Session,熟吧?玩过Web开发的朋友肯定在梦中都会见到她。

在Web中为什么要会话呢?毕竟每个用户在一个Web应用中可能不止进行一次操作,比如,某二手飞机交易网站,用户A登陆后,可能他会修改他的个人 信息,他也有可能看好了一架二手飞机,打算入手,就把商品放到他的“购物车”中,这些过程中,都会产生许多与用户A相关的数据,这些数据只是对A有效,而 当用户B登陆后,对于B,又会有他自己的数据,总的一句话就是,每个客户端在服务器上都有其的独立数据存储区,互不相干,就好像A和服务器在单独谈话一样,所以叫会话。

在WCF中,会话的含义与Web中的会话概念是差不多的,就是客户端与服务器端在“私聊”,这便是存在会话的调用;那么,没有会话的调用呢,就是“群聊”;通信过程中,数据都以明文显示,不进行加密保护,这叫“裸*聊”。

好了,现在大家对于会话,肯定有点理解了。但是,会话是看不见摸不着的,怎么通过实例来检验它呢?

下面,我们写一个例子,看看在不支持会话的绑定上连续调用两个有关联的代码,会发生什么情况。

namespace Server
{
[ServiceContract]
public interface IService
{
[OperationContract(IsOneWay = true)]
void SetValue(int n);
[OperationContract]
int GetValue();
}
}
namespace Server
{
[ServiceBehavior(IncludeExceptionDetailInFaults = true)]
public class MyService : IService
{ public MyService()
{
Console.WriteLine("-------------------------------");
Console.WriteLine("{0} - 服务被实例化。", DateTime.Now.ToLongTimeString());
}
// 在析构函数中也输出信息
~MyService()
{
Console.WriteLine("{0} - 服务实例被释放。", DateTime.Now.ToLongTimeString());
Console.WriteLine("--------------------------------");
} /// <summary>
/// 私有字段
/// </summary>
private int mValue = int.MinValue; public void SetValue(int n)
{
this.mValue = n;
Console.WriteLine("会话ID:{0}", OperationContext.Current.SessionId);
} public int GetValue()
{
Console.WriteLine("会话ID:{0}", OperationContext.Current.SessionId);
return this.mValue;
}
}
}

在服务类中,我们分别在构造函数和析构函数中输出一些内容,以便于在运行时看到服务什么时候被实例化,什么时候被释放。同时,在调用每个操作协定时,我们也输出当前会话的ID,如果空就说明当前没有启用会话。

接着,实现服务主机,我们使用不支持的Bindding。

namespace Server
{
class Program
{
static void Main(string[] args)
{
Console.Title = "WCF服务器端";
using (ServiceHost host = new ServiceHost(typeof(MyService), new Uri("http://127.0.0.1:1211/sv")))
{
// 绑定
BasicHttpBinding binding = new BasicHttpBinding();
binding.Security.Mode = BasicHttpSecurityMode.None;//不需要安全模式
host.AddServiceEndpoint(typeof(IService), binding, "/ep");
// 服务元数据
ServiceMetadataBehavior mb = new ServiceMetadataBehavior();
mb.HttpGetEnabled = true;
mb.HttpGetUrl = new Uri("http://127.0.0.1:8008/meta");
host.Description.Behaviors.Add(mb);
host.Opened += (sender, arg) =>
{
Console.WriteLine("服务已启动。");
};
try
{
host.Open();//打开服务
Console.Read();
}
catch (Exception ex)
{
Console.WriteLine(ex.Message);
}
}
}
}
}

然后,实现客户端,为了使演示操作更方便,客户端使用wpf项目,界面大致如下图所示。

服务有两个操作,从前面服务类的定义我们知道,我在服务类内部定义了一个私有字段,而公开的两个操作协定,分别是设置这个值和获取这个值。

namespace Client
{
/// <summary>
/// Interaction logic for MainWindow.xaml
/// </summary>
public partial class MainWindow : Window
{
ServiceReference1.ServiceClient client = null; public MainWindow()
{
InitializeComponent(); client = new ServiceReference1.ServiceClient();
// 关闭通道
this.Closing += (frmsender, frmargs) => client.Close();
} private void button1_Click(object sender, RoutedEventArgs e)
{
int v;
if (!int.TryParse(this.textBox1.Text, out v))
{
return;
}
client.SetValue(v);
} private void button2_Click(object sender, RoutedEventArgs e)
{
int v = client.GetValue();
this.textBox2.Text = v.ToString();
}
}
}

我们现在要测试一下,看看先后调用这两个方法,能不能得到我们预期的值。即如果我调用SetValue(100),那么,在调用GetValue时应该返回100,事实是不是这样呢?

八格牙路,我没有看到预期的值。

我输入了100,可是取出来的是Min int,再看一看服务器端输出了哪些信息。

很明显,每调用一次操作,服务类就被实例化一次,意思就是:调用SetValue时是实例A,而调用GetValue时可能是实例B了,所以,私有字段的值没有被保存。

那么,如何证明两次调用的操作不属于同一个服务实例呢?还记得GetHashCode吗?对的,只要在内存中不是同一个实例的,其哈希值肯定不同。是不是这样呢,我们把上面的服务代码改一下。

        public void SetValue(int n)
{
this.mValue = n;
Console.WriteLine("------------------------");
Console.WriteLine("当前实例的哈希值:{0}", this.GetHashCode().ToString("x"));
Console.WriteLine("会话ID:{0}", OperationContext.Current.SessionId);
} public int GetValue()
{
Console.WriteLine("------------------------");
Console.WriteLine("当前实例的哈希值:{0}", this.GetHashCode().ToString("x"));
Console.WriteLine("会话ID:{0}", OperationContext.Current.SessionId);
return this.mValue;
}

那么,这次又会发生什么事呢,看结果。

这个结果证实了我之前的推断,先后调用的两个方法不是同一个实例的。

那么,如果启用了会话,结果又会如何呢?

    [ServiceContract(SessionMode = SessionMode.Required)]
public interface IService
{
。。。。。
}

在定义服务协定的时候,设置SessionMode可以让服务要求客户端启用会话。

接下来,要使用支持会话的绑定。

                // 绑定
//BasicHttpBinding binding = new BasicHttpBinding();
//binding.Security.Mode = BasicHttpSecurityMode.None;//不需要安全模式
//host.AddServiceEndpoint(typeof(IService), binding, "/ep");
NetTcpBinding binding = new NetTcpBinding();
binding.Security.Mode = SecurityMode.None;
host.AddServiceEndpoint(typeof(IService), binding, "net.tcp://127.0.0.1:2377/ep");

把客户端的服务引用更新一下。然后看看这回会不会达到我们的目的。

怎么样,高兴吧?终于看到我们要的效果了,我输入了100,取出来的还是100,这一回从服务器端的输出看,服务类只被实例化了一次,而且看看两个哈希值是相同的,这证明了确实是同一个实例,同时,我们也看到了两次调用的会话ID是一样的,这也说明了,客户端两次调用都基于同一个会话进行的,这么一来,输进去的100就能顺利取出来了。

你不妨多开几个客户端来试试。

看到那个不同的会话ID,哈希值和实例化时间了吧?这表明了:服务器独立维护着与每个客户端的会话。

下面还要对我们的解决方案进行修改。

服务器端,把服务协定改为:

    [ServiceContract(SessionMode = SessionMode.Required)]
public interface IService
{
[OperationContract(IsOneWay = true, IsInitiating = true, IsTerminating = false)]
void SetValue(int n); [OperationContract]
int GetValue(); [OperationContract(IsInitiating = false, IsTerminating = true)]
void EndSession();
}

在服务类中增加对EndSession方法的实现。

        public void EndSession()
{
Console.WriteLine("会话结束。");
}

看到变化了吗?

我们在使用OperationContractAttribute定义操作协定时,设置了两个属性:

a、IsInitiating:如果为真,则当调用该操作时就启用会话。

b、IsTerminating:如果为真,则说明当该用该操作时终结会话。

所以,上面的例子是,当调用SetValue时开始会话,当调用EndSession方法后会话结束,在选择作为结束会话的方法时,最好使用返回值为void或者单向通讯(One Way)的方法,这样,不用等待客户结束才结束会话,因为单向通讯,不需要向客户端回复消息,因为它被调用后就可以马上终止会话了。

        ServiceReference1.ServiceClient client = null;

        public MainWindow()
{
InitializeComponent(); client = new ServiceReference1.ServiceClient();
// 关闭通道
//this.Closing += (frmsender, frmargs) => client.Close();
}

然后,在调用完GetValue后马上就EndSession,看看这回又会发生什么。

        private void button2_Click(object sender, RoutedEventArgs e)
{
int v = client.GetValue();
this.textBox2.Text = v.ToString();
client.EndSession();
}

这样,一旦三个方法调用完之后,会话结束,服务实例也解放了。

再说明一下,IsInitiating = true的操作开始新会话,IsTerminating = true的操作终结会话。

传说中的WCF(11):会话(Session)的更多相关文章

  1. 重温WCF之会话Session(九)

    转载地址:http://blog.csdn.net/tcjiaan/article/details/8281782 每个客户端在服务器上都有其的独立数据存储区,互不相干,就好像A和服务器在单独谈话一样 ...

  2. WCF会话(Session)与实例(Instance)管理

    一.理解Session 1.Session的作用:保留Client和Service之间交互的状态,确保Client与Service之间交互唯一性(SessionId),即:多个Client同时访问Se ...

  3. jfc在jsp页面画图,不将图片存在服务器端,只存入会话session(可用)

    jfc在jsp页面画图,不将图片存在服务器端,只存入会话session.其中主要用到jfc的一个servlet类. <%@ page contentType="text/html;ch ...

  4. 简单PHP会话(session)说明

    现在程序员愈发的不容易了,想要精通,必然要寻本溯源,这其实与目前泛滥的愈发高级的语言以及众多的框架刚好相反,因为它们在尽可能的掩盖本源使其简单,个人称之为程序员学习悖论. 注:作者接触web开发和ph ...

  5. http 会话(session)详解

    会话(session)是一种持久网络协议,在用户(或用户代理)端和服务器端之间创建关联,从而起到交换数据包的作用机制 一.查看session id 可利用相关工具,比如firebug,httpwatc ...

  6. Tensorflow会话Session

    转载自: http://blog.csdn.net/Hanging_Gardens/article/details/72784392 https://www.cnblogs.com/hypnus-ly ...

  7. 网络基础 http 会话(session)详解

    http 会话(session)详解 by:授客 QQ:1033553122 会话(session)是一种持久网络协议,在用户(或用户代理)端和服务器端之间创建关联,从而起到交换数据包的作用机制 一. ...

  8. Java通过遍历sessionId获取服务器所有会话session

    Servlet2.1之后不支持SessionContext里面getSession(String id)方法,也不存在遍历所有会话Session的方法.但是,我们可以通过HttpSessionList ...

  9. Oracle数据库的连接模式connection Mode、连接connection与会话session

    数据库的连接模式Connection Mode: Dedicated Server Mode(专有模式) 当用户发出请求时,如远程的client端通过监听器连接数据库上,ORACLE的服务器端会启用一 ...

  10. MyBatis之会话Session原理

    MyBatis 之会话 Session 执行逻辑 1.SQL 会话工厂构建器类 SqlSessionFactoryBuilder 的 build 方法用于构建 SqlSessionFactory 类的 ...

随机推荐

  1. JavaScript高级程序设计之location对象

    location对象用来处理URL的相关信息 1.获取查询字符串 // 获取查询字符串对象 var getQueryStringArgs = function () { ? location.sear ...

  2. win7虚拟打印驱动开发注意事项

    win7虚拟打印驱动开发注意事项 通过控制面板安装遇到以下问题:错误1.提示“Printer driver was not installed. Operation could not be comp ...

  3. 实体框架 (EF) 入门 => 一、我该用哪个工作流?

    入门的参考资料:http://msdn.microsoft.com/zh-cn/data/ee712907 本篇的参考资料:http://msdn.microsoft.com/zh-cn/data/j ...

  4. 失败的数据库迁移UDB

    公司采用的是ucloud的云主机,数据库也是架设在云主机上.由于数据越来越多数据查询数据越来越慢,所以我决定往 UDB上迁移.当时考虑的理由如下: (1)云主机底层架设在虚拟机上IO性能有折损,而UD ...

  5. Ubuntu 关闭锁屏界面的 on-screen keyboard

    试了试屏幕键盘,在 系统设置里开启了,又关了,但是在屏幕解锁时总是出现 screen keyboard,老烦人了,不知到在哪里关闭了,系统设置里面都关了,网上搜了解决办法,原来在这里 把 show w ...

  6. UserDefault 用户首选项读写 swift

    // // ViewController.swift // 首选项数据读写 // // Created by mac on 15/7/12. // Copyright (c) 2015年 fangyu ...

  7. C++变量的存储类别与作用域

    总结一下C++中变量的存储类别以及变量的作用域. (1)标示符的存储类别决定了标示符在内存中存在的时间(我们可以理解标示符就是确定一个变量的符号,也就是我们所说的变量名) 二:存储类别 (1)静态存储 ...

  8. Java Day 16

    基本数据类型包装类 Integer.MAX_VALUE  Integer.parseInt(); intValue(); valueOf(); 自动装拆箱 如果是一个字节范围,数据共享 字符串中 数值 ...

  9. C++ 学习笔记(一)

    只是记录自己学习C++ 笔记实例来自<c++ primer> 1.static: static 局部对象确保不迟于在程序执行流程第一次经过该对象的定义语句时进行初始化.这种对象一旦被创建, ...

  10. objective-c自学总结(一)---面向对象

    本人大二本科在读,利用一个月多一点的时间对OC语言基础进行了自学,在下一阶段UI学习开始之前, 对这一阶段的自学进行一些总结.在此特别感谢刘晓斌学长和无线互联3G学院 首先说一下对OC的整体感觉,这是 ...