php 面向对象笔记

WBOY
WBOYasal
2016-06-23 13:04:451004semak imbas

===================================================
PHP的面向对象:

对象是客观存在的一个实体。
类是对对象抽象的一个描述。

概念:对象(实体)、类、 类与对象的关系。

oop面向对象编程的特点:封装、继承、多态

类和对象的关系:
类的实例化结果就是一个对象(使用new关键字)
对对象的抽象描述就是一个类

===================================================
一、如何定义一个类,和使用

1.1 语法格式:
[修饰符] class 类名 [extends 父类] [implements 接口1[,接口2...]]{
【成员属性】定义变量
【成员方法】定义函数
}

1.2 成员属性格式:
修饰符 $变量名[=默认值]; 如:public $name="zhangsan";
注意:成员属性不可以是带运算符的表达式、变量、方法或函数的调用。 如:
public $var3 = 1+2;
public $var4 = self::myStaticMethod();
public $var5 = $myVar;
正确定义方式:
public $var6 = 100; //普通数值(4个标量:整数、浮点数、布尔、字串)
public $var6 = myConstant; //常量
public $var7 = self::classConstant; //静态属性
public $var8 = array(true, false); //数组

常用的属性修饰符:public、protected、private、static、var

1.3 成员方法格式:
[修饰符] function 方法名(参数..){
[方法体]
[return 返回值]
}

常用的方法修饰符:public、protected、private、static、abstract、final

1.4 对象:类的实例化就会产生一个对象。
如: $p = new Person();

1.5 $this 关键字: 表示自己,表示当前使用对象,*哪个对象调用代表哪个
1.我们在类中调用自己的成员属性或方法都是使用 $this->调用。
2.注意:静态方法不可以使用$this关键字

===================================================
二、构造方法和析构方法

2.1. 构造方法:
1.当我们通过new关键字来创建一个对象时,第一个自动执行的方法称为构造方法。

2.方法名__construct(); 主要用于初始化对象。
(在php4.0时可使用与类名重名的方法作为构造方法)

2.2. 析构方法:当这个对象被销毁时最后自动调用的方法,称为析构方法。
1.__destruct(); 目的是释放资源(如关闭连接、文件,释放资源)

2.对象被销毁时最后自动调用

===================================================
三、封装(访问控制)

3.1 封装:就是将"属性" private或protected 私有,并提供公有的setter放置与getter取值方法

public(公有) protected(受保护) private(私有)
===========================================================
在本类中 Y Y Y
在家族中 Y Y N
在类外边 Y N N

public 类外需要调用此类内的方法或属性时使用
protected 类外不需要调用,而家族内需要使用
private 用于安全信息,只用于当前类

===================================================
四、 重载

4.1 属性重载中的四个魔术方法:__set() __get() __isset() __unset()

__get():当我们直接输出一个对象中的非公有属性时会自动调用的方法,
并将属性名以第一个参数传进去。
__get($name){...} //外部:$m->priAge;

__set():当我们直接设置一个对象中的非公有属性时会自动调用的方法,
并将属性名以第一个参数,值作为第二参数传进去。
__set($name,$value){...} //外部:$m->priAge = 100;

__isset():验证一个对象属性isset($this->name) 或 empty($this->name)时, $this->name是一个非公有属性时,自动调用此方法。
public function __isset($param){
return isset($this->$param);
}

__unset()当销毁一个对象属性unset($this->name), $this->name是一个非公有属性时,自动调用此方法。
public function __unset($param){
unset($this->$param);
}

4.2 方法的重载:
mixed __call ( string $name , array $arguments )
mixed __callStatic ( string $name , array $arguments ) php5.3.0支持

1.当调用一个不可访问方法(如未定义,或者不可见)时,__call() 会被调用。
2.当在静态方法中调用一个不可访问方法(如未定义,或者不可见)时,__callStatic() 会被调用。

第一个参数表示方法名,第二参数表示调用时的参数列表(数组类型)

class A {
//调用不存在的方法
public function __call($methodName,$args) {
echo $methodName;
print_r($args);
}

//调用不存在的静态方法
static public function __callStatic($methodName,$args) {
echo '
'.$methodName;
print_r($args);
}
}
$a = new A;
$a->say('a','b');
A::test('c','d');

===================================================
五、 继承

5.1 继承:extends
*继承的目的:为了减少重复的代码。

假如B类继承A类,那么就继承了A中所有非私有属性和方法.其中A叫父类(基类)B叫子类(派生类)。
class B extends A{
....
}

1.*parent:若子类出现覆盖父类的方法,那么想调用被覆盖掉的父类方法,并可以跟子类的方法融合为一起

格式:parent::父类方法 | 父类名称::父类方法

class A{
public function __construct(){
//....
}
}
class B extends A{
public function __construct(){
parent::__construct();//注意要调用一下父类的构造方法
//....
}
}

2.继承的特性
1、PHP支持单继承,一类只能继承一个父类
2、PHP支持多层继承,可以有子孙类
3、子类会继承父类所有属性和方法,但是私有的属性和方法不能调用

3.属性的继承
1、私有的属性不会被覆盖,其他会被覆盖。
2、属性的修饰只能更加松散,不能更加严谨。 松散性:public > protected > private
3、如果子类中有与父类同名的属性,父类有public $name 子类只能使用public修饰
3、如果子类中有与父类同名的属性,父类有protected $name 子类能使用protected,public修饰

4.方法的继承
1、所有方法可以被继承,但是私有的方法不能被使用
2、如果子类有与父类同名的方法,那么除了私有的方法,其他都被覆盖掉。
3、如果子类有与父类同名的方法,方法的修饰只能更加松散,不能更严谨。

===================================================
六、 final、static和const
*内存逻辑上:常量区(常量、静态属性)、代码区(类、函数)、栈内存(普通的小的变量)、堆内存(大的变量)

6.1 final关键字:主要用于修饰类与成员方法
目的:一是为了安全,二是没有必要

1.使用final关键字修饰类,表示这个类不能被继承,防止有子类的覆盖

2.使用final关键字修饰的方法,不可以在子类中被覆盖(重写)。

3.使用final关键字不能修饰属性

final class B {
final public function A {
..
}
}
---------------------------------------------------------------
6.2 static关键字:表示静态的意思,用于修饰类的属性和方法

static关键字修饰 属性/方法 称为 静态属性/静态方法,
注意:静态方法在实例化后的对象不可以访问 //$对象名->静态方法名

1.可以不用new(实例化)就可以直接使用方法或属性
*外部调用:类名::方法名 | 类名::属性名
*类中调用:self::静态属性名 | self::静态方法名

*2.静态属性是共享的。也就是调用多个对象也是指向同一个地址

3.在静态方法中不可以使用非静态的内容。就是不让使用$this

4.在一个类的方法中若没有出现$this的调用,默认此方法为静态方法。

5.static不能修饰类

---------------------------------------------------------------
6.21 设计模式。目的:节省资源

1、单例设计模式 2、工厂设计模式 3、 抽象工厂设计模式

class Single
{
//定义一个静态属性,用来存放对象
static private $obj = null;

//私有化构造方法。不能在类的外部实例化对象
private function __construct() {}

//实例化对象
static public function test() {
//如果没有对象,我们就允许它new一个对象
if(!self::$obj) {
self::$obj = new Single();//创建对象
}

//有了对象,就直接返回之前的对象
return self::$obj;
}

//单例不允许克隆
public function __clone() {
trigger_error('错误', E_USER_ERROR);
}
}

$s1 = Single::test(); var_dump($s1);
$s2 = Single::test(); var_dump($s2); //调用同一个对象

$s3 = clone $s2; //报错

---------------------------------------------------------------
6.3. const关键字: 在类中修饰成员属性,将其定义成常量(不可修改的),

1.一般要求常量名都是大写的,没有“$”符 没有其他修饰符(public)

2.定义格式:const 成员常量名="值";

3.使用:
*外部调用:类名::成员常量名;
*类中调用:self::成员常量名;

---------------------------------------------------------------
6.4. 检测当前对象是属于谁的类(家族类也是)

1.instanceof 用于确定一个 PHP 变量是否属于某一类 class 的实例:

class A {}
class B extends A {}
$b = new B;
if($b instanceof A) {echo 'y';}

2.is_a — 如果对象属于该类或该类是此对象的父类则返回 TRUE

===================================================
七、 类型约束

1. 类型约束可以使用的类型是:数组,对象,回调函数。
若指定的一个类名,那么可传入本类及子类的对象进去。
可以使用的约束类型:(复合类型)数组array,类名、抽象类名、接口名

class MyClass
{
//第一个参数必须为类OtherClass的一个对象
public function test(OtherClass $otherclass) {
**可以传入别的类,调用别的类的方法和属性
echo $otherclass->var;

//同这里
if( $otherclass instanceof Hanzi){
$otherclass->qinqin();
}
}

//第一个参数必须为数组
public function test_array(array $input_array) {
print_r($input_array);
}
}

===================================================
八、其他魔术方法:

1. 对象复制clone 克隆一个对象,因为对象属于引用类型,普通的“=”号属于引用赋值,
所有需要clone来复制一份。
魔术方法:__clone() 当执行clone克隆时会自动调用的方法。

class A {
public function __clone() {
//自动执行
}
}

$a = new A;
$b = clone $a; //克隆,是两个对象

-----------------------------------------------------------------
2. __toString()方法:魔术方法,当我们直接要输出一个对象时,如echo $a,print $a,
那么会自动调用的方法。
注意:__toString()方法必须返回一个字串类型的值。

class A {
public function __toString() {
return '这是一个对象,不允许ehco,print...';
}
}
$a = new A;
echo $a; //报错

-----------------------------------------------------------------
3. *自动加载"类函数"__autoload(类名):
当new 实例化一个对象时,这个类若不存在,则自动调用此函数,并将类名存入参数
我可以使用这个实现类的自动加载。

function __autoload($className) {
include './autoload/'.$className.'.class.php';
}

$a = new A;
$a->say(); //say

//在 autoload/A.class.php下
class A{
public function say() {
echo 'say';
}
}

-----------------------------------------------------------------
4. __invoke把对象当成函数使用的时候自动调用

class A {
public function __invoke() {
echo 'aa';
}
}

$a = new A;
$a(); //aa

===================================================
九、 对象序列化(串行化)(webservice XML)(在对象持久化存储、和传输中使用)

serialize() -- 序列化
unserialize() -- 反序列化

1.php里面的值都可以使用函数serialize()来返回一个包含字节流的字符串来表示。
2.unserialize()函数能够重新把字符串变回php原来的值。
3.序列化一个对象将会保存对象的所有变量,但是不会保存对象的方法,只会保存类的名字。

1.__sleep(): 是执行串行化时自动调用的方法,目的是实现资源类型的属性的关闭操作。
@@sleep方法需要返回一个数组,其中数组中的值是序列化时要保留的属性名
2.__wakeup():是在执行反串行化时自动调用的方法,目的是实现资源属性的打开(sleep方法关闭的资源)

//实例:
class A {
public $name1 = 'xiaohei';
public $name2 = 'xiaobai';

public function demo() {
return 'aa';
}

//序列化自动调用的方法
public function __sleep() {
return array('name2'); //只想存储name2,就返回name2
}

//反序列化自动调用的方法
public function __wakeup() {
echo '###';
}
}

$a = new A;

//序列化
$data = serialize($a);

//存入文件
file_put_contents('./1.txt', $data);

//获得数据
$str = file_get_contents('./1.txt');

echo $str; //O:1:"A":1:{s:4:"name";s:7:"xiaohei";}

//反序列化
$tmp = unserialize($str);

print_r($tmp); //A Object ( [name] => xiaohei )

-------------------------------------------------------------------------------------
JSON格式的字符串进行编码/解码

json_encode:对变量进行 JSON 编码
json_decode:对 JSON 格式的字符串进行编码

$arr = array('id'=>1, 'name'=>'jack');

$str1 = json_encode($arr);
echo $str1; //{"id":1,"name":"jack"}

$a = json_decode($str1, true); //加true返回array,否则返回object
echo $a; //Array ( [id] => 1 [name] => jack )

===================================================
十、 抽象类:

目的:抽象类对于子类(实现类),有一个约束的作用,

特点:
//1.定义抽象类
abstract class A {
//2.定义抽象方法
abstract public function test1();
abstract public function test2();

//3.可以有普通方法和属性,不需要子级实现
public function demo() {}
public $name = 'aa';

//4.不能定义抽象属性 Fatal error
abstract public $age = 18;
}

class B extends A {
//5.@@继承了抽象类的"普通类"必须实现抽象类的"所有抽象方法"
public function test1() {}
public function test2() {}

//6.包含了抽象方法的类必须是抽象类,Fatal error
abstract function test3() {}
}

abstract class C extends A {
//7.不能覆盖抽象方法,Fatal error
abstract public function test1() {}

//8.可以继承抽象类的同时添加抽象方法
abstract public function test3();
}

//8.不能实例化,Fatal error
$c = new C;

===================================================
十一、 接口:

目的:同抽象类相同,是更具体的抽象类
假如一个抽象类中所有的方法都是抽象的,那么我们可以使用另外一种方式定义:接口
接口使用关键字interface来定义,接口中只能有常量与抽象方法。

//1.定义接口
interface A {
//2.接口不能有普通属性,Fatal error
public $name = 'aa';

//3.接口不能有普通方法,Fatal error
public function test() {}

//3.但可以定义常量
const HOST = 'aa';

//4.定义接口方法,不需要abstract
public function test1();

}

interface B {
//5.接口之间是继承关系,重复定义Fatal error
const HOST = 'bb';

public function test2();
}

class C {}

//6.可以同时继承"单"类、实现"多"接口(先继承,后实现)
class D extends C implements A,B {
//7.@@继承了抽象类的"普通类"必须实现抽象类的"所有抽象方法"
public function test1() {}
public function test2() {}
}

抽象类和接口总结:
1.目的:制定规范,为了限制其子类
2.选择:如果规范中需要有"普通方法"就选择抽象类
3.区别:
1.定义关键字不同。抽象类:abstract 接口:interface
2.继承/实现关键字不同。抽象类:extends 接口:implements
3.抽象类可以有普通方法,接口不能有普通方法
4.抽象类只能被单继承,接口可以多实现
5.接口不能有成员属性,只有常量与抽象方法

===================================================
十二、多态

定义:对于“同一个方法”,传入“不同对象”,实现了不同的效果

class Meizi {
public function meet($obj) {
//限制了对象必须有qinqin()这个方法
if( $obj instanceof Hanzi){
//同一个方法
$obj->qinqin();
}else{
echo '你不会亲亲这个功能!';
}
}
}

//抽象类
abstract class Hanzi
{
abstract public function qinqin();
}

//大屌丝类
class DDS extends Hanzi {
public function qinqin() {
echo '被妹子打了一巴掌,这就是DDS的命运';
}
}

//高富帅类
class GFS extends Hanzi {
public function qinqin() {
echo '不仅仅亲亲,还啪啪啪,这是高富帅的命运';
}
}

//萌妹子类
class MengMeizi {}

$m = new Meizi;
$dds = new DDS;
$gfs = new GFS;
$mmz = new MengMeizi;

//传入对象
echo $m->meet($dds); //实现了DDS的效果
echo $m->meet($gfs); //实现了GFS的效果

//没有限制
echo $m->meet($mmz); //你不会亲亲!

@@多态的两个实例:
./1.通过接口实现多态的应用.php
./2.通过普通类实现多态的应用.php

===================================================
十三、异常处理:
在php5中有一种新的错误处理机制--异常处理:(采用面向对象方式的)
涉及的类:Exception异常类
Exception类结构:
class Exception
{
protected $message = 'Unknown exception'; // 异常信息
protected $code = 0; // 用户自定义异常代码
protected $file; // 发生异常的文件名
protected $line; // 发生异常的代码行号

function __construct($message = null, $code = 0);

final function getMessage(); // 返回异常信息
final function getCode(); // 返回异常代码
final function getFile(); // 返回发生异常的文件名
final function getLine(); // 返回发生异常的代码行号
final function getTrace(); // backtrace() 数组
final function getTraceAsString(); // 已格成化成字符串的 getTrace() 信息

/* 可重载的方法 */
function __toString(); // 可输出的字符串
}

使用:
try{
throw new Exception("年龄不可以为负数"); //异常抛出
}catch(Exception $e){
//异常处理
echo $e->getmessage();
}

实例:
function addUser($username) {
if($username="xiaobai") {
//添加成功
}else {
//添加失败
throw new Exception('添加失败');
}
}

function updateUser($username) {
if($username = "xiaohei") {
//修改成功
}else {
//修改失败
throw new Exception('修改失败');
}
}

try {
addUser('xiaobaia');

updateUser('xiaohei');
}catch(Exception $e) {

echo '失败信息是:'.$e->getMessage();
}

@@当捕获到一个异常后,try()块里面的后续代码不继续执行

Kenyataan:
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn