CI学习总结

没有评论

2010 年 12 月 10 日 at 下午 10:57分类:PHP | WEB开发

CI学习总结
1、什么是框架?
框架就是一个没写完的网站,提供了一些基础的功能,比如安全检查,表单难,数据库连接等。其它人可以在上面开发自己的东西。框架把一些烦琐重复而没有创造性的工作帮你做好,而让你从事具有创造性的工作。不过使用框架对执行速度有影响。
2、什么是CodeIgniter?
CodeIgniter是一个小巧的php框架,名称由两个单词组成,code代码Igniter点火器,连起来意思是代码点火器(什么意思?),简称CI
3、Application文件夹可以移动么?
可以移动或改名,修改后需要在index.php里把$application_folder = “application”;这句修改成相应的路径和名称。但是application下的controllers、models、views不能动。
4、什么是MVC?
复杂的不说了,人们都在讲这个,简单点儿几句话:m跟数据打交道,V跟界面打交道,C把M和V连起来,把从M取到的数据交给V显示。
5、CI中怎么连接数据库?
在application/config/database.php里面修改连接参数,然后在需要连接数据库的时候载入数据库就可以了。

$this->load->database();

6、如何取得post数据?
可以使用传统的php方法,

$_POST['name']
或者$this->input->post('name');

7、CI可以使用$_GET么?
不可以,如要实现类似功能,参考下面一问。
8、如何得到URL地址中的参数?
第一种办法:$this->uri->segment(n),n 为你想要得到的段数,具体可以查手册。
第二种办法:假设URL是这样:http://example.com/index.php/blog/list/3/myname
在blog的list方法可以这样写:

function list($id,$name)
{
//$id=3
//$name=myname
}

$id和$name得到链接中的参数,如果参数个数不一样会提示404错误。
9、VIEW怎么相互引用?
view也是php文件,可以直接include,也可以在view里使用load:

$this->load->view('View_name');

这个以实现VIew共用,分享网页头和尾。
10、View里怎么调用CSS和JS?
在VIEW头部这么写:

<link href='<?= base_url();?>css/tabs.css' rel='stylesheet' type='text/css' />
<script type='text/javascript' src='<?= base_url();?>js/jquery.js'></script>

base_url是网站根目录地址。
11、可以给一个view传递多个参数吗?
只能传递一个数组或对象参数,但是数组或对象可以想多复杂就多复杂。
12、controllers下可以建子目录吗?
可以,但只能建一级子目录
13、Controller是否可以互相调用?
不可以,可以用以下方法替代解决:
1.代码写到model里,调用model。
2.代码写到其他类里,比如库、插件、helper等。
3.扩展Controller,代码写到My_Controller里,继承。(sinopf in codeigniter.org.cn)
我的学习方法
第一天看手册
第二天逛CI论坛,搜索一些关键词,比如:VIEW 数据 安全 登录 验证等。
一方面回顾手册里的知识,另一方面,了解大家以前遇到过的问题及解决办法,避免以后走弯路。
比如我搜索到CI自带的SESSION不好用,我以后就不会在SESSION这个问题上折腾了,直接使用kndb session第三方类库。
第三天,看codeigniter.org.cn里的视频,照着做一个blog。
接下来,看源码,下载一个使用CI的开源系统看看,然后动手自己写。

有效防御PHP木马攻击的技巧

没有评论

2010 年 12 月 10 日 at 下午 10:40分类:WEB开发 | 乱七八糟

1、防止跳出web目录
  首先修改httpd.conf,如果你只允许你的php脚本程序在web目录里操作,还可以修改httpd.conf文件限制php的操作路径。比如你的web目录是/usr/local/apache/htdocs,那么在httpd.conf里加上这么几行:
  php_admin_value open_basedir /usr/local/apache
  /htdocs
  这样,如果脚本要读取/usr/local/apache/htdocs以外的文件将不会被允许,如果错误显示打开的话会提示这样的错误:
  Warning: open_basedir restriction in effect. File is in wrong directory in
  /usr/local/apache/htdocs/open.php on line 4
  等等。
  2、防止php木马执行webshell
  打开safe_mode,
  在,php.ini中设置
  disable_functions= passthru,exec,shell_exec,system
  二者选一即可,也可都选
  3、防止php木马读写文件目录
  在php.ini中的
  disable_functions= passthru,exec,shell_exec,system
  后面加上php处理文件的函数
  主要有
  fopen,mkdir,rmdir,chmod,unlink,dir
  fopen,fread,fclose,fwrite,file_exists
  closedir,is_dir,readdir.opendir
  fileperms.copy,unlink,delfile
  即成为
  disable_functions= passthru,exec,shell_exec,system,fopen,mkdir,rmdir,chmod,unlink,dir
  ,fopen,fread,fclose,fwrite,file_exists
  ,closedir,is_dir,readdir.opendir
  ,fileperms.copy,unlink,delfile
  ok,大功告成,php木马拿我们没辙了,遗憾的是这样的话,利用文本数据库的那些东西就都不能用了。
  如果是在windos平台下搭建的apache我们还需要注意一点,apache默认运行是system权限,这很恐怖,这让人感觉很不爽.那我们就给apache降降权限吧。
  net user apache fuckmicrosoft /add
  net localgroup users apache /del
  ok.我们建立了一个不属于任何组的用户apche。
  我们打开计算机管理器,选服务,点apache服务的属性,我们选择log on,选择this account,我们填入上面所建立的账户和密码,重启apache服务,ok,apache运行在低权限下了。
  实际上我们还可以通过设置各个文件夹的权限,来让apache用户只能执行我们想让它能干的事情,给每一个目录建立一个单独能读写的用户。这也是当前很多虚拟主机提供商的流行配置方法哦,不过这种方法用于防止这里就显的有点大材小用了。

程序员的技术烦恼

没有评论

2010 年 12 月 10 日 at 下午 10:23分类:乱七八糟

今天我们来看看程序员的烦恼。每个程序员都有自己烦心事,不论这事指的是范围蠕变(scope creep),还是指匈牙利变量命名 (Hungarian notation),我们都明白,这是我们有我们行业里的特定的烦恼。 下面要说的就是让程序员们烦恼的十件事情。

10. 注释 — 只解释了“how”却没有解释“why”

入门级的编程课程通常会教育学生们写代码前先写注释、而且要尽量多注释。 这种教育的出发点是“多注释肯定比少注释好、少注释肯定比没注释好”。可不幸的是,很多的程序员把这当成了一种任务,对每一行代码都注释一下。


r = n / 2; 

// 让 r 等于 n 除以 2 

 // 当 r - (n/r) 大于 t 时进行循环 

while ( abs( r - (n/r) ) &gt; t ) { 

 r = 0.5 * ( r + (n/r) ); 

// 设置 r 等于 r + (n/r) 的一半 

} 

经过这样的注释,你否明白了这段代码是干什么的?的确,我也没明白。 问题就在于,虽然有大量的注释,可它们只是描述了代码是干什么了,却没有说明代码为什么要这样写。

现在,请看一下我们采用另外一种方式对同一段代码进行的注释:


// 使用牛顿-Raphson算法求n的平方根近似值  

r = n / 2; 

while ( abs( r - (n/r) ) &gt; t ) { 

r = 0.5 * ( r + (n/r) ); 

} 

这就好多了!也许我们还是不能完全明白这段代码的作用,但至少是有了一点方向了。

注释是用来帮助读者理解代码的,不是用来解释语法的。 我可以大胆的认为,读者对for循环的工作原理是了解的;所以没必要写这样的注释:“// 对客户列表进行for循环操作”。读者不明白的是你的代码是做什么用的,你为什么要采用这种方式实现它。

9. 干扰

很少有程序员能在眨眼之间从一种活动中转换到编程的状态中。通常情况下,我们更类似于需要慢慢启动的火车,而不是能突然加速的 法拉利; 我们需要一定的时间才能进入工作状态,一旦我们进入稳定有效的工作状态,我们的工作效果和产出会很丰硕。 不幸的是,当思路不断的被客户、经理、以及你的同事打断时,你的大脑很难进入编程的状态。

当我们干一件事情时,有太多的琐事需要我们放在心里,我们需要先放下这个事情,处理那个人事情,回头又干这个事情,还不能有差错。这些干扰会中 断我们的思路,而重新整理清楚思路又要你花费大量的时间,这是让人懊恼的、没有比这更让人泄气、让人有挫折感的过程了。

8. 范围蠕变(Scope creep)

范围蠕变(Scope creep) (也称作焦点蠕变(focus creep), 需求蠕变(requirement creep), 功能蠕变(feature creep),以及其它一些乱七八糟的演变词语),指在项目管理里项目的需求变更失控。 当一个项目的范围没有明确的定义清楚、没有文档化、不受控时就会出现这种现象。 这通常被认为是一种有负面影响的事情,应该尽力避免。

范围蠕变通常会把一个简单的需求变成一个复杂惊人的需要大量时间的巨无霸。 那些负责需求调研的家伙们只需要敲几下无辜的键盘就能把事情变成这样:

◆版本 1: 显示这个地区的地图

◆版本 2: 显示这个地区的地图,要三维立体的

◆版本 3: 显示这个地区的地图,要三维立体的,而且能够使用它作为飞行导航图

一个本来30分钟能完成的任务变成了一项要几百人/天才能完成的超级复杂的系统。更糟糕的是,大多数情况下,需求变更是发生在开发阶段 的,这样一来你需要重写代码,重新回归,有时要把前几天才开发的代码删除。

7. 管理者 — 完全不懂编程

管理工作不是一种简单的工作。人是一种让人很讨厌的动物; 我们善变、喜怒无常,我们都自以为天下第一。想让这样的一群人都感到满意和团结,你需要付出像山一样大的努力。 然而,这并不意味着管理者就可以在对下属的工作毫不理解的情况下进行管理。 当管理者对我们的工作没有一点知识概念时,后果只会是需求频繁变动,不现实的工期,普遍的挫折感(管理者和开发人员)。程序员们对此的抱怨相当普遍,这也是产生争执不合的根源。

6. 写文档

在说这个条目之前我先承认,我们确实有很多的文档生成工具,但据我的经验,这些工具都是只适合生成API文档,以供其他程序员参考。如果你开发 的软件是平时人们每天都要用的,你必须要写一些外行人(例如你的实施,客服等)都能理解的文档手册。

我们可以很容易的看出,有些事情程序员们极不愿意去做。 你可以简单的回顾一下所有的开源项目。 人们百折不挠的对这些项目的一个索求是什么:文档。
我敢打保票的说,不管在哪里,至少会有一半的程序员当要求写文档时会说:“不能让其他人去写吗?“。

5. 程序 — 缺少文档

我可从来没说过我们程序员是说一套做一套的人。 程序员们经常会在他们的项目里用到第三方的类库和应用。 于是,我们需要文档。 很不幸呀,就像我在第6条里说的那样,程序员们痛恨写文档。这戏剧性的事情发生在我们自己身上。

当你需要使用一个第三方类库时发现,至少有一半的API无从知道是干什么好用的,没有任何事情比这个更打击人的了。 函数 poorlyNamedFunctionA() 和函数 poorlyButSimilarlyNamedFunctionB() 有什么区别? 在我使用 PropertyX 属性前是否需要测试一下它是不是 null 值?我估计只有通过自己的测试和报错才能弄清楚!。

4. 硬件

任何一个曾经被叫去调试一个数据库服务器上奇怪的宕机现象,或是被叫去解决RAID驱动器不能正确的工作的问题的程序员,当发现是硬件问题时, 都会痛苦不已。 人们有一种普遍的误解,认为程序员就是搞电脑的,他们肯定知道如何修理电脑。 不可否认,有些程序员确实是个全才,但我估计,绝大部分程序员都不知道,或者根本不关心当程序被编译成机器码后如何工作的。我们只关心做出来的东西是否符 合需求文档,这样我们才能集中精力去解决这上层的任务。

3. 含糊不清

“网站宕机了”. “XX功能工作不正常”。 处理含糊不清的任务是种痛苦。 每次当非程序员被要求重现他们所遇到的问题时表现出的愤怒都让我吃惊不已。 他们似乎不太明白,仅仅一句”它宕机了,修复它!”是无法让我们开始工作的,我们需要更多的信息。

软件的运行是(大部分情况下)有迹可寻的。我们也乐见与此。 请迁就我们,帮我们指出是在哪个阶段,什么情况下出的问题,而不是简单的说一句”修复它“。

2. 其他程序员

程序员经常和其他程序员合不来。诧异吗,但这是真的。 这方面的事情我可以轻松的列出十大条,讲细点甚至可以单独写篇博客,所以这里我只列出几个常见的、让其他同事感到懊恼的程序员的特征:

◆脾气暴躁以至态度极不友好。

◆不能明白什么时候该去讨论系统的架构,什么时候是应该去动手去做。

◆无法进行有效的沟通,使用易于误解的专业术语。

◆自己的事情处理不好。

◆对要做的程序和项目缺乏兴趣。

那么,这最后的,但不是最糟糕的,序号为1的让程序员们烦恼的…

1. 自己写的代码 — 6个月以后的

Don’t sneeze, I think I see a bug.

回顾一下自己以前写的代码,是否也会愁眉苦脸?当时怎么会这么愚蠢!怎么能编写成这样的东西!烧掉!丢到火里!

现实是,软件技术界是一个不断变化的世界。 今天被看成是最好的方式,明天也许就会过时。 我们不可能写出完美的代码,因为判断我们的程序好坏的标准日新月异。 这令人很不爽,你的作品,今天看来是那么的完美,但也许不久之后就会变成被人嘲笑的对象了。 真是让人沮丧,因为不论我们如何努力的学习最新最棒的开发工具,设计,框架,以及开发方法,我们总是比最新的技术发展趋势慢了一拍。 对于我来说,这是做一个程序员最苦恼的事情了。我们不断的升级技术,是为了让软件更好,但却禁不住感到,我就像一个做沙毯(sand-painting)的和尚。

TP中连贯操作的实现原理

没有评论

2010 年 12 月 10 日 at 下午 4:40分类:PHP | WEB开发

之前在用TP框架的时候,感觉到TP的那个关于数据库操作的连贯操作很是喜欢,最近在学习CI的时候发现在CI中也用到了连贯操作,但是还不是很多,
但是在即将面世的CI2.0中好像基本上都是采用连贯操作实现的额!今天有时间看了TP的源代码,弄懂了所谓神奇的连贯操作,其实主要依靠一个在
面向对象中作为错误处理的函数__call();下面是自己写了一个实现连贯操作的代码实例,只是为了说明原理:

<?php
class Users
{
	protected $options = array();

	public function select()
	{
		$fields = $this->options['field'] ? $this->options['field'] : "*";
		$query = "select ".$fields." from ";
		$query .= $this->options['table']." where ";
		$query .= $this->options['where']." limit 0,";
		$query .= $this->options['limit']." order by ";
		$query .= $this->options['order'].";";;
		echo $query;
		//do something for select data from database;
	}
	public function __call($methods,$vars)
	{
		$this->options[$methods] = $vars[0];
		return $this;			//关键在这里是返回对象
	}
}
$p = new Users();
$p->table('users')->field('username,passwd')->where("uid = 12")->limit('10')->order("uid asc")->select();

OOP中__clone(),__toString(),__autoload(),__destruct的使用

没有评论

2010 年 12 月 10 日 at 下午 3:49分类:PHP | WEB开发

有的时候我们需要在一个项目里面,使用两个或多个一样的对象,如果你使用“new”关键字重新创建对象的话,再赋值上相同的属性,这样做比较烦琐而且也容易出错,所以要根据一个对象完全克隆出一个一模一样的对象,是非常有必要的,而且克隆以后,两个对象互不干扰。
在PHP5中我们使用”clone”这个关键字克隆对象;

<? 
    class Person 
    { 
        //下面是人的成员属性 
        var $name;  //人的名子 
        var $sex;    //人的性别 
        var $age;    //人的年龄 
        //定义一个构造方法参数为属性姓名$name、性别$sex和年龄$age进行赋值 
        function __construct($name="", $sex="", $age="") 
        { 
            $this->name=$name; 
            $this->sex=$sex; 
            $this->age=$age; 
        } 
        //这个人可以说话的方法, 说出自己的属性 
        function say()   
        { 
            echo "我的名子叫:".$this->name." 性别:".$this->sex." 我的年龄是:".$this->age."<br>"; 
        } 
    } 
    $p1=new Person("张三", "男", 20); 
    //使用“clone”克隆新对象p2,和p1对象具有相同的属性和方法。 
    $p2=clone $p1; 
    $p2->say(); 
?> 

PHP5定义了一个特殊的方法名“__clone()”方法,是在对象克隆时自动调用的方法,用“__clone()”方法将建立一个与原对象拥有相同属性和方法的对象,如果想在克隆后改变原对象的内容,需要在__clone()中重写原本的属性和方法, ”__clone()”方法可以没有参数,它自动包含$this和$that两个指针,$this指向复本,而$that指向原本;

<?
class Person 
{ 
    //下面是人的成员属性 
    var $name;  //人的名子 
    var $sex;    //人的性别 
    var $age;    //人的年龄 
    //定义一个构造方法参数为属性姓名$name、性别$sex和年龄$age进行赋值 
    function __construct($name="", $sex="", $age="") 
    { 
        $this->name=$name; 
        $this->sex=$sex; 
        $this->age=$age; 
    } 
    //这个人可以说话的方法, 说出自己的属性 
    function say()   
    { 
        echo "我的名子叫:".$this->name." 性别:".$this->sex." 我的年龄是:".$this->age."<br>"; 
    } 
    //对象克隆时自动调用的方法, 如果想在克隆后改变原对象的内容,需要在__clone()中重写原本的属性和方法 
    function __clone() 
    { 
        //$this指的复本p2, 而$that是指向原本p1,这样就在本方法里,改变了复本的属性。  
        $this->name="我是假的$that->name"; 
        $this->age=30; 
    } 
} 
$p1=new Person("张三", "男", 20); 
$p2=clone $p1; 
$p1->say(); 
$p2->say(); 
?> 

上例输出:

我的名子叫:张三 性别:男 我的年龄是:20
我的名子叫:我是假的张三 性别:男 我的年龄是:30  

__toString()”方法也是一样自动被调用的,是在直接输出对象引用时自动调用的, 前面我们讲过对象引用是一个指针,比如说:“$p=new Person()“中,$p就是一个引用,我们不能使用echo 直接输出$p, 这样会输出”Catchable fatal error: Object of class Person could not be converted to string“这样的错误,如果你在类里面定义了“__toString()”方法,在直接输出对象引用的时候,就不会产生错误,而是自动调用了”__toString()”方法, 输出“__toString()”方法中返回的字符,所以“__toString()”方法一定要有个返回值(return 语句).

<?php
    // Declare a simple class
    class TestClass
    {
        public $foo;
        public function __construct($foo) {
        $this->foo = $foo;
        }
       //定义一个__toString方法,返加一个成员属性$foo
        public function __toString() {
        return $this->foo;
        }
    }
    $class = new TestClass('Hello'); 
    //直接输出对象
    echo $class;
?> 

上例输出:Hello
__autoload的用法相当的简单,我们一般在实例化一个类时,都要include一个类文件,那么要是我们在所有实例化之前先定义这么一个方法的的话,
那么就不用每次都include了 代码如下:

function __autoload($className){
        include($className.".php");            //先后加载MyPc.php和FBB.php这两个类文件
}
$pp = new MyPc();
$bb = new FBB();

__destruct()函数是在该类执行所有的操作动作之后被自动执行的,用来销毁该类的对象的!
只要我们在类中定义了,就不用管他了,在所有结束后,该类会自动执行!

OOP中__set() __get() __isset() __unset()四个方法的应用

没有评论

2010 年 12 月 10 日 at 下午 3:27分类:PHP | WEB开发

一般来说,总是把类的属性定义为private,这更符合现实的逻辑。但是, 对属性的读取和赋值操作是非常频繁的,因此在PHP5中,预定义了两个函数”__get()”和”__set()”来获取和赋值其属性,以及检查属性的”__isset()”和删除属性的方法”__unset()”。
”__set()”和“__get()”这两个方法不是默认存在的, 而是我们手工添加到类里面去的,像构造方法(__construct())一样, 类里面添加了才会存在,可以按下面的方式来添加这两个方法,当然也可以按个人的风格来添加:

//__get()方法用来获取私有属性
function __get($property_name)
{
    if(isset($this->$property_name)) {
        return($this->$property_name);
    }else {
        return(NULL);
    }
} 
//__set()方法用来设置私有属性
function __set($property_name, $value)
{
    $this->$property_name = $value;
}

__get()方法:这个方法用来获取私有成员属性值的,有一个参数, 参数传入你要获取的成员属性的名称,返回获取的属性值, 这个方法不用我们手工的去调用, 是在直接获取私有属性的时候自动调用的。因为私有属性已经被封装上了,是不能直接获取值的(比如:”echo $p1->name” 这样直接获取是错误的),但是如果你在类里面加上了这个方法,在使用”echo $p1->name” 这样的语句直接获取值的时候就会自动调用__get($property_name)方法,将属性name传给参数$property_name,通过这个方法的内部执行,返回我们传入的私有属性的值。
__set()方法:这个方法用来为私有成员属性设置值的,有两个参数,第一个参数为你要为设置值的属性名,第二个参数是要给属性设置的值,没有返回值。这个方法同样不用我们手工去调用,是在直接设置私有属性值的时候自动调用的,同样属性私有的已经被封装上了, 如果没有__set()这个方法,是不允许的, 比如:”$this->name=’zhangsan’ , 这样会出错,但是如果你在类里面加上了__set($property_name, $value)这个方法,在直接给私有属性赋值的时候,就会自动调用它,把属性比如name传给$property_name, 把要赋的值”zhangsan”传给$value,通过这个方法的执行,达到赋值的目的, 为了不传入非法的值, 还可以在这个方法给做一下判断。代码如下:

<?php
    class Person
    {
    //下面是人的成员属性, 都是封装的私有成员
    private $name; //人的名子
    private $sex; //人的性别
    private $age; //人的年龄

    //__get()方法用来获取私有属性
    function __get($property_name)
    {
        echo "在直接获取私有属性值的时候,自动调用了这个__get()方法";
        if(isset($this->$property_name)) {
            return($this->$property_name);
        }else {
            return(NULL);
        }
    } 
    //__set()方法用来设置私有属性
    function __set($property_name, $value)
    {
        echo "在直接设置私有属性值的时候,自动调用了这个__set()方法为私有属性赋值";
        $this->$property_name = $value;
    } 
    }
    $p1=new Person();
    //直接为私有属性赋值的操作, 会自动调用__set()方法进行赋值
    $p1->name="张三";
    $p1->sex="男";
    $p1->age=20;
    //直接获取私有属性的值, 会自动调用__get()方法,返回成员属性的值
    echo "姓名:".$p1->name;
    echo "性别:".$p1->sex;
    echo "年龄:".$p1->age;
?>

程序执行结果:(省略)
在直接设置私有属性值的时候, 自动调用了这个__set()方法为私有属性赋值,在直接获取私有属性值的时候,自动调用了这个__get()方法姓名:张三在直接获取私有属性值的时候,自动调用了这个__get()方法获取性别:男在直接获取私有属性值的时候,自动调用了这个__get()方法年龄:20
以上代码如果不加上__get()和__set()方法,程序就会出错,因为不能在类的外部操作私有成员,而上面的代码是通过自动调用__get()和__set()方法来帮助我们直接存取封装的私有成员的。
__isset() 方法:检查属性是否存在的方法。
__unset()方法:删除指定属性的方法。
最后补充说明:
  __set(), __get() 是专门为类的私有属性设立的,对于类的公开(public)属性,外面是可以直接访问与设置的(如:$p1->name)!!!

此外还要补充一下:在OOP中我们还用到的函数有
__call()函数:这个函数用来处理错误调用
程序开发中,如果在使用对象调用对象内部方法时候,调用的这个方法不存在那么程序就会出错,然后程序退出不能继续执行。那么可不可以在程序调用对象内部不存在的方法时,提示我们调用的方法及使用的参数不存在,但程序还可以继续执行,这个时候我们就要使用在调用不存在的方法时自动调用的方法”__call()”.

<?php 
    //这是一个测试的类,里面没有属性和方法 
    class Test 
    { 
    } 
    //产生一个Test类的对象 
    $test=new Test(); 
    //调用对象里不存在的方法
    $test->demo("one", "two", "three"); 
    //程序不会执行到这里 
    echo "this is a test<br>"; 
?> 

上例出现如下错误,程序通出不能继续执行;

Fatal error: Call to undefined method Test::demo() 

下面我们加上“__call()”方法,这个方法有2个参数,第一个参数为调用不存在的方法过程中,自动调用__call()方法时,把这个不存在的方法的方法名传给第一个参数,第二个参数则是把这个方法的多个参数以数组的形式传进来。

<?php 
    //这是一个测试的类,里面没有属性和方法 
    class Test 
    { 
        //调用不存的方法时自动调用的方法,第一个参数为方法名,第二个参数是数组参数 
        function __call($function_name, $args) 
        {  
            print "你所调用的函数:$function_name(参数:";  
            print_r($args);  
            echo "不存在!<br>\n"; 
        } 
    } 
    //产生一个Test类的对象 
    $test=new Test(); 
    //调用对象里不存在的方法 
    $test->demo("one", "two", "three"); 
    //程序不会退出可以执行到这里 
    echo "this is a test<br>"; 
?> 

上例输出结果为:
你所调用的函数: demo(参数:Array ( [0] => one [1] => two [2] => three ) )不存在!
上面讲的就是一个简单的测试例子!

Apache和Nginx下面如何做301重定向。

没有评论

2010 年 12 月 10 日 at 上午 11:47分类:Nginx | PHP | WEB开发

在Apache和Nginx下面如何做301重定向。

1,Apache下面可以用.htaccess设置301重定向

例如把 www.a.com 和 a.com 都301重定向到 www.b.com

RewriteEngine on
RewriteBase /
RewriteCond %{HTTP_HOST} ^www.a.com [NC]
RewriteRule ^(.*)$ http://www.b.com/$1 [R=301,L]
RewriteCond %{HTTP_HOST} ^a.com [NC]
RewriteRule ^(.*)$ http://www.b.com/$1 [R=301,L]

2,Nginx中进行301重定向也是非常容易的

比如域名 a.com 要重定向到 www.a.com

server {
server_name a.com;
rewrite ^(.*) http://www.a.com$1 permanent;
}

或者不是访问www.a.com的域名都转向到www.a.com

可以用:

server {
server_name a.com c.com;
if ($host != ‘www.a.com’ ) {
rewrite ^/(.*)$ http://www.a.com/$1 permanent;
}