当前位置: 编程技术>php
本页文章导读:
▪php面向对象全攻略 (六)__set() __get() __isset() __unset()的用法
10.__set() __get() __isset() __unset()四个方法的应用 一般来说,总是把类的属性定义为private,这更符合现实的逻辑。但是,对属性的读取 和赋值操作是非常频繁的,因此在PHP5 中,预定义了两个函.........
▪php面向对象全攻略 (五) 封装性
9.封装性 封装性是面向对象编程中的三大特性之一,封装性就是把对象的属性和服务结合成一个 独立的相同单位,并尽可能隐蔽对象的内部细节,包含两个含义:1.把对象的全部属性和全 部.........
▪php面向对象全攻略 (四)构造方法与析构方法
8.构造方法与析构方法 大多数类都有一种称为构造函数的特殊方法。当创建一个对象时,它将自动调用构造函 数,也就是使用new 这个关键字来实例化对象的时候自动调用构造方法。 构造函.........
[1]php面向对象全攻略 (六)__set() __get() __isset() __unset()的用法
来源: 互联网 发布时间: 2013-11-30
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);
}else {
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()这四个方法都是我们添加到对象里面的,在需要时
自动调用的,来完成在对象外部对对象内部私有属性的操作。
一般来说,总是把类的属性定义为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);
}else {
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()这四个方法都是我们添加到对象里面的,在需要时
自动调用的,来完成在对象外部对对象内部私有属性的操作。
[2]php面向对象全攻略 (五) 封装性
来源: 互联网 发布时间: 2013-11-30
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;
}
function getAge(){ //为外部提供一个公有获取年龄的方法
return($this->age);
}
上面的方法是为一个成员属性设置和获取值,当然你也可以为每个属性用同样的方法对
其进行赋值和取值的操作,完成在类外部的存取工作
封装性是面向对象编程中的三大特性之一,封装性就是把对象的属性和服务结合成一个
独立的相同单位,并尽可能隐蔽对象的内部细节,包含两个含义: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;
}
function getAge(){ //为外部提供一个公有获取年龄的方法
return($this->age);
}
上面的方法是为一个成员属性设置和获取值,当然你也可以为每个属性用同样的方法对
其进行赋值和取值的操作,完成在类外部的存取工作
[3]php面向对象全攻略 (四)构造方法与析构方法
来源: 互联网 发布时间: 2013-11-30
8.构造方法与析构方法
大多数类都有一种称为构造函数的特殊方法。当创建一个对象时,它将自动调用构造函
数,也就是使用new 这个关键字来实例化对象的时候自动调用构造方法。
构造函数的声明与其它操作的声明一样,只是其名称必须是__construct( )。这是PHP5 中
的变化,以前的版本中,构造函数的名称必须与类名相同,这种在PHP5 中仍然可以用,但
现在以经很少有人用了,这样做的好处是可以使构造函数独立于类名,当类名发生改变时不
需要改相应的构造函数名称了。为了向下兼容,如果一个类中没有名为__construct( )的方法,
PHP 将搜索一个php4 中的写法,与类名相同名的构造方法。
格式:function __construct ( [参数] ) { ... ... }
在一个类中只能声明一个构造方法,而是只有在每次创建对象的时候都会去调用一次构
造方法,不能主动的调用这个方法,所以通常用它执行一些有用的初始化任务。比如对成属
性在创建对象的时候赋初值。
代码片段
<?
//创建一个人类
class Person{
//下面是人的成员属性
var $name; //人的名字
var $sex; //人的性别
var $age; //人的年龄
//定义一个构造方法参数为姓名$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
如图:
析构函数:
与构造函数相对的就是析构函数。析构函数是PHP5 新添加的内容,在PHP4 中没有析
构函数。析构函数允许在销毁一个类之前执行的一些操作或完成一些功能,比如说关闭文件,
释放结果集等,析构函数会在到某个对象的所有引用都被删除或者当对象被显式销毁时执行,
也就是对象在内存中被销毁前调用析构函数。与构造函数的名称类似,一个类的析构函数名
称必须是__destruct( )。析构函数不能带有任何参数。
格式:function __destruct ( ) { ... ... }
代码片段
<?
//创建一个人类
class Person{
//下面是人的成员属性
var $name; //人的名字
var $sex; //人的性别
var $age; //人的年龄
//定义一个构造方法参数为姓名$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>";
}
//这是一个析构函数,在对象销毁前调用
function __destruct(){
echo “再见”.$this->name.”<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
再见张三
再见李四
再见王五
大多数类都有一种称为构造函数的特殊方法。当创建一个对象时,它将自动调用构造函
数,也就是使用new 这个关键字来实例化对象的时候自动调用构造方法。
构造函数的声明与其它操作的声明一样,只是其名称必须是__construct( )。这是PHP5 中
的变化,以前的版本中,构造函数的名称必须与类名相同,这种在PHP5 中仍然可以用,但
现在以经很少有人用了,这样做的好处是可以使构造函数独立于类名,当类名发生改变时不
需要改相应的构造函数名称了。为了向下兼容,如果一个类中没有名为__construct( )的方法,
PHP 将搜索一个php4 中的写法,与类名相同名的构造方法。
格式:function __construct ( [参数] ) { ... ... }
在一个类中只能声明一个构造方法,而是只有在每次创建对象的时候都会去调用一次构
造方法,不能主动的调用这个方法,所以通常用它执行一些有用的初始化任务。比如对成属
性在创建对象的时候赋初值。
代码片段
代码如下:
<?
//创建一个人类
class Person{
//下面是人的成员属性
var $name; //人的名字
var $sex; //人的性别
var $age; //人的年龄
//定义一个构造方法参数为姓名$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
如图:
析构函数:
与构造函数相对的就是析构函数。析构函数是PHP5 新添加的内容,在PHP4 中没有析
构函数。析构函数允许在销毁一个类之前执行的一些操作或完成一些功能,比如说关闭文件,
释放结果集等,析构函数会在到某个对象的所有引用都被删除或者当对象被显式销毁时执行,
也就是对象在内存中被销毁前调用析构函数。与构造函数的名称类似,一个类的析构函数名
称必须是__destruct( )。析构函数不能带有任何参数。
格式:function __destruct ( ) { ... ... }
代码片段
代码如下:
<?
//创建一个人类
class Person{
//下面是人的成员属性
var $name; //人的名字
var $sex; //人的性别
var $age; //人的年龄
//定义一个构造方法参数为姓名$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>";
}
//这是一个析构函数,在对象销毁前调用
function __destruct(){
echo “再见”.$this->name.”<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
再见张三
再见李四
再见王五
最新技术文章: