>  기사  >  백엔드 개발  >  동적으로 객체를 인스턴스화하고 매개변수를 생성자에 전달하는 PHP의 사례 분석

동적으로 객체를 인스턴스화하고 매개변수를 생성자에 전달하는 PHP의 사례 분석

黄舟
黄舟원래의
2017-07-02 09:49:232362검색

프레임워크 개발, 모듈식 개발 등에서 PHP가 실행되는 동안 객체를 동적으로 인스턴스화해야 할 수도 있습니다.

동적 인스턴스화 개체란 무엇인가요? 먼저 PHP에서 변수 함수(Variable function)의 개념을 살펴보겠습니다. 예를 들어 다음 코드는

function foo() {
    echo 'This is the foo function';
}
$bar = 'foo';
$bar();

위 코드를 실행하면 "This is the foo function"이 출력됩니다. 자세한 내용은 PHP 매뉴얼: 변수 함수를 참조하세요. 물론, 동적으로 호출해야 한다면 call_user_func 또는 call_user_func_array 함수를 사용하세요. 이 두 기능의 사용법은 이 글의 초점이 아닙니다. 이해가 되지 않는다면 다른 정보를 확인하시기 바랍니다. 이 기사의 주제로 돌아가서: 동적으로 인스턴스화된 개체란 무엇입니까? 객체의 동적 인스턴스화는 인스턴스화해야 할 객체가 런타임에 동적으로 결정(변수에 의해 결정)되고, 인스턴스화해야 하는 객체의 종류가 코드에 직접 작성되지 않는다는 것을 의미한다고 생각합니다.

위의 예를 통해 우리는 런타임에 함수를 동적으로 호출하는 방법을 이미 알고 있습니다. 오늘날 객체 지향이 널리 사용되는 경우 일부 코드에서는 클래스를 동적으로 인스턴스화해야 합니다.

상황 1: 클래스의 생성자에 매개변수가 없거나 매개변수 개수가 결정됩니다.

클래스의 생성자에 매개변수가 없거나 인스턴스화하려는 클래스에 생성자가 전혀 없는 경우 더 간단하게 위의 내용을 따를 수 있습니다. 예제를 변경하면 됩니다. 음, 동일한 예제를 따르십시오. 누가 할 수 없습니까?

코드 예제: (생성자에는 매개변수가 없습니다)

class FOO {
    private $a, $b;
    public function construct() {
        $this->a = 1;
        $this->b = 2;
    }
    public function test() {
        echo &#39;This is the method test of class FOO<br />&#39;;
        echo &#39;$this->a=&#39;, $this->a, &#39;, $this->b=&#39;, $this->b;
    }
}
$bar = &#39;FOO&#39;;
$foo = new $bar();
$foo->test();

실행하면 다음 출력이 표시됩니다.

This is the method test of class FOO
$this->a=1, $this->b=2

글쎄, 매개변수를 전달해야 합니다. 그렇다면 다음을 수행하세요.

class FOO {
    private $a, $b;
    public function construct($a, $b) {
        $this->a = $a;
        $this->b = $b;
    }
    public function test() {
        echo &#39;This is the method test of class FOO<br />&#39;;
        echo &#39;$this->a=&#39;, $this->a, &#39;, $this->b=&#39;, $this->b;
    }
}
$bar = &#39;FOO&#39;;
$foo = new $bar(&#39;test&#39;, 5);
$foo->test();

도 비슷한 결과를 얻을 수 있습니다.

This is the method test of class FOO
$this->a=test, $this->b=5

가 이상적입니다.

상황 2: 클래스 생성 개수 함수 매개변수 가 불확실함

이 상황은 훨씬 더 번거롭겠지만, 좀 더 보편적으로 작성하려면 이 상황을 고려해야 합니다. 예를 들어, 다음 두 클래스

class FOO {
    public function test() {
        echo &#39;This is the method test of class FOO&#39;;
    }
}
class BAR {
    private $a, $b;
    public function construct($a, $b) {
        $this->a = $a;
        $this->b = $b;
    }
    public function test() {
        echo &#39;This is the method test of class BAR<br />&#39;;
        echo &#39;$this->a=&#39;, $this->a, &#39;, $this->b=&#39;, $this->b;
    }
}

가 있고 이 두 클래스를 인스턴스화하는 일반적인 방법을 원합니다. FOO 클래스에는 생성자가 없습니다. 또는 BAR 클래스의 생성자에는 매개변수가 있는 반면 FOO 클래스 생성자의 매개변수 개수는 0이라고 간주할 수 있습니다. 다행스럽게도 PHP5는 충분히 강력하며 리플렉션 개념을 도입하고 있습니다. 자세한 내용은 PHP 매뉴얼: 리플렉션을 참조하세요. 매뉴얼에는 참고할 내용이 없습니다. :) 다행스럽게도 이름은 잘 작성되어 있으므로 클래스 이름과 메서드 이름을 보면 이미 대략적인 내용을 알 수 있으므로 많은 단어가 필요하지 않습니다.

그렇다면 PHP5의 리플렉션을 사용하여 이 문제를 시작하겠습니다.

(아직 PHP4를 사용하고 있는 학생은 가지 마십시오. 리플렉션이 없는 PHP 버전이 있거나 리플렉션을 원하지 않는 경우) 호환성, 업그레이드해도 상관없어요 어차피 리플렉션은 안쓰고 싶고 아래에 해결방법이 있어요)

$class = new ReflectionClass(&#39;FOO&#39;);
$foo = $class->newInstance(); //或者是$foo = $class->newInstanceArgs();
$foo->test();

뭐 보셨나요? 다음:

$class = new ReflectionClass(&#39;BAR&#39;);
$bar = $class->newInstanceArgs(array(55, 65));
$bar->test();

괜찮은 것 같으니 정리해서 일반적인 함수를 만들어 보겠습니다. 이 함수의 첫 번째 함수는 인스턴스화할 클래스의 이름입니다. 두 번째 매개변수입니다. 클래스 생성자의 매개변수를 인스턴스화하려면 가능한 한 많이 작성하세요. 그렇지 않은 경우에는 작성하지 마세요. 가변 개수의 매개변수를 사용하여 함수를 구현하려면 다음 두 가지 방법이 있습니다.

첫 번째는

function foo($arg1, $arg2 = 123, $arg3 = &#39;test&#39;, $arg4 = null, ....... ) {
    //some code;
}

와 유사한 방법입니다. 첫 번째는 100을 전달해야 하는 경우 그냥 써야 한다는 것입니다. 매개변수에 대한 매개변수가 100개인가요? 두 번째는 프로그램의 어떤 매개변수가 null이거나 다른 기본값인지 결정해야 한다는 것입니다. (여담: 이런 식으로 하면 매개변수의 기본값은 맨 뒤에 와야 합니다. 기본값이 있는 매개변수의 중간이나 앞에는 기본값이 없는 매개변수를 삽입할 수 없습니다. 그렇지 않은 경우에도 명시적으로 입력해야 합니다.

변수 개수를 구현하는 또 다른 방법은 PHP의 내장 함수 func_get_args(매뉴얼을 읽으려면 여기를 클릭하세요)를 사용하여 매개변수를 전달받는 것입니다. 함수 속의 함수. 유사한 함수로는 func_get_num 및 func_get_arg가 있습니다. 잊어버리세요. 매뉴얼을 찾아서 직접 읽어보세요.

그래서 이 함수를 사용하는 것이 훨씬 편리한 것 같습니다. 우리가 상상한 함수 매개변수의 배열을 바탕으로 코드는 다음과 같아야 합니다.

function newInstance() {
    $arguments = func_get_args();
    $className = array_shift($arguments);
    $class = new ReflectionClass($className);
    return $class->newInstanceArgs($arguments);
}

자, 효과를 살펴보겠습니다.

$foo = newInstance(&#39;FOO&#39;);
$foo->test();
//输出结果:
//This is the method test of class FOO

$bar = newInstance(&#39;BAR&#39;, 3, 5);
$bar->test();
//输出结果:
//This is the method test of class BAR
//$this->a=3, $this->b=5

단 4줄의 코드로 효과가 아주 완벽합니다. 그러면 이 아이디어를 수업에 적용하면 마법의 메소드로 직접 작성해서 수업을 더욱 멋지게 만들 수 있어요!

class INSTANCE {
    function call($className, $arguments) {
        $class = new ReflectionClass($className);
        return $class->newInstanceArgs($arguments);
    }
}
$inst = new INSTANCE();
$foo = $inst->foo();
$foo->test();
//输出结果:
//This is the method test of class FOO

$bar = $inst->bar(&#39;arg1&#39;, &#39;arg2&#39;);
$bar->test();
//输出结果:
//This is the method test of class BAR
//$this->a=3, $this->b=5

카카, 기분 좋아요.

接下来讨论一下不使用反射类的情况。例如PHP4中就没有反射,而一些老项目就是运行在PHP4上面的。或者是要保证项目对未知环境的兼容性,Whatever,来关心一下怎么动态传参吧。PHP中动态传参的函数只有一个:call_user_func_array(轻击此处查看手册)。这是一个动态调用函数的函数,作用是可以将函数的参数以数组的形式传递给要调用的函数。好吧,我自己也被自己绕晕了,直接来看实例:

function foo($a, $b) {
    echo &#39;$a=&#39;, $a, &#39;<br />&#39;;
    echo &#39;$b=&#39;, $b;
}
call_user_func_array(&#39;foo&#39;, array(1, &#39;string&#39;));
//本例输出结果:
//$a=1
//$b=string

那么,要实现用这种方法来动态实例化对象并传参,呃……,只有曲线救国了,我们得先写一个函数,让这个函数来实例化对象,而这个函数的参数就原原本本地传给要实例化对象的类的构造函数就好了。打住!那这个函数得有几个参数啊?怎么实现传递不同个数的参数呢?嘿嘿,我一声冷笑,你忘了PHP里提供一个创建匿名函数的函数吗?(又开始绕起来了……)create_function(手册在此),照着手册里面的例子直接画就可以了,我也懒得打字了,直接看下面的代码,注释我写清楚点大家都明白了:

function newInst() {
    //取得所有参数
    $arguments = func_get_args();

    //弹出第一个参数,这是类名,剩下的都是要传给实例化类的构造函数的参数了
    $className = array_shift($arguments);

    //给所有的参数键值加个前缀
    $keys = array_keys($arguments);
    array_walk($keys, create_function(&#39;&$value, $key, $prefix&#39;, &#39;$value = $prefix . $value;&#39;), &#39;$arg_&#39;);

    //动态构造实例化类的函数,主要是动态构造参数的个数
    $paramStr = implode(&#39;, &#39;,$keys);
    $newClass=create_function($paramStr, "return new {$className}({$paramStr});");

    //实例化对象并返回
    return call_user_func_array($newClass, $arguments);
}

好了,至于效果是什么,就麻烦各位看官自己动动手,运行一下看看,是不是自己期望的结果。

如果改写成类里面的魔术方法,哼哼,威力嘛,你懂的!然后呢,我还是懒了,如果要写成魔术方法的话,相信这么easy的事情你很轻松就办到了。就当一个作业吧。另,本文的代码都是本人运行过的,但是,写文章的时候没有使用复制/粘贴功能,所以,你最好是也不要复制粘贴。如果从本文中的代码copy下来运行出错的话,还烦请各位自己debug一下,编程不就是要自己写么。

本来这个话题到这里就应该可以结束了,但是想到我在想到这个办法之前用的方法(好绕……),本着重在交流的态度,一起放出来。我例两个关键函数吧:extract和eval。只是我个人觉得用eval函数比较山寨,能不用最好不用。于是又想出了上面的办法,哈哈,编程最重要的是思想,不是吗?好,又是一道作业题了,大家可以试试用这两个函数(当然也会用到别的函数)来实现带不定数量的参数动态实例化对象的函数。

위 내용은 동적으로 객체를 인스턴스화하고 매개변수를 생성자에 전달하는 PHP의 사례 분석의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.