trait的使用技巧
trait是php5.4以后新增加的一个功能,可以将多个类中,共用的一些属性和方法提取出来做来公共trait类,就像是装配汽车的配件,如果你的类中要用到这些配件,就直接用use导入就可以了,相当于把trait中的代码复制到当前类中.
因为trait不是类,所以不能有静态成员,类常量,当然也不可能被实例化。

其实一个类中的代码,可以分为二大部分:一是我们自己写的代码,暂且叫私有代码吧,还有一部分就是公共代码了,之前主要是由父类代码组成。现在你的类中的公共代码又多一个新成员:trait类代码。
如果说:继承可以纵向扩展一个类,那么trait就是横向扩展一个类功能

下面以实例进行演示:
//1创建一个trait类Test1

<?php
trait Test1
{
public $name = 'PHP中文网'; //trait类中可以用属性
public function hello1() //trait类中主要成员是方法
{
return 'Test1::hello1()';
}
}
//2.创建triat类Test2
trait Test2
{
function hello2()
{
return 'Test2::hello2()';
}
}
//3.创建Demo1类
class Demo1
{
use Test1, Test2;
}
//进行测试
$obj = new Demo1;
echo $obj->hello1(); //访问trait类Test1中的hello1()
echo '<hr>';
echo $obj->name; //访问ttrait类Test1中的$name属性
echo '<hr>';
echo $obj->hello2(); //访问ttrait类Test1中的hello2()

trait可以互相嵌套,一个trait类中可以用use导入另一个trait类,理解成代码复制就可以了.
例如本例中,在Test2中要用到Test1中的代码,我们只要改动二个地方就可以了。
一是在Test2中用use Test1;导入Test1中的代码,
二是在Demo1类中的,去掉对Test1的引用,只保留对Test2的引用,想想这是为什么?给大家当作一个思考题吧~
修改后的代码如下:
//1创建一个trait类Test1

<?php
trait Test1
{
public $name = 'PHP中文网'; //trait类中可以用属性
public function hello1() //trait类中主要成员是方法
{
return 'Test1::hello1()';
}
}
//2.创建triat类Test2
trait Test2
{
use Test1;
function hello2()
{
//在Test2中访问Test1中的属性name,注意语法与普通类是一样的
return 'Test2::hello2()'.$this->name;
}
}
//3.创建Demo1类
class Demo1
{
// use Test1, Test2;
use Test2;
}
//进行测试
$obj = new Demo1;
echo $obj->hello1(); //访问trait类Test1中的hello1()
echo '<hr>';
echo $obj->name; //访问ttrait类Test1中的$name属性
echo '<hr>';
echo $obj->hello2(); //访问ttrait类Test1中的hello2()

刚才说过,类中导入的公共代码,除了trait方法集,还可以有父类,如果在子类中访问父类中的成员,大家应该很熟悉了,现在一个类除了可以从父类继承成员,还可以从trait类中继承,那么有一个问题就不可避免了,如果父类和trait类中的成员命名冲突怎么办?说人话,就是重名了怎么办?下面我们以方法重名来演示一下处理方案。
再创建一个类Demo,做为Demo1类的父类。
//3.创建父类Demo

class Demo
{
//在父类中创建一个与Test2重名的方法hello2()
public function hello2()
{
return '父类Demo::hello2()';
}
}

代码如下:

//1创建一个trait类Test1
trait Test1
{
public $name = 'PHP中文网'; //trait类中可以用属性
public function hello1() //trait类中主要成员是方法
{
return 'Test1::hello1()';
}
}
//2.创建triat类Test2
trait Test2
{
use Test1;
function hello2()
{
//在Test2中访问Test1中的属性name,注意语法与普通类是一样的
return 'Test2::hello2()'.$this->name;
}
}
//3.创建父类Demo
class Demo
{
public function hello2()
{
return '父类Demo::hello2()';
}
}
//4.创建Demo1类
class Demo1 extends Demo
{
// use Test1, Test2;
use Test2;
}
//进行测试
$obj = new Demo1;
echo $obj->hello1(); //访问trait类Test1中的hello1()
echo '<hr>';
echo $obj->name; //访问ttrait类Test1中的$name属性
echo '<hr>';
echo $obj->hello2(); //访问ttrait类Test1中的hello2()

再次访问,会发现,结果与之前完全一样没有任何变化,父类Demo中的hello2方法好像隐身了,压根不存在一样的。事实上,父类Demo中的hello2方法当然是存在的,只是被trat类Test2中的同名方法hello2覆盖掉了,原因就是:trait类中的同名方法,访问优先级大于父类的同名方法。
如果我们就想访问父类中的hello2方法,怎么办呢?只有一个办法,要么父类方法改名,要么Test2中的方法改名,我们把Test2中的hello2方法改成hello3,再次访问,就可以看到父类的执行结果了。

那么,我们再进一点想一下,如果在子类也有一个hello2方法呢?那么结果会是什么样?
我们来试一下,在Demo1类中添加如下代码:

//4.创建Demo1类
class Demo1 extends Demo
{
// use Test1, Test2;
use Test2;
//在Demo1类中创建与Test2和父类Demo中同名的方法hello2()
public function hello2()
{
return 'Demo1::hello()';
}
}

在浏览器再次方法,果然不出所料,子类Demo1中的hello2方法的执行结果覆盖掉了Test2中的同名方法
现在我们总结一下在同一个类中,同名方法的优先级:子类>Trait类>父类,与就是说,谁离调用者越近,谁的优先级就越高。

下面我们再讨论最后一个问题:如果trait类中方法重名了,怎么办?如果是trait类中被所有类共享的方法集,重名的可能性是非常大的。

下面我们修改一下代码,删除一些用不到代码:

//1创建一个trait类Test1
trait Test1
{
public function hello()
{
return 'Test1::hello()';
}
}
//2.创建triat类Test2
trait Test2
{
function hello()
{
return 'Test2::hello()';
}
}
//3.创建类Demo
class Demo
{
use Test1, Test2{
//用Test1中的hello()方法替代Test2中的同名方法
Test1::hello insteadof Test2;
//Test2中的hello()方法用别名访问
Test2::hello as test2Hello;
} //这里千万不要加分号 ;
} //进行测试
$obj = new Demo;
echo $obj->hello(); //访问Test1中的hello()
echo '<hr>';
echo $obj->test2Hello();//别名访问Test2中的hello()

php面向对象之trait的更多相关文章

  1. Scala入门系列(八):面向对象之trait

    基础知识 1 将trait作为接口使用 此时Trait就与Java中的接口非常类似,不过注意,在Scala中无论继承还是trait,统一都是extends关键字. Scala跟Java 8前一样不支持 ...

  2. Scala基础:面向对象之trait

    trait类似于java中的interface,但是有所不同 Scala中的trait是一种特殊的概念: 首先先将trait作为接口使用,此时的trait就与Java中的接口 (interface)非 ...

  3. scala(三)

    一.面向对象编程——类 1.定义一个简单的类 class HelloWorld { private var name = "leo" def sayHello() { print( ...

  4. 第2节 Scala中面向对象编程:12、13、14、15、16、trait

    6.4.  Scala中面向对象编程之trait 6.4.1.    将trait作为接口使用 Scala中的trait是一种特殊的概念: 首先先将trait作为接口使用,此时的trait就与Java ...

  5. Atitit usbQb212 oo 面向对象封装的标准化与规范解决方案java c# php js

    Atitit usbQb212 oo 面向对象封装的标准化与规范解决方案java c# php js 1.1. 封装性是面象对象编程中的三大特性之一  三个基本的特性:封装.继承与多态1 1.2. 魔 ...

  6. PHP类与面向对象

    PHP常量PHP常量大写PHP常量用define函数或const关键字定义一个常量一旦被定义,就不能再改变或者取消定义.在 PHP 5.3.0 以后,可以使用 const 关键字在类定义之外定义常量. ...

  7. Scala中的语言特性是如何实现的(3) -- Trait

    我的新博客地址:http://cuipengfei.me/blog/2013/10/13/scala-trait/ 我在Coursera上跟了一门叫做Functional Programming Pr ...

  8. trait与policy模板技术

    trait与policy模板技术 我们知道,类有属性(即数据)和操作两个方面.同样模板也有自己的属性(特别是模板参数类型的一些具体特征,即trait)和算法策略(policy,即模板内部的操作逻辑). ...

  9. PHP面向对象的小总结

     面向对象特性: *重用性 (每个模块都可以在项目中重复使用) *灵活性 (每个模块都很轻松被替换更改的) *拓展性(在模块上添加新功能是很方便的) 类和对象的关系(类生成对象) 物以类聚:相同特性的 ...

随机推荐

  1. MapReduce(一)

    MapReduce(一) 一.介绍 百度百科: MapReduce是一种编程模型,用于大规模数据集(大于1TB)的并行运算.概念"Map(映射)"和"Reduce(归约) ...

  2. 一步步构建iOS路由

    什么是移动端路由层: 路由层的概念在服务端是指url请求的分层解析,将一个请求分发到对应的应用处理程序.移动端的路由层指的是将诸如App内页面访问.H5与App访问的访问请求和App间的访问请求,进行 ...

  3. CURD插件(仿Django-admin版)

    前言 如何提升自己的开发效率? 每个新项目都是自己经做过的项目(经验所致),在项目开发过程中不断总结.封装属于自己的组件, 例如:每个web项目大部分都涉及增删改查,分页显示,搜素,CRM就是这样的组 ...

  4. kali-rolling安装openvas 9并创建扫描任务教程

    一. 前置说明 官方:漏洞扫描中最常用和强大的是某个”N“开头的漏洞扫描器(nessus),但由于许可证限制,kali中并未安装该漏洞扫描器.取而代之安装了nessus收费之后发起的开源版本openv ...

  5. weblogic修改安装路径教程

    我们有一个安装好的weblogic,我们想再装一个weblogic或者想把weblogic装到别的目录去,最直接的做法是从头装一个. 但是从头装一个是比较费时费力的,尤其是打补丁环节和创domain环 ...

  6. Eclipse导入Oracle/MySQL数库驱动包教程

    在操作数据库时除了import相关的SQL类外,还得在项目中导入数据库的驱动才能连接和操作数据库. 而数据库驱动jar包在默认Java的lib里是没有的,要自己到官网下载导入:本教程以Oracle为例 ...

  7. Struts 2 初步入门(五)之接受参数

    1.使用action的属性接受参数 执行顺序为:前端提交参数--->LoginAction.do进行处理--->处理成功后,跳转到sucess.jsp文件. (1)新建login.jsp文 ...

  8. linux系统管理 vi编辑器

    Vim是vi improved的缩写是vi的改进版本,vi被认为是事实上的标准编辑器 所有版本的Linux都带有vi编辑器 占用的资源少 与ed,ex等其他编辑器相比,vi对用户更加友好 进入vi编辑 ...

  9. JavaScript创建对象(三)——原型模式

    在JavaScript创建对象(二)——构造函数模式中提到,构造函数模式存在相同功能的函数定义多次的问题.本篇文章就来讨论一下该问题的解决方案——原型模式. 首先我们来看下什么是原型.我们在创建一个函 ...

  10. java基础学习之final关键字

    final可以修饰类.方法.变量,一旦使用了final则将不能改变被修饰的对象的引用; 被final修饰的类不可以被继承 被final修饰的方法不可以被覆盖 被final修饰的变量一般为常量,只允许对 ...