微信公众平台开发教程(八)Session处理问题

在微信窗口,输入的信息有限,我们需要将一些信息分多次请求。

比如:在进行用户绑定时,我们需要输入用户的相关信息,比如:用户名、密码,或者姓名、电话号码,服务端验证通过,即可将系统用户与微信用户绑定。

然后,此微信账户就有一定的功能权限了,可以查积分,消费记录等。服务号:招商银行信用卡,就有很多功能。

微信客户端无法缓存信息,而且输入信息有限,需要进行多次请求,在服务端保存当前会话状态。这就需要Session。

本文以用户认证,绑定账号为例,来说明具体处理。

一、创建通用的Session处理机制。

为了更好的说明原理,便于扩展,我们来自己设计Session。当然,这里也可以使用System.Web.SessionState.HttpSessionState,这是Web常用的Session机制。

1、自定义Session

用于存储会话片段以及相关数据。

class Session
  {
    /// <summary>
    /// 缓存hashtable
    /// </summary>
    private static Hashtable mDic = new Hashtable();
    /// <summary>
    /// 添加
    /// </summary>
    /// <param name="key">key</param>
    /// <param name="value">value</param>
    public static void Add(string key, object value)
    {
      mDic[key] = value;
    }
    /// <summary>
    /// 移除
    /// </summary>
    /// <param name="key">key</param>
    public static void Remove(string key)
    {
      if (Contains(key))
      {
        mDic.Remove(key);
      }
    }
    /// <summary>
    /// 设置值
    /// </summary>
    /// <param name="key"></param>
    /// <param name="value"></param>
    public static void Set(string key, object value)
    {
      mDic[key] = value;
    }
    /// <summary>
    /// 获取值
    /// </summary>
    /// <param name="key"></param>
    /// <returns></returns>
    public static object Get(string key)
    {
      return mDic[key];
    }
    /// <summary>
    /// 是否含有
    /// </summary>
    /// <param name="key">key</param>
    /// <returns>bool</returns>
    public static bool Contains(string key)
    {
      return mDic.ContainsKey(key);
    }
    /// <summary>
    /// 清空所有项
    /// </summary>
    public static void Clear()
    {
      mDic.Clear();
    }
  }

2、操作类型

记录具体的操作类型,标识当前会话的具体操作

/// <summary>
  /// 操作类型
  /// </summary>
  enum Operation
  {
    /// <summary>
    /// 认证
    /// </summary>
    Auth,
    /// <summary>
    /// 添加用户
    /// </summary>
    CreateUser
  }

3、操作过程枚举

用于标识当前操作,处于哪一个阶段,不同阶段做不同的处理。

/// <summary>
  /// 操作过程
  /// </summary>
  enum OperationStage
  {
    /// <summary>
    /// 默认
    /// </summary>
    Default,
    /// <summary>
    /// 第一步
    /// </summary>
    First,
    /// <summary>
    /// 第二步
    /// </summary>
    Second,
    /// <summary>
    /// 第三步
    /// </summary>
    Third
  }

4、Session缓存项

缓存记录的项,这里面记录了操作类型、操作步骤以及会话对象。为了便于进行Session管理,还增加了最后访问时间,是否自动清除标识。

class SessionItem
  {
    /// <summary>
    /// 操作类型
    /// </summary>
    public Operation Oper { get; set; }
    /// <summary>
    /// 当前步骤
    /// </summary>
    public OperationStage Stage { get; set; }
    /// <summary>
    /// 数据对象
    /// </summary>
    public object Data { get; set; }
    /// <summary>
    /// 是否自动删除
    /// </summary>
    public bool AutoRemove
    {
      get;
      set;
    }
    /// <summary>
    /// 最后更新时间
    /// </summary>
    public DateTime UpdateTime { get; set; }
  }

二、就要在消息处理中,加入Session处理。

1、增加缓存项数据对象

这个对象,记录用户在会话过程中,录入的相关信息。也是作为业务处理数据提供对象。

class AuthSessionItem
  {
    /// <summary>
    /// 用户名
    /// </summary>
    public string FromUserName { get; set; }
    /// <summary>
    /// 账号
    /// </summary>
    public string Code { get; set; }
    /// <summary>
    /// 唯一标识
    /// </summary>
    public string ID { get; set; }
  }

 2、认证处理过程

1)开始进入认证,根据认证关键字进行标识,启动会话,并缓存相关数据

2)提示录入个人账号信息

3)微信用户录入个人账号,服务端记录账号信息,并提示录入员工卡号

4)微信用户录入卡号信息,服务端记录卡号信息,并调用具体的认证逻辑

5)用户认证通过,绑定微信OpenId,提示成功绑定信息,并清除会话。

在认证过程中,需要对用户录入信息进行合法性验证,而且在会话过程中,支持用户退出当前操作。

/// <summary>
    /// 认证用户信息
    /// </summary>
    /// <param name="tm"></param>
    /// <returns></returns>
    private bool Auth(TextMessage tm, ref string response)
    {
      SessionItem sessionItem = null;
      if (string.Equals(tm.Content, "Auth", StringComparison.OrdinalIgnoreCase))
      {
        //检查是否已经认证,业务组件验证
        if (UserManager.IsAuth(tm.FromUserName))
        {
          //如果已经认证,提示
          tm.Content = "您已经认证过了,无需再次认证!";
        }
        else
        {
          AuthSessionItem authSessionItem = new AuthSessionItem();
          authSessionItem.FromUserName = tm.FromUserName;

          sessionItem.Oper = Operation.Auth;
          sessionItem.Stage = OperationStage.First;
          sessionItem.Data = authSessionItem;
          Session.Set(tm.FromUserName, sessionItem);

          //输入账号,并将数据和步骤,写入缓存
          tm.Content = "请输入您的个人账号";
        }

        response = ResponseText(tm);
        return false;
      }

      //从Session获取用户信息
      sessionItem = Session.Get(tm.FromUserName) as SessionItem;
      //如果会话存在,且当前操作为用户认证
      if (sessionItem != null && sessionItem.Oper == Operation.Auth)
      {
        if (sessionItem.Stage == OperationStage.First)
        {
          tm.Content = tm.Content.Trim();
          if (string.IsNullOrEmpty(tm.Content) || tm.Content.Length > 20)
          {
            tm.Content = "输入的个人账号不合法,请重新输入。";
            response = ResponseText(tm);
            return false;
          }
          AuthSessionItem authSessionItem = sessionItem.Data as AuthSessionItem;
          if (authSessionItem != null)
          {
            authSessionItem.Code = tm.Content;
          }

          //更新缓存
          sessionItem.Stage = OperationStage.Second;
          Session.Set(tm.FromUserName, sessionItem);
          tm.Content = "请输入您的员工卡号!\n退出认证请输入Exit。";
          response = ResponseText(tm);
        }
        else if (sessionItem.Stage == OperationStage.Second)
        {
          string cardNum = null;
          if (!Common.TryConvertToCardNum(tm.Content, out cardNum))
          {
            tm.Content = "员工卡号不合法,请重新输入。\n退出认证请输入Exit。";
            response = ResponseText(tm);
            return false;
          }
          AuthSessionItem authSessionItem = sessionItem.Data as AuthSessionItem;
          if (authSessionItem != null)
          {
            authSessionItem.ID = cardNum;
          }
          //认证
          string message;
          if (UserManager.Authenticate(authSessionItem, out message))
          {
            tm.Content = "祝贺您,已经认证成功,可以使用通讯录的查询功能呢。";
            //清理缓存
            Session.Remove(tm.FromUserName);
            response = ResponseText(tm);
            return true;
          }
          else if (!string.IsNullOrEmpty(message))
          {
            tm.Content = message;
          }
          else
          {
            tm.Content = "您输入的信息有误。\n重新认证请输入:Auth!";
          }
          //过程结束:清理Session
          Session.Remove(tm.FromUserName);
          response = ResponseText(tm);
          return false;
        }
      }

      return false;
    }

3、退出会话,清理Session

在认证过程中,用户可以通过命令,强制退出当前操作,在退出当前操作时,需要清理会话信息。

/// <summary>
    /// 退出,并清理Session
    /// </summary>
    /// <param name="tm"></param>
    /// <param name="response"></param>
    /// <returns></returns>
    private bool Exit(TextMessage tm, ref string response)
    {
      //退出
      if (string.Equals(tm.Content, "Exit", StringComparison.OrdinalIgnoreCase))
      {
        //清除Session
        Session.Remove(tm.FromUserName);
        tm.Content = "您已退出当前操作,请执行其他操作。";
        response = ResponseText(tm);
        return true;
      }

      return false;
    }

 三、用户认证通过,绑定微信账户

用户认证通过,并绑定微信OpenId,通过OpenId即可查询通讯录、查询个人积分以及消费记录等操作了。用户认证是一个身份认证过程,也是一个用户绑定过程。用户身份认证通过,即可通过微信账号查询具体信息了。这时候业务层可以根据微信分配的OpenId直接查询用户相关信息。

四、后记

通过这种方法,公众账号,可以通过小小的文本输入框,实现更多、更复杂的业务应用。当然,还是通过提供网页来进行信息录入,更直观便捷。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • ASP.NET微信公众号添加菜单

    本文实例为大家分享了微信公众号添加菜单的具体代码,供大家参考,具体内容如下 testjs.aspx代码: <%@ Page Language="C#" AutoEventWireup="true" CodeBehind="testjs.aspx.cs" Inherits="MyTest.WebUI.Manager.Main.testjs" %> <!DOCTYPE html> <html xmln

  • ASP.NET微信公众号查看粉丝信息接口

    本文实例为大家分享了ASP.NET微信粉丝信息接口查看代码,供大家参考,具体内容如下 微信Token实体类: /// <summary> /// 微信Token实体类 /// </summary> public class WeChatTokenEntity { public string Access_token { get; set; } public string Expires_in { get; set; } } 用户信息实体类 /// <summary> /

  • ASP.NET微信公众号之用户分组管理web页面

    本文实例为大家分享了ASP.NET微信用户分组管理的具体代码,供大家参考,具体内容如下 Model层实体类: public class UserList { public string total { get; set; } public string count { get; set; } public userlistopenid data { get; set; } public string next_openid { get; set; } } public class userlis

  • ASP.NET微信公众号客服接口

    本文实例为大家分享了ASP.NET微信客服接口的具体代码,供大家参考,具体内容如下 Kf_account.cs代码: public partial class Kf_account : Form { private readonly DataTable adt_user = new DataTable(); private readonly string as_INIFile = Application.StartupPath + "\\user.ini"; public Kf_acc

  • 微信公众平台开发教程②微信端分享功能图文详解

    本文实例讲述了微信公众平台微信端分享功能.分享给大家供大家参考,具体如下: 背景 初次尝试微信公众号的开发,对于学习方法的探索都是来源于网上的博客.问答,对于参差不齐的信息,自己也是有苦说不出,抽出一点时间写点文章,既是对自己的学习总结,也希望给予同是菜鸟的小白一点帮助. 今天想添加微信分享的功能,如果不进行自定义设计,那么当我们点击分享朋友圈.好友或者QQ好友.空间时,默认的标题就是<title>标签中的信息,而显示的描述信息就是链接,图片多是默认为页面中显示的第一张图片,显然这样的处理是不

  • 微信公众平台开发教程(四) 实例入门:机器人回复(附源码)

    上一篇文章,写了基本框架,可能很多人会觉得晕头转向,这里提供一个简单的例子来予以说明,希望能帮你解开谜团. 一.功能介绍 通过微信公众平台实现在线客服机器人功能.主要的功能包括:简单对话.查询天气等服务. 这里只是提供比较简单的功能,重在通过此实例来说明公众平台的具体研发过程.只是一个简单DEMO,如果需要的话可以在此基础上进行扩展. 当然后续我们还会推出比较复杂的应用实例. 二.具体实现 1.提供访问接口 这里不再赘述,参照上一章,微信公众账号开发教程(二) 基础框架搭建 http://www

  • 微信公众平台开发教程⑥ 微信开发集成类的使用图文详解

    本文实例讲述了微信公众平台开发之微信开发集成类的使用.分享给大家供大家参考,具体如下: 背景 这几天又在接触微信PHP方面的开发,认为之前写的文章确实有些乱,刚好发现了一个不错的集成类(看原始代码,出自"云知梦军哥",不算打广告,只是尊重别人的开发成果,谢谢)里面涉及了非常全面的函数,在此针对我自己的实际使用过程简单的描述一下希望能给同道小白们一个诚恳的引导 ... 框架:ThinkPHP 3.2.3 前期准备: ①.微信公众平台的配置 这里不赘述,具体的可以参考我前面所写的1.2章,

  • 微信公众平台开发教程(五)详解自定义菜单

    一.概述: 如果只有输入框,可能太简单,感觉像命令行.自定义菜单,给我们提供了很大的灵活性,更符合用户的操作习惯.在一个小小的微信对话页面,可以实现更多的功能.菜单直观明了,不仅能提供事件响应,还支持URL跳转,如果需要的功能比较复杂,我们大可以使用URL跳转,跳转至我们的网页即可. 注意:自定义菜单,只有服务号才有此功能 接着我们详细介绍,如何实现自定义菜单? 二.详细步骤: 1.首先获取access_token access_token是公众号的全局唯一票据,公众号调用各接口时都需使用acc

  • 微信公众平台开发教程①获取用户Openid及个人信息图文详解

    本文实例讲述了微信公众平台开发获取用户Openid及个人信息.分享给大家供大家参考,具体如下: 前言: 初次尝试微信公众号的开发,对于学习方法的探索都是来源于网上的博客.问答,对于参差不齐的信息,自己也是有苦说不出,抽出一点时间写点文章,既是对自己的学习总结,也希望给予同是菜鸟的学渣一点帮助 背景介绍: 我需要用户接收微信分享的链接后,点击进入给参加活动的用户[点赞],然后需要后台获取该微信用户的 openid 作为唯一的标记信息,以便保证该用户下次进入后进行数据库的比对,直接提取其对应的操作信

  • node.js微信公众平台开发教程

    用nodejs怎样来实现对微信公众平台的开发呢? 别的就不多说了,先来简单介绍微信公众平台的基本原理. 微信服务器就相当于一个转发服务器,终端(手机.Pad等)发起请求至微信服务器,微信服务器,然后将请求转发给自定义服务(这里就是我们的具体实现).服务处理完毕,然后转发给微信服务器,微信服务器再将具体响应回复到终端:通信协议为:HTTP:数据格式为:XML. 具体的流程如下图所示: 其实,我们需要做的事情,就是对HTTP请求,做出响应.具体的请求内容,我们按照特定的XML格式去解析,处理完毕后,

  • 微信公众平台开发教程④ ThinkPHP框架下微信支付功能图文详解

    本文实例讲述了ThinkPHP框架下微信支付功能.分享给大家供大家参考,具体如下: 声明:原文主要摘自白俊遥博客 ,部分内容针对个人事例已作修改,主要用于自己的参考,欢迎指正. 注意:微信公众号支付,强烈建议使用外网可访问的链接测试,否则即便代码正确也无法调出支付界面,可使用草料二维码生成可扫描图片,也可直接在微信中输入网址. 使用框架:ThinkPHP 3.2.3 一.微信公众平台信息配置 1).进入微信公众平台 由左侧的"微信支付"进入配置界面,添加或修改正确的支付授权目录,注意该

  • 微信公众平台开发教程⑤ 微信扫码支付模式介绍

    本文实例讲述了微信扫码支付模式.分享给大家供大家参考,具体如下: 背景:因为微信占据众多的用户群,作为程序开发,自然而然也成了研究的重点.毕竟个人能力有限,很难想象设计的复杂性,多数时间接触起来,各种蒙圈,在此笔记自己的操作流程,仅做参考,欢迎指正. 一.微信扫码支付模式 1.附带微信公众号"微信开发"中,对微信扫码支付的两种模式流程图以作"膜拜". 2.具体的操作,可详细参考官方开发文档 文档有强调: 模式一开发前,商户必须在公众平台后台设置支付回调URL.URL

  • 微信公众平台开发教程(二) 基本原理及消息接口总结

    一.基本原理 在开始做之前,大家可能对这个很感兴趣,但是又比较茫然.是不是很复杂?很难学啊? 其实恰恰相反,很简单.为了打消大家的顾虑,先简单介绍了微信公众平台的基本原理. 微信服务器就相当于一个转发服务器,终端(手机.Pad等)发起请求至微信服务器,微信服务器,然后将请求转发给自定义服务(这就里就是我们的具体实现). 服务处理完毕,然后挥发给微信服务器,微信服务器再将具体响应回复到终端. 通信协议为:HTTP 数据格式为:XML 具体的流程如下图所示: 其实,我们需要做的事情,就是对HTTP请

  • 微信公众平台开发教程③ PHP实现微信公众号支付功能图文详解

    本文实例讲述了PHP实现微信公众号支付功能.分享给大家供大家参考,具体如下: 直言无讳,我就是一个初涉微信开发的小白,写这篇博客的原因:一是为了给自己做下备忘记录,以便以后能回忆这条程序猿的坎坷路:二是希望能帮助到同是自学开发的小白们:三是对那些不屑一顾于我等尘埃的大牛们的控诉,小白的道路坎坷,你们凭什么总要一副高高在上的样子?我等敬而不畏... 背景介绍: 随着智能手机的普及,移动支付下的微信.支付宝所提供的便利需求不言而喻,好吧,至少我周围连个小摊贩的早餐都可以微信支付,而且人家手机还比我高

随机推荐