[TOC]
# 反射
反射API功能很强大,甚至能还原这个类的原型,包括方法的访问权限
~~~
<?php
class Person
{
public $gender;
public $name;
public function __get($param)
{
}
public function __set($key, $value)
{
}
public function say()
{
}
}
//反射获取类的原型
$obj = new ReflectionClass('Person');
$className = $obj->getName();
$methods = $Properties = [];
foreach ($obj->getProperties() as $key => $value) {
$Properties[$value->getName()]=$value;
}
foreach ($obj->getMethods() as $key => $value) {
$Methods[$value->getName()] = $value;
}
echo "Class {$className}".PHP_EOL."{".PHP_EOL;
is_array($Properties) && ksort($Properties);
foreach ($Properties as $key => $value) {
echo PHP_EOL;
echo $value->isPublic()? 'public':'',$value->isPrivate()?'private':'',
$value->isProtected()?'protected':'',
$value->isStatic()?'static':'';
echo PHP_EOL,$key,PHP_EOL;
}
echo PHP_EOL;
if (is_array($Methods)) ksort($Methods);
foreach ($Methods as $key => $value) {
echo PHP_EOL,"function {$key} () {}",PHP_EOL;
}
echo '}',PHP_EOL;
~~~
# 动态代理
~~~
<?php
class mysql
{
public function connect ($db)
{
echo "连接到数据库 ${db[0]}",PHP_EOL;
}
}
class sqlproxy
{
//保存连接数据库
private $target;
public function __construct($tar)
{
//把连接数据库的保存到这里
$this->target[] = new $tar;
}
public function __call($name, $args)
{
//使用魔术方法动态代理不需要实现和目标对象一样的方法
//因为自己这个__call就代表所有方法(除静态的)
foreach($this->target as $obj) {
$r = new ReflectionClass($obj);
if($method = $r->getMethod($name)) {
if($method->isPublic() && !$method->isAbstract()) {
//可以在前置和后置做一些你自定义的事情
echo '方法前置拦截记录---log',PHP_EOL;
//可以调用你原来mysql类的方法
$method->invoke($obj, $args);
echo '方法后置拦截记录---log',PHP_EOL;
}
}
}
}
}
$obj = new sqlproxy('mysql');
$obj->connect('member');
~~~
反射能保持代码的简洁,但是他违背了封装性
php有token函数,可以通过这一机制实现一些反射功能
注意这边的count函数,php7.2有变化注意
~~~
<?php
$tokens = token_get_all('<?php echo(123);');
// print_r($tokens);die;
for($i=0; $i < count($tokens); $i++){
for($j=1; $j < count($tokens[$i]); $j++){
echo PHP_EOL;
//token_name可以将数字表示的字符串的类型转化为php中固定名称,此函数PHP自带
$token_name = token_name($tokens[$i][0]);
echo $token_name;
echo PHP_EOL;
//把预定义的字符转换为 HTML 实体
echo htmlspecialchars($tokens[$i][$j]);
}
}
$tokens = token_get_all('<?php echo(123); ?>');
for($i=0 ; $i<count($tokens);$i++){
for($j=0 ; $j<count($tokens[$i]);$j++){
echo PHP_EOL;
echo htmlspecialchars($tokens[$i][$j]);
}
}
~~~
- 书列表
- laravel框架关键技术
- 第一章 组件化开发与composer使用
- 简介
- composer
- 添加路由组件
- 添加控制器模块
- 添加模型组件
- 添加视图组件
- 第三章 laravel框架中常用的php语法
- 匿名函数
- 文件包含
- 魔术方法
- 魔术常量
- 反射
- 后期静态绑定
- traits
- 第四章 laravel框架中使用的HTTP协议基础
- HTTP协议
- 数据库
- 数据迁移
- 第六章 laravel框架中的设计模式
- IOC模式
- php核心技术与最佳实践
- 第一章面向对象核心
- 反射
- 简单ORM
- 异常和错误
- 接口
- 第二章,面向对象设计
- 设计原则
- 单一职责
- 接口隔离
- 开放封闭
- 替换原则
- 依赖倒置
- linux是怎么写的呢?
- 第三章 正则表达
- 认识正则
- 第四章 php网络技术应用
- HTTP协议详解
- php和http相关函数
- 垃圾信息防御措施
- 现代操作系统
- 引论
- sql必知必会
- 限制结果
- 按位置排序
- where求职顺序
- IN操作符
- like
- 函数
- group by
- 组合查询
- 插入检索出的数据
- 视图
- 高性能mysql
- 第一章节 mysql架构与历史
- mysql架构逻辑图
- 连接与管理
- 优化与运行
- 读写锁
- 锁粒度
- 表锁(table lock)
- 行级锁(row lock)
- ACID
- 隔离级别
- 死锁
- 隐式和显式锁定
- 多版本并发控制
- Innodb概览
- 第四章节 Schema与数据类型优化
- 选择优化的数据类型
- 日期和时间类型
- 标识列
- 特殊类型数据
- 表设计中的缺陷
- 范式
- 计数器表
- 第五章 创建高性能索引
- 索引基础
- 索引类型
- 索引的优点
- 高性能索引策略
- 选择合适的索引列顺序
- 聚簇索引
- 顺序的主键什么时候会造成更坏的后果
- 覆盖索引
- 使用索引扫描来做排序
- 压缩索引
- 冗余和重复索引
- 索引和锁
- 支持多种过滤条件
- 什么是范围条件
- 优化排序
- 维护索引和表
- 表损坏
- 减少索引和数据的碎片
- 第六章 查询性能优化
- 扫描的行数和访问类型
- 重构查询方式
- 查询执行的基础
- 重构-改善既有代码设计
- 第一章-重构
- 什么是重构
- 第一个案列
- 重构第一步
- 王垠博客
- 多态取代价格相关逻辑