이 글은 주로 ThinkPHP5 인스턴스의 간단한 구현을 소개하며 관심 있는 친구들이 참고할 수 있습니다. 그것이 모두에게 도움이 되기를 바랍니다.
최근 ThinkPHP5를 배웠는데, TestClass 인스턴스를 생성할 수 있는 메서드인 TestClass::instance()를 처음 보았습니다. 저는 매우 궁금했습니다. ThinkPHP의 소스 코드를 살펴보고 그 디자인 아이디어가 전반적으로 이해되었습니다.
기존 규칙, 코드로 직접 이동:
<?php class TestClass { public static function instance() { return new self(); } public $data = []; public function __set($name, $val) { return $this->data[$name] = $val; } public function __get($name) { return $this->data[$name]; } } $app1 = TestClass::instance(); $app1->key = 'Application 1'; echo $app1->key . '<br />'; ?>
통화를 용이하게 하기 위해 ThinkPHP도 모방하고 도우미 함수를 작성했습니다.
<?php function app() { return TestClass::instance(); } $app2 = app(); $app2->key = 'Application 2'; echo $app2->key . '<br />'; ?>
이렇게 하면 인스턴스가 간단하게 구현됩니다.
그러나 이 방법에는 작은 문제가 있습니다. 100번 호출하면 100개의 인스턴스를 생성해야 한다는 것을 생각하면 무섭습니다.
Test 클래스에 정적 속성을 추가하고 생성된 인스턴스를 여기에 저장하세요. 다음에 호출해야 하는 경우 이 인스턴스를 직접 호출하세요.
<?php class TestClass { public static $instance; //用于缓存实例 public $data = []; public static function instance() { //如果不存在实例,则返回实例 if (empty(self::$instance)) { self::$instance = new self(); } return self::$instance; } public function __set($name, $val) { return $this->data[$name] = $val; } public function __get($name) { return $this->data[$name]; } } function app($option = []) { return TestClass::instance($option); } header('content-type:text/plain'); $result = []; $app1 = app(); $app1->key = "Application 1"; //修改 key 为 Application 1 $result['app1'] = [ 'app1' => $app1->key, //实例中 key 为 Application 1 ]; // 创建 app2,因为 instance 已经存在实例,直接返回 缓存的实例 $app2 = app(); $result['app2'] = [ 'setp1' => [ 'app1' => $app1->key, // Application 1 'app2' => $app2->key, //因为直接调用的实例的缓存,所以 key 也是 Application 1 ], ]; // 无论 app1,app2 都对在内存中 对应的同一个实例,无论通过谁修改,都能改变值 $app1->key = "Application 2"; $result['app2']['setp2'] = [ 'app1' => $app1->key, // Application 2 'app2' => $app2->key, // Application 2 ]; print_r($result); ?>
위의 실험을 통해 몇 번을 호출해도 동일한 인스턴스가 사용되는 것을 알 수 있습니다. 이는 낮은 효율성 문제를 해결합니다.
지금까지는 기본적으로 대부분의 상황을 충족합니다. 유일한 작은 결함은 인스턴스의 초기 매개변수가 다를 수 있으므로 유연하게 호출할 수 없다는 것입니다(일반적으로 동일한 프로그램이 두 개의 데이터베이스를 호출함). 이 문제는 위의 예제를 약간 수정하여 들어오는 매개변수를 키로 사용하고 불합리한 인스턴스를 배열에 캐시함으로써 해결할 수 있습니다.
<?php class TestClass { public static $instance = []; //用于缓存实例数组 public $data = []; public function __construct($opt = []) { $this->data = $opt; } public static function instance($option = []) { // 根据传入的参数 通过 serialize 转换为字符串,md5 后 作为数组的 key $instance_id = md5(serialize($option)); //如果 不存在实例,则创建 if (empty(self::$instance[$instance_id])) { self::$instance[$instance_id] = new self($option); } return self::$instance[$instance_id]; } public function __set($name, $val) { return $this->data[$name] = $val; } public function __get($name) { return $this->data[$name]; } } function app($option = []) { return TestClass::instance($option); } header('content-type:text/plain'); $result = []; //传入 初始数据 $app1 = app(['key' => '123']); $result['init'] = $app1->key; // 使用 传入的数据,即:123 $app1->key = "app1"; $result['app'] = $app1->key; // 现在值改为了 自定义的 app1了 print_r($result); $result = []; // 创建 app2,注意 初始参数不一样 $app2 = app(); // 因为初始参数不一样,所以还是创建新的实例 $app2->key = "app2"; $result['app1'] = $app1->key; // app1 $result['app2'] = $app2->key; // app2 print_r($result); $result = []; // 创建 app3,传入的参数 和 app1 一样,所以会直接返回 和app1相同 的 实例 $app3 = app(['key' => '123']); $result['log'] = [ 'app1' => $app1->key, // app1 'app2' => $app2->key, // app2 'app3' => $app3->key, // app1 ]; // 设置 app3 的key,会自动修改 app1 的值,因为他们两个是同一个实例 $app3->key = 'app3'; $result['app3_set'] = [ 'app1' => $app1->key, // app3 'app2' => $app2->key, // app2 'app3' => $app3->key, // app3 ]; // 同理,设置 app1 的key,app3 的 key 也会修改 $app1->key = 'app1'; $result['app1_set'] = [ 'app1' => $app1->key, // app1 'app2' => $app2->key, // app2 'app3' => $app3->key, // app1 ]; print_r($result); ?>
관련 추천:
가장 상세한 ThinkPHP5 사용자 정의 포인트 페이지 튜토리얼
위 내용은 ThinkPHP5 인스턴스 구현에 대한 자세한 설명의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!