最近公司换了yaf框架,突然对用c实现php拓展感兴趣了,如果一个功能已经很稳定很成熟而且用的地方很多,那么我们就可以尝试用拓展实现(不一定每种情况都可以写成拓展),写成拓展后就不用每次用都包含一下,工具类直接随php启动加载进内存里。

我这次是把用户会话加密类写成了php的拓展,用户类是基于des加密的,主要实现了,

  1. isLogin //判断是否登录
  2. setLogin //就是种个加密的cookie
  3. getUid
  4. getUsername

我给这个项目起名slime(史莱姆),勇者斗恶龙里的小怪,黏黏的。。。。。。

ok,先来看下php的实现,因为我们是php -> c,所以先看下php的实现,其实就是很普通用户类

 <?php
class Session_DES
{
var $key;
var $iv; //偏移量 function Session_DES( $key="AAAAAAAA", $iv="BBBBBBBB") {
//key长度8例如:1234abcd
$this->key = $key;
if( $iv == "" ) {
$this->iv = $key; //默认以$key 作为 iv
} else {
$this->iv = $iv; //mcrypt_create_iv ( mcrypt_get_block_size (MCRYPT_DES, MCRYPT_MODE_CBC), MCRYPT_DEV_RANDOM );
}
} function encrypt($str) {
//加密,返回大写十六进制字符串
$size = mcrypt_get_block_size ( MCRYPT_DES, MCRYPT_MODE_CBC );
$str = $this->pkcs5Pad ( $str, $size );
return strtoupper( bin2hex( mcrypt_cbc(MCRYPT_DES, $this->key, $str, MCRYPT_ENCRYPT, $this->iv ) ) );
} function decrypt($str) {
//解密
$strBin = $this->hex2bin( strtolower( $str ) );
$str = mcrypt_cbc( MCRYPT_DES, $this->key, $strBin, MCRYPT_DECRYPT, $this->iv );
$str = $this->pkcs5Unpad( $str );
return $str;
} function hex2bin($hexData) {
$binData = "";
for($i = 0; $i < strlen ( $hexData ); $i += 2) {
$binData .= chr ( hexdec ( substr ( $hexData, $i, 2 ) ) );
}
return $binData;
} function pkcs5Pad($text, $blocksize) {
$pad = $blocksize - (strlen ( $text ) % $blocksize);
return $text . str_repeat ( chr ( $pad ), $pad );
} function pkcs5Unpad($text) {
$pad = ord ( $text {strlen ( $text ) - 1} );
if ($pad > strlen ( $text ))
return false;
if (strspn ( $text, chr ( $pad ), strlen ( $text ) - $pad ) != $pad)
return false;
return substr ( $text, 0, - 1 * $pad );
}
}

再来看Session_User

 <?php
define ('BACK_AUTH_NAME','xxxx'); //用到的cookie名
class Session_User {
static $obj;
private $uid;
private $username;
private $chineseName;
public $auth_name = BACK_AUTH_NAME;
private $login_url = null;
public $domain = null; private function __construct($login_url = null, $domain = null){
$host = $_SERVER["HTTP_HOST"];
if (!$login_url) {
$this->login_url = "http://{$host}/login";
}
else {
$this->login_url = $login_url;
} if (!$domain) {
$domain = $_SERVER["SERVER_NAME"];
$this->domain = $domain;
} if (empty ( $_COOKIE [$this->auth_name] )) {
return;
} list ( $uid, $username, $ua, $tm, $chineseName ) = @$this->decodeAuth ($_COOKIE [$this->auth_name]); //ua检验
if (empty ( $uid ) || $ua !== md5($_SERVER ['HTTP_USER_AGENT'])) {
return;
} //TODO:过期时间检验 $this->uid = $uid;
$this->username = $username;
$this->chineseName = $chineseName;
} static public function instance($login_url = null, $domain = null){
if(self::$obj)
return self::$obj;
else{
self::$obj = new Session_User($login_url, $domain);
}
return self::$obj;
} /**
* 用户是否登陆
* */
public function isLogin(){
if(! empty($this->uid))
return true;
else
return false;
}
/**
*
* 跳转到登录页面
* @param unknown_type $forward
* @param unknown_type $exit
*/
public function requireLogin($forward = '', $exit = true){
if(! $this->isLogin()){
if($forward === null)
{
header("location: " . $this->login_url); }
else
{
if(empty($forward))
{
$forward = 'http://'.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI'];
}
$forward = urlencode($forward);
header("location: ". $this->login_url . "?forward=$forward");
}
if($exit)
exit;
}
}
/**
*
*设置登录状态
* @param unknown_type $uid
* @param unknown_type $username
* @param unknown_type $ua
* @param unknown_type $outtime
*/ public function setLogin($uid, $username, $ua = null,$outtime = null, $chineseName = null){
if(empty($ua)){
$ua = $_SERVER['HTTP_USER_AGENT'];
} $str = $this->encodeAuth($uid, $username, $ua, $chineseName);
setcookie($this->auth_name,urlencode($str),$outtime,'/',$this->domain);
}
/**
* 用户退出
*/
public function setLogout(){
setcookie($this->auth_name,'',-1,'/',$this->domain);
} public function __get($key){
if('uid' == $key){
return $this->uid;
}elseif ('username' == $key) {
return $this->username;
}elseif ('chineseName' == $key) {
return $this->chineseName;
}
return ;
} public function getUid(){
return $this->uid;
} public function getUserName(){
return $this->username;
} public function getChineseName(){
return $this->chineseName;
} /**
* 生成加密的登陆cookie
*/
private function encodeAuth($uid,$username,$ua,$chineseName=null){
$tm = time();
$ua = md5($ua);
$info = "$uid\t$username\t$ua\t$tm\t$chineseName";
$des = new Session_DES();
$str = $des->encrypt($info);
return $str;
} /**
* 解析加密cookie
*/
private function decodeAuth($str){
$des = new Session_DES();
$info = explode("\t",@$des->decrypt($str));
if(is_array($info)){
return $info;
}else{
return array();
}
} public function auth($controller,$action)
{
if(!in_array($controller,$conArr)){
return false;
}
if(!in_array($action,$actArr)){
return false;
}
return true;
}
}

结合上面两段代码,分离出来其实得到的会话类(常常做的就是把一样的的跟不一样的分离出来),只需要3个变量两个是des用到的iv偏移量,key加密秘钥,跟cookie名,然后我们就可以写我们的拓展了,其中踩到了很多坑,一些makefile,config.w4等小语法,lib依赖,总之还是学到了很多

代码地址:https://github.com/lietdai/ldclass

编写php拓展实例--slime项目(用户登录会话类)的更多相关文章

  1. IDEA项目搭建十二——站点用户登录会话实现

    一.简介 前两天写了一篇用户登录会话设计的脑图,这次就把这个引入到项目中实现,总体来说需要几步先罗列一下: 1.需要一个Cookie工具类用于读写cookie 2.需要一个Cache工具类用于在服务端 ...

  2. Intellij IDEA 构建Spring Web项目 — 用户登录功能

    相关软件: 1.Intellij IDEA14:http://pan.baidu.com/s/1nu16VyD 2.JDK7:http://pan.baidu.com/s/1dEstJ5f 3.Tom ...

  3. DRF 商城项目 - 用户( 登录, 注册,登出,个人中心 ) 逻辑梳理

    用户登录 自定义用户登录字段处理 用户的登录时通过 手机号也可以进行登录 需要重写登录验证逻辑 from django.contrib.auth.backends import ModelBacken ...

  4. 36 Flutter仿京东商城项目 用户登录 退出登录 事件广播更新状态

    Login.dart import 'dart:convert'; import 'package:dio/dio.dart'; import 'package:flutter/material.da ...

  5. vue项目用户登录状态管理,vuex+localStorage实现

    安装vuex cnpm install vuex --save-dev

  6. php 获取ip地址的5种方法,插入用户登录日志实例

    php 获取ip地址的5种方法,插入用户登录日志实例,推荐使用第二种方法 <?php //方法1: $ip = $_SERVER["REMOTE_ADDR"]; echo $ ...

  7. SpringMVC+Apache Shiro+JPA(hibernate)案例教学(二)基于SpringMVC+Shiro的用户登录权限验证

    序: 在上一篇中,咱们已经对于项目已经做了基本的配置,这一篇文章开始学习Shiro如何对登录进行验证. 教学: 一.Shiro配置的简要说明. 有心人可能注意到了,在上一章的applicationCo ...

  8. JavaWeb实现用户登录注册功能实例代码(基于Servlet+JSP+JavaBean模式)

    一.Servlet+JSP+JavaBean开发模式(MVC)介绍 Servlet+JSP+JavaBean模式(MVC)适合开发复杂的web应用,在这种模式下,servlet负责处理用户请求,jsp ...

  9. vue新手入门之使用vue框架搭建用户登录注册案例,手动搭建webpack+Vue项目(附源码,图文详解,亲测有效)

    前言 本篇随笔主要写了手动搭建一个webpack+Vue项目,掌握相关loader的安装与使用,包括css-loader.style-loader.vue-loader.url-loader.sass ...

随机推荐

  1. 精通C#(第6版)

    <精通C#(第6版)> 基本信息 原书名:Pro C# 5.0 and the .NET 4.5 framework,sixth edition 作者: (美)Andrew Troelse ...

  2. CSS/CSS3 如何实现元素水平居中

    更新:可直接访问 [CSS/CSS3 如何实现元素水平居中] 查看效果,右键查看源代码 -------------------------------------------------分割线---- ...

  3. linux 文件系统(inode和block)

    linux文件系统(inode block superblock)   先说一下格式化:每种操作系统所设置的文件属性/权限并不相同,为了存放这些文件所需的数据,因此就需要将分区格式化,以成为操作系统能 ...

  4. [CAMCOCO][C#]我的系统架构 总图

    之前写的感觉有点乱,把架构的设计图先放上来吧,对照着说. CAMCOCO架构能够支持的模型: 1.B/S程序,比如CRM什么的,和访问普通网站没什么区别,都是从WEB服务器上进行操作: 2.APP的服 ...

  5. Android实现抽奖转盘

    一.SurfaceView认识及的应用的思路 SurfaceView继承自(extends)View,View是在UI线程中进行绘制: 而SurfaceView是在一个子线程中对自己进行绘制,优势:避 ...

  6. T-SQL openquery 删除报错 “键列信息不足或不正确。更新影响到多行”

    DELETE OPENQUERY (VERYEAST_MAIN_MYSQL_CONN, 'SELECT college_userid FROM college_student_information ...

  7. asp.net ajax 调用错误解决

    ajax调用aspx页面出现如下错误 前台源代码: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" ...

  8. js 动态添加元素(div、li、img等)及设置属性

    把一串 html 标签赋给一个 javascript 变量,除属性的值要用转义的双引号外,某些时候字符串还很长,显得有些复杂.如果用 js 动态添加元素,就不会有那么复杂的字符串出现,代码阅读性强一点 ...

  9. 20150515--关于IIS的备忘(WIN7)

    一.IIS服务位置: 1)控制面板--程序和功能 2)点击打开或关闭Windows功能, 3)Internet服务信息(英文:internet information services)--Web管理 ...

  10. AMQ学习笔记 - 05. 客户端模板化

    概述 客户端编程模型中,大部分的步骤都是相同的.将相同的部分做成模板,将不同的部分预留接口,实现者就只需要针对不同的部分提供实现. 设计 类图 发送方客户端 说明: 基于模板的思想,SendTempl ...