PHP - 使用大量参数和默认值初始化对象的最佳方法

Tom*_*ger 30 php constructor idioms class instantiation

我正在设计一个类,它定义了一个高度复杂的对象,其中包含大量可选参数(50+),其中许多参数都有默认值(例如:) $type = 'foo'; $width = '300'; $interactive = false;.我正在尝试确定设置构造函数和实例/类变量的最佳方法,以便能够:

  • 让这个课程变得简单易用
  • 使类自动记录变得容易(即:使用phpDocumentor)
  • 优雅地编码

鉴于上述情况,我不想让构造函数传递大量的论据.我将传递一个包含初始化值的哈希,例如:$foo = new Foo(array('type'=>'bar', 'width'=>300, 'interactive'=>false));

在编写课程方面,我仍然觉得我宁愿......

class Foo {
    private $_type = 'default_type';
    private $_width = 100;
    private $_interactive = true;

    ...
}
Run Code Online (Sandbox Code Playgroud)

...因为我相信这将有助于文档生成(您可以获得类属性的列表,这可以让API用户知道他们必须使用的'选项'),并且"感觉"就像正确的方法一样它.

但是你遇到了将构造函数中的传入参数映射到类变量的问题,并且在没有利用符号表的情况下,你进入了一种"暴力"的方法,这让我无法实现目的(尽管我对其他人开放)意见).例如:

function __construct($args){
    if(isset($args['type'])) $_type = $args['type']; // yuck!
}
Run Code Online (Sandbox Code Playgroud)

我考虑过创建一个单独的类变量,它本身就是一个关联数组.初始化这将非常简单,例如:

private $_instance_params = array(
    'type' => 'default_type',
    'width' => 100,
    'interactive' => true
);

function __construct($args){
    foreach($args as $key=>$value){
        $_instance_params[$key] = $value;
    }
}
Run Code Online (Sandbox Code Playgroud)

但这似乎我没有利用私有类变量等本机功能,而且感觉文档生成不适用于这种方法.

感谢您阅读这篇文章; 我可能在这里问了很多,但我是PHP的新手,我真的只是寻找这种惯用/优雅的方式.你最好的做法是什么?


附录(有关此特定班级的详细信息)

这个类很可能试图做太多,但它是一个旧的Perl库的端口,用于创建和处理表单.可能有一种划分配置选项以利用继承和多态的方法,但它实际上可能适得其反.

根据请求,这里是一些参数的部分列表(Perl代码).您应该看到这些不能很好地映射到子类.

这个类肯定有许多这些属性的getter和setter,所以用户可以覆盖它们; 这篇文章的目标(以及原始代码做得很好)是提供一种使用已设置的必需参数实例化这些Form对象的简洁方法.它实际上使代码非常易读.

# Form Behaviour Parameters
        # --------------------------
        $self->{id}; # the id and the name of the <form> tag
        $self->{name} = "webform"; # legacy - replaced by {id}
        $self->{user_id} = $global->{user_id}; # used to make sure that all links have the user id encoded in them. Usually this gets returned as the {'i'} user input parameter
        $self->{no_form}; # if set, the <form> tag will be omitted
        $self->{readonly}; # if set, the entire form will be read-only
        $self->{autosave} = ''; # when set to true, un-focusing a field causes the field data to be saved immediately
        $self->{scrubbed}; # if set to "true" or non-null, places a "changed" radio button on far right of row-per-record forms that indicates that a record has been edited. Used to allow users to edit multiple records at the same time and save the results all at once. Very cool.
        $self->{add_rowid}; # if set, each row in a form will have a hidden "rowid" input field with the row_id of that record (used primarily for scrubbable records). If the 'scrubbed' parameter is set, this parameter is also automatically set. Note that for this to work, the SELECT statement must pull out a unique row id. 
        $self->{row_id_prefix} = "row_"; # each row gets a unique id of the form id="row_##" where ## corresponds to the record's rowid. In the case of multiple forms, if we need to identify a specific row, we can change the "row_" prefix to something unique. By default it's "row_"

        $self->{validate_form}; # parses user_input and validates required fields and the like on a form
        $self->{target}; # adds a target window to the form tag if specified
        $self->{focus_on_field}; # if supplied, this will add a <script> tag at the end of the form that will set the focus on the named field once the form loads.
        $self->{on_submit}; # adds the onSubmit event handler to the form tag if supplied
        $self->{ctrl_s_button_name}; # if supplied with the name of the savebutton, this will add an onKeypress handler to process CTRL-S as a way of saving the form

        # Form Paging Parameters
        # ----------------------
        $self->{max_rows_per_page}; # when displaying a complete form using printForm() method, determines the number of rows shown on screen at a time. If this is blank or undef, then all rows in the query are shown and no header/footer is produced.
        $self->{max_pages_in_nav} = 7; # when displaying the navbar above and below list forms, determines how many page links are shown. Should be an odd number
        $self->{current_offset}; # the current page that we're displaying
        $self->{total_records}; # the number of records returned by the query
        $self->{hide_max_rows_selector} = ""; # hide the <select> tag allowing users to choose the max_rows_per_page
        $self->{force_selected_row} = ""; # if this is set, calls to showPage() will also clear the rowid hidden field on the form, forcing the first record to be displayed if none were selected
        $self->{paging_style} = "normal"; # Options: "compact"
Run Code Online (Sandbox Code Playgroud)

当然,我们可以让自己陷入围绕编程风格的更长时间的辩论中.但我希望避免它,因为所有参与者的理智!这里(Perl代码,再次)是一个用一组相当大的参数实例化这个对象的例子.

my $form = new Valz::Webform (
            id                      => "dbForm",
            form_name               => "user_mailbox_recip_list_students",
            user_input              => \%params,
            user_id                 => $params{i},
            no_form                 => "no_form",
            selectable              => "checkbox",
            selectable_row_prefix   => "student",
            selected_row            => join (",", getRecipientIDsByType('student')),
            this_page               => $params{c},
            paging_style            => "compact",
            hide_max_rows_selector  => 'true',
            max_pages_in_nav        => 5
        );
Run Code Online (Sandbox Code Playgroud)

Daf*_*aff 7

我可以想到两种方法.如果要保留实例变量,只需遍历传递给构造函数的数组并动态设置实例变量:

    <?php

    class Foo {
        private $_type = 'default_type';
        private $_width = 100;
        private $_interactive = true;

        function __construct($args){
            foreach($args as $key => $val) {
                $name = '_' . $key;
                if(isset($this->{$name})) {
                    $this->{$name} = $val;
                }
            }
        }
    }

    ?>
Run Code Online (Sandbox Code Playgroud)

使用数组方法时,您不必放弃文档.只需在类体中使用@property注释:

<?php

/**
 * @property string $type
 * @property integer $width
 * @property boolean $interactive
 */
class Foo {
    private $_instance_params = array(
        'type' => 'default_type',
        'width' => 100,
        'interactive' => true
    );

    function __construct($args){
        $this->_instance_params = array_merge_recursive($this->_instance_params, $args);
    }

    public function __get($name)
    {
        return $this->_instance_params[$name];
    }

    public function __set($name, $value)
    {
        $this->_instance_params[$name] = $value;
    }
}

?>
Run Code Online (Sandbox Code Playgroud)

也就是说,一个包含50个成员变量的类要么仅用于配置(可以拆分),要么只是做得太多而你可能想要考虑重构它.

  • `private $ foo = null;`和`private $ foo;`完全相同,isset将返回false.如果要检查是否在类中定义了属性,请使用[property_exists](http://fi2.php.net/property_exists). (3认同)

Lui*_*mim 7

另一种方法是使用FooOptions对象实例化类,仅作为选项容器:

<?php
class Foo 
{
    /*
     * @var FooOptions
     */
    private $_options;

    public function __construct(FooOptions $options) 
    {
        $this->_options = $options;
    }
}


class FooOptions
{
    private $_type = 'default_type';
    private $_width = 100;
    private $_interactive = true;

    public function setType($type);
    public function getType();

    public function setWidth($width);
    public function getWidth();

    // ...
}
Run Code Online (Sandbox Code Playgroud)

您的选项已有详细记录,您可以轻松设置/检索它们.这甚至可以帮助您进行测试,因为您可以创建和设置不同的选项对象.

我不记得这个模式的确切名称,但我认为它是BuilderOption模式.