技术相关

PHP 检测是否 Ajax 请求

PHP如何判断ajax请求。这个问题要解决有一个先决条件,那就是你使用ajax请求网页内容时,它会向服务器传递一个 HTTP_X_REQUESTED_WITH 的参数,你可以利用如下方法判断某个请 求是ajax请求

1
1 $ajax = false;
2	 
3	if(!empty($_SERVER['HTTP_X_REQUESTED_WITH']) &&
4	    strtolower($_SERVER['HTTP_X_REQUESTED_WITH']) =='xmlhttprequest') {
5	    $ajax = true;
6	}

利用这个来进行判断操作,可以使网页端的URL保持一致,同时能够对两种不同的请求得到不同内容的网页。即实现了用户操作优化,又不影响搜索引擎收录,我觉得是一个很棒的解决方案!

这里有一个另外需要注意的问题,就是如果你的jquery请求是通过iframe打开网页的,那么HTTP_X_REQUESTED_WITH参数不会被传递,也就是说你没有办法判断请求的类型。

谈谈$_SERVER

$_SERVER是一个包含诸如头信息(header),路径(path)和脚本位置(script locations)的数组.这是PHP手册的原话.

有一些头信息是系统本身就存在的,且不能更改,比如说HOST, USER_AGENT等等,注意我们常用到的是$_SERVER[‘HTTP_HOST’], $_SERVER[‘HTTP_USER_AGENT’],这里面都有HTTP前缀,实现上在 header信息里面是没有HTTP前缀的,信息名称只是HOST之类的,$_SERVER自动的为他们加上HTTP前缀。为了是:

防止与本身的其它信息相冲突
是为了表示他们是header头部的信息

所以$_SERVER中的变量凡是带有HTTP前缀的均是header信息,没有的则不是, 如:$_SERVER[‘QUERY_STRING’], $_SERVER[‘PHP_SELF’]等.

自定义header头部信息

用户可以创建自定义的header头部信息,并把这些信息发送给服务器端, 服务器端就会记录这些header信息,并把他们存储到$_SERVER变量中,当然实现的变量名称变成了”HTTP_”连上用户的变量名称.比如上面的判断ajax请求.实际上是在向服务器发送异步请求之前,向header中加入一条header信息,实际的语句应该是:HttpRequest 对象.

setRequestHeader(  "X_REQUESTED_WITH"  ,  "任意字符串"  ); 

服务器端通过$_SERVER[‘HTTP_X_REQUESTED_WITH’]这个变量是否存在,或具体的值来判断是否是ajax请求或具体是哪一条ajax请求. 所以利用这个不但可以验证请求是否是ajax请求,还可以获取更多的信息.

同样我们可以在header中任意设置其它的信息,用于跟服务器端进行数据交流.

比如:

xmlHttpRequest.setRequestHeader(  "test"  ,  "just a test"  ); 

那么被请求的服务器页面中的变量$_SERVER[‘HTTP_TEST’]的值就是”just a test”; 注意这里面前面的test不固定大小写.后面的$_SERVER[‘HTTP_TEST’]必须是大写.




PHP加密解密函数实例教程

PHP加密解密函数分享,一个是 Discuz! 的 authcode 加密函数,一个是encrypt()函数。

需要使用PHP将特定的信息进行加密,也就是通过加密算法生成一个加密字符串,这个加密后的字符串可以通过解密算法进行解密,便于程序对解密后的信息进行处理。

最常见的应用在用户登录以及一些API数据交换的场景。

加密解密原理一般都是通过一定的加密解密算法,将密钥加入到算法中,最终得到加密解密结果。

1、非常给力的authcode加密函数,Discuz!经典代码(带详解):

function authcode($string, $operation = 'DECODE', $key = '', $expiry = 0) {
	// 动态密匙长度,相同的明文会生成不同密文就是依靠动态密匙 
	$ckey_length = 4; 
	// 密匙 
	$key = md5($key ? $key : $GLOBALS['discuz_auth_key']); 
	// 密匙a会参与加解密 
	$keya = md5(substr($key, 0, 16)); 
	// 密匙b会用来做数据完整性验证 
	$keyb = md5(substr($key, 16, 16)); 
	// 密匙c用于变化生成的密文 
	$keyc = $ckey_length ? ($operation == 'DECODE' ? substr($string, 0, $ckey_length): 
	substr(md5(microtime()), -$ckey_length)) : ''; 
	// 参与运算的密匙 
	$cryptkey = $keya.md5($keya.$keyc); 
	$key_length = strlen($cryptkey); 
	// 明文,前10位用来保存时间戳,解密时验证数据有效性,10到26位用来保存$keyb(密匙b), 
	//解密时会通过这个密匙验证数据完整性 
	// 如果是解码的话,会从第$ckey_length位开始,因为密文前$ckey_length位保存 动态密匙,以保证解密正确 
	$string = $operation == 'DECODE' ? base64_decode(substr($string, $ckey_length)) : sprintf('%010d', $expiry ? $expiry + time() : 0).substr(md5($string.$keyb), 0, 16).$string; 
	$string_length = strlen($string); 
	$result = ''; 
	$box = range(0, 255); 
	$rndkey = array(); 
	// 产生密匙簿 
	for($i = 0; $i <= 255; $i++) {
		$rndkey[$i] = ord($cryptkey[$i % $key_length]); 
	}
	// 用固定的算法,打乱密匙簿,增加随机性,好像很复杂,实际上对并不会增加密文的强度 
	for($j = $i = 0; $i < 256; $i++) {
		$j = ($j + $box[$i] + $rndkey[$i]) % 256; 
		$tmp = $box[$i]; 
		$box[$i] = $box[$j]; 
		$box[$j] = $tmp; 
	}
	// 核心加解密部分 
	for($a = $j = $i = 0; $i < $string_length; $i++) {
		$a = ($a + 1) % 256; 
		$j = ($j + $box[$a]) % 256; 
		$tmp = $box[$a]; 
		$box[$a] = $box[$j]; 
		$box[$j] = $tmp; 
		// 从密匙簿得出密匙进行异或,再转成字符 
		$result .= chr(ord($string[$i]) ^ ($box[($box[$a] + $box[$j]) % 256])); 
	}
	if($operation == 'DECODE') {
		// 验证数据有效性,请看未加密明文的格式 
		if((substr($result, 0, 10) == 0 || substr($result, 0, 10) - time() > 0) && 
			substr($result, 10, 16) == substr(md5(substr($result, 26).$keyb), 0, 16)) {
			return substr($result, 26); 
		} else {
			return ''; 
		}
	} else {
		// 把动态密匙保存在密文里,这也是为什么同样的明文,生产不同密文后能解密的原因 
		// 因为加密后的密文可能是一些特殊字符,复制过程可能会丢失,所以用base64编码 
		return $keyc.str_replace('=', '', base64_encode($result)); 
	}
}

函数 authcode($string, $operation, $key, $expiry)中的$string:字符串,明文或密文;$operation:DECODE表示解密,其它表示加密;$key:密匙;$expiry:密文有效期。

用法:

$str = 'yangjunwei.com'; 
$key = 'yangjunwei'; 
echo authcode($str,'ENCODE',$key,0)."<br>"; //加密 
$str = '4d23H0IN8iczYa1KporUtOdgnBwuEFYM0RXgA+HQh/eXfxjZurkcepuGAw'; 
echo authcode($str,'DECODE',$key,0)."<br>"; //解密

2、加解密函数encrypt()

function encrypt($string,$operation,$key=''){
	$key=md5($key); 
	$key_length=strlen($key); 
	$string=$operation=='D'?base64_decode($string):substr(md5($string.$key),0,8).$string; 
	$string_length=strlen($string); 
	$rndkey=$box=array(); 
	$result=''; 
	for($i=0;$i<=255;$i++){
		$rndkey[$i]=ord($key[$i%$key_length]); 
		$box[$i]=$i; 
	}
	for($j=$i=0;$i<256;$i++){
		$j=($j+$box[$i]+$rndkey[$i])%256; 
		$tmp=$box[$i]; 
		$box[$i]=$box[$j]; 
		$box[$j]=$tmp; 
	}
	for($a=$j=$i=0;$i<$string_length;$i++){
		$a=($a+1)%256; 
		$j=($j+$box[$a])%256; 
		$tmp=$box[$a]; 
		$box[$a]=$box[$j]; 
		$box[$j]=$tmp; 
		$result.=chr(ord($string[$i])^($box[($box[$a]+$box[$j])%256])); 
	}
	if($operation=='D'){
		if(substr($result,0,8)==substr(md5(substr($result,8).$key),0,8)){
			return substr($result,8); 
		}else{
			return''; 
		}
	}else{
		return str_replace('=','',base64_encode($result)); 
	}
}

函数encrypt($string,$operation,$key)中$string:需要加密解密的字符串;$operation:判断是加密还是解密,E表示加密,D表示解密;$key:密匙。

用法:

$str = 'yangjunwei.com';
$key = 'yangjunwei';
$token = encrypt($str, 'E', $key); //加密 
echo $token."<br>";
echo encrypt($token, 'D', $key)."<br>"; //解密

前端优化之图片小述

<img src="small.jpg" data-fullsrc="large.jpg">

Date-fullsrc是html5中的一个定义文件的属性,宽度超过480px的屏幕,就会加载较大分辨率的图片(large.jpg),小屏幕分辨率的就会加载较小的图片(small.jpg)。

提升页面大小的效率,不仅仅是取决于使用精灵或是压缩代码,给定页面的请求数量在前端性能中也占有了很不小的重量。减少请求可以让你的网站加载更快,而其中一种减少页面请求的方法就是用Data URI代替图片的src属性:

<!-- 以前的写法 -->
<img src="http://images.cnblogs.com/logo.png" />

<!-- 使用data URI的写法 -->
<img src="data: image/jpeg;base64,/9j/4AAQSkZJRgABAgAAZABkAAD/7AARRHVja3kAAQAEAAAAPAAA/+4ADkFkb2JlAGTAAAAAAf/bAIQABgQEBAUEBgUFBgkGBQYJCwgGBggLDAoKCwoKDBAMDAwMDAwQDA4PEA8ODBMTFBQTExwbGxscHx8fHx8fHx8fHwEHBwcNDA0YEBAYGhURFRofHx8fHx8fHx8fHx8fHx8fH ...." />

<-- 范例:  http://davidwalsh.name/demo/data-uri-php.php -->

当然页面大小会增加(如果你的服务器使用适当的gzip内容,这个增加会很小),但是你减少了潜在的请求,同时也在过程中减少了服务器请求的数量。现在大多数浏览器都支持Data URI,在CSS中的背景骨片也可以使用Data URI,因此这个策略现在已经可以在应用层级,广泛应用。

根据屏幕宽度自动缩放图片大小

在开发中,我们经常遇到需要适配屏幕宽度的应用,比如图片自动缩放到适合尺寸。

<script>
    window.onload = function () {
        ReImgSize();
    }
</script>
<script type="text/javascript" defer>
    function ReImgSize() {
        var clientWidth = document.body.clientWidth;// window.innerWidth; //
        //首页主体背景
        var indexH = 747 * clientWidth / 640;   
		$(".default img").css("width", clientWidth * changeTwoDecimal(600 / 640));		
    }

    function getElementsByClassName(classname) {
        var children = document.all; 
        var elements = new Array(); 
        for (var i = 0; i < children.length; i++) {
            var child = children[i];
            var classNames = child.className.split(' ');
            for (var j = 0; j < classNames.length; j++) {
                if (classNames[j] == classname) {
                    elements.push(child);
                    break;
                }
            }
        }
        return elements;
    }

    function changeTwoDecimal(x) {
        var f_x = parseFloat(x);
        if (isNaN(f_x)) {
            alert('function:changeTwoDecimal->parameter error');
            return false;
        }
        var f_x = Math.round(x * 100) / 100;
        return f_x;
    }
</script>

在网页中嵌入任意字体以及解决访问速度慢的方案—google在线字体库应用

字体使用是网页设计中不可或缺的一部分。经常地,我们希望在网页中使用某一特定的字体, 但是该字体并非主流操作系统的内置字体,我们常用的方法就是把特殊字体处理成图片,这样做有很多缺陷, 但是谷歌发布了Google在线字体库,这样在网站设计的时候只要调用Google在线字体库就可以让网页在客户端很好的显示,而不用只用枯版的字体了,实现字体多样化。

特殊字体处理成图片弊端

1、不可能大范围使用
2、不易修改
3、不利于seo优化
利用google在线的字体库优势
1、可继续使用html文本比起自己自做字体库上传到自己的服务器可以减轻服务器负担
2、兼用性好,不需担心像各个浏览器对 CSS 解析不同导致这样那样的 bug、HACK。
3、无需引用js
但是也存在一定的弊端,
1、只能引进应为的字体库,中中文不支持
2、随时有被墙外的可能性
 
虽然有一定的缺陷,但是google在线字体库的有时还是很明显的,下面就介绍下具体的使用方法:
header部分引用要用的字体

<link rel=’stylesheet’ type=’text/css’ href=’http://fonts.googleapis.com/css?family=Font name’ >

在css中使用

css selector{
   Font-family: “font name”, serif;
}

就是这么简单

在引用时注意一定要始终有个字体的备份方案,上面的代码你可能也发现了我使用了serif 来作为后备的字体,这样做是为了以防万一

如果你想一次行载入多个字体

http://fonts.googleapis.com/css?family=Vollkorn|Yanone|Droid+Sans

字体名称之间用 | 隔开,并且不留空格,还要注意Droid Sans字体的使用,对于字体名称中有空格的,需用 + 符号来代替空格

 Google在线字体库更多字体: http://code.google.com/webfonts

现在谷歌被封禁,访问速度很慢,我们可以使用360网站卫代理访问Google免费字体,在html中加入即可。

<link href='http://fonts.useso.com/css?family=Open+Sans:300,400,600&subset=latin,latin-ext' rel='stylesheet'> 

使用PHP的框架ThinkPHP必须要掌握的程序开发调试手段

使用ThinkPHP应该掌握的调试手段

经常看到有人问到findAll的返回数据类型是什么之类的问题,以及出错了不知道什么原因的情况,其实还是没有熟悉ThinkPHP内置的调试手段和方 法,抛开IDE本身自带的调试方式不说,如果你正在用或者打算用ThinkPHP开发的话,那么下面一些和调试相关的方法你是应该要了解和掌握的:
1、在项目配置文件里面打开调试模式DEBUG_MODE,这样能够让你发现大部分的错误原因
2、如果不想使用调试模式,可以单独开启页面Trace显示。发现很多人不想使用调试模式的原因居然是因为有页面Trace信息的输出,其实这里面有一个 误区,以为调试模式就一定会有页面Trace,但其实调试模式和页面Trace没有必然的关系,只是因为打开调试模式后,系统默认的调试配置文件会开启页 面 Trace显示,所以你完全可以给项目单独定义调试配置文件。
3、使用系统定义的dump函数,该方法同var_dump可以输出任何类型的变量信息,而且更加有利于在浏览器里面查看,例如:

    $User = D(“User”);$list    = $User->findAll();dump($list);

4、页面Trace信息只能显示当前页面执行的sql语句,但无法查看ajax方式执行的后台操作里面的sql语句,所以你还可以开启sql日志记录 SQL_DEBUG_LOG 来记录每一条执行的sql语句,并且可以查看到每条sql语句的执行时间 sql日志文件位于Logs目录下面,会自动按日期来区分每天的sql日志。
5、另外一个是在执行某个数据操作后怀疑sql执行有错误的话,可以使用模型类的getLastSql方法来查看上次执行的sql语句,以便分析具体的错误原因。例如:

    $User = D(“User”);$User->id = 3;$User->name = ‘ThinkPHp’;$User->save();echo $User->getLastSql();// 输出 update think_user set name=’ThinkPHP’ where id=3;

6、当需要调试某段代码的运行时间的时候,可以使用系统提供的debug_start($label)和debug_end($label)方法,例如:

    debug_start(‘demo’);// 这里是你的代码段…….debug_end(‘demo’);

thinkphp的各种内部函数 D()、F()、S()、C()、L()、A()、I()详解

D、F、S、C、L、A、I 他们都在functions.php这个文件家
下面我分别说明一下他们的功能

D() 加载Model类
M() 加载Model类

A() 加载Action类
L() 获取语言定义
C() 获取配置值    用法就是   C(”这里填写在配置文件里数组的下标”)
S() 全局缓存配置 用法S(“这里相当于一个唯一的标识”)
F() 快速文件数据读取和保存 针对简单类型数据 字符串、数组
I()   快速创建一个对象实例

以上如果有遗漏的 内部函数

ThinkPHP值得大家学习的源码,特别是他们的中文注释让我们很容易就了解他源码中的内容

在用thinkphp做程序时经常能用到的内部函数,对于换成这一块用S()比较会减轻对数据库的负担!

在thinkphp中出了有内部还是,thinkphp还把配置文件保存为静态变量,在模板里方便快速调用

如果{$Thinkphp.config.参数名称}

如果主键是自动增长的,用add()的时候他会自动的返回一个插入新记录的ID值

我们在做用户金币或者是积分或者是投票的时候,需要对指定的字段自加1这时候我可以可以这样写

$s= D(’admin’);$s->score='(score+1)’;$s->save(); 这样我们会剩下许多的步骤。

如果要修改指定的一个字段用简单的办法  D(‘admin’)->setField(‘name’,’hehe’,’id=2′);

我们要灵活的应用  thinkphp 查询语句

一篇对ThinkPHP的总结文章 全面概括了ThinkPHP的基础知识点

php的开发框架是比多的,最近我们针对zend Framework、CakePHP等框架进行了简单的比较后,认为ThinkPHP的开发效率最高,代码最为简洁,最适合中小型项目开发。

 
1. 模板中不能使用的标签
{$content} {$i}
 
2. If标签
如: <if condition=”$name eq 1 “>
试验后总是有想不到的错误, 这样,还不如直接用<?php if(…){ …?>来得快些呢.
约定:
1.所有类库文件必须使用.class.php作为文件后缀,并且类名和文件名保持一致
2.控制器的类名以Action为后 缀
3.模型的类名以Model为后缀,类名第一个字母须大写
4.数据库表名全部采用小写,
 
如:
数据表名: 前缀_表名
模型类名: 表名Model 注:这里的表名第一个字母要大写
创建对象: D(‘表名’) 注:这里的表名第一个字母要大写
 
定义控制器类
class IndexAction extends Action{
public function show(){
echo ‘这是新的 show 操作’;
}
}
然后在浏览器里面输入
http://localhost/myApp/index.php/Index/show/
 
定义模型类:
class 表名Model extends Model{
[//手动定义字段[可选]
protected $fields = array(
‘id’,
‘username’,
’email’,
‘age’,
‘_pk’=>’id’, //主键
‘_autoInc’=>true //是否自增
)
]
}
 
记录的修改:
$User = D(“User”) // 实例化 User 对象
$User->find(1) // 查找 id 为 1 的记录
$User->name = ‘ThinkPHP’ // 把查找到的记录的名称字段修改为 ThinkPHP
$User->save() // 保存修改的数据
更新特定字段的值
$User->setField(‘name’,’TopThink’,’id=1′)
同 样可以支持对字段的操作
$User->setField(‘score’,'(score+1)’,’id=1′)
 
新建记录,方法1:
$User = new UserModel() //实例化 User 对象
$User->字 段名 = 字段值 //给字段赋值
$User->add() //添加记录
新建记录,方法2:
$data[‘字段名’] = 字段值; //给字段赋值
$User = D(‘User’); //实例化 User 对象
$User->add($data); //$insertId,Add 方法的返回值就是最新插入的主键值,可以直接获取。
新增多条记录:
$User = new UserModel()
$data[0][‘name’] = ‘ThinkPHP’
$data[0][’email’] = ‘sjolzy@chen.com’
$data[1][‘name’] = ‘流年’
$data[1][’email’] = ‘chen@sjolzy.cn’
$User>addAll($data)
 
删除记录
$User->find(2)
$User->delete() // 删除查找到的记录
$User->delete(‘5,6’) // 删除主键为 5、6 的数据
$User->deleteAll() // 删除查询出来的所有数据
 
记录查询
$User->getDbFields() //获取当前数据字段
$User->findAll(); //查找所有记录
$User->findAll(‘1,3,8’) //查询主键为1,3,8的记录集
$User->count() // 获取记录数
$User->max(‘score’) // 获取用户的最大积分
$User->min(‘score’,’score>0′) // 获取积分大于 0 的用户的最小积分
$User->avg(‘字段名’) // 获取所有记录的字段值的平均值
$User->sum(‘字段名 ‘) // 统计字段值
$User->getN(2,’score>80′,’score desc’) // 返回符合条件的第 2 条记录
$User->getN(2,’score>80′,’score desc’) //还可以获取最后第二条记录
$User->first(‘score>80′,’score desc’) //如果要查询第一条记录,还可以使用
$User->last(‘score>80′,’score desc’) // 获取最后一条记录
$User->top(5,”,’score desc’) // 获取积分最高的前 5 条记录
$User->getBy(‘name’,’liu21st’) //跟据字段的字段值来查询记录
$Model = new Model() // 实例化一个 model 对象 没有对应任何数据表
$Model->query(“select * from think_user where status=1”)
$objrs = $Model->query(“select * from think_user where status=1”) //自定义查询
$Model->execute(“update think_user set name=’thinkPHP’ where status=1”) //用于更新和写入数据的 sql 操作,返回影响的记录数
 
$User->startTrans() // 启动事务
$User->commit() // 提交事务
$User->rollback() // 事务回滚
模板:
$this->assign(‘name’,$value); //在 Action 类里面使用 assign 方法对模板变量赋值,无论何种变量类型都统一使用 assign 赋值
$this->display() // 输出模版文件
 
批量赋值
$array[‘name’] = ‘thinkphp’
$array[’email’] = ‘chen@sjolzy.cn’
$array[‘phone’] = ‘12335678’
$this->assign($array)
 
$this->display() // 调用 User 模块的 read 操作模版
$this->display(‘edit’) // 调用 User 模块的 edit 操作模版
$this->display(‘Member:read’) // 调用 Member 模块的 read 操作模版
$this->display(‘Xp@User:edit’) // 调用 Xp 主题的 User 模块的 edit 操作模版
$this->display(‘../Member/read.html’) // 直接指定模版文件的全名
 
模板标签:
{ } 或 {// 注释内容 } //模板注释
{$user[‘name’]} //输出数组变量
{$user:name} //输出对象的属性
为了方便模板定义,无论输出的模板变量是数组还是对象,都可以用下列统一方式输出:
{$user.name}
如果是多维数组或者多 层对象属性的输出,请使用下面的定义方式:
{$user[‘sub’][‘name’]}
{$user:sub:name}
 
使用函数:
格式:{$varname|function1|function2=arg1,arg2,### }
说明:
{ 和 $ 符号之间不能有空格 ,后面参数的空格就没有问题
###表示模板变量本身的参数位置
 
系统变量
{$Think.server.script_name } //取得$_SERVER 变量
{$Think.session.session_id|md5 } // 获取$_SESSION 变量
{$Think.get.pageNumber } //获取$_GET 变量
{$Think.cookie.name } //获取$_COOKIE 变量
 
系统常量
{$Think.const.__FILE__ }
{$Think.const.MODULE_NAME }
特殊变量 ,由 ThinkPHP 系统定义的常量
{$Think.version } //版本
{$Think.now } //现在时间
 
快捷输出
{:function(…)} //执行方法并输出返回值
{~function} //执行方法不输出
{@var} //输出 Session 变量
{&var} //输出配置参数
{%var} //输出语言变量
{.var} //输出 GET 变量
{^var} //输出 POST 变量
{*var} //输出常量
 
包含外部文件
<include file=”$tplName” /> // 用变量控制要导入的模版
<include file=”../Public/header.html” /> // 使用一个完整的文件名包含
 
循环输出
iterate 还有其它的别名,包括 volist,resultset,sublist
 
模版赋值:
$User = D(‘User’)
$list = $User->findAll()
$this->assign(‘list’,$list)
 
模版定义:
<iterate name=”list” id=”vo”>
{$vo.name}
</iterate>
注意 name 和 id 表示的含义
// 输出 list 的第 5~15 条记录
<iterate name=”list” id=”vo” offset=”5″ length=’10’>
{$vo.name}
</iterate>
// 输出偶数记录
<iterate name=”list” id=”vo” mod=”2″ >
<eq name=”mod” value=”1″>
{$vo.name}
</eq>
</iterate>
// 输出 key
<iterate name=”list” id=”vo” key=”k” >
{$k}.{$vo.name}
</iterate>
//子循环输出
<volist name=”list” id=”vo”>
<sublist name=”vo[‘sub’]” id=”sub”>
{$sub.name}
</sublist>
</volist>
 
Switch 标签
<switch name=”name”>
<case value=”1″>value1</case>
<case value=”2″>value2</case>
<default />default
</switch>
其 中 name 属性可以使用函数以及系统变量,例如:
<switch name=”Think.get.userId|abs”>
<case value=”1″>admin</case>
<default />default
</switch>
也 可以对 case 的 value 属性使用变量,例如:
<switch name=”userId”>
<case value=”$adminId”>admin</case>
<case value=”$memberId”>member</case>
<default />default
</switch>
 
比较标签
<eq name=”name” value=”value”>value</eq> // name 变量的值等于 value 就输出
<neq name=”name” value=”value”>value</neq> // name 变量的值不等于 value 就输出
<gt name=”name” value=”5″>value</gt> // name 变量的值大于 5 就输出
<egt name=”name” value=”5″>value</egt> // name 变量的值大于等于 5 就输出
<lt name=”name” value=”5″>value</lt> // name 变量的值小于 5 就输出
<elt name=”name” value=”5″>value</elt> // name 变量的值小于等于 5 就输出
 
//其实上面的所有标签都是 compare 标签的别名
// 其中 type 属性的值就是上面列出的判断标签名称
<compare name=”name” value=”5″ type=”eq”>value</compare> // name 变量的值等于 5 就输出
If标签
<if condition=”$name eq 1 “> value1
<elseif condition=”$name eq 2″ />value2
<else /> value3
</if>
 
C操作
操作(动态)配置: 主要用于Action方法里面
获取:
C(‘配置参数’)
设置:
C(‘配置参数 ‘,新值)
 
A操作
快速创建Action对象:
$action = A(‘User’);
等效于
$action = new UserAction();
 
D操作
快速创建模型数据对象:
$model = D(‘User’);
等效于
$model = new UserModel();
 
S操作
快速操作缓存方法
获取:
S(‘name’)
设置:
S(‘name’,’value’);
删 除:
S(‘name’,NULL);
 
F操作
快速文件数据保存方法
使用方法与S操作一样
 
L操作
快速操作语言变量
获取:
L(‘语言变量’);
设置:
L(‘语言变量’,’值’);
如: L(‘USER_INFO’,’用户信息’); //设置名称为USER_INFO的语言变量
批量赋值:
$arr[‘语言变量1’] = ‘值1’;
$arr[‘语言变量2’] = ‘值2’;
L($arr);
 
Create PROCEDURE procedure1
(IN parameter1 INTEGER)
BEGIN
DECLARE variable1 CHAR(10);
IF parameter1 = 17 THEN
SET variable1 = ‘birds’;
ELSE
SET variable1 = ‘beasts’;
END IF;
Insert INTO table1 VALUES (variable1);
END
 
ThinkPHP系统常量
THINK_PATH   // ThinkPHP 系统目录
APP_PATH   // 当前项目目录
APP_NAME   // 当前项目名称
MODULE_NAME   //当前模块名称
ACTION_NAME   // 当前操作名称
TMPL_PATH   // 项目模版目录
LIB_PATH   // 项目类库目录
CACHE_PATH   // 项目模版缓存目录
CONFIG_PATH   //项目配置文件目录
LOG_PATH   // 项目日志文件目录
LANG_PATH   // 项目语言文件目录
TEMP_PATH   //项目临时文件目录
PLUGIN_PATH   // 项目插件文件目录
VENDOR_PATH   // 第三方类库目录
DATA_PATH   // 项目数据文件目录
IS_APACHE   // 是否属于 Apache
IS_IIS    //是否属于 IIS
IS_WIN    //是否属于Windows 环境
IS_LINUX   //是否属于 Linux 环境
IS_FREEBSD   //是否属于 FreeBsd 环境
NOW_TIME   // 当前时间戳
MEMORY_LIMIT_ON // 是否有内存使用限制
MEMORY_LIMIT_ON // 是否有内存使用限制
OUTPUT_GZIP_ON   // 是否开启输出压缩
MAGIC_QUOTES_GPC // MAGIC_QUOTES_GPC
THINK_VERSION   //ThinkPHP 版本号
LANG_SET   // 浏览器语言
TEMPLATE_NAME   //当前模版名称
TEMPLATE_PATH   //当前模版路径
__ROOT__   // 网站根目录地址
__APP__   // 当前项目(入口文件)地址
__URL__   // 当前模块地址
__ACTION__   // 当前操作地址
__SELF__   // 当前 URL 地址
TMPL_FILE_NAME   //当前操作的默认模版名(含路径)
WEB_PUBLIC_URL   //网站公共目录
APP_PUBLIC_URL   //项目公共模版目录
 
预定义常量
WEB_LOG_ERROR=0     // 错误日志类型
WEB_LOG_DEBUG=1 // 调试日志类型
SQL_LOG_DEBUG=2   // SQL 日志类型
SYSTEM_LOG=0   // 系统方式记录日志
MAIL_LOG=1   // 邮件方式记录日志
TCP_LOG=2   // TCP 方式记录日志
FILE_LOG=3   // 文件方式记录日志
DATA_TYPE_OBJ=1 // 对象方式返回
DATA_TYPE_ARRAY=0 // 数组方式返回
URL_COMMON=0   // 普通模式 URL
URL_PATHINFO=1   // PATHINFO URL
URL_REWRITE=2   // REWRITE URL
HAS_ONE=1   // HAS_ONE 关联定义
BELONGS_TO=2   // BELONGS_TO 关联定义
HAS_MANY=3   // HAS_MANY 关联定义
MANY_TO_MANY=4   // MANY_TO_MANY 关联定义
EXISTS_TO_VAILIDATE = 0 // 表单存在字段则验证
MUST_TO_VALIDATE = 1 // 必须验证
VALUE_TO_VAILIDATE = 2 // 表单值不为空则验证

讲得知识比较多,而且比较杂,但是这些都是ThinkPHP的基础知识点。所以大家如果对上面的都详细的熟悉的去掌握的话,对于大家去Thinkphp开发将会大大的有所帮助。

使用Thinkphp框架时,他的httpd返回状态码的X-Powered-By为ThinkPHP有修改方法

以前用ThinkPHP框架开发了一个小网站,前几天查询页面HTTP状态发现,里面有一项:
X-Powered-By: ThinkPHP 2.0
 
这样虽然没什么,但感觉如果别有用心的人查询会知道你是用这个框架开发的,有可能针对该框架的漏洞进行相关活动。
 
如果要修改这个X-Powered-By,怎么办呢?
 
其实很简单,我们打开视图类文件:
 
ThinkPHP/Lib/Think/Core/View.class.php
 
找到这个方法fetch()
可以看到:
 
header(“X-Powered-By:ThinkPHP”.THINK_VERSION);
 
将这行删除掉,或者修改成你自己的。

如果你使用的是ThinkPHP3.0以上的话,那他的修改地方将是:
render 函数里的:header(‘X-Powered-By:ThinkPHP’);
我是修改成:
header(‘X-Powered-By:www.itokit.com’);

分享php中四种webservice实现的简单架构方法及实例

本人所了解的webservice有以下几种:PHP本身的SOAP,开源的NUSOAP,商业版的PHPRPC,以及使用二进制传输数据流的HessianPHP,那么一下就简单的介绍下这几种webservice在php中的使用,虽然网上也有很多这方面的资料,但是这是我个人实践所得,当然也是从网上找的资料,在此简单的做个笔记。
一:PHP本身的SOAP
所有的webservice都包括服务端(server)和客户端(client)。
要使用php本身的soap首先要把该拓展安装好并且启用。下面看具体的code

首先这是服务端实现:


<?php   
class test   
{   
    function show()   
    {   
        return 'the data you request!';   
    }   
}   
function getUserInfo($name)   
{   
    return 'fbbin';   
}   
//实例化的参数手册上面有,这个是没有使用wsdl的,所以第一个参数为null,如果有使用wsdl,那么第一个参数就是这个wsdl文件的地址。   
$server = new SoapServer(null, array('uri' ='http://soap/','location'='http://localhost/test/server.php'));   
$server->setClass('test');   
//$server->addFunction('getUserInfo');   
$server->handle();   
?>  

然后是客户端

$soap = new SoapClient(null, array('location'='http://localhost/test/server.php','uri' ='http://soap/'));     
  
echo $soap->show();   
//得到:'the data you request!'   
  
//echo $soap->getUserInfo('sss');  

就这么简单,当时这只是一个很简单的例子,其实很多的通信机制都是这么去实现的!
////////////////////////////////////////////////////////////////////////////////

二:PHPRPC
首先到官网(http://www.phprpc.org/zh_CN/ )上面去下载最新版的phprpc,解压之后会有相关的文件,我们把文件进行划分(服务端和客户端文件)如下:
服务端文件:


dhparams   
dhparams.php   
phprpc_server.php   
bigint.php   
compat.php   
phprpc_date.php   
xxtea.php  

客户端文件:

phprpc_client.php   
bigint.php   
compat.php   
phprpc_date.php   
xxtea.php  

我们把服务端文件放在服务端文件夹中,然后把客户端文件放在客户端文件夹中,之后再服务端文件夹中新建个文件(server.php)作为服务,然后再客户端新建个文件(client.php)作为客户端,各自代码如下:
server端:

<?php   
include_once"phprpc_server.php"; //加载phprpc文件   
$server = new PHPRPC_Server();   
$server->add('getUser');   
$server->setDebugMode(true);   
$server->start();   
function getUser( )   
{   
    return ‘the data you request!’;   
}   
  
client端:   
[code][/code]   
<?php   
include_once "phprpc_client.php";   
$client = new PHPRPC_Client('http://127.0.0.1/phpservice/phprpcserver/server.php');   
$data = $client->getUser();   
var_dump($data);   
//得到:the data you request!  

这上面提到wsdl之后会讲到如何生成。
////////////////////////////////////////////////////////////////////////////////

三:开源的NUSOAP
首先到网上去下载最新版的nusoap,现在的最新版本是0.9.5的,解压之后会得到一个lib文件夹,把这个文件分别放到服务端和客户端各一份,然后再服务端和客户端分别建立server.php和client.php文件,作为通信文件。
服务端文件如下:

<?php   
ini_set("soap.wsdl_cache_enabled", 0);//关闭缓存   
require_once("lib/nusoap.php"); //加载nusoap文件   
$server = new soap_server;   
$server->configureWSDL('nusoasp');//设定服务的名称,使用的wsdl来通信,如果不适用wsdl将会更简单,网上有很多的例子   
$server->register('getUserInfo', array('name'="xsd:string", 'email'="xsd:string"), array('return'="xsd:string"));   
$HTTP_RAW_POST_DATA = isset($HTTP_RAW_POST_DATA) ? $HTTP_RAW_POST_DATA : '';   
$server->service( $HTTP_RAW_POST_DATA );   
function getUserInfo($name, $email)   
{   
    return ‘the data you request!’;   
}  

客户端文件如下:

require_once("lib/nusoap.php");   
$client = new soapclient('http://localhost/phpservice/nusoapserver/server.php?wsdl');   
$pagram = array('fbbin', 'fbbin@foxmail.com');   
$string = $client->call('getUserInfo', $pagram);   
//得到:the data you request!  

四:HessianPHP
hessian其实我个人认为他不是一个webservice,只能说是类似而已。因为它不具备webservice的那些特性。它支持的语言比较多我们现在只需要研究php版本的HessianPHP就行了,下载最新版本是v2.0.3的,解压之后会得到一个src的目录,这个目录使我们需要使用的一个核心文件夹。
我们把名字重命名为HessianPHP然后分别分别放到server和client端,然后分别建立server.php和client.php文件。
server端:

<?php   
include_once 'HessianPHP/HessianService.php';//加载核心文件   
class TestService   
{   
    public function __construct()   
    {   
  
    }   
  
    public function add($numa, $numb)   
    {   
        return $numa + $numb;   
    }   
  
    public function check()   
    {   
        return 'fbbiin@gmail.com';   
    }   
}   
$test = new TestService();   
$hessian = new HessianService( $test, array('displayInfo' => true) );   
$hessian->handle();//注意这里不是网上的$hessian->service(),可能是版本不一样,改了吧!我也是看了源码才知道!   
?>  

client 端:

<?php   
include_once 'HessianPHP/HessianClient.php';   
$url = "http://localhost/phpservice/hessianserver/server.php";   
$options = new HessianOptions();   
$client = new HessianClient( $url, $options );   
$num = $client->add( 3, 5 );   
echo $num;//得到:8;   
echo $client->check();//得到:fbbiin@gmail.com;  

////////////////////////////////////////////////////////////////////////////////
以上四种方式为web开发过程中常用到的webservice通信方式。用的最多的是nusoap,个人感觉phprpc其实也不错,这个在性能上面和nusoap基本上差不多,只不过,phprpc在商业上是收费的。还有个hessianPHP好像是用java的,采用的二进制方式传输数据流,其实也是各有千秋啊。更多的详细资料去找百度和谷歌吧。
下面说下生成wsdl文件。
我们在webservice上面进行通信用的最多的和相对来说比较安全的就是使用wsdl了,这种文件可以自己书写,但是不是一定的大牛好像写不出来啊,因此我们需要借助一个工具zend studio来生成wsdl文件。
下面我们就来生成WSDL文件了,File->New—>Other—>Web Service—>WSDL,这样就可以新建一个WSDL文件了,如图。
分享php中四种webservice实现的简单架构方法及实例
分享php中四种webservice实现的简单架构方法及实例
分享php中四种webservice实现的简单架构方法及实例
然后我们就来修改WSDL文件,zeng studio为我们提供了可视化的操作,当然如果你牛的话,你当然是可以改文件代码的,其实也就几个东西, 弄懂了的话也不会太难。分享php中四种webservice实现的简单架构方法及实例
分享php中四种webservice实现的简单架构方法及实例
做完这一步,这个WSDL文件就基本可用了,但又两个问题需要注意:
做到这一步,有可能会测试失败,可能会因为没有进行binding,这个东西有时是需要手动来完成的,在binding上右键选择Generate Binding Content(就是两个大框中间那个小框)就行了。
第二个要注意的是php的WSDL缓存,在做测试时,一般要将WSDL缓存关闭,否则你使用的有可能是原来的WSDL文件,而不是更新过的。关闭缓存有两种方法,第一种就是直接到php.ini中设置soap.wsdl_cache_enabled = 0;第二种就是在php文件中添加一条语句,ini_set(“soap.wsdl_cache_enabled”, “0”);
做到这里,你就可以放心地测试,调用你的server程序了。
说完了,OK!

资料参考:
1:http://winphfar.iteye.com/blog/559347;
2:http://www.iteye.com/blogs/tag/phprpc
3:http://yeyuan.iteye.com/blog/1261491
4:http://www.cnblogs.com/helloxyz/archive/2011/11/5.html
5:http://www.phpx.com/happy/thread-122030-1-1.html