PHP Conference Japan 2024

ReflectionClass::newInstanceWithoutConstructor

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

ReflectionClass::newInstanceWithoutConstructor在不调用构造函数的情况下创建一个新的类实例

描述

public ReflectionClass::newInstanceWithoutConstructor(): object

在不调用构造函数的情况下创建一个新的类实例。

参数

返回值

错误/异常

如果该类是无法在不调用构造函数的情况下实例化的内部类,则会抛出 ReflectionException 异常。此异常仅限于 final 的内部类。

参见

添加注释

用户贡献注释 7 条注释

11
tom at r dot je
11 年前
应该明确的是,从 OOP 理论的角度来看,使用此方法与 goto、eval 和单例一样,是一种非常糟糕的实践。如果您发现需要在生产代码中使用它,那么您几乎肯定在某个地方做错了什么。它偶尔可能对调试有用,但即使那样也暗示了最初代码的质量不高。

问题是什么?它破坏了封装。对象可能存在于应用程序中,但可能无法履行其职责,因为它缺少依赖项。使用此方法使得系统中可能存在不完整的对象成为可能;对象可能处于其作者从未预料到的状态。这是不好的,因为它会导致意想不到的事情发生!OOP 中的一项基本原则是对象完全控制其状态,使用此方法会阻止这种保证。

注意:下面列出的基于注释的“依赖注入”也不是此方法的解决方案或有效用例,因为它破坏了封装(以及其他方面!),并且正在构造的类需要通过提供注释来了解容器。
9
oliver at ananit dot de
13 年前
如果您的 PHP 版本中没有此方法可用,您可以使用一种技巧来创建实例而不调用构造函数。
使用反射获取类的属性和默认值,并创建一个伪造的“序列化”字符串。

<?php
function createInstanceWithoutConstructor($class){
$reflector = new ReflectionClass($class);
$properties = $reflector->getProperties();
$defaults = $reflector->getDefaultProperties();

$serealized = "O:" . strlen($class) . ":\"$class\":".count($properties) .':{';
foreach (
$properties as $property){
$name = $property->getName();
if(
$property->isProtected()){
$name = chr(0) . '*' .chr(0) .$name;
} elseif(
$property->isPrivate()){
$name = chr(0) . $class. chr(0).$name;
}
$serealized .= serialize($name);
if(
array_key_exists($property->getName(),$defaults) ){
$serealized .= serialize($defaults[$property->getName()]);
} else {
$serealized .= serialize(null);
}
}
$serealized .="}";
return
unserialize($serealized);
}
?>

示例

<?php
class foo
{
public
$a = 10;
protected
$b = 2;
private
$c = "default";
protected
$d;
public function
__construct(){
$this->a = null;
$this->b = null;
$this->c = "constructed";
$this->d = 42;
}
}

var_dump(createInstanceWithoutConstructor('foo'));
?>

输出
object(foo)#6 (4) {
["a"]=>
int(10)
["b":protected]=>
int(2)
["c":"foo":private]=>
string(7) "default"
["d":protected]=>
NULL
}

我希望这可以帮助到某人。
Oliver Anan
4
260 at ciemnosc
8 年前
抱歉回复如此旧的评论,但我有一些想指出的内容。

@ tom at r dot je
虽然我总体上同意您所说的,但确实存在一些情况并非如此,并且由于 PHP 不允许使用多个构造函数,因此没有其他好的解决方法。

> 问题是什么?它破坏了封装。
> 对象可能存在于应用程序中,但可能无法履行其
> 职责,因为它缺少依赖项。
> 使用此方法使得系统中可能存在不完整的对象
> 成为可能;
> 对象可能处于其作者从未预料到的状态。

如果你从某个工厂方法中使用此方法,并以构造函数之外的其他方式手动初始化对象,那么这一点将不再有效。

考虑一个例子,你使用构造函数在从数据库获取对象后设置对象(例如,你需要根据某个 id 参数从另一个表中提取一些数组)。但你也希望能够手动创建对象(例如,用于插入数据库)。
最好的方法是使用两个不同的构造函数,但由于 PHP 中不允许这样做,因此你需要其他方法来创建对象。

示例

<?php
// 通常情况下,这会在某个类中作为静态缓存,或由方法 getMeta($id) 等返回的数组。
$meta = array(1337 => array('key1' => 'value1', 'key2' => 'value2'));

class
Test {
public
$id;
public
$data;
public
$meta;

public function
__construct() {
global
$meta;
if(
is_int($this->id)) $this->meta = $meta[$this->id];
}
public static function
create_empty() {
$r = new ReflectionClass(__CLASS__);
return
$r->newInstanceWithoutConstructor();
}
}
echo
"模拟 PDO::FETCH_CLASS 行为: ";
$t = Test::create_empty();
$t->meta = 1337;
$t->__construct();
var_dump($t);

echo
"空类: ";
$testItem = Test::create_empty();
// ... 在这里你可以开始设置项目,例如从 XML 中。
var_dump($testItem);

$testItem->id = 0;
$testItem->data = "some data";
$testItem->meta = array("somekey" => "somevalue");

echo
"设置后:";
var_dump($testItem);
?>

当然,你也可以创建一个空的构造函数,并创建一个 init() 方法,但随后你必须记住在任何地方都调用 init()。
你还可以创建其他方法将项目添加到数据库中,但随后你需要两个类来处理相同的数据——一个用于检索,另一个用于存储。

如果你使用某个工厂类(或只是一个工厂方法,如上例所示的简化示例),那么能够创建完全空的 object 非常有用。使用完整的工厂方法,你首先会使用某个 TestFactory->prepare() 方法,然后调用方法来设置你需要的内容,工厂会在调用 TestFactory->get() 获取准备好的对象时将所有未初始化的变量设置为某个默认值。
3
alejosimon at gmail
13 年前
此新方法的一个很好的初始用途。

它实现了一个透明的解析器构造函数参数,以实现 99% 的可重用组件。

<?php

use ReflectionClass ;

trait
TSingleton
{
/**
* 构造函数。
*/
protected function __construct() {}

/**
* 禁止克隆单例对象。
*/
protected function __clone() {}

/**
* 获取唯一实例。
*
* @params 可选,构造函数的参数。
* @return 调用类的实例对象。
*/
public static function getInstance()
{
static
$instance = null ;

if ( !
$instance )
{
$ref = new ReflectionClass( get_called_class() ) ;
$ctor = $ref->getConstructor() ;

// 感谢 PHP 5.4
$self = $ref->newInstanceWithoutConstructor() ;

// 魔法时刻。
$ctor->setAccessible( true ) ;
$instance = $ctor->invokeArgs( $self, func_get_args() ) ;
}

return
$instance ;
}
}

?>
1
me [ata] thomas-lauria.de
12 年前
此新功能支持基于注解的依赖注入。
<?php

// 要注入的依赖项
class dep {}

class
a {
/**
* @inject
* @var dep
*/
protected $foo;
}

class
b extends a {
/**
* @inject
* @var dep
*/
protected $bar;

public function
__construct() {
echo
"CONSTRUCTOR CALLED\n";
}
}

$ref = new ReflectionClass('b');
$inst = $ref->newInstanceWithoutConstructor();

$list = $ref->getProperties();
foreach(
$list as $prop){
/* @var $prop ReflectionProperty */
$prop->getDocComment(); // 检查 @inject 和 @var 类的名称
$prop->setAccessible(true);
$prop->setValue($inst, new dep());
}
if(
$const = $ref->getConstructor()) {
$constName = $const->getName();
$inst->{$constName}(); // 使用 call_user_func_array($function, $param_arr); 来传递参数
}

print_r($inst);
print_r($inst->foo); // 属性仍然无法访问

输出:

CONSTRUCTOR CALLED
b Object
(
[
bar:protected] => dep Object
(
)

[
foo:protected] => dep Object
(
)

)
PHP 致命错误: 无法访问受保护的属性 b::foo 在 diTest.php 的第 42 行。
0
benoit dot wery at online dot fr
1 年前
使用 ReflectionClass::newInstanceWithoutConstructor 和 ReflectionProperty::setValue 可以为只读提升属性设置值。例如,此代码有效(在 PHP 8.2 上测试过)。

<?php

class Test
{
public function
__construct(public readonly string $name)
{}
}

$test1 = new Test('test1');
$reflectionProperty = new ReflectionProperty(Test::class, 'name');
// 下一行会抛出致命错误
$reflectionProperty->setValue($test1, 'error');

$reflectionClass = new ReflectionClass(Test::class);
$test2 = $reflectionClass->newInstanceWithoutConstructor();
$reflectionProperty->setValue($test2, 'test2');

echo
$test2->name; // 将输出 "test2"
0
ben at NOSPAM dot fanmade dot de
3 年前
此类函数绝对应该尽可能少地出现在生产代码中。
当然,也有例外情况 :)
我们的库中有一些 UseCase 类扩展了一个抽象的 UseCase,后者有一个构造函数,用于进行一些设置,如果某些特定条件尚未满足,它甚至会抛出异常。
在该 UseCase 中,还定义了权限。


简化示例
<?php
抽象类 UseCase 实现 IUseCase
{
/** @var string[] */
受保护的数组 $requiredPermissions = [];

最终公共函数
__construct(IPresenter $out)
{
// 一些启动检查,包括授权
}
}
?>
我们已经有了一个更好的解决方案,但重构它需要一些时间,我们至少需要忍受几个星期。
现在我想到了一个主意,在初始化单个 UseCase 之前,在应用程序的传入请求中检查权限。
所以在看到这里的方法后,我只是在我们的抽象 UseCase 中引入了这个小助手
<?php
公共静态函数 getRequiredPermissions(): array
{
$class = new ReflectionClass(get_called_class());

返回
$class->newInstanceWithoutConstructor()->requiredPermissions;
}
?>
现在我能够在请求中检查权限,而无需真正修改库中的任何内容
<?php
公共函数 authorize(): bool
{
如果 (!
$this->user->hasPermissions(CreatePost::getRequiredPermissions())) {
返回
false;
}
返回
true;
}
?>
这也有一个好处,那就是为尽快切换到我们的新 UseCase 类增加了另一个理由 :)
To Top