9.封裝性
封裝性是物件導向程式設計中的三大特性之一,封裝性就是把物件的屬性和服務結合成一個獨立的
相同單位,並儘可能隱藏物件的內部細節,包含兩個意義:1.把物件的全部屬性和全部服務結合在一
起,形成一個不可分割的獨立單位(即物件)。 2.資訊隱蔽,即盡可能隱藏物件的內部細節,對外形
成一個邊界〔或者說形成一道屏障〕,只保留有限的對外接口使之與外部發生聯繫。
封裝的原則在軟體上的反映是:要求使物件以外的部分不能隨意訪問物件的內部資料(屬性),
從而有效的避免了外部錯誤對它的"交叉感染",使軟體錯誤能夠局部化,大幅減少查錯和排錯的難
度。
用個實例來說明吧,假如某個人的物件中有年齡和工資等屬性,像這樣個人隱私的屬性是不想
讓其它人隨意就能獲得到的,如果你不使用封裝,那麼別人想知道就能得到,但是如果你封裝上之
後別人就沒有辦法獲得封裝的屬性,除非你自己把它說出去,否則別人沒有辦法得到。
再比如說,個人電腦都有一個密碼,不想讓其它人隨意的登陸,在你的電腦裡面拷貝和粘貼。
還有就是像人這個對象,身高和年齡的屬性,只能是自己來增長,不可以讓別人隨意的賦值等等。
使用private 這個關鍵字來封裝屬性和方法:
原來的成員:
var $name; //声明人的姓名 var $sex; //声明人的性别 var $age; //声明人的年龄 function run(){… … .}
改成封裝的形式:
private $name; //把人的姓名使用private 关键字进行封装 private $sex; //把人的性别使用private 关键字进行封装 private $age; //把人的年龄使用private 关键字进行封装 private function run(){… … } //把人的走路方法使用private 关键字进行封装
注意:只要是成員必須到前面它的屬性就要去原來有掉原有掉原有掉原有掉原有的關鍵字“var”。
透過private 就可以把人的成員(成員屬性和成員方法)封裝上了。封裝上的成員就不能被類外
面直接存取了,只有物件內部自己可以存取;下面的程式碼會產生錯誤:
class Person { //下面是人的成员属性 private $name; //人的名字,被private封装上了 private $sex; //人的性别, 被private封装上了 private $age; //人的年龄, 被private封装上了 //这个人可以说话的方法 function say() { echo "我的名字叫:".$this->name." 性别:".$this->sex." 我的年龄是: ".$this->age."<br>"; }/ /这个人可以走路的方法, 被private封装上了 private function run() { echo "这个人在走路"; } } //实例化一个人的实例对象 $p1=new Person(); //试图去给私有的属性赋值, 结果会发生错误 $p1->name="张三"; $p1->sex="男"; $p1->age=20; //试图去打印私有的属性, 结果会发生错误 echo $p1->name.”<br>”; echo $p1->sex.”<br>”; echo $p1->age.”<br>” //试图去打印私有的成员方法, 结果会发生错误 $p1->run();
輸出結果為:
Fatal error: Cannot access private property Person::$name Fatal error: Cannot access private property Person::$sex Fatal error: Cannot access private property Person::$age Fatal error: Cannot access private property Person::$name Fatal error: Call to private method Person::run() from context ''
從上面的實例可以看到,私有的成員是不能被外部訪問的,因為私有成員只能在本對象內部自
己訪問,比如,$p1 這個對象自己想把他的私有屬性說出去,在say()這個方法裡面訪問了私有屬性,
這樣是可以。 (沒有增加任何存取控制,預設的是public 的,任何地方都可以存取)
//这个人可以说话的方法, 说出自己的私有属性,在这里也可以访问私有方法 function say() { echo "我的名字叫:".$this->name." 性别:".$this->sex." 我的年龄是: ".$this->age."<br>"; //在这里也可以访问私有方法 //$this->run(); } 因为成员方法say()是公有的, 所以我们在类的外部调用say()方法是可以的,改变上面的代码; class Person { //下面是人的成员属性 private $name; //人的名字,被private封装上了 private $sex; //人的性别, 被private封装上了 private $age; //人的年龄, 被private封装上了 //定义一个构造方法参数为私有的属性姓名$name、性别$sex和年龄$age进行赋值 function __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>"; } } //通过构造方法创建3个对象$p1、p2、$p3,分别传入三个不同的实参为姓名、性别和年龄 $p1=new Person(“张三”,”男”, 20); $p2=new Person(“李四”,”女”, 30); $p3=new Person(“王五”,”男”, 40); //下面访问$p1对象中的说话方法 $p1->say(); //下面访问$p2对象中的说话方法 $p2->say(); //下面访问$p3对象中的说话方法 $p3->say();
輸出結果為:
我的名字叫:张三性别:男我的年龄是:20 我的名字叫:李四性别:女我的年龄是:30 我的名字叫:王五性别:男我的年龄是:40
因為構造方法是預設的公有方法(構造方法不要設定成私有的),所以在類別的外面可以訪問到,
這樣就可以使用構造方法創建對象, 另外構造方法也是類別裡面的函數,所以可以用構造方法給私
有的屬性賦初值。 Say()的方法是預設公有的, 所以在外面也可以存取的到, 說出他自己的私有屬
性。
從上面的例子我們可以看到,私有的成員只能在類別的內部使用,不能被類別外部直接來訪問,
但是在類別的內部是有權限存取的,所以有時我們需要在類的外面給私有屬性賦值和讀取出來,也
就是給類的外部提供一些可以訪問的接口,上例中構造方法就是一種賦值的形式,但是構造方法只
是在創建對象的時候賦值,如果我們已經有一個存在的對象了,想對這個存在的對象賦值,這個時
候,如果你還使用構造方法傳值的形式傳值,那麼就創建了一個新的對象,並不是這個已存在的對
象了。所以我們要對私有的屬性做一些可以被外部訪問的接口,目的就是可以在對象存在的情況下,
改變和訪問屬性的值,但要注意,只有需要讓外部改變的屬性才這樣做,不想讓外面存取的屬性是
不做這樣的介面的,這樣就能達到封裝的目的,所有的功能都是物件自己來完成,給外面提供盡量
少的操作。
如果給類外部提供接口,可以為私有屬性在類外部提供設置方法和獲取方法,來操作私有屬性.
例如:
prvate $age; //私有的属性年龄 function setAge($age) //为外部提供一个公有设置年龄的方法 { if($age<0 || $age>130) //在给属性赋值的时候,为了避免非法值设置给属性 return; $this->age=$age; }f unction getAge() //为外部提供一个公有获取年龄的方法 { return($this->age); }
上面的方法是为一个成员属性设置和获取值, 当然你也可以为每个属性用同样的方法对其进行
赋值和取值的操作,完成在类外部的存取工作。
10.__set() __get() __isset() __unset()四个方法的应用
一般来说,总是把类的属性定义为private,这更符合现实的逻辑。但是,对属性的读取和赋值
操作是非常频繁的,因此在PHP5 中,预定义了两个函数“__get()”和“__set()”来获取和赋值其
属性,以及检查属性的“__isset()”和删除属性的方法“__unset()”。
上一节中,我们为每个属性做了设置和获取的方法,在PHP5 中给我们提供了专门为属性设置
值和获取值的方法,“__set()”和“__get()”这两个方法,这两个方法不是默认存在的,而是我们手
工添加到类里面去的,像构造方法(__construct())一样, 类里面添加了才会存在,可以按下面的方式
来添加这两个方法,当然也可以按个人的风格来添加:
//__get()方法用来获取私有属性
private function __get($property_name) { if(isset($this->$property_name)) { return($this->$property_name); }else { return(NULL); } }/ /__set()方法用来设置私有属性 private 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()方法用来获取私有属性 private function __get($property_name) { echo "在直接获取私有属性值的时候,自动调用了这个__get()方法<br>"; if(isset($this->$property_name)) { return($this->$property_name); }e lse { return(NULL); } }/ /__set()方法用来设置私有属性 private function __set($property_name, $value) { echo "在直接设置私有属性值的时候,自动调用了这个__set()方法为私有属性赋值<br>"; $this->$property_name = $value; } } $p1=new Person(); //直接为私有属性赋值的操作, 会自动调用__set()方法进行赋值 $p1->name="张三"; $p1->sex="男"; $p1->age=20; //直接获取私有属性的值, 会自动调用__get()方法,返回成员属性的值 echo "姓名:".$p1->name."<br>"; echo "性别:".$p1->sex."<br>"; echo "年龄:".$p1->age."<br>"; ?>
程序执行结果:
在直接设置私有属性值的时候,自动调用了这个__set()方法为私有属性赋值 在直接设置私有属性值的时候,自动调用了这个__set()方法为私有属性赋值 在直接设置私有属性值的时候,自动调用了这个__set()方法为私有属性赋值 在直接获取私有属性值的时候,自动调用了这个__get()方法 姓名:张三 在直接获取私有属性值的时候,自动调用了这个__get()方法 性别:男 在直接获取私有属性值的时候,自动调用了这个__get()方法 年龄:20
以上代码如果不加上__get()和__set()方法,程序就会出错,因为不能在类的外部操作私有成员,
而上面的代码是通过自动调用__get()和__set()方法来帮助我们直接存取封装的私有成员的。
__isset() 方法:在看这个方法之前我们看一下“isset()”函数的应用,isset()是测定变量是否设
定用的函数,传入一个变量作为参数,如果传入的变量存在则传回true,否则传回false。那么如果
在一个对象外面使用“isset()”这个函数去测定对象里面的成员是否被设定可不可以用它呢?分两种
情况,如果对象里面成员是公有的,我们就可以使用这个函数来测定成员属性,如果是私有的成员
属性,这个函数就不起作用了,原因就是因为私有的被封装了,在外部不可见。那么我们就不可以
在对象的外部使用“isset()”函数来测定私有成员属性是否被设定了呢?可以,你只要在类里面加上
一个“__isset()”方法就可以了,当在类外部使用”isset()”函数来测定对象里面的私有成员是否被设
定时, 就会自动调用类里面的“__isset()”方法了帮我们完成这样的操作,“__isset()”方法也可以
做成私有的。你可以在类里面加上下面这样的代码就可以了:
private function __isset($nm) { echo "当在类外部使用isset()函数测定私有成员$nm时,自动调用<br>"; return isset($this->$nm); }
__unset()方法:
看这个方法之前呢,我们也先来看一下“unset()”这个函数,“unset()”这个函
数的作用是删除指定的变量且传回true,参数为要删除的变量。那么如果在一个对象外部去删除对
象内部的成员属性用“unset()”函数可不可以呢,也是分两种情况,如果一个对象里面的成员属性
是公有的,就可以使用这个函数在对象外面删除对象的公有属性,如果对象的成员属性是私有的,
我使用这个函数就没有权限去删除,但同样如果你在一个对象里面加上“__unset()”这个方法,就
可以在对象的外部去删除对象的私有成员属性了。在对象里面加上了“__unset()”这个方法之后,
在对象外部使用“unset()”函数删除对象内部的私有成员属性时,自动调用“__unset()”函数来帮
我们删除对象内部的私有成员属性,这个方法也可以在类的内部定义成私有的。在对象里面加上下
面的代码就可以了:
private function __unset($nm) { echo "当在类外部使用unset()函数来删除私有成员时自动调用的<br>"; unset($this->$nm); } 我们来看一个完整的实例: <?php class Person { //下面是人的成员属性 private $name; //人的名字 private $sex; //人的性别 private $age; //人的年龄 //__get()方法用来获取私有属性 private function __get($property_name) { if(isset($this->$property_name)) { return($this->$property_name); }else { return(NULL); } }/ /__set()方法用来设置私有属性 private function __set($property_name, $value) { $this->$property_name = $value; }/ /__isset()方法 private function __isset($nm) { echo "isset()函数测定私有成员时,自动调用<br>"; return isset($this->$nm); }/ /__unset()方法 private function __unset($nm) { echo "当在类外部使用unset()函数来删除私有成员时自动调用的<br>"; unset($this->$nm); } } $p1=new Person(); $p1->name="this is a person name"; //在使用isset()函数测定私有成员时,自动调用__isset()方法帮我们完成,返回结果为true echo var_dump(isset($p1->name))."<br>"; echo $p1->name."<br>"; //在使用unset()函数删除私有成员时,自动调用__unset()方法帮我们完成,删除name私有属性 unset($p1->name); //已经被删除了, 所这行不会有输出 echo $p1->name; ?>
输出结果为:
isset()函数测定私有成员时,自动调用 bool(true) this is a person name 当在类外部使用unset()函数来删除私有成员时自动调用的 __set()、__get()、__isset()、__unset() 这四个方法都是我们添加到对象里面的,在需要时自动调 用的,来完成在对象外部对对象内部私有属性的操作
以上就是php面向对象学习教程5的内容,更多相关内容请关注PHP中文网(www.php.cn)!