最近公司换了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. MVC+easyui 完整实现

    学习mvc半月有余,趁几天假期,没事做就动手做一个完整的网站玩玩,顺便和上家公司的方法做个比较.页面引擎采用mvc自带的功能,建立视图,交给.net自带渲染出页面(对比:上家公司采用的第三方组件渲染的 ...

  2. 【Android 界面效果15】Android UI 之一步步教你自定义控件(自定义属性、合理设计onMeasure、合理设计onDraw等)

        Android开发做到了一定程度,多少都会用到自定义控件,一方面是更加灵活,另一方面在大数据量的情况下自定义控件的效率比写布局文件更高.     一个相对完善的自定义控件在布局文件中和java ...

  3. Java优先级队列

    package com.lk.A; import java.util.PriorityQueue; public class Test5 { public static void main(Strin ...

  4. Java Script基础(十二) 正则表达式

    一.正则表达式中常用的符号 虽然可以使用string函数来完成验证,但是这种验证方式不够严谨,而且编写比较麻烦.而正则表达式是一种描述字符模式的对象,由一些特殊的符号组成,其组成的字母模式用来匹配各种 ...

  5. 剑指Offer03 逆序输出链表&链表逆序

    多写了个逆序链表 /************************************************************************* > File Name: ...

  6. 转:从三层架构到MVC-MVP

    当然这种架构模式本身的一些问题也会在接下来的内容就加以介绍,另外就是如果大家有什么不同观点的话,欢迎拍砖(只要不打脸就行,呵呵). 一. MVC是谁提出的 模型-视图-控制器(MVC)是Xerox P ...

  7. asp结合ajax中文乱码问题

    XMLHttpRequest 在w3c标准中这样提到: 如果响应包含了为响应体指定字符编码的头部,就使用该编码.否则,假定使用 Unicode UTF-8. 前端页面sele.asp <&quo ...

  8. Ubuntu 16.04 - python3 安装mysql驱动

    阿西吧,今天碰到一件特别蛋疼的事,给Ubuntu安装Python的MySQL驱动,驱动显示安装成功了 pip install mysql-connector 但是 在程序中导入,老是报错. Trace ...

  9. SQL Server2000安装

    本篇文章介绍了安装SQL Server 2000各版本的软硬件配置要求,企业版安装过程的详细步骤,以及需要注意的事项.其他版本可以参考安装 注意:Windows XP不能装企业版.win2000\wi ...

  10. PB出现中文乱码

    在design>option>font把字体都设定为宋体 tools>system option>font 把字体都设定为宋体 然后重启PB即可