PHP Conference Japan 2024

ReflectionClass::newInstanceArgs

(PHP 5 >= 5.1.3, PHP 7, PHP 8)

ReflectionClass::newInstanceArgs使用给定参数创建一个新的类实例

描述

public ReflectionClass::newInstanceArgs(array $args = []): ?object

创建一个新的类实例,给定的参数将传递给类的构造函数。

参数

args

作为 array 传递给类构造函数的参数。

返回值

返回一个新的类实例,或者在失败时返回 null

错误/异常

如果类的构造函数不是公共的,则会抛出 ReflectionException

如果类没有构造函数并且 args 参数包含一个或多个参数,则会抛出 ReflectionException

示例

示例 #1 ReflectionClass::newInstanceArgs() 的基本用法

<?php
$class
= new ReflectionClass('ReflectionFunction');
$instance = $class->newInstanceArgs(array('substr'));
var_dump($instance);
?>

以上示例将输出

object(ReflectionFunction)#2 (1) {
  ["name"]=>
  string(6) "substr"
}

参见

添加笔记

用户贡献笔记 8 条笔记

kirillsaksin at no-spam dot yandex dot ru
9 年前
正确实例化具有私有构造函数的类的技巧

<?php

class TestClass
{
private
$property;
private function
__construct($argument)
{
$this->property = $argument;
}
}

$ref = new ReflectionClass(TestClass::class);
$instance = $ref->newInstanceWithoutConstructor();
var_dump($instance);
echo
PHP_EOL . '------------------------' . PHP_EOL . PHP_EOL;
$constructor = $ref->getConstructor();
$constructor->setAccessible(true);
$constructor->invokeArgs($instance, ['It works!']);
var_dump($instance);

// 输出:
// class TestClass#3 (1) {
// private $property =>
// NULL
// }
//
// ------------------------
//
// class TestClass#3 (1) {
// private $property =>
// string(9) "It works!"
// }

?>
sarfraznawaz2005 at gmail dot com
14 年前
我使用反射类并检测参数是按引用传递还是按值传递
然后使用这些参数成功地初始化/调用方法

<?php
if (count($args) > 1)
{
if (
method_exists($class_name, '__construct') === false)
{
exit(
"类 <strong>$class_name</strong> 的构造函数不存在,您不应该将参数传递给此类的构造函数!" );
}

$refMethod = new ReflectionMethod($class_name, '__construct');
$params = $refMethod->getParameters();

$re_args = array();

foreach(
$params as $key => $param)
{
if (
$param->isPassedByReference())
{
$re_args[$key] = &$args[$key];
}
else
{
$re_args[$key] = $args[$key];
}
}

$refClass = new ReflectionClass($class_name);
$class_instance = $refClass->newInstanceArgs((array) $re_args);
}
?>
richardcook at gmail dot com
15 年前
如果类的构造函数的参数中有引用,newInstanceArgs 函数将无法调用该类的构造函数,因此请注意传递给它的内容

<?php
Foo {
函数
__construct (&$arr) {
$this->arr = &$arr;
}
函数
createInstance () {
$reflectionClass = new ReflectionClass("Bar");

返回
$reflectionClass->newInstanceArgs(数组($this, $this->arr));
}
函数
mod($key, $val) {
$this->arr[$key] = $val;
}
}

Bar {
函数
__construct (&$foo, &$arr) {
$this->foo = &$foo;
$this->arr = &$arr;
}
函数
mod($key, $val) {
$this->arr[$key] = $val;
}
}

$arr = 数组();

$foo = new Foo($arr);

$arr["x"] = 1;

$foo->mod("y", 2);

$bar = $foo->createInstance();

$bar->mod("z", 3);

echo
"<pre>";
print_r($arr);
print_r($foo);
print_r($bar);
echo
"</pre>";

/*
输出:
警告:在 [代码路径] 的第 31 行调用 Bar 的构造函数失败

致命错误:在 [代码路径] 的第 58 行调用非对象的 mod() 方法
*/
?>
talk at stephensugden dot com
14 年前
这是我在轻量级 IoC 容器中动态实例化对象的方式

<?php

SimpleContainer {

// ...

// 使用提供的参数数组创建对象实例
保护函数 instantiate($name, $args=数组()){
如果(空(
$args))
返回 new
$name();
否则 {
$ref = new ReflectionClass($name);
返回
$ref->newInstanceArgs($args);
}
}
// ...
}
?>

我明确地**不**处理用户为无构造函数的类传递构造函数参数的情况,因为这**应该**会失败。
dev at yopmail dot com
9 年前
使用 PHP 5.6,我们可以使用 ... (T_ELLIPSIS) 运算符

<?php

Test {
公共函数
__construct($a, $b) {
echo
$a . ' ' . $b;
}
}

$args = 数组(12, 34);
new
Test(... $args); // 显示 "12 34"

?>
sausage at tehsausage dot com
14 年前
烦人的是,即使您为参数传递空数组,对于没有构造函数的类,这也会抛出异常。对于泛型编程,您应该避免使用此函数,而应使用带有 newInstance 的 call_user_func_array。
kevinpeno at gmail dot com
15 年前
我误解了这个函数,以为它是一种以数组形式设置 Reflection::newInstance() 参数的 setter,而不是创建新实例本身。

此函数等效于 call_user_func_array(),而 Reflection::newInstance() 等效于 call_user_func()
gromit at mailinator dot com
16 年前
请注意,使用空数组调用 newInstanceArgs 方法仍将调用不带参数的构造函数。如果类没有构造函数,则会生成异常。

在调用此方法之前,您需要检查是否存在构造函数,或者使用 try 和 catch 来处理异常。
To Top