欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  后端开发

虚代理实现延迟加载

程序员文章站 2022-05-01 22:33:45
...

这货是从 Martin 大神的《企业应用架构模式》中学到的,辅助 PHP 动态语言的特性,可以比 Java 轻松很多的实现延迟加载(LazyLoad)。基本原理是通过一个虚代理(Virtual Proxy)做占位符,一旦访问代理对象的某成员(方法或属性),加载就被触发。

不过我实现的这个版本有局限性:

  1. 只适用于对象,无法代理数组等基本数据类型(需要用 ArrayObject 一类的内置对象封装)
  2. 被代理之后,一些带有操作符重载性质的接口实现就失效了,例如 ArrayAccess 的索引器、Itreator 的迭代器,如果是用该代理来处理集合类型的延迟加载,还需要继承一个子类做特殊处理,以便外部用 foreach 迭代

详见我的博客:http://tonyseek.tumblr.com/post/6166066775/virtual-proxy-lazy-load

  1. // 测试
  2. $v = new VirtualProxy(function(){
  3. echo 'Now, Loading', "\n";
  4. $a = new ArrayObject(range(1,100));
  5. $a->abc = 'a';
  6. // 实际使用中,这里调用的是 DataMapper 的 findXXX 方法
  7. // 返回的是领域对象集合
  8. return $a;
  9. });
  10. // 代理对象直接当作原对象访问
  11. // 而此时构造方法传入的 callback 函数才被调用
  12. // 从而实现加载对象操作的延迟
  13. echo $v->abc . $v->offsetGet(50);
复制代码
  1. /**
  2. * 虚代理,只有在被访问成员时才调用闭包函数生成目标对象。
  3. *
  4. * @author tonyseek
  5. *
  6. */
  7. class VirtualProxy
  8. {
  9. private $holder = null;
  10. private $loader = null;
  11. /**
  12. * 虚代理,只有在被访问成员时才调用闭包函数生成目标对象。
  13. *
  14. * @param Closure $loader 生成被代理对象的闭包函数
  15. */
  16. public function __construct(Closure $loader)
  17. {
  18. $this->loader = $loader;
  19. }
  20. /**
  21. * 代理成员方法的调用
  22. *
  23. * @param string $method
  24. * @param array $arguments
  25. * @throws BadMethodCallException
  26. * @return mixed
  27. */
  28. public function __call($method, array $arguments = null)
  29. {
  30. $this->check();
  31. if (!method_exists($this->holder, $method)) {
  32. throw new BadMethodCallException();
  33. }
  34. return call_user_func_array(
  35. array(&$this->holder, $method),
  36. $arguments);
  37. }
  38. /**
  39. * 代理成员属性的读取
  40. *
  41. * @param string $property
  42. * @throws ErrorException
  43. * @return mixed
  44. */
  45. public function __get($property)
  46. {
  47. $this->check();
  48. if (!isset($this->holder->$property)) {
  49. throw new ErrorException();
  50. }
  51. return $this->holder->$property;
  52. }
  53. /**
  54. * 代理成员属性的赋值
  55. *
  56. * @param string $property
  57. * @param mixed $value
  58. */
  59. public function __set($property, $value)
  60. {
  61. $this->check();
  62. $this->holder->$property = $value;
  63. }
  64. /**
  65. * 检查是否已经存在被代理对象,不存在则生成。
  66. */
  67. private function check()
  68. {
  69. if (null == $this->holder) {
  70. $loader = $this->loader;
  71. $this->holder = $loader();
  72. }
  73. }
  74. }
复制代码