Maison  >  Article  >  développement back-end  >  php面向对象方法实例详解

php面向对象方法实例详解

怪我咯
怪我咯original
2017-07-11 11:48:502467parcourir

面向对象编程(OOP)是我们编程的一项基本技能,PHP5对OOP提供了良好的支持。如何使用OOP的思想来进行PHP的高级编程,对于提高PHP编程能力和规划好Web开发构架都是非常有意义的。下面我们就通过实例来说明使用PHP的OOP进行编程的实际意义和应用方法。

  我们通常在做一个有数据库后台的网站的时候,都会考虑到程序需要适用于不同的应用环境。和其他编程语言有所不同的是,在PHP中,操作数据库的是一系列的具体功能函数(如果你不使用ODBC接口的话)。这样做虽然效率很高,但是封装却不够。如果有一个统一的数据库接口,那么我们就可以不对程序做任何修改而适用于多种数据库,从而使程序的移植性和跨平台能力都大大提高。

一、php面向对象的简单列子
以人为列子:

<?phpheader("Content-type: text/html; charset=utf-8");class person{//下面是人的成员属性var $name;//人的名字var $sex;//人的性别var $age;//人的年龄//定义一个构造方法参数为姓名$name,性别$sex和年龄$agefunction construct($name,$sex,$age){//通过构造方法传进来的$name给成员属性$this->name赋初始值$this->name=$name;//通过构造方法传进来的$sex给成员属性$this->sex赋初始值$this->sex=$sex;//通过构造方法传进来的$age给成员属性$this->age赋初始值$this->age="$age";
}//下面是人的成员方法function say()//这个人可以说话的方法
{echo "我的名字叫:".$this->name."性别;".$this->sex."我的年龄是:".$this->age."<br>";
}function run() //这个人可以走路的方法
{echo "这个人在走路";
}//这是一个析构函数,在对象销毁前调用function destruct(){echo "再见".$this->name."<br>";
}
}//通过构造方法创建3个对象$p1,$p2,$p3,分别传入三个不同的实参为姓名性别和年龄$p1=new person("小明","男",20);$p2=new person("熊","女",30);$p3=new person("向日葵","男",25);$p1->say();$p2->say();$p3->say();//下面访问3个对象的说话方式$p1->say();$p2->say();$p3->say();?>

二、php面向对象的几个步骤
第一类的定义:

<?php
  Class myobject{
    //……}?>

第二成员方法:

<?phpclassmyobject{   function getobjectname($name){
      echo "商品名称为:".$name;
   }
}?>

第三类的实例化:

<?phpclass myobject{
  function getobjectname($name){
     echo "商品名称为:".$name;
   }
}$c_book=new myobject();           //实例化对象echo $c_book->getobjectname("计算机图书");  //调用getbookname()方法?>

第四成员变量:

<?phpclass myobject{
  public $object_name;
  functionsetobjectname($name){    $this->object_name=$name;
  }
  functiongetobjectname(){    return$this->object_name;
  }
}$c_book=new myobject();$c_book->setobjectname("计算机图书");echo $c_book->object_name."<br>";echo $c_book->getobjectname();?>

第五常量类:
既然有变量,当然也会有常量了。常量就是不会改变的量,是一个恒值。众所周知的一个常量就是圆周率Pi。定义常量使用关键字const如:
ConstPI=3.14159;
例:

<?phpclass myobject{                        
  const book_type="计算机图书";             //声明常量book_type
  public $object_name;                                    //声明变量
  functionsetobjectname($name){                     //声明方法setobjectname()
      $this->object_name=$name;                       //设置成员的变量值
  }
  functiongetobjectname(){                //声明方法getobject()
    return$this->object_name;                        
  }
}$c_book=new myobject();                                 //实例化对象$c_book->setobjectname("PHP的类");              //调用方法setobjectnameecho myobject::book_type."<br>";              //输出常量的值echo $c_book->getobjectname();                 //调用方法getobjectname?>

第六面向对象类的构造方法

<?phpclass myobject{                        
   public $object_name;   //商品名称
   public $object_price;              //商品价格
   public $object_num;        //商品数量
   public $object_agio;        //商品折扣
   …………
}            
?>

声明一个myobject类的对象,并对这个类的一些成员变量赋初值。代码如下:

<?php
class myobject{                        
   public $object_name;   public $object_price;   public $object_num;   public $object_agio;
   functiongetobjectname(){      return$this->object_name;        return$this->object_price;        return $this->object_num;        return $this->object_agio;
   }
}         
$dress=new myobject();$dress->object_name="western-style clothes";$dress->object_price=1500;$dress->object_num=5;$dress->object_agio=8;
echo $dress->getobjectname();  
?>Voidconstruect([mixed args,[……]])

注意:函数中的是两条下划线,不是一条。
实例2:

<?phpclass myobject{                        
  public $object_name;  public $object_price;  public $object_num;  public $object_agio;
functionconstruct($name,$price,$num,$agio){   //通过参数给成员变量赋值
      $this->object_name=$name;          $this->object_price=$price;          $this->object_num=$num;          $this->object_agio=$agio;
   }function setobjectname($name){
    $this->object_name=$name;
}function getobjectname1(){
    return $this->object_name;
}function getobjectname2(){
        return $this->object_price;
}
}            
$c_book=new myobject("western-styleclothes",1500,5,8);echo $c_book->getobjectname1();echo "<br>";echo $c_book->getobjectname2();?>

第七析构方法:
概念
析构方法的作用和构造方法正好相反,是对象被销毁时被调用的,作用是释放内存。析构方法的格式为:
Voiddestruct(void)
例:

<?phpclass myobject{                        
  public $object_name;  public $object_price;  public $object_num;  public $object_agio;
functionconstruct($name,$price,$num,$agio){   //通过参数给成员变量赋值
      $this->object_name=$name;          $this->object_price=$price;          $this->object_num=$num;          $this->object_agio=$agio;
   }function setobjectname($name){
    $this->object_name=$name;
}function getobjectname1(){
    return $this->object_name;
}function getobjectname2(){
        return $this->object_price;
}function destruct(){
  echo "<p><b>对象被销毁,调用析构函数。</b></p>";
}
}            
$c_book=new myobject("western-styleclothes",1500,5,8);echo $c_book->getobjectname1();echo "<br>";echo $c_book->getobjectname2();unset($c_book);?>

PHP使用的是一种“垃圾回收”机制,自动清除不再使用的对象,释放内存。就是说即使不使用unset函数,析构方法也会自动被调用,这里只是明确一下析构函数在何时被调用。一般情况下是不需要手动创建析构方法的。

<?phpclass myobject{                        
   public $object_name;   public $object_price;   public $object_num;   public $object_agio;function construct($name,$price,$num,$agio){    //通过参数给成员变量赋值
       $this->object_name=$name;          $this->object_price=$price;          $this->object_num=$num;          $this->object_agio=$agio;
  }function setobjectname($name){
    $this->object_name=$name;
}function getobjectname1(){
     return$this->object_name;
}function getobjectname2(){
        return $this->object_price;
}function destruct(){
   echo"<p><b>对象被销毁,调用析构函数。</b></p>";
}
}            
$c_book=new myobject("western-styleclothes",1500,5,8);echo $c_book->getobjectname1();echo "<br>";echo $c_book->getobjectname2();?>

第八继承和多状态的实现

Class subclass extends superclass{
   ……
}

说明:subclass为子类的名称,superclass为父类名称。
例:

<?phpclass myobject{                        
  public $object_name;  public $object_price;  public $object_num;  public $object_agio;
functionconstruct($name,$price,$num,$agio){   //通过参数给成员变量赋值
      $this->object_name=$name;          $this->object_price=$price;          $this->object_num=$num;          $this->object_agio=$agio;
   }function showme(){
    echo "这句话会输出吗?答案是不会。";
}
}class book extends myobject{         
   public $book_type;
       functionconstruct($type,$num){         $this->book_type=$type;         $this->object_num=$num;  
       }
       functionshowme(){                            //重写父类中的showme()方法。
         return "本次新进".$this->book_type."图书".$this->object_num."本"."<br>";
       }
}class elec extends myobject{
   function showme(){                             //重写父类中的showme()方法
         return "热卖商品:".$this->object_name."<br>"."原 价:".$this->object_price."<br>"."特 价".$this->object_price*$this->object_agio;
       }
}$c_book=new book("计算机类",1000);   //声明一个book子类对象。$h_elec=new elec("待机王XX系列",1200,3,0.8);    //声明一个elec子类对象。echo$c_book->showme()."<br>";   //输出book子类的showme()方法echo $h_elec->showme();          //输出elec子类的是showme()方法?>

子类继承了父类的所有成员变量和方法,包括构造函数。这就是继承的实现。
当子类被创建时,PHP会先在子类中查找构造方法。如果子类有自己的构造方法,PHP会先调用子类中的方法,当子类中没有时,PHP则会去调用父类中的构造方法。
两个子类重写了父类的方法showme(),所以两个对象虽然调用的都是showme()方法,但返回的却是两段不同的信息。这就是多态性的实现。

第九

this->
在 前面类的实例化中,对如何调用成员方法有了基本的了解,那就是用对象名加方法名,格式为“对象名->方法名”。但在定义类时(如 myobject),根本无法得知对象的名称是什么。这时如果调用类中的方法,就要用伪变量

this的意思就是本身,所 以$this->只可以在类的内部使用。
例:


<?phpclassexample{   //创建类example
  function exam(){   //创建成员方法
     if(isset($this)){            //判断变量$this是否存在
           echo "\$this的值为:".get_class($this);    //如果存在,输出$this所属类的名字
        }else{           echo "$this未定义。";
        }
  }
}$class_name=newexample();   //实例化对象$class_name->exam();          //调用方法exam()?>

Get_class函数返回对象所属类的名字,如果不是对象,则返回false。
2、操作符::
相比伪变量$this只能在类的内部使用,操作符::可是真正的强大。操作符::可以在没有声明任何实例的情况下访问类中的成员方法和成员变量。使用::操作符的通用格式为:
      关键字::变量名/常量名/方法名
这里的关键字分为3种情况:
Parent关键字:可以调用父类中的成员变量、成员方法和常量。
Self关键字:可以调用当前类中的静态成员和常量。
类名:可以调用本类中的变量、常量和方法。
例:

   <?phpclass book{const name="conputer";      //声明常量namefunction construct(){        //构造方法
     echo "本月图书类冠军为:".book::name."<br>";   //输出默认值
  }
}class l_book extends book{          //book类的子类
  const name="foreign language";      //声明常量
  function construct(){                  //子类的构造方法
     parent::construct();       //调用父类的构造方法
         echo "本月图书类冠军为:".self::name;    //输出本类中的默认值
   }
}$obj=new l_book();?>

说明:关于静态方法(变量)的声明及使用可参考以下内容。

第十公共、私有和保护
1、 public公共成员
顾名思义,就是可以公开的、没有必要隐藏的数据信息。可以在程序的任何地点(类内、类外)被其他的类和对象调用。子类可以继承和使用父类中所有的公共成员。
所有的变量都被声明为public,而所有的方法在默认的状态下也是public。所以对变量
和方法的调用显示得十分混乱。为了解决这个问题,就需要使用第二个关键字:private。
2、 private私有成员
被private关键字修饰的变量和方法,只能在所属类的内部被调用和修改,不可以在类外被访问。在子类中也不可以。
例:

<?phpclass book{private $name="computer";public function setname($name){
   $this->name=$name;
  }public function getname(){
   return $this->name;
  }
}class lbook extends book{}$lbook=new lbook();echo "正确操作私有变量的方法:";$lbook->setname("PHP应用开发!");echo $lbook->getname();echo "直接操作私有变量的结果:";echo book:name;?>

对于成员方法,如果没有写关键字,那么默认就是public。从本节开始,以后所有的方法及变量都会带个关键字,这是一种良好的书写习惯。
3、 protected保护成员
private 关键字可以将数据完全隐藏起来,除了在本类外,其他地方都不可以调用。子类也不可以。但对于有些变量希望子类能够调用,但对另外的类来说,还要做到封装。 这时,就可以使用protected。被protected修改的类成员,可以在类和子类中被调用,其他地方则不可以被调用。
例:

<?phpclass book{
  protected $name="computer";
}class lbook extends book{
  public function showme(){
      echo "对于protected修饰的变量,在子类中是可以直接调用的。如:\$name=".$this->name."<br>";
   }
}$lbook=new lbook();$lbook->showme();echo "但在其他的地方是不可以调用的,否则:";$lbook->name="history";?>

第十一  PHP对象的高级应用
1.        final关键字
final,中文含义是最终的、最后的。被final修饰过的类和方法就是“最终版本”;
如果有一个类的格式为:

  Final class class_name{}

说明该类不可以再被继承,也不能再有子类。
如果有一个方法的格式为:
Final function method_name();
说明该方法在子类中不可以进行重写,也不可以被覆盖。
例:

<?phpfinal class myobject{
  function construct(){
      echo "initialize object";
   }
}class mybook extends myobject{
  static function exam(){
      echo "you can&#39;t see me.";
   }
}
mybook::exam();?>

输出的结果是:

Fatal error: Class mybookmay not inherit from final class (myobject) in C:\webserver\wwwroot\2.phpon line 19
  1. 抽象类
    抽象类是一种不能被实例化的类,只能作为其他类的父类来使用。抽象类使用abstract关键字来声明,格式为:

    Abstract class abstractname{}

抽象类和普通类相似,都包含成员变量、成员方法。两者的区别在于,抽象类至少要包含一个抽象方法。抽象方法没有方法体,其功能的实现只能在子类中完成。抽象方法也是使用abstract关键字修饰的。它的格式为:
Abstractfunction abstractname();
注意:在抽象方法后面要有分号“;”
抽象类和抽象方法主要应用于复杂的层次关系中,这种层次关系要求每一个子类都包含并重写某些特定的方法。举一个例子:中国的美食是多种多样的,有吉菜、鲁菜、川菜、粤菜等。每种菜系使用的都是煎、炒、炸等手法,只是在具体的步骤上,各有各的不同。如果把中国美食当作一个大类cate,下面的各大菜系就是cate的子类,而煎炒烹炸则是每个类中都有的方法。每个方法在子类中的实现都是不同的,在父类中无法规定。为了统一规范,不同子类的方法要有一个相同的方法名:decoct(煎)、stir_fry(炒)、cook(烹)、fry(炸)。
例:

<?php
    /* 抽象类myobject */
        abstract class myobject{    //抽象方法servie
            abstract functionservice($getname,$price,$num);
        }        //声明抽象类的子类mybook
        class mybook extends myobject{    //实现抽象方法service
            function service($getname,$price,$num){
                  echo "您购买的商品是".$getname.",该商品的价格是:".$price."元";                      echo "您购买的数量为:".$num."本。";                      echo "如发现缺页、损坏、请在3日内更换。";
               }
        }        //声明另一个抽象类的子类mycomputer
        class mycomputer extends myobject{   //实现抽象方法service
            function service($getname,$price,$num){
                  echo "您购买的商品是:".$getname.",该商品的价格是:".$price."元。";                      echo "您购买的数量为:".$num."台。";                      echo "如发生非人为质量问题,请在3个月内更换。";
               }
        }        $book=new mybook();    //实例化对象$book
        $computer=new mycomputer();   //实例化对象$computer
        $book->service("《PHP从入门到精通》",85,3);   //调用$book对象的service方法
        echo "<br />";   
        $computer->service("XX笔记本",8500,1);    //调用computer对象的service方法?>
  1. 接口的使用
    继承特性简化了对象、类的创建,增加了代码的可重性。但PHP只支持单继承。如果想实现多重继承,就要使用接口。PHP可以实现多个接口。
    接口类通过interface关键字来声明,并且类中只能包含未实现的方法和一些成员变量,格式如下:

Interface interfacename{
  Function interfacename1();
  Function interfacename2();
  ………………
}

注意:不要用public以外的关键字来修饰接口中的类成员,对于方法,不写关键字也可以。这是一个接口类自身的天性决定的。
子类是通过implements关键字来实现接口的,如果要实现多个接口,那么每个接口之间使用逗号“,”连接。而且所有未实现的方法需要在子类中全部实现,否则PHP将会出现错误。格式如下:

Class subclass implementsinterfacename1,interfacename2{
    Function interfacename1(){
       //功能实现
     }    Function interfacename2(){
       //功能实现
     }
     …………
}

例:本例首先声明了两个接口mpopedom和mpurview,接着声明了两个类member和manager,其中member类继承了mpopedom接口;manager继承了mpopedom和mpurview接口。分别实现各自的成员方法后,实例化两个对象

manager。最后调用实现的方法。


<?php
     interface mpopedom{
           function popedom();
        }        interface mpurview{
           function purview();
        }        class member implements mpurview{
            function purview(){
                    echo "会员拥有的权限";
                }
        }        class manager implements mpurview,mpopedom{
            function purview(){
                echo "管理员拥有会员的全部权限。";
                }                function popedom(){
                    echo "管理员还有会员没有的权限。";
                }
        }        $member=new member();        $manager=new manager();        $member->purview();        echo "<br />";        $manager->purview();        $manager->popedom();?>

通过上面的实例可以发现,抽象类和接口实现的功能十分相似。抽象类的优点是可以在抽象类中实现公共的方法,而接口则可以实现多继承。至于何时使用抽象类和接口就要看具体实现了。
4.        克隆对象
  (1)、克隆对象
   在PHP4中,对象被当作普通的数据类型来使用。如果想引开对象,需要使用&来声明,否则会按照PHP4的默认方式来按值传递对象。
例:本实例首先实例化一个myobject类的对象

book1的默认值是book,然后将对象

book2赋值。改变

book1的值。


   <?php
     classmyobject{             private $object_type="book";               public function settype($type){
                   $this->object_type=$type;  
                }               public function gettype(){
                   return $this->object_type;
                }
        }        $book1=new myobject();        $book2=$book1;        $book2->settype("computer");        echo "对象\$book1的值为:".$book1->gettype();?>

上面的实例,在PHP5中的返回值为:对象

book2只是

book1的值为:book,因为对象

book1的一个备份。
在PHP5中如果需要将对象复制,也就是克隆一个对象。需要使用关键字clone来实现。克隆一个对象的格式为:


     $object1=new classname();     $object2=clone $object1;
 (2)、clone()方法

有时除了单纯地克隆对象外,还需要克隆出来的对象可以拥有自己的属性和行为。这时就可以使用_clone()方法来实现。clone()方法的作用是:在克隆对象的过程中,调用_clone()方法,可以使用克隆出来的对象保持自己的一些行为及属性。
例:

<?php
     class myobject{
             private $object_type="book";                 public function settype($type){
                      $this->object_type=$type;
                 }                 public function gettype(){
                      return $this->object_type;
                 }                 public function clone(){
                     $this->object_type="computer";
                                            }      
        }        $book1=new myobject();        $book2=clone $book1;        echo "对象$book1的变量值为:".$book1->gettype();        echo"<br />";        echo "对象$book2的变量值为:".$book2->gettype();?>
  1. 对象比较
    通过克隆对象,相信读者已经理解表达式

    object1和

    object1所表示的不同含义。但在实际开发中,应如何判断两个对象之间的关系是克隆还是引用?
    这是可以使用比较运算符==和===。两个等号是比较两个对象的内容,3个等号===是比较对象的引用地址。
    例:

<?php
     classmyobject{            private $name;               function construct($name){
                   $this->name=$name;
                }
        }        $book=new myobject(book);        $clonebook=clone $book;        $referbook=$book;        if($clonebook==$book){            echo "两个对象的内容相等。<br />";
        }        if($referbook===$book){            echo "两个对象的引用地址相等。<br />";
        }?>
  1. 对象类型检测
    Instanceof操作符可以检测当前对象是属于哪个类的。一般格式如为:
    Objectname instanceof classname
    例:本实例首先创建两个类,一个基类(myobject)与一个子类(mybook)。实例化一个子类对象,判断对象是否属于该子类,再判断对象是否属于基类。
    例:

<?php
     classmyobject{}        class mybook extends myobject{
            private $type;
     }        $cbook=new mybook();        if($cbook instanceof mybook){            echo "对象\$cbook属于mybook类<br />";
        }        if($cbook instanceof myobject){            echo "对象\$book属于myobject类<br />";
        }?>
  1. 魔术方法(_)
    PHP中有很多以两个下划线开头的方法,如前面已经介绍过的construct()、destruct()和clone(),这些方法被称为魔术方法。下在我们将学习其他一些魔术方法。
    注意:PHP中保留了所有以””开头的方法,所以只能使用在PHP文档中有的这些方法,不要自己创建。
    (1)、set()和get()方法
    这两个魔术方法的作用为:
    当程序试图写入一个不存在或不可见的成员变量时,PHP就会执行set()方法。set()方法包含两个参数,分别表示变量名称和变量值。两个参数不可省略。
    当程序调用一个未定义或不可见的成员变量时,可以通过get()方法来读取变量值。get()方法有一个参数,表示要调用的变量名。
    注意:如果希望PHP调用这些魔术方法,那么首先必须在类中进行定义;否则PHP不会执行未创建的魔术方法。
    例:

<?php
     class myobject{
           private $type="";               private function get($name){
                  if(isset($this->$name)){                        echo "变量".$name."的值为:".$this->$name."<br />";
                     }                     else{                        echo "变量".$name."未定义,初始化为0<br />";                            $this->$name=0;
                     }
               }               private function set($name,$value){
                   if(isset($this->$name)){                         $this->$name=$value;                             echo "变量".$name."赋值为:".$value."<br />";
                      }else{                          $this->$name=$value;                             echo "变量".$name."被初始化为:".$value."<br />";
                      }
               }
        }        $mycomputer=new myobject();        $mycomputer->type="DIY";        $mycomputer->type;        $mycomputer->name;?>

(2)、call()方法
call()方法的作用是:当程序试图调用不存在或不可见的成员方法时,PHP会先调用call()方法来存储方法名及其参数。call()方法包含两个参数,即方法名和方法参数。其中,方法参数是以数组形式存在的。
例:

<?php
     class myobject{
            public function mydream(){
                     echo "调用的方法存在,直接执行此方法:<br />";
                }                public function call($method,$parameter){
                     echo "如果方法不存在,则执行call()方法。<br/>";                       echo "方法名为:".$method."<br />";                       echo "参数有:";
                       var_dump($parameter);
                }
        }        $exam=new myobject();        $exam->mydream();        $exam->mdreadm(&#39;how&#39;,&#39;what&#39;,&#39;why&#39;);?>

(3)、sleep()和wakeup()方法
使用serialize()函数可以实现序列化对象。就是将对象中的变量全部保存下来,对象中的类则只保存类名。在使用serialize()函数时,如果实例化的对象包含sleep()方法,则会执行sleep()方法。该方法可以清除对象并返回一个该对象中所有变量的数组。使用sleep()方法的目的是关闭对象可能具有的数据库连接等类似的善后工作。
Unserialize()函数可以重新还原一个被serialize()函数序列化的对象,wakeup()方法则是恢复在序列化中可能丢失的数据库连接及相关工作。
例:

<?php
     class myobject{
           private $type="DIY";               public function gettype(){
                   return $this->type;
               }               public function sleep(){
                   echo "使用serialize()函数将对象保存起来,可以存放到文本文件、数据库等地方。<br />";                      return $this;
               }               public function wakeup(){
                   echo "当需要该数据时,使用unserialize()函数对已序列化的字符串进行操作,将其转换回来对象。<br />";
               }
        }       $mybook=new myobject();       $i=serialize($mybook);       echo "序列化后的字符串:".$i."<br />";       $rebook=unserialize($i);       echo "还原后的成员变量:".$rebook->gettype();?>

(4)、tostring()方法
魔术方法这个作用是:当使用echo 或print输出对象时,将对象转化为字符串。
例:

<?php
     class myobject{
            private $type="DIY";                public function tostring(){
                    return $this->type;
                }
        }        $mycomputer=new myobject();        echo "对象\$mycomputer的值为:";        echo $mycomputer;?>

(5)、autoload()方法
   将一个独立、完整的类保存到一个PHP页中,并且文件名和类名保持一致,这是每个开发人员者需要养成的良好习惯。这样,在下次重复使用某个类时即能很轻易地找到它。但还有一个问题是让开发人员头疼不已的,如果要在一个页面中引进很多的类,需要使用include_once()函数或require_once()函数一个一个地引入。
   PHP5解决了这个问题,autoload()方法可以自动实例化需要使用的类。当程序要用到一个类,但该类还没有被实例化时,PHP5将使用autoload()方法,在指定的路径下自动查找和该类名称相同的文件。如果找到,程序则继续执行,否则,报告错误。
例:
Myobject.class.php页面代码如下:

<?php
     classmyobject{              private $cont;                public function construct($cont){
                    $this->cont=$cont;
                }                public function tostring(){
                    return $this->cont;
                }
        }?>

Index.php页面代码如下:

<?php
     functionautoload($class_name){                                  $class_path=$class_name.&#39;.class.php&#39;;                                    if(file_exists($class_path)){                                       include_once($class_path);
                                    }else{                                       echo "类路径错误。";
                                    }
                              }                              $mybook=new myobject("江山代有人才出    各领风骚数百年");                              echo $mybook;?>

一、php面向对象的简单列子
以人为列子:

<?phpheader("Content-type: text/html; charset=utf-8");class person{//下面是人的成员属性var $name;//人的名字var $sex;//人的性别var $age;//人的年龄//定义一个构造方法参数为姓名$name,性别$sex和年龄$agefunction construct($name,$sex,$age){//通过构造方法传进来的$name给成员属性$this->name赋初始值$this->name=$name;//通过构造方法传进来的$sex给成员属性$this->sex赋初始值$this->sex=$sex;//通过构造方法传进来的$age给成员属性$this->age赋初始值$this->age="$age";
}//下面是人的成员方法function say()//这个人可以说话的方法
{echo "我的名字叫:".$this->name."性别;".$this->sex."我的年龄是:".$this->age."<br>";
}function run() //这个人可以走路的方法
{echo "这个人在走路";
}//这是一个析构函数,在对象销毁前调用function destruct(){echo "再见".$this->name."<br>";
}
}//通过构造方法创建3个对象$p1,$p2,$p3,分别传入三个不同的实参为姓名性别和年龄$p1=new person("小明","男",20);$p2=new person("熊","女",30);$p3=new person("向日葵","男",25);$p1->say();$p2->say();$p3->say();//下面访问3个对象的说话方式$p1->say();$p2->say();$p3->say();?>

二、php面向对象的几个步骤
第一类的定义:

<?php
  Class myobject{
    //……}?>

第二成员方法:

<?phpclassmyobject{   function getobjectname($name){
      echo "商品名称为:".$name;
   }
}?>

第三类的实例化:

<?phpclass myobject{
  function getobjectname($name){
     echo "商品名称为:".$name;
   }
}$c_book=new myobject();           //实例化对象echo $c_book->getobjectname("计算机图书");  //调用getbookname()方法?>

第四成员变量:

<?phpclass myobject{
  public $object_name;
  functionsetobjectname($name){    $this->object_name=$name;
  }
  functiongetobjectname(){    return$this->object_name;
  }
}$c_book=new myobject();$c_book->setobjectname("计算机图书");echo $c_book->object_name."<br>";echo $c_book->getobjectname();?>

第五常量类:
既然有变量,当然也会有常量了。常量就是不会改变的量,是一个恒值。众所周知的一个常量就是圆周率Pi。定义常量使用关键字const如:
ConstPI=3.14159;
例:

<?phpclass myobject{                        
  const book_type="计算机图书";             //声明常量book_type
  public $object_name;                                    //声明变量
  functionsetobjectname($name){                     //声明方法setobjectname()
      $this->object_name=$name;                       //设置成员的变量值
  }
  functiongetobjectname(){                //声明方法getobject()
    return$this->object_name;                        
  }
}$c_book=new myobject();                                 //实例化对象$c_book->setobjectname("PHP的类");              //调用方法setobjectnameecho myobject::book_type."<br>";              //输出常量的值echo $c_book->getobjectname();                 //调用方法getobjectname?>

第六面向对象类的构造方法

<?phpclass myobject{                        
   public $object_name;   //商品名称
   public $object_price;              //商品价格
   public $object_num;        //商品数量
   public $object_agio;        //商品折扣
   …………
}            
?>

声明一个myobject类的对象,并对这个类的一些成员变量赋初值。代码如下:

<?php
class myobject{                        
   public $object_name;   public $object_price;   public $object_num;   public $object_agio;
   functiongetobjectname(){      return$this->object_name;        return$this->object_price;        return $this->object_num;        return $this->object_agio;
   }
}         
$dress=new myobject();$dress->object_name="western-style clothes";$dress->object_price=1500;$dress->object_num=5;$dress->object_agio=8;
echo $dress->getobjectname();  
?>Voidconstruect([mixed args,[……]])

注意:函数中的是两条下划线,不是一条。
实例2:

<?phpclass myobject{                        
  public $object_name;  public $object_price;  public $object_num;  public $object_agio;
functionconstruct($name,$price,$num,$agio){   //通过参数给成员变量赋值
      $this->object_name=$name;          $this->object_price=$price;          $this->object_num=$num;          $this->object_agio=$agio;
   }function setobjectname($name){
    $this->object_name=$name;
}function getobjectname1(){
    return $this->object_name;
}function getobjectname2(){
        return $this->object_price;
}
}            
$c_book=new myobject("western-styleclothes",1500,5,8);echo $c_book->getobjectname1();echo "<br>";echo $c_book->getobjectname2();?>

第七析构方法:
概念
析构方法的作用和构造方法正好相反,是对象被销毁时被调用的,作用是释放内存。析构方法的格式为:
Voiddestruct(void)
例:

<?phpclass myobject{                        
  public $object_name;  public $object_price;  public $object_num;  public $object_agio;
functionconstruct($name,$price,$num,$agio){   //通过参数给成员变量赋值
      $this->object_name=$name;          $this->object_price=$price;          $this->object_num=$num;          $this->object_agio=$agio;
   }function setobjectname($name){
    $this->object_name=$name;
}function getobjectname1(){
    return $this->object_name;
}function getobjectname2(){
        return $this->object_price;
}function destruct(){
  echo "<p><b>对象被销毁,调用析构函数。</b></p>";
}
}            
$c_book=new myobject("western-styleclothes",1500,5,8);echo $c_book->getobjectname1();echo "<br>";echo $c_book->getobjectname2();unset($c_book);?>

PHP使用的是一种“垃圾回收”机制,自动清除不再使用的对象,释放内存。就是说即使不使用unset函数,析构方法也会自动被调用,这里只是明确一下析构函数在何时被调用。一般情况下是不需要手动创建析构方法的。

<?phpclass myobject{                        
   public $object_name;   public $object_price;   public $object_num;   public $object_agio;function construct($name,$price,$num,$agio){    //通过参数给成员变量赋值
       $this->object_name=$name;          $this->object_price=$price;          $this->object_num=$num;          $this->object_agio=$agio;
  }function setobjectname($name){
    $this->object_name=$name;
}function getobjectname1(){
     return$this->object_name;
}function getobjectname2(){
        return $this->object_price;
}function destruct(){
   echo"<p><b>对象被销毁,调用析构函数。</b></p>";
}
}            
$c_book=new myobject("western-styleclothes",1500,5,8);echo $c_book->getobjectname1();echo "<br>";echo $c_book->getobjectname2();?>

第八继承和多状态的实现

Class subclass extends superclass{
   ……
}

说明:subclass为子类的名称,superclass为父类名称。
例:

<?phpclass myobject{                        
  public $object_name;  public $object_price;  public $object_num;  public $object_agio;
functionconstruct($name,$price,$num,$agio){   //通过参数给成员变量赋值
      $this->object_name=$name;          $this->object_price=$price;          $this->object_num=$num;          $this->object_agio=$agio;
   }function showme(){
    echo "这句话会输出吗?答案是不会。";
}
}class book extends myobject{         
   public $book_type;
       functionconstruct($type,$num){         $this->book_type=$type;         $this->object_num=$num;  
       }
       functionshowme(){                            //重写父类中的showme()方法。
         return "本次新进".$this->book_type."图书".$this->object_num."本"."<br>";
       }
}class elec extends myobject{
   function showme(){                             //重写父类中的showme()方法
         return "热卖商品:".$this->object_name."<br>"."原 价:".$this->object_price."<br>"."特 价".$this->object_price*$this->object_agio;
       }
}$c_book=new book("计算机类",1000);   //声明一个book子类对象。$h_elec=new elec("待机王XX系列",1200,3,0.8);    //声明一个elec子类对象。echo$c_book->showme()."<br>";   //输出book子类的showme()方法echo $h_elec->showme();          //输出elec子类的是showme()方法?>

子类继承了父类的所有成员变量和方法,包括构造函数。这就是继承的实现。
当子类被创建时,PHP会先在子类中查找构造方法。如果子类有自己的构造方法,PHP会先调用子类中的方法,当子类中没有时,PHP则会去调用父类中的构造方法。
两个子类重写了父类的方法showme(),所以两个对象虽然调用的都是showme()方法,但返回的却是两段不同的信息。这就是多态性的实现。

第九

this->
在 前面类的实例化中,对如何调用成员方法有了基本的了解,那就是用对象名加方法名,格式为“对象名->方法名”。但在定义类时(如 myobject),根本无法得知对象的名称是什么。这时如果调用类中的方法,就要用伪变量

this的意思就是本身,所 以$this->只可以在类的内部使用。
例:


<?phpclassexample{   //创建类example
  function exam(){   //创建成员方法
     if(isset($this)){            //判断变量$this是否存在
           echo "\$this的值为:".get_class($this);    //如果存在,输出$this所属类的名字
        }else{           echo "$this未定义。";
        }
  }
}$class_name=newexample();   //实例化对象$class_name->exam();          //调用方法exam()?>

Get_class函数返回对象所属类的名字,如果不是对象,则返回false。
2、操作符::
相比伪变量$this只能在类的内部使用,操作符::可是真正的强大。操作符::可以在没有声明任何实例的情况下访问类中的成员方法和成员变量。使用::操作符的通用格式为:
      关键字::变量名/常量名/方法名
这里的关键字分为3种情况:
Parent关键字:可以调用父类中的成员变量、成员方法和常量。
Self关键字:可以调用当前类中的静态成员和常量。
类名:可以调用本类中的变量、常量和方法。
例:

   <?phpclass book{const name="conputer";      //声明常量namefunction construct(){        //构造方法
     echo "本月图书类冠军为:".book::name."<br>";   //输出默认值
  }
}class l_book extends book{          //book类的子类
  const name="foreign language";      //声明常量
  function construct(){                  //子类的构造方法
     parent::construct();       //调用父类的构造方法
         echo "本月图书类冠军为:".self::name;    //输出本类中的默认值
   }
}$obj=new l_book();?>

说明:关于静态方法(变量)的声明及使用可参考以下内容。

第十公共、私有和保护
1、 public公共成员
顾名思义,就是可以公开的、没有必要隐藏的数据信息。可以在程序的任何地点(类内、类外)被其他的类和对象调用。子类可以继承和使用父类中所有的公共成员。
所有的变量都被声明为public,而所有的方法在默认的状态下也是public。所以对变量
和方法的调用显示得十分混乱。为了解决这个问题,就需要使用第二个关键字:private。
2、 private私有成员
被private关键字修饰的变量和方法,只能在所属类的内部被调用和修改,不可以在类外被访问。在子类中也不可以。
例:

<?phpclass book{private $name="computer";public function setname($name){
   $this->name=$name;
  }public function getname(){
   return $this->name;
  }
}class lbook extends book{}$lbook=new lbook();echo "正确操作私有变量的方法:";$lbook->setname("PHP应用开发!");echo $lbook->getname();echo "直接操作私有变量的结果:";echo book:name;?>

对于成员方法,如果没有写关键字,那么默认就是public。从本节开始,以后所有的方法及变量都会带个关键字,这是一种良好的书写习惯。
3、 protected保护成员
private 关键字可以将数据完全隐藏起来,除了在本类外,其他地方都不可以调用。子类也不可以。但对于有些变量希望子类能够调用,但对另外的类来说,还要做到封装。 这时,就可以使用protected。被protected修改的类成员,可以在类和子类中被调用,其他地方则不可以被调用。
例:

<?phpclass book{
  protected $name="computer";
}class lbook extends book{
  public function showme(){
      echo "对于protected修饰的变量,在子类中是可以直接调用的。如:\$name=".$this->name."<br>";
   }
}$lbook=new lbook();$lbook->showme();echo "但在其他的地方是不可以调用的,否则:";$lbook->name="history";?>

第十一  PHP对象的高级应用
1.        final关键字
final,中文含义是最终的、最后的。被final修饰过的类和方法就是“最终版本”;
如果有一个类的格式为:

  Final class class_name{}

说明该类不可以再被继承,也不能再有子类。
如果有一个方法的格式为:
Final function method_name();
说明该方法在子类中不可以进行重写,也不可以被覆盖。
例:

<?phpfinal class myobject{
  function construct(){
      echo "initialize object";
   }
}class mybook extends myobject{
  static function exam(){
      echo "you can&#39;t see me.";
   }
}
mybook::exam();?>

输出的结果是:

Fatal error: Class mybookmay not inherit from final class (myobject) in C:\webserver\wwwroot\2.phpon line 19
  1. 抽象类
    抽象类是一种不能被实例化的类,只能作为其他类的父类来使用。抽象类使用abstract关键字来声明,格式为:

    Abstract class abstractname{}

抽象类和普通类相似,都包含成员变量、成员方法。两者的区别在于,抽象类至少要包含一个抽象方法。抽象方法没有方法体,其功能的实现只能在子类中完成。抽象方法也是使用abstract关键字修饰的。它的格式为:
Abstractfunction abstractname();
注意:在抽象方法后面要有分号“;”
抽象类和抽象方法主要应用于复杂的层次关系中,这种层次关系要求每一个子类都包含并重写某些特定的方法。举一个例子:中国的美食是多种多样的,有吉菜、鲁菜、川菜、粤菜等。每种菜系使用的都是煎、炒、炸等手法,只是在具体的步骤上,各有各的不同。如果把中国美食当作一个大类cate,下面的各大菜系就是cate的子类,而煎炒烹炸则是每个类中都有的方法。每个方法在子类中的实现都是不同的,在父类中无法规定。为了统一规范,不同子类的方法要有一个相同的方法名:decoct(煎)、stir_fry(炒)、cook(烹)、fry(炸)。
例:

<?php
    /* 抽象类myobject */
        abstract class myobject{    //抽象方法servie
            abstract functionservice($getname,$price,$num);
        }        //声明抽象类的子类mybook
        class mybook extends myobject{    //实现抽象方法service
            function service($getname,$price,$num){
                  echo "您购买的商品是".$getname.",该商品的价格是:".$price."元";                      echo "您购买的数量为:".$num."本。";                      echo "如发现缺页、损坏、请在3日内更换。";
               }
        }        //声明另一个抽象类的子类mycomputer
        class mycomputer extends myobject{   //实现抽象方法service
            function service($getname,$price,$num){
                  echo "您购买的商品是:".$getname.",该商品的价格是:".$price."元。";                      echo "您购买的数量为:".$num."台。";                      echo "如发生非人为质量问题,请在3个月内更换。";
               }
        }        $book=new mybook();    //实例化对象$book
        $computer=new mycomputer();   //实例化对象$computer
        $book->service("《PHP从入门到精通》",85,3);   //调用$book对象的service方法
        echo "<br />";   
        $computer->service("XX笔记本",8500,1);    //调用computer对象的service方法?>
  1. 接口的使用
    继承特性简化了对象、类的创建,增加了代码的可重性。但PHP只支持单继承。如果想实现多重继承,就要使用接口。PHP可以实现多个接口。
    接口类通过interface关键字来声明,并且类中只能包含未实现的方法和一些成员变量,格式如下:

Interface interfacename{
  Function interfacename1();
  Function interfacename2();
  ………………
}

注意:不要用public以外的关键字来修饰接口中的类成员,对于方法,不写关键字也可以。这是一个接口类自身的天性决定的。
子类是通过implements关键字来实现接口的,如果要实现多个接口,那么每个接口之间使用逗号“,”连接。而且所有未实现的方法需要在子类中全部实现,否则PHP将会出现错误。格式如下:

Class subclass implementsinterfacename1,interfacename2{
    Function interfacename1(){
       //功能实现
     }    Function interfacename2(){
       //功能实现
     }
     …………
}

例:本例首先声明了两个接口mpopedom和mpurview,接着声明了两个类member和manager,其中member类继承了mpopedom接口;manager继承了mpopedom和mpurview接口。分别实现各自的成员方法后,实例化两个对象

manager。最后调用实现的方法。


<?php
     interface mpopedom{
           function popedom();
        }        interface mpurview{
           function purview();
        }        class member implements mpurview{
            function purview(){
                    echo "会员拥有的权限";
                }
        }        class manager implements mpurview,mpopedom{
            function purview(){
                echo "管理员拥有会员的全部权限。";
                }                function popedom(){
                    echo "管理员还有会员没有的权限。";
                }
        }        $member=new member();        $manager=new manager();        $member->purview();        echo "<br />";        $manager->purview();        $manager->popedom();?>

通过上面的实例可以发现,抽象类和接口实现的功能十分相似。抽象类的优点是可以在抽象类中实现公共的方法,而接口则可以实现多继承。至于何时使用抽象类和接口就要看具体实现了。
4.        克隆对象
  (1)、克隆对象
   在PHP4中,对象被当作普通的数据类型来使用。如果想引开对象,需要使用&来声明,否则会按照PHP4的默认方式来按值传递对象。
例:本实例首先实例化一个myobject类的对象

book1的默认值是book,然后将对象

book2赋值。改变

book1的值。


   <?php
     classmyobject{             private $object_type="book";               public function settype($type){
                   $this->object_type=$type;  
                }               public function gettype(){
                   return $this->object_type;
                }
        }        $book1=new myobject();        $book2=$book1;        $book2->settype("computer");        echo "对象\$book1的值为:".$book1->gettype();?>

上面的实例,在PHP5中的返回值为:对象

book2只是

book1的值为:book,因为对象

book1的一个备份。
在PHP5中如果需要将对象复制,也就是克隆一个对象。需要使用关键字clone来实现。克隆一个对象的格式为:


     $object1=new classname();     $object2=clone $object1;
 (2)、clone()方法

有时除了单纯地克隆对象外,还需要克隆出来的对象可以拥有自己的属性和行为。这时就可以使用_clone()方法来实现。clone()方法的作用是:在克隆对象的过程中,调用_clone()方法,可以使用克隆出来的对象保持自己的一些行为及属性。
例:

<?php
     class myobject{
             private $object_type="book";                 public function settype($type){
                      $this->object_type=$type;
                 }                 public function gettype(){
                      return $this->object_type;
                 }                 public function clone(){
                     $this->object_type="computer";
                                            }      
        }        $book1=new myobject();        $book2=clone $book1;        echo "对象$book1的变量值为:".$book1->gettype();        echo"<br />";        echo "对象$book2的变量值为:".$book2->gettype();?>
  1. 对象比较
    通过克隆对象,相信读者已经理解表达式

    object1和

    object1所表示的不同含义。但在实际开发中,应如何判断两个对象之间的关系是克隆还是引用?
    这是可以使用比较运算符==和===。两个等号是比较两个对象的内容,3个等号===是比较对象的引用地址。
    例:

<?php
     classmyobject{            private $name;               function construct($name){
                   $this->name=$name;
                }
        }        $book=new myobject(book);        $clonebook=clone $book;        $referbook=$book;        if($clonebook==$book){            echo "两个对象的内容相等。<br />";
        }        if($referbook===$book){            echo "两个对象的引用地址相等。<br />";
        }?>
  1. 对象类型检测
    Instanceof操作符可以检测当前对象是属于哪个类的。一般格式如为:
    Objectname instanceof classname
    例:本实例首先创建两个类,一个基类(myobject)与一个子类(mybook)。实例化一个子类对象,判断对象是否属于该子类,再判断对象是否属于基类。
    例:

<?php
     classmyobject{}        class mybook extends myobject{
            private $type;
     }        $cbook=new mybook();        if($cbook instanceof mybook){            echo "对象\$cbook属于mybook类<br />";
        }        if($cbook instanceof myobject){            echo "对象\$book属于myobject类<br />";
        }?>
  1. 魔术方法(_)
    PHP中有很多以两个下划线开头的方法,如前面已经介绍过的construct()、destruct()和clone(),这些方法被称为魔术方法。下在我们将学习其他一些魔术方法。
    注意:PHP中保留了所有以””开头的方法,所以只能使用在PHP文档中有的这些方法,不要自己创建。
    (1)、set()和get()方法
    这两个魔术方法的作用为:
    当程序试图写入一个不存在或不可见的成员变量时,PHP就会执行set()方法。set()方法包含两个参数,分别表示变量名称和变量值。两个参数不可省略。
    当程序调用一个未定义或不可见的成员变量时,可以通过get()方法来读取变量值。get()方法有一个参数,表示要调用的变量名。
    注意:如果希望PHP调用这些魔术方法,那么首先必须在类中进行定义;否则PHP不会执行未创建的魔术方法。
    例:

<?php
     class myobject{
           private $type="";               private function get($name){
                  if(isset($this->$name)){                        echo "变量".$name."的值为:".$this->$name."<br />";
                     }                     else{                        echo "变量".$name."未定义,初始化为0<br />";                            $this->$name=0;
                     }
               }               private function set($name,$value){
                   if(isset($this->$name)){                         $this->$name=$value;                             echo "变量".$name."赋值为:".$value."<br />";
                      }else{                          $this->$name=$value;                             echo "变量".$name."被初始化为:".$value."<br />";
                      }
               }
        }        $mycomputer=new myobject();        $mycomputer->type="DIY";        $mycomputer->type;        $mycomputer->name;?>

(2)、call()方法
call()方法的作用是:当程序试图调用不存在或不可见的成员方法时,PHP会先调用call()方法来存储方法名及其参数。call()方法包含两个参数,即方法名和方法参数。其中,方法参数是以数组形式存在的。
例:

<?php
     class myobject{
            public function mydream(){
                     echo "调用的方法存在,直接执行此方法:<br />";
                }                public function call($method,$parameter){
                     echo "如果方法不存在,则执行call()方法。<br/>";                       echo "方法名为:".$method."<br />";                       echo "参数有:";
                       var_dump($parameter);
                }
        }        $exam=new myobject();        $exam->mydream();        $exam->mdreadm(&#39;how&#39;,&#39;what&#39;,&#39;why&#39;);?>

(3)、sleep()和wakeup()方法
使用serialize()函数可以实现序列化对象。就是将对象中的变量全部保存下来,对象中的类则只保存类名。在使用serialize()函数时,如果实例化的对象包含sleep()方法,则会执行sleep()方法。该方法可以清除对象并返回一个该对象中所有变量的数组。使用sleep()方法的目的是关闭对象可能具有的数据库连接等类似的善后工作。
Unserialize()函数可以重新还原一个被serialize()函数序列化的对象,wakeup()方法则是恢复在序列化中可能丢失的数据库连接及相关工作。
例:

<?php
     class myobject{
           private $type="DIY";               public function gettype(){
                   return $this->type;
               }               public function sleep(){
                   echo "使用serialize()函数将对象保存起来,可以存放到文本文件、数据库等地方。<br />";                      return $this;
               }               public function wakeup(){
                   echo "当需要该数据时,使用unserialize()函数对已序列化的字符串进行操作,将其转换回来对象。<br />";
               }
        }       $mybook=new myobject();       $i=serialize($mybook);       echo "序列化后的字符串:".$i."<br />";       $rebook=unserialize($i);       echo "还原后的成员变量:".$rebook->gettype();?>

(4)、tostring()方法
魔术方法这个作用是:当使用echo 或print输出对象时,将对象转化为字符串。
例:

<?php
     class myobject{
            private $type="DIY";                public function tostring(){
                    return $this->type;
                }
        }        $mycomputer=new myobject();        echo "对象\$mycomputer的值为:";        echo $mycomputer;?>

(5)、autoload()方法
   将一个独立、完整的类保存到一个PHP页中,并且文件名和类名保持一致,这是每个开发人员者需要养成的良好习惯。这样,在下次重复使用某个类时即能很轻易地找到它。但还有一个问题是让开发人员头疼不已的,如果要在一个页面中引进很多的类,需要使用include_once()函数或require_once()函数一个一个地引入。
   PHP5解决了这个问题,autoload()方法可以自动实例化需要使用的类。当程序要用到一个类,但该类还没有被实例化时,PHP5将使用autoload()方法,在指定的路径下自动查找和该类名称相同的文件。如果找到,程序则继续执行,否则,报告错误。
例:
Myobject.class.php页面代码如下:

<?php
     classmyobject{              private $cont;                public function construct($cont){
                    $this->cont=$cont;
                }                public function tostring(){
                    return $this->cont;
                }
        }?>

Index.php页面代码如下:

<?php
     functionautoload($class_name){                                  $class_path=$class_name.&#39;.class.php&#39;;                                    if(file_exists($class_path)){                                       include_once($class_path);
                                    }else{                                       echo "类路径错误。";
                                    }
                              }                              $mybook=new myobject("江山代有人才出    各领风骚数百年");                              echo $mybook;?>

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Déclaration:
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn