[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與數據類型優化
- 選擇優化的數據類型
- 日期和時間類型
- 標識列
- 特殊類型數據
- 表設計中的缺陷
- 范式
- 計數器表
- 第五章 創建高性能索引
- 索引基礎
- 索引類型
- 索引的優點
- 高性能索引策略
- 選擇合適的索引列順序
- 聚簇索引
- 順序的主鍵什么時候會造成更壞的后果
- 覆蓋索引
- 使用索引掃描來做排序
- 壓縮索引
- 冗余和重復索引
- 索引和鎖
- 支持多種過濾條件
- 什么是范圍條件
- 優化排序
- 維護索引和表
- 表損壞
- 減少索引和數據的碎片
- 第六章 查詢性能優化
- 掃描的行數和訪問類型
- 重構查詢方式
- 查詢執行的基礎
- 重構-改善既有代碼設計
- 第一章-重構
- 什么是重構
- 第一個案列
- 重構第一步
- 王垠博客
- 多態取代價格相關邏輯