构造函数和析构函数

构造函数
PHP 5 允行开发者在一个类中定义一个方法作为构造函数。具有构造函数的类会在每次创建新对象时先调用此方法,所以非常适合在使用对象之前做一些初始化工作。
如果子类中定义了构造函数则不会隐式调用其父类的构造函数。要执行父类的构造函数,需要在子类的构造函数中调用parent::__construct()。

class BaseClass {
    function __construct()
    {
        print 'In BaseClass construct<br/>';
    }
}

class SubClass extends BaseClass {
    function __construct()
    {
        parent::__construct();
    }
}

$base = new BaseClass();
$sub = new SubClass();

析构函数
析构函数会在到某个对象的所有引用都被删除或者当对象被显式销毁时执行。

class MyDestructableClass {
    function __construct()
    {
        print 'in construct<br/>';
        $this->name = 'MyDestructableClass';
    }

    function __destruct()
    {
        // TODO: Implement __destruct() method.
        print 'Destroying ' . $this->name . '<br/>';
    }
}

$obj = new MyDestructableClass();

访问控制
由 public 所定义的类成员可以在任何地方被访问;
由 protected 所定义的类成员则可以被其所在类的子类和父类访问(当然,该成员所在的类也可以访问);
而由 private 定义的类成员则只能被其所在类访问。

class MyClass {
    public $public = 'Public';
    protected $protected = 'Protected';
    private $private = 'Private';

    public function printHello()
    {
        echo $this->public;
        echo '<br>';
        echo $this->protected;
        echo '<br>';
        echo $this->private;
    }
}

/*$obj = new MyClass();
echo $obj->public;
echo '<br>';
$obj->printHello();*/

class MyClass2 extends MyClass {
    // 可以对 public 和 protected 进行重定义,但private不能
    protected $protected = 'Protected2';

    public function printHello()
    {
        parent::printHello(); // TODO: Change the autogenerated stub
    }
}

$obj2 = new MyClass2();
echo $obj2->printHello();

对方法访问的控制

class MyClass {
    // 构造函数必须是 public
    public function __construct()
    {}

    public function MyPublic()
    {}

    protected function MyProtected()
    {}

    private function MyPrivate()
    {}

    public function foo ()
    {
        $this->MyPublic();
        $this->MyProtected();
        $this->MyPrivate();
    }
}

$obj = new MyClass();
$obj->MyPublic();
$obj->foo();

class MyClass2 extends MyClass {
    public function foo()
    {
        parent::foo(); // TODO: Change the autogenerated stub
/*        $this->MyPublic();
        $this->MyProtected();
        $this->MyPrivate(); // 这行会产生一个致命错误*/
    }
}

$obj2 = new MyClass2();
$obj2->foo();

对象继承
当扩展一个类,子类就会继承父类的所有公有和保护方法。但是子类的方法会覆盖父类的方法。

class foo
{
    public function printItem($string)
    {
        echo 'Foo: ' . $string . '<br/>';
    }

    public function printPHP()
    {
        echo 'PHP is great<br/>';
    }
}

class bar extends foo
{
    public function printItem($string)
    {
        echo 'Bar: ' . $string . '<br/>';
    }
}

$bar = new bar();
$bar->printItem('bazz');
$bar->printPHP();

范围解析操作符(::)
用于访问静态成员、方法和常量,或者覆盖类中的成员和方法。
当在类的外部访问这些静态成员、方法和常量时,必须使用类的名字。
self 和 parent 这两个特殊的关键字是用于在类的内部对成员或方法进行访问的。

// 在类的外部使用范围解析操作符
class MyClass {
    const CONST_VALUE = 'A constant value';
}

echo MyClass::CONST_VALUE;
class OtherClass extends MyClass {
    public static $my_static = 'static var';

    public static function doubleColon ()
    {
        echo self::$my_static;
        echo '<br/>';
        echo parent::CONST_VALUE;
    }
}

OtherClass::doubleColon();
class MyClass {
    protected function myFunc()
    {
        echo 'MyClass::myFunc()<br/>';
    }
}

class OtherClass extends MyClass {
    public function myFunc()
    {
        parent::myFunc(); // TODO: Change the autogenerated stub
        echo 'OtherClass::myFunc()';
    }
}

$other = new OtherClass();
$other->myFunc();

作业:

// 请定义一个动物类,然后定义一个小狗类继承动物类,里面定义一个吠叫的方法,最后实例化一条小狗,调用它吠叫的方法。
class Animal
{
    public function call()
    {
        echo 'Animal<br/>';
    }
}

class Dog extends Animal
{
    public function call()
    {
        parent::call();
        echo 'wang';
    }
}

$dog = new Dog();
$dog->call();

static关键字
声明类成员或方法为static,就可以不实例化类而直接访问
由于静态方法不需要通过对象即可调用,所以伪变量$this在静态方法中不可用。
静态属性不可以由对象通过->操作符来访问。
用::方式调用一个非静态方法会导致一个E_STRICT级别的错误。
静态属性只能被初始化为一个字符值或一个常量,不能使用表达式。 所以你可以把静态属性初始化为整型或数组,但不能指向另一个变量或函数返回值,也不能指向一个对象。

class Foo {
    public static $my_static = 'foo';

    public function staticValue() {
        return self::$my_static;
    }
}

$foo = new Foo();
echo $foo->staticValue();

class Bar extends Foo {
    public function fooStatic() {
        return parent::$my_static;
    }
}

$bar = new Bar();
echo $bar->fooStatic();
echo '<br/>';
echo Bar::$my_static;

抽象类

abstract class AbstractClass
{
    // 强制要求子类定义这些方法
    abstract protected function getValue();
    abstract protected function prefixValue($prefix);

    // 普通方法(非抽象方法)
    public function printOut() {
        print $this->getValue() . '<br/>';
    }
}

class ConcreteClass1 extends AbstractClass {
    public function getValue()
    {
        // TODO: Implement getValue() method.
        echo 'ConcreteClass1<br/>';
    }

    public function prefixValue($prefix)
    {
        // TODO: Implement prefixValue() method.
        echo "{$prefix}ConcreteClass1<br/>";
    }
}

class ConcreteClass2 extends AbstractClass {
    public function getValue()
    {
        // TODO: Implement getValue() method.
        echo 'ConcreteClass2<br/>';
    }

    public function prefixValue($prefix)
    {
        // TODO: Implement prefixValue() method.
        echo "{$prefix}ConcreteClass2<br/>";
    }
}

$concrete1 = new ConcreteClass1();
$concrete1->printOut();
$concrete1->prefixValue('FOO__');

$concrete2 = new ConcreteClass2();
$concrete2->printOut();
$concrete2->prefixValue('FOO__');
// 第二个抽象类的例子
abstract class AbstractClass
{
    // 我们的抽象方法仅需要定义需要的参数
    abstract protected function prefixName($name);
}

class ConcreteClass extends AbstractClass {
    // 我们的子类可以定义父类签名中不存在的可选参数
    public function prefixName($name, $separator = '.')
    {
        // TODO: Implement prefixName() method.
        if ($name == 'Pacman') {
            $prefix = 'Mr';
        } elseif ($name == 'Pacwoman') {
            $prefix = 'Mrs';
        } else {
            $prefix = '';
        }

        return "{$prefix}{$separator}{$name}<br/>";
    }
}

$concrete = new ConcreteClass();
echo $concrete->prefixName('Pacman');
echo $concrete->prefixName('Pacwoman');

接口
指定某个类必须实现哪些方法,但不需要定义这些方法的具体内容。
就像定义一个标准的类一样,但其中定义所有的方法都是空的。 接口中定义的所有方法都必须是public,这是接口的特性。

要实现一个接口,可以使用implements操作符。类中必须实现接口中定义的所有方法,否则会报一个fatal错误。
如果要实现多个接口,可以用逗号来分隔多个接口的名称。
实现多个接口时,接口中的方法不能有重名。
接口也可以继承,通过使用extends操作符。

PHP类是单继承,也就是不支持多继承,当一个类需要多个类的功能时,继承就无能为力了,为此PHP引入了接口技术

PHP类与面向对象(二)的更多相关文章

  1. 101 01 Android 零基础入门 02 Java面向对象 03 综合案例(学生信息管理) 02 案例分析及实现 05 通过方法实现学生类与专业类关联——方案二

    101 01 Android 零基础入门 02 Java面向对象 03 综合案例(学生信息管理) 02 案例分析及实现 05 通过方法实现学生类与专业类关联--方案二 本文知识点:通过方法实现学生类与 ...

  2. 2. 星际争霸之php面向对象(二)

    题记==============================================================================本php设计模式专辑来源于博客(jymo ...

  3. Javascript面向对象二

    Javascript面向对象二 可以通过指定原型属性来对所有的对象指定属性, Object.prototype.name="zhangsan"; Object.prototype. ...

  4. php入门 数据类型 运算符 语言结构语句 函数 类与面向对象

    php PHP-enabled web pages are treated just like regular HTML pages and you can create and edit them ...

  5. 第四节:详细讲解Java中的类和面向对象思想

    前言 大家好,给大家带来详细讲解Java中的类和面向对象思想的概述,希望你们喜欢 类和面向对象 在Java中怎样理解对象,创建对象和引用:什么是引用,对于基础学习的同学,要深入了解引用.示例:Stri ...

  6. ansible笔记(8):常用模块之系统类模块(二)

    ansible笔记():常用模块之系统类模块(二) user模块 user模块可以帮助我们管理远程主机上的用户,比如创建用户.修改用户.删除用户.为用户创建密钥对等操作. 此处我们介绍一些user模块 ...

  7. PHP之旅6 php的类与面向对象

    对于类与面向对象的问题其实刚开始不用非要弄懂原因,直接先这样用,用多了你就发现你会了.所以 学习面向对象就先去做.用的多了你就发现这个就是这样的. 对于类的理解,我自己的理解就是:有一个叫做类的东西里 ...

  8. python 教程 第九章、 类与面向对象

    第九章. 类与面向对象 1)    类 基本类/超类/父类被导出类或子类继承. Inheritance继承 Inheritance is based on attribute lookup in Py ...

  9. java - day005 - 数组工具类, 数组复制,二维数组,变量,方法, 面向对象

    1. java.util.Arrays  数组工具类    Arrays.toString (数组) 数组值链接字符串 Arrays.sort(数组) 基本类型: 优化的快速排序 引用类型: 优化的合 ...

  10. python 面向对象二 类和实例

    一.类和实例 面向对象最重要的概念就是类(Class)和实例(Instance),必须牢记类是抽象的模板,比如Student类,而实例是根据类创建出来的一个个具体的“对象”,每个对象都拥有相同的方法, ...

随机推荐

  1. ZT 趋势移动安全apk

    趋势移动安全 应用截图   应用简介 趋势移动安全( Mobile Security) 是一款专业的Android移动安全软件.利用趋势科技世界领先的云安全技术,保护用户避免被移动恶意程序骚扰,避免个 ...

  2. [转] swf文件加密基础

    本来打算下班回来就写这个东西,一方面算是对今天学习的一个笔记记录,另外一方面,给一些朋友普及一些swf文件加密基础知识.之所以说是基础,那是因为我也是刚学习了一点,灰常的基础.不过晚上看了一会我是传奇 ...

  3. 亿级Web系统的高容错性实践

    亿级Web系统的高容错性实践 背景介绍 大概三年前,我在腾讯负责的活动运营系统,因为业务流量规模的数倍增长,系统出现了各种各样的异常,当时,作为开发的我,7*24小时地没日没夜处理告警,周末和凌晨也经 ...

  4. 怎样按xc或yc转正视图

    extern void create_view(void) { tag_t wcs_id,matrix_id; double mtx[9],wcs_pt[3]; double x_axis[3]={1 ...

  5. java常用类

    String 字符串类 System 可得到系统信息 Runtime类 StringBuilder(StringBuffer)类 Thread 线程类 Math 与数学有关的工具类 Date 日期类( ...

  6. nginx 从vagant挂载目录中加载nginx.conf配置进行开机启动

    nginx从vagrant挂载目录中读取配置启动,将nginx加入开机启动项!开机启动的时候nginx会因为加载不了配置导致启动失败! 原因是开机启动nginx服务在vagrant挂载之前,导致无法正 ...

  7. Jade之Includes

    Includes jade允许利用include将其他文件(支持filters所支持的类型)中的代码嵌入当前代码中. jade: //- index.jade doctype html html in ...

  8. 理解RHEL上安装oracle的配置参数

    无论安装什么版本的oracle,在安装之前,都需要配置 /etc/pam.d/login   /etc/profile   /etc/security/limits.conf这三个文件 那这三个文件究 ...

  9. 斯坦福第十九课:总结(Conclusion)

    19.1  总结和致谢 欢迎来到<机器学习>课的最后一段视频.我们已经一起学习很长一段时间了.在最后视频中,我想快速地回顾一下这门课的主要内容,然后简单说几句想说的话. 作为这门课的结束时 ...

  10. github 仓库管理

    一.远程仓库有master和dev分支1. 克隆代码 git clone https://github.com/master-dev.git # 这个git路径是无效的,示例而已 2. 查看所有分支 ...