php面向对象 static、 const、final关键字的使用

 更新时间:2016年11月25日 15:06  点击:1841
Static关键字是在类中描述成员属性和成员方法是静态的,final 关键字的应用 这个关键字只能用来定义类和定义方法, 不能使用final这个关键字来定义成员属性,因为final是常量的意思,我们在PHP里定义常量使用的是define()函数,所以不能使用final来定义成员属性

Static关键字是在类中描述成员属性和成员方法是静态的;静态的成员好处在那里呢?前面我们声明了“Person”的人类,在”Person”这个类里如果我们加上一个“人所属国家”的属性,这样用”Person”这个类实例化出几百个或者更多个实例对象,每个对象里面就都有“所属国家”的属性了,如果开发的项目就是为中国人而开发的,那么每个对象里面就都有一个国家的属性是“中国“其它的属性是不同的,如果我们把“国家”的属性做成静态的成员,这样国家的属性在内存中就只有一个,而让这几百个或更多的对象共用这一个属性,static成员能够限制外部的访问,因为static的成员是属于类的,是不属于任何对象实例,是在类第一次被加载的时候分配的空间,其他类是无法访问的,只对类的实例共享,能一定程度对类该成员形成保护。

从内存的角度我们来分析一下,内存从逻辑上被分为四段,其中对象是放在“堆内存”里面,对象的引用被放到了“栈内存“里,而静态成员则放到了“初始化静态段”,在类第一次被加载的时候放入的,可以让堆内存里面的每个对象所共享,如下图;

 

类的静态变量,非常类似全局变量,能够被所有类的实例共享,类的静态方法也是一样的,类似于全局函数。

 代码如下 复制代码

class Person
    {
    //下面是人的静态成员属性
    public static $myCountry="中国";
    // var $name;  //人的名子

    //这是人的静态成员方法
    public static function say()
    {
        echo "我是中国人
";
    }
}

//输出静态属性
echo Person::$myCountry;

//访问静态方法
Person::say();

//重新给静态属性赋值
Person::$myCountry="美国";
echo Person::$myCountry;

"PHP5面向对象详解(高洛峰)" 这教程写得很不错,对于PHP菜鸟还是老鸟来说,貌似都是值得一看的,附上word文档php5面向对象详解。

static 和 const 关键字的使用
Static关键字是在类中描述成员属性和成员方法是静态的;静态的成员好处在那里呢?前面我们声明了“Person”的人类,在”Person”这个类里如果我们加上一个“人所属国家”的属性,这样用”Person”这个类实例化出几百个或者更多个实例对象,每个对象里面就都有“所属国家”的属性了,如果开发的项目就是为中国人而开发的,那么每个对象里面就都有一个国家的属性是“中国“其它的属性是不同的,如果我们把“国家”的属性做成静态的成员,这样国家的属性在内存中就只有一个,而让这几百个或更多的对象共用这一个属性,static成员能够限制外部的访问,因为static的成员是属于类的,是不属于任何对象实例,是在类第一次被加载的时候分配的空间,其他类是无法访问的,只对类的实例共享,能一定程度对类该成员形成保护。

从内存的角度我们来分析一下,内存从逻辑上被分为四段,其中对象是放在“堆内存”里面,对象的引用被放到了“栈内存“里,而静态成员则放到了“初始化静态段”,在类第一次被加载的时候放入的,可以让堆内存里面的每个对象所共享,如下图;

 

类的静态变量,非常类似全局变量,能够被所有类的实例共享,类的静态方法也是一样的,类似于全局函数。

 

 代码如下 复制代码

class Person  
    {  
    //下面是人的静态成员属性  
    public static $myCountry="中国";  
    // var $name;  //人的名子  
 
    //这是人的静态成员方法  
    public static function say()  
    {  
        echo "我是中国人 
";  
    }  
}  
 
//输出静态属性  
echo Person::$myCountry;  
 
//访问静态方法  
Person::say();  
 
//重新给静态属性赋值  
Person::$myCountry="美国";  
echo Person::$myCountry; 

class Person
    {
    //下面是人的静态成员属性
    public static $myCountry="中国";
    // var $name;  //人的名子

    //这是人的静态成员方法
    public static function say()
    {
        echo "我是中国人
";
    }
}

//输出静态属性
echo Person::$myCountry;

//访问静态方法
Person::say();

//重新给静态属性赋值
Person::$myCountry="美国";

echo Person::$myCountry;因为静态成员是在类第一次加载的时候就创建的,所以在类的外部不需要对象而使用类名就可以访问的到静态的成员;上面说过,静态成员被这个类的每个实例对象所共享,那么我们使用对象可不可以访问类中的静态成员呢?从上图中我们可以看到,静态的成员不是在每个对象内部存在的,但是每个对象都可以共享,所以我们如果使用对象访问成员的话就会出现没有这个属性定义,使用对象访问不到静态成员的,在其它的面向对象的语言中,比如Java是可以使用对象的方式访问静态成员的,如果PHP中可以使用对象访问静态成员的话,我们也尽量不要去使用,因为静态的成员我们在做项目的时候目的就是使用类名去访问。

类里面的静态方法只能访问类的静态的属性,在类里面的静态方法是不能访问类的非静态成员的,原因很简单,我们要想在本类的方法中访问本类的其它成员,我们需要使用$this这个引用,而$this这个引用指针是代表调用此方法的对象,我们说了静态的方法是不用对象调用的,而是使用类名来访问,所以根本就没有对象存在,也就没有$this这个引用了,没有了$this这个引用就不能访问类里面的非静态成员,又因为类里面的静态成员是可以不用对象来访问的,所以类里面的静态方法只能访问类的静态的属性,即然$this不存在,在静态方法中访其它静态成员我们使用的是一个特殊的类”self”; self和$this相似,只不过self是代表这个静态方法所在的类。所以在静态方法里,可以使用这个方法所在的类的“类名“,也可以使用“self“来访问其它静态成员,如果没有特殊情况的话,我们通常使用后者,即 "self::成员属性"的方式。

 代码如下 复制代码

class Person
{
    //下面是人的静态成员属性
    public static $myCountry="中国";

    //这是人的静态成员方法, 通过self访问其它静态成员
    public static function say()
    {
        echo "我是".self::$myCountry."";
    }
}
//访问静态方法
Person::say();

在非静态方法里可不可以访问静态成员呢,当然也是可以的了,但是也不能使用”$this”引用也要使用类名或是”self::成员属性的形式”。 const是一个定义常量的关键字,在PHP中定义常量使用的是”define()”这个函数,但是在类里面定义常量使用的是”const”这个关键字,类似于C中的#define如果在程序中改变了它的值,那么会出现错误,用”const”修饰的成员属性的访问方式和”static”修饰的成员访问的方式差不多,也是使用”类名”,在方法里面使用”self”关键字。但是不用使用”$”符号,也不能使用对象来访问。

 代码如下 复制代码

class MyClass
{
    //定义一个常量constant
    const constant = 'constant value';

    function showConstant() {
        echo  self::constant . "n";  //使用self访问,不要加”$”
    }
}

echo MyClass::constant . "n";  //使用类名来访问,也不加”$”

$class = new MyClass();
$class->showConstant();
//echo $class::constant;  是不允许的


final 关键字的应用
这个关键字只能用来定义类和定义方法, 不能使用final这个关键字来定义成员属性,因为final是常量的意思,我们在PHP里定义常量使用的是define()函数,所以不能使用final来定义成员属性。

使用final关键标记的类不能被继承;

 代码如下 复制代码

final class Person
{
}

class Student extends Person
{

//会出现下面错误:
Fatal error: Class Student may not inherit from final class (Person)

使用final关键标记的方法不能被子类覆盖,是最终版本;

class Person
{
 final function say() {}
}

class Student extends Person
{
 function say() {}
}

//会出现下面错误:
Fatal error: Cannot override final method Person::say()

PHP中的方法是不能重载的,所谓的方法重载就是定义相同的方法名,通过“参数的个数“不同或“参数的类型“不同,来访问我们的相同方法名的不同方法,继承是php5面象对象程序设计的重要特性之一,它是指建立一个新的派生类,从一个或多个先前定义的类中继承数据和函数,而且可以重新定义或加进新数据和函数,从而建立了类的层次或等级。

重载新的方法


在学习PHP 这种语言中你会发现,PHP中的方法是不能重载的,所谓的方法重载就是定义相同的方法名,通过“参数的个数“不同或“参数的类型“不同,来访问我们的相同方法名的不同方法。但是因为PHP是弱类型的语言,所以在方法的参数中本身就可以接收不同类型的数据,又因为PHP的方法可以接收不定个数的参数,所以通过传递不同个数的参数调用不相同方法名的不同方法也是不成立的。所以在PHP里面没有方法重载。不能重载也就是在你的项目中不能定义相同方法名的方法。另外,因为PHP没有名子空间的概念,在同一个页面和被包含的页面中不能定义相同名称的方法,也不能定义和PHP给我提供的方法的方法重名,当然在同一个类中也不能定义相同名称的方法。

我们这里所指的重载新的方法所指的是什么呢?其实我们所说的重载新的方法就是子类覆盖父类的已有的方法,那为什么要这么做呢?父类的方法不是可以继承过来直接用吗?但有一些情况是我们必须要覆盖的,比如说我们前面提到过的例子里面,“Person”这个人类里面有一个“说话”的方法,所有继承“Person”类的子类都是可以“说话”的,我们“Student”类就是“Person”类的子类,所以“Student”的实例就可以“说话“了,但是人类里面“说话”的方法里面说出的是“Person”类里面的属性,而“Student”类对“Person”类进行了扩展,又扩展出了几个新的属性,如果使用继承过来的“say()”说话方法的话,只能说出从“Person”类继承过来的那些属性,那么新扩展的那些属性使用这个继承过来的“say()”的方法就说不出来了,那有的人就问了,我在“Student”这个子类中再定义一个新的方法用于说话,说出子类里面所有的属性不就行了吗?一定不要这么做,从抽象的角度来讲,一个“学生”不能有两种“说话”的方法,就算你定义了两个不同的说话的方法,可以实现你想要的功能,被继承过来的那个“说话“方法可能没有机会用到了,而且是继承过来的你也删不掉。这个时候我们就要用到覆盖了。

虽然说在PHP里面不能定义同名的方法,但是在父子关系的两个类中,我们可以在子类中定义和父类同名的方法,这样就把父类中继承过来的方法覆盖掉了。

 代码如下 复制代码

//定义一个“人”类做为父类
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>";
    }
}

class Student extends Person
{
    var $school;  //学生所在学校的属性

    //这个学生学习的方法
    function study()
    {
        echo "我的名子叫:".$this->name." 我正在”.$this->school.” 学习<br>";
    }

    //这个学性可以说话的方法, 说出自己所有的属性,覆盖了父类的同名方法
    function say()
    {
        echo "我的名子叫:".$this->name." 性别:".$this->sex." 我的年龄是:".$this->age."我在".$this->school."上学.<br>";
    }
}

上面的例子, 我们就在“Student”子类里覆盖了继承父类里面的”say()”的方法,通过覆盖我们就实现了对“方法”扩展。

但是,像这样做虽然解决了我们上面说的问题,但是在实际开发中,一个方法不可能就一条代码或是几条代码,比如说“Person”类里面的“say()”方法有里面有100条代码,如果我们想对这个方法覆盖保留原有的功能外加上一点点功能,就要把原有的100条代码重写一次, 再加上扩展的几条代码,这还算是好的,而有的情况,父类中的方法是看不见原代码的,这个时候你怎么去重写原有的代码呢?我们也有解决的办法,就是在子类这个方法中可以调用到父类中被覆盖的方法, 也就是把被覆盖的方法原有的功能拿过来再加上自己的一点功能,可以通过两种方法实现在子类的方法中调用父类被覆盖的方法:

一种是使用父类的 "类名::" 来调用父类中被覆盖的方法;
一种是使用 "parent::" 的方试来调用父类中被覆盖的方法;

 代码如下 复制代码

class Student extends Person
{
    var $school;  //学生所在学校的属性

    //这个学生学习的方法
    function study()
    {
        echo "我的名子叫:".$this->name." 我正在”.$this->school.” 学习<br>";
    }

    //这个学性可以说话的方法, 说出自己所有的属性,覆盖了父类的同名方法
    function say()
    {
        //使用父类的“类名::“来调用父类中被覆盖的方法;
        //Person::say();

        //或者使用“parent::”的方试来调用父类中被覆盖的方法;
        parent::say();

        //加上一点自己的功能
        echo “我的年龄是:".$this->age."我在".$this->school."上学.<br>";
    }
}

现在用两种方式都可以访问到父类中被覆盖的方法,我们选那种方式最好呢?用户可能会发现自己写的代码访问了父类的变量和函数。如果子类非常精炼或者父类非常专业化的时候尤其是这样。 不要用代码中父类文字上的名字,应该用特殊的名字 parent,它指的就是子类在 extends 声明中所指的父类的名字。这样做可以避免在多个地方使用父类的名字。如果继承树在实现的过程中要修改,只要简单地修改类中 extends 声明的部分。

同样,构造方法在子类中如果没有声明的话,也可以使用父类中的构造方法,如果子类中重新定义了一个构造方法也会覆盖掉父类中的构造方法,如果想使用新的构造方法为所有属性赋值也可以用同样的方式。

 代码如下 复制代码

class Student extends Person
{
    var $school;  //学生所在学校的属性

    function __construct($name, $sex, $age, $school)
    {
        //使用父类中的方法为原有的属性赋值
        parent::__construct($name, $sex, $age);
        $this->school=$school;
    }

    //这个学生学习的方法
    function study()
    {
        echo "我的名子叫:".$this->name." 我正在”.$this->school.” 学习<br>";
    }

    //这个人可以说话的方法, 说出自己的属性
    function say()
    {
        parent::say();
        //加上一点自己的功能
        echo “我的年龄是:".$this->age."我在".$this->school."上学.<br>";
    }
}

类的继承


继承作为面向对象的三个重要特性的一个方面,在面向对象的领域有着及其重要的作用,好像没听说哪个面向对象的语言不支持继承。继承是php5面象对象程序设计的重要特性之一,它是指建立一个新的派生类,从一个或多个先前定义的类中继承数据和函数,而且可以重新定义或加进新数据和函数,从而建立了类的层次或等级。说的简单点就是,继承性是子类自动共享父类数据结构和方法的机制,这是类之间的一种关系。在定义和实现一个类的时候,可以在一个已经存在的类的基础之上来进行,把这个已经存在的类所定义的内容作为自己的内容,并加入若干新的内容。比如你现在已经有一个“人”这个类了,这个类里面有两个成员属性“姓名和年龄”以及还有两个成员方法“说话的方法和走路的方法“,如果现在程序需要一个学生的类,因为学生的也是人,所以学生也有成员属性“姓名和年龄”以及成员方法“说话的方法和走路的方法“,这个时候你就可以让学生类来继承人这个类,继承之后,学生类就会把人类里面的所有的属性都继承过来,就不用你再去重新声明一遍这些成员属性和方法了,因为学生类里面还有所在学校的属性和学习的方法,所以在你做的学生类里面有继承自人类里面的属性和方法之外在加上学生特有的”所在学校属性“和”学习的方法“,这样一个学生类就声明完成了,继函我们也可以叫作“扩展”,从上面我们就可以看出,学生类对人类进行了扩展,在人类里原有两个属性和两个方法的基础上加上一个属性和一个方法扩展出来一个新的学生类。

通过继承机制,可以利用已有的数据类型来定义新的数据类型。所定义的新的数据类型不仅拥有新定义的成员,而且还同时拥有旧的成员。我们称已存在的用来派生新类的类为基类,又称为父类以及超类。由已存在的类派生出的新类称为派生类,又称为子类。

在软件开发中,类的继承性使所建立的软件具有开放性、可扩充性,这是信息组织与分类的行之有效的方法,它简化了对象、类的创建工作量,增加了代码的可重性。采用继承性,提供了类的规范的等级结构。通过类的继承关系,使公共的特性能够共享,提高了软件的重用性。

在C++语言中,一个派生类可以从一个基类派生,也可以从多个基类派生。从一个基类派生的继承称为单继承;从多个基类派生的继承称为多继承。

 

但是在PHP和Java语言里面没有多继承,只有单继承,也就是说,一个类只能直接从一个类中继承数据, 这就是我们所说的单继承。

 

例如:下面是“人”类的抽象

 代码如下 复制代码

//定义一个“人”类做为父类
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>";
    }
}

下面我们做一个”学生类”,如果不是用继承如下:

 代码如下 复制代码

//定义一个“学生类”类做为父类
class Student
{
    //下面是人的成员属性
    var $name;  //人的名子
    var $sex;    //人的性别
    var $age;    //人的年龄
    var $school;  //学生所在学校的属性

    //定义一个构造方法参数为属性姓名$name、性别$sex和年龄$age进行赋值
    function __construct($name="", $sex="", $age="", $school="")
    {
        $this->name=$name;
        $this->sex=$sex;
        $this->age=$age;
        $this->school=$school;
    }

    //这个人可以说话的方法, 说出自己的属性
    function say()
    {
        echo "我的名子叫:".$this->name." 性别:".$this->sex." 我的年龄是:".$this->age."<br>";
    }  

    //这个学生学习的方法
    function study()
    {
        echo "我的名子叫:".$this->name." 我正在".$this->school." 学习<br>";
    }
}


//定义一个子类“学生类“使用”extends”关键字来继承”人”类  
class Student extends Person  
{  
    //学生所在学校的属性  
    var $school;    
 
    //这个学生学习的方法  
    function study()  
    {  
        echo "我的名子叫:".$this->name." 我正在".$this->school." 学习<br>";  
    }  

通过上面“Student“类的定义,Student类通过使用”extends”这个关键字把Person类里的所有成员属性和成员方法都继承过来了,并扩展了一个所在学校成员属性”school”,和一个学习方法“study()”.现在子类”Student”里面和使用这个类实例出来的对象都具有如下的属性和方法:

学生类”Student”里面的成员属性有:
姓名:name;
年龄:age;
性别:sex;
学校:school;
学生类”Student”里面的成员方法有:
说话方法:say();
学习方法:study();

本文章介绍了关于php中面向对象的抽象方法和抽象类 __call 克隆对象用法,有需要发解的朋友可以学学。

抽象方法和抽象类


在OOP语言中,一个类可以有一个或多个子类,而每个类都有至少一个公有方法做为外部代码访问其的接口。而抽象方法就是为了方便继承而引入的,我们先来看一下抽象类和抽象方法的定义再说明它的用途。

什么是抽象方法?我们在类里面定义的没有方法体的方法就是抽象方法,所谓的没有方法体指的是,在方法声明的时候没有大括号以及其中的内容,而是直接在声明时在方法名后加上分号结束,另外在声明抽象方法时还要加一个关键字“abstract”来修饰;例如:

 代码如下 复制代码

abstract function fun1();
abstract function fun2();

上例是就是“abstract”修饰的没有方法体的抽象方法“fun1()”和“fun2()”,不要忘记抽象方法后面还要有一个分号;那么什么是抽象类呢?只要一个类里面有一个方法是抽象方法,那么这个类就要定义为抽象类,抽象类也要使用“abstract”关键字来修饰;在抽象类里面可以有不是抽象的方法和成员属性,但只要有一个方法是抽象的方法,这个类就必须声明为抽象类,使用”abstract”来修饰。例如:

 代码如下 复制代码

abstract class Demo
{
    var $test;

    abstract function fun1();
    abstract function fun2();
    function fun3()
    {
        ... ...
    }
}

上例中定义了一个抽象类“Demo”使用了”abstract”来修饰, 在这个类里面定义了一个成员属性“$test”,和两个抽象方法“fun1”和“fun2”还有一个非抽象的方法fun3();那么抽象类我们怎么使用呢?最重要的一点就是抽象类不能产生实例对象,所以也不能直接使用,前面我们多次提到过类不能直接使用,我们使用的是通过类实例化出来的对象,那么抽象类不能产生实例对象我们声明抽象类有什么用呢?我们是将抽象方法是做为子类重载的模板使用的,定义抽象类就相当于定义了一种规范,这种规范要求子类去遵守,子类继函抽象类之后,把抽象类里面的抽象方法按照子类的需要实现。子类必须把父类中的抽象方法全部都实现,否则子类中还存在抽象方法,那么子类还是抽象类,还是不能实例化对;为什么我们非要从抽象类中继承呢?因为有的时候我们要实现一些功能就必须从抽象类中继承,否则这些功能你就实现不了,如果继承了抽象类,就要实现类其中的抽象方法;

 代码如下 复制代码

abstract class Demo
    {
    var $test;

    abstract function fun1();
    abstract function fun2();
    function fun3()
    {
        ... ...
    }
}

//抽象类为能产生实例对象,所以这样做是错的,实例化对象交给子类

 代码如下 复制代码

$demo=new Demo();

class Test extends Demo
{
    function fun1()
    {
        ... ...
    }

    function fun2()
    {
        ... ...
    }
}

//子类可以实例化对象,因为实现了父类中所有抽象方法

 代码如下 复制代码
$test=new Test();

__call处理调用错误


在程序开发中,如果在使用对象调用对象内部方法时候,调用的这个方法不存在那么程序就会出错,然后程序退出不能继续执行。那么可不可以在程序调用对象内部不存在的方法时,提示我们调用的方法及使用的参数不存在,但程序还可以继续执行,这个时候我们就要使用在调用不存在的方法时自动调用的方法”__call()”。

 代码如下 复制代码

//这是一个测试的类,里面没有属性和方法
class Test
{
}

//产生一个Test类的对象
$test=new Test();

//调用对象里不存在的方法
$test->demo("one", "two", "three");

//程序不会执行到这里
echo "this is a test";

上例出现如下错误,程序通出不能继续执行;
Fatal error: Call to undefined method Test::demo()

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

 代码如下 复制代码

//这是一个测试的类,里面没有属性和方法
class Test
{
    //调用不存的方法时自动调用的方法,第一个参数为方法名,第二个参数是数组参数
    function __call($function_name, $args)
    {
        print "你所调用的函数:$function_name(参数:";
        print_r($args);
        print ")不存在!n";
    }
}

//产生一个Test类的对象
$test=new Test();

//调用对象里不存在的方法
$test->demo("one", "two", "three");

//程序不会退出可以执行到这里
echo "this is a test";

上例输出结果为:
你所调用的函数: demo(参数:Array ( [0] => one [1] => two [2] => three ) )不存在! this is a test.


克隆对象


有的时候我们需要在一个项目里面,使用两个或多个一样的对象,如果你使用“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."";
    }
}

$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."";
    }

    //对象克隆时自动调用的方法, 如果想在克隆后改变原对象的内容,需要在__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

PHP 5也支持接口的概念,并为之引入了interface和implements关键字。和Java一样,PHP 5使用接口也实现类似于“多重继承”的效果。

PHP5接口语法如下:
声明一个类的时候我们使用的关键字是”class”,而接口一种特殊的类,使用的关键字是“interface”;

类的定义:class 类名{ … },接口的声明:interface 接口名{ …}

具体PHP5抽象类的代码如下:

 代码如下 复制代码

abstract class AbstractClass {     

   abstract public function test();     

}     

    

class ImplementedClass extends AbstractClass {     

   public function test() {     

       echo "ImplementedClass::test() called. ";     

   }     

}     

    

$o = new ImplementedClass;     

$o->test();    


实例二

 代码如下 复制代码

interface displayable {     

  function display();     

}     

interface printable {     

  function doprint();     

}     

    

class foo implements displayable,printable {     

  function display() {     

    // code     

  }   function doprint() {     

    // code     

  }     

}    

 


抽象类和PHP5接口的引入使PHP成了一个完全面向对象的语
//定义一个接口使用interface关键字,“One”为接口名称

 代码如下 复制代码

interface One
{
    //定义一个常量
    const constant = 'constant value';

    //定义了一个抽象方法”fun1”
    public function fun1();

    //定义了抽象方法”fun2”
    public function fun2();
}

上例中定义了一个接口”one”,里面声明了两个抽象方法“fun1”和”fun2”,因为接口里面所有的方法都是抽象方法,所以在声明抽象方法的时候就不用像抽象类那样使用”abstract”这个关键字了,默认的已经加上这个关键字,另外在接口里边的”public”这个访问权限也可以去掉,因为默认就是public的,因为接口里所有成员都要是公有的,所在对于接口里面的成员我们就不能使用“private”的和”protected”的权限了,都要用public或是默认的。另外在接口里面我们也声明了一个常量“constant“, 因为在接口里面不能用变量成员,所以我们要使用const这个关键字声明。

因为接口是一种特殊的抽象类,里面所有的方法都是抽象方法,所以接口也不能产生实例对象; 它也做为一种规范,所有抽象方法需要子类去实现。

我们可以使用”extends”关键字让一个接口去继承另一个接口;

 代码如下 复制代码

//使用”extends”继承另外一个接口
interface Two extends One
{
    function fun3();
    function fun4();
}

而我们定义一接口的子类去实现接口中全部抽象方法使用的关键字是”implements”,而不是我们前面所说的”extends”;

//使用“implements”这个关键字去实现接口中的抽象方法

 代码如下 复制代码

class Three implements One
{
    function fun1()
    {
        … …
    }

    function fun2()
    {
        … …
    }
}

//实现了全部方法,我们去可以使用子类去实例化对象了

 代码如下 复制代码
$three=new Three();

我们也可以使用抽象类,去实现接口中的部分抽象方法,但要想实例化对象,这个抽象类还要有子类把它所有的抽象方法都实现才行;

在前面我们说过,PHP是单继承的,一个类只能有一父类,但是一个类可以实现多个接口,就相当于一个类要遵守多个规范,就像我们不仅要遵守国家的法律,如果是在学校的话,还要遵守学校的校规一样;

 代码如下 复制代码

//使用implements实现多个接口
class Four implemtns 接口一,接口二,……
{
    //必须把所有接口中的方法都要实现才可以实例化对象。
}

PHP中不仅一个类可以实现多个接口,也可以在继承一个类的同时实现多个接口,一定要先继承类再去实现接口;

 代码如下 复制代码

//使用extends继承一个类,使用implements实现多个接口
class Four extends 类名一 implemtns 接口一,接口二,……
{
    //所有接口中的方法都要实现才可以实例化对象
    ………
}

讲了这么多最后看看一个实例


VideoCard.php 接口文件(显卡的功能接口定义)

 代码如下 复制代码
<?php
interface VideoCardInter{
function Display();
function getName();
}
?>

Dmeng.php 实现接口(帝盟的厂家实现了这些接口,怎么实现的,主板厂家不用管)

 代码如下 复制代码
<?php
include_once("VideoCard.php");
class Dmeng implements VideoCardInter {
function Display(){
echo "Display";
}
function getName(){
return "Dmeng VideoCard";
}
}
?>

Mainboard.php  应用接口(把显卡插到主板上,主板只要用这些接口就行了,也可以不用)

 代码如下 复制代码
<?php
include_once("VideoCard.php");
include_once("Dmeng.php");
class Mainboard{
var $vc;
function run(VideoCardInter $vc){  //定义VideoCardInter接口类型参数,这时并不知道是谁来实现。
$this->vc=$vc;
$this->vc->Display();
echo "主板运行!";
}
}
$conputer=new Mainboard();
$conputer->run(new Dmeng); 

//用的时候把实现接口类的名称写进来,(现在是帝盟的显卡,也可以换成别的场家的,只要他们都实现了接口)
?>上面是转自的,然后自己试着写了点东西:把自己的思想总结一下
这里我想是分成4个方面

1,静态链接数据库类(基本的链接数据库,查询记录数,返回结果集等等)

2,每个模块的接口(例如新闻类接口  NewsInterface)

3,实现接口的类(新闻实现类  NewsInterfaceImpl 这里面写一些页面中用到的方法,然后new这个对象 调用这里么的方法)

4,模块类(新闻类  News)

本文章详细的介绍了在php学习中的最基础的内容包括了基本数据类型、变量类型、操作符等入门都会要接触到的东西。

php支持的基本数据类型:
Integer:整数
Float(Double,双精度值):浮点数,表示所有实数
String:字符串
Boolean:布尔值,true/false
Array:保存具有相同类型的多个数据项
Object:对象,保存类的实例
NULL:没有被赋值、已经被重置或被赋值为特殊值NULL的变量
特定的内置函数(例如数据库函数)将返回resource类型的变量

四种标量类型:

string(字符串)
integer(整型)
float(浮点型,也作 double )
boolean(布尔型)
两种复合类型:

array(数组)
object(对象)
两种特殊类型:

resource(资源)
NULL(空)
查看变量类型
通过 gettype() 函数可以方便的查看某个变量的类型:

 代码如下 复制代码

<?php
$var_bool = TRUE;     // a boolean
$var_str  = "foo";     // a string
$var_int  = 12;      // an integer

echo gettype($var_bool);    // 输出 boolean
echo gettype($var_str);     // 输出 string
echo gettype($var_int);     // 输出 integer
?>

提示
由于历史原因,如果是 float 类型数据,gettype() 函数返回的是 double,而不是 float 。
如果想查看某个表达式的值和类型,请使用用 var_dump() 函数。
判断变量类型
如果想通过判断变量类型来确定下一步逻辑动作,不要使用 gettype() ,而使用 is_type 系列函数:

 代码如下 复制代码

<?php
$var_int  = 12;

// 如果 $var_int 是 int 类型,这进行加法
if (is_int($var_int)) {
    $var_int = $var_int+4;
}
echo $var_int;  // 输出 16
?>


NULL和resource都代表外部资源,例如数据库连接
基本上不能直接操作一个resource变量,但是通常它们都将被函数返回,而且必须作为参数传递给其他函数。

在php中,变量的类型是由赋给变量的值确定的。(动态类型语言)
PHP可以在任何时间根据保存在变量中的值来确定变量的类型,也就是具有在任何时间透明地改变变量类型的功能。
PHP将“自动地”获得输入的数据类型,一旦从变量中检索变量值,它将返回具有相同数据类型的数据。
养成习惯:常量名称都是由大写字母组成的。这样容易区分变量和常量。
常量和变量之间的一个重要不同点在于引用一个常量的时候,它前面并没有$符号。
常量和变量的另一个差异在于常量只可以保存布尔值、整数、浮点数或字符串数据,这些类型都是标量数据。

变量类型变化(juggling)
PHP不需要(或者说不支持)在声名变量明确他的变量类型;一个变量的类型是由这个变量被使用的前后关系来决定的,也就是说,如果你给一个变量var赋了 一个字符串的值,var就变成了字符串变量。如果你再将一个整数赋值给var,他又变成了一个整数型变量。 一个PHP自动转化变量类型的例子是加法的操作符’+’。如果任何一个操作数都是双精度数,那么所有的操作数被当作双精度数求值,并且结果也是双精度数。 否则,操作数将被认为是整数,结果也将会是整数。注意这并不影响每个操作数本身的变量类型,唯一的变化就是操作数在计算的过程中被如何处理。 $foo  =  "0";  //  $foo  是一个字符串  ,值是“0”(ASCII  48)

PHP代码

 代码如下 复制代码
$foo  =  "0";  //  $foo  是一个字符串  ,值是“0”(ASCII  48)     
$foo++;  //  $foo  是一个字符串,值是“1”(ASCII  49)     
$foo  +=  1;  //  $foo  现在是一个整数(2)了     
$foo  =  $foo  +  1.3;  //  $foo  现在是一个双精度数(3.3)了     
$foo  =  5  +  "10  Little  Piggies";  //  $foo  是一个整数  (15)     
$foo  =  5  +  "10  Small  Pigs";  //  $foo  是一个整数  (15)  


如果你觉得上面例子中最后两个表达式看起来有点奇怪的话,请看 “字符串的转化”部分。 如果你希望强制一个变量被当作某种固定的类型来计算,请看 “类型强制(casting)” 一节。如果你希望转变一个变量的类型,请看函数“settype()”的说明。
 确定一个变量的类型
因为PHP自己决定变量的类型并且一般按照需要转化他们,一个特定的变量的类型不是在任何时间都非常明显。PHP包括一些函数可以找出这个变量的类型。这 些函数是gettype(), is_long(), is_double(), is_string(), is_array(), 和 is_object().
类型强制(type casting)
在PHP中类型强制同在C语言中大抵相似:将需要的类型的种类写在要强制的变量的前面的圆括号中。
 

PHP代码

 代码如下 复制代码
$foo  =  10;  //  $foo  是一个整数    
$bar  =  (double)  $foo;  //  $bar  是一个双精度数   


以下的强制方式是允许的: (int), (integer) – 强制成整数 (real), (double), (float) – 强制成双精度数 (string) – 强制成字符串 (array) – 强制成数组 (object) – 强制成对象 注意在圆括号中允许制表符(tabs)和空格(spaces) ,因此下面的语句是等价的: $foo = (int) $bar; $foo = ( int ) $bar;
字符串转化
当一个字符串被当作数字型的值来计算时,他的结果和类型如下面所述的决定。 如果这个字符串含有字符’.’,’e’,或者’E’,则被当作双精度类型变量来处理,否则当作是整数。 这个字符串的值由词首部分决定。如果这个字符串以任何有效的数字的数据开始,那么这个数字数据就是这个字符串参加运算的值。否则,值为零(zero)。有 效的数字数据是遵循下面的这些标记的,跟随在一个或多个数字后面(可以包含小数点),跟随在一个可选的指数后面。指数是由一个或多个数字跟在’e’或 ’E’后面构成的。
 

PHP代码

 代码如下 复制代码
$foo  =  1  +  "10.5";  //  $foo  是双精度数  (11.5)     
$foo  =  1  +  "-1.3e3";  //  $foo  是双精度数(-1299)     
$foo  =  1  +  "bob-1.3e3";  //  $foo  是整数  (1)     
$foo  =  1  +  "bob3";  //  $foo  是整数  (1)     
$foo  =  1  +  "10  Small  Pigs";  //  $foo  是整数  (11)   
$foo  =  1  +  "10  Little  Piggies";  //  $foo  是整数  (11);     
//  这个字符串包括字符'e'  


标识符


操作符:
1)赋值操作符:=

2)算术操作符:+,-,*,/,%(取模)

3)连接操作符:. ,无论操作数是什么,都当成String,结果返回String

4)合计赋值操作符(Combined Assignment Operators):+=,*=,/=,-=,%=,.=

5)自动增减操作符(Automatically Incrementing and Decrementing):

(1)$variable+=1 ⇔$variable++;$variable-=1 ⇔$variable-,跟c语言一样,先做其他操作,后++或-

(2)++$variable,-$variable,先++或-,再做其他操作

6)比较操作符:

= =(左边等于右边),!=(左边不等于右边),
= = =(左边等于右边,且数据类型相同),
>=,>,<,<=

7)逻辑操作符:|| or,&& and,xor(当左右两边有且只有一个是true,返回true),!

一个参考表

运算符(Arithmetic Operators)

Operator
符号
Description
描述
Example
案例
Result
结果
+ Addition
加号
x=2
x+2
4
- Subtraction
减号
x=2
5-x
3
* Multiplication
乘号
x=4
x*5
20
/ Division
除号
15/5
5/2
3
2.5
% Modulus (division remainder)
求模(余数)
5%2
10%8
10%2
1
2
0
++ Increment
自加
x=5
x++
x=6
-- Decrement
自减
x=5
x--
x=4

Assignment Operators
分配符(Assignment Operators)

Operator
符号
Example
案例
Is The Same As
等同于
= x=y x=y
+= x+=y x=x+y
-= x-=y x=x-y
*= x*=y x=x*y
/= x/=y x=x/y
%= x%=y x=x%y

Comparison Operators
比较符(Comparison Operators)

Operator
符号
Description
具体描述
Example
案例
== is equal to
等于
5==8 返回 false
!= is not equal
不等于
5!=8 返回  true
> is greater than
大于
5>8 返回  false
< is less than
小于
5<8 返回  true
>= is greater than or equal to
大于等于
5>=8 返回  false
<= is less than or equal to
小于等于
5<=8 返回  true

Logical Operators
逻辑判断符(Logical Operators)

Operator
符号
Description
描述
Example
案例
&& and
x=6
y=3

(x < 10 && y > 1) 返回 true

|| or
x=6
y=3

(x==5 || y==5) 返回 false

! not
x=6
y=3

!(x==y)返回true

其他操作符:

三元操作符、错误抑制操作符、执行操作符、数组操作符、类型操作符

运算符"?"前的表达式

如果一个包含二元运算符的表达式出现在三元运算符" ? : "的"?"之前,那么应该给表达式添上一对圆括号。例如:
(x >= 0) ? x : -x;

下面我们来看一个简单的?号表达式实例

 代码如下 复制代码

<?

$a=1;

echo $a?'true':'false';

输出结果为 trur;

再看

if( $a )

{

   echo 'true';

}

else

{

   echo 'false';

}

[!--infotagslink--]

相关文章

  • 图解PHP使用Zend Guard 6.0加密方法教程

    有时为了网站安全和版权问题,会对自己写的php源码进行加密,在php加密技术上最常用的是zend公司的zend guard 加密软件,现在我们来图文讲解一下。 下面就简单说说如何...2016-11-25
  • ps怎么使用HSL面板

    ps软件是现在很多人都会使用到的,HSL面板在ps软件中又有着非常独特的作用。这次文章就给大家介绍下ps怎么使用HSL面板,还不知道使用方法的下面一起来看看。 &#8195;...2017-07-06
  • Plesk控制面板新手使用手册总结

    许多的朋友对于Plesk控制面板应用不是非常的了解特别是英文版的Plesk控制面板,在这里小编整理了一些关于Plesk控制面板常用的使用方案整理,具体如下。 本文基于Linu...2016-10-10
  • C# 中的var关键字详细介绍

    这篇文章主要介绍了C# 中的var关键字详细介绍的相关资料,需要的朋友可以参考下...2020-06-25
  • 使用insertAfter()方法在现有元素后添加一个新元素

    复制代码 代码如下: //在现有元素后添加一个新元素 function insertAfter(newElement, targetElement){ var parent = targetElement.parentNode; if (parent.lastChild == targetElement){ parent.appendChild(newEl...2014-05-31
  • jQuery 1.9使用$.support替代$.browser的使用方法

    jQuery 从 1.9 版开始,移除了 $.browser 和 $.browser.version , 取而代之的是 $.support 。 在更新的 2.0 版本中,将不再支持 IE 6/7/8。 以后,如果用户需要支持 IE 6/7/8,只能使用 jQuery 1.9。 如果要全面支持 IE,并混合...2014-05-31
  • 使用percona-toolkit操作MySQL的实用命令小结

    1.pt-archiver 功能介绍: 将mysql数据库中表的记录归档到另外一个表或者文件 用法介绍: pt-archiver [OPTION...] --source DSN --where WHERE 这个工具只是归档旧的数据,不会对线上数据的OLTP查询造成太大影响,你可以将...2015-11-24
  • 使用GruntJS构建Web程序之构建篇

    大概有如下步骤 新建项目Bejs 新建文件package.json 新建文件Gruntfile.js 命令行执行grunt任务 一、新建项目Bejs源码放在src下,该目录有两个js文件,selector.js和ajax.js。编译后代码放在dest,这个grunt会...2014-06-07
  • 如何使用php脚本给html中引用的js和css路径打上版本号

    在搜索引擎中搜索关键字.htaccess 缓存,你可以搜索到很多关于设置网站文件缓存的教程,通过设置可以将css、js等不太经常更新的文件缓存在浏览器端,这样访客每次访问你的网站的时候,浏览器就可以从浏览器的缓存中获取css、...2015-11-24
  • MySQL日志分析软件mysqlsla的安装和使用教程

    一、下载 mysqlsla [root@localhost tmp]# wget http://hackmysql.com/scripts/mysqlsla-2.03.tar.gz--19:45:45-- http://hackmysql.com/scripts/mysqlsla-2.03.tar.gzResolving hackmysql.com... 64.13.232.157Conn...2015-11-24
  • C#注释的一些使用方法浅谈

    C#注释的一些使用方法浅谈,需要的朋友可以参考一下...2020-06-25
  • 安装和使用percona-toolkit来辅助操作MySQL的基本教程

    一、percona-toolkit简介 percona-toolkit是一组高级命令行工具的集合,用来执行各种通过手工执行非常复杂和麻烦的mysql和系统任务,这些任务包括: 检查master和slave数据的一致性 有效地对记录进行归档 查找重复的索...2015-11-24
  • php语言中使用json的技巧及json的实现代码详解

    目前,JSON已经成为最流行的数据交换格式之一,各大网站的API几乎都支持它。我写过一篇《数据类型和JSON格式》,探讨它的设计思想。今天,我想总结一下PHP语言对它的支持,这是开发互联网应用程序(特别是编写API)必须了解的知识...2015-10-30
  • 使用jquery修改表单的提交地址基本思路

    基本思路: 通过使用jquery选择器得到对应表单的jquery对象,然后使用attr方法修改对应的action 示例程序一: 默认情况下,该表单会提交到page_one.html 点击button之后,表单的提交地址就会修改为page_two.html 复制...2014-06-07
  • PHP实现无限级分类(不使用递归)

    无限级分类在开发中经常使用,例如:部门结构、文章分类。无限级分类的难点在于“输出”和“查询”,例如 将文章分类输出为<ul>列表形式; 查找分类A下面所有分类包含的文章。1.实现原理 几种常见的实现方法,各有利弊。其中...2015-10-23
  • php类的使用实例教程

    php类的使用实例教程 <?php /** * Class program for yinghua05-2 * designer :songsong */ class Template { var $tpl_vars; var $tpl_path; var $_deb...2016-11-25
  • 双冒号 ::在PHP中的使用情况

    前几天在百度知道里面看到有人问PHP中双冒号::的用法,当时给他的回答比较简洁因为手机打字不大方便!今天突然想起来,所以在这里总结一下我遇到的双冒号::在PHP中使用的情况!双冒号操作符即作用域限定操作符Scope Resoluti...2015-11-08
  • 浅析Promise的介绍及基本用法

    Promise是异步编程的一种解决方案,在ES6中Promise被列为了正式规范,统一了用法,原生提供了Promise对象。接下来通过本文给大家介绍Promise的介绍及基本用法,感兴趣的朋友一起看看吧...2021-10-21
  • C#中的yield关键字的使用方法介绍

    yield这个关键字是和迭代器挂钩的,而且是与return一起以yield return的形式合用的,用来返回迭代器中的条目。...2020-06-25
  • PHP mysql与mysqli事务使用说明 分享

    mysqli封装了诸如事务等一些高级操作,同时封装了DB操作过程中的很多可用的方法。应用比较多的地方是 mysqli的事务。...2013-10-02