concept-behaviors.md 9.09 KB
Newer Older
1
行为
2 3
=========

4
行为是 [[yii\base\Behavior]] 或其子类的实例。行为,也称为 [mixins](http://en.wikipedia.org/wiki/Mixin),可以无须改变类继承关系即可增强一个已有的 [[yii\base\Component|组件]] 类功能。当行为附加到组件后,它将“注入”它的方法和属性到组件,然后可以像访问组件内定义的方法和属性一样访问它们。此外,行为通过组件能响应被触发的[事件](basic-events.md),从而自定义或调整组件正常执行的代码。
5

6

7 8
定义行为
-----------
9

10
要定义行为,通过继承 [[yii\base\Behavior]] 或其子类来建立一个类。如:
11

12 13
```php
namespace app\components;
14

15 16
use yii\base\Model;
use yii\base\Behavior;
17

18 19 20
class MyBehavior extends Behavior
{
    public $prop1;
21

22
    private $_prop2;
23

24 25 26 27
    public function getProp2()
    {
        return $this->_prop2;
    }
28

29 30 31 32 33 34 35 36 37 38
    public function setProp2($value)
    {
        $this->_prop2 = $value;
    }

    public function foo()
    {
        // ...
    }
}
39 40
```

41
以上代码定义了行为类 `app\components\MyBehavior` 并为要附加行为的组件提供了两个属性 `prop1``prop2` 和一个方法 `foo()` 。注意属性 `prop2` 是通过 getter `getProp2()` 和 setter `setProp2()` 定义的。能这样用是因为 [[yii\base\Object]] 是 [[yii\base\Behavior]] 的祖先类,此祖先类支持用 getter 和 setter 方法定义[属性](basic-properties.md)
42

43
> 提示:在行为内部可以通过 [[yii\base\Behavior::owner]] 属性访问行为已附加的组件。
44

45 46 47 48 49

处理事件
-------------

如果要让行为响应对应组件的事件触发,就应覆写 [[yii\base\Behavior::events()]] 方法,如:
50 51

```php
52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72
namespace app\components;

use yii\db\ActiveRecord;
use yii\base\Behavior;

class MyBehavior extends Behavior
{
    // 其它代码

    public function events()
    {
        return [
            ActiveRecord::EVENT_BEFORE_VALIDATE => 'beforeValidate',
        ];
    }

    public function beforeValidate($event)
    {
        // 处理器方法逻辑
    }
}
73 74
```

75 76 77 78 79 80 81
[[yii\base\Behavior::events()|events()]] 方法返回事件列表和相应的处理器。上例声明了 [[yii\db\ActiveRecord::EVENT_BEFORE_VALIDATE|EVENT_BEFORE_VALIDATE]] 事件和它的处理器 `beforeValidate()` 。当指定一个事件处理器时,要使用以下格式之一:

* 指向行为类的方法名的字符串,如上例所示;
* 对象或类名和方法名的数组,如 `[$object, 'methodName']`
* 匿名方法。

处理器的格式如下,其中 `$event` 指向事件参数。关于事件的更多细节请参考[事件](basic-events.md)
82 83

```php
84 85
function ($event) {
}
86 87 88
```


89 90
附加行为
----------
91

92
可以静态或动态地附加行为到[[yii\base\Component|组件]]。前者在实践中更常见。
93

94
要静态附加行为,覆写行为要附加的组件类的 [[yii\base\Component::behaviors()|behaviors()]] 方法即可。[[yii\base\Component::behaviors()|behaviors()]] 方法应该返回行为[配置](basic-configs.md)列表。每个行为配置可以是行为类名也可以是配置数组。如:
95 96 97 98 99 100 101 102 103 104 105 106

```php
namespace app\models;

use yii\db\ActiveRecord;
use app\components\MyBehavior;

class User extends ActiveRecord
{
    public function behaviors()
    {
        return [
107
            // 匿名行为,只有行为类名
108 109
            MyBehavior::className(),

110
            // 命名行为,只有行为类名
111 112
            'myBehavior2' => MyBehavior::className(),

113
            // 匿名行为,配置数组
114 115 116 117 118 119
            [
                'class' => MyBehavior::className(),
                'prop1' => 'value1',
                'prop2' => 'value2',
            ],

120
            // 命名行为,配置数组
121 122 123 124 125 126 127 128 129 130
            'myBehavior4' => [
                'class' => MyBehavior::className(),
                'prop1' => 'value1',
                'prop2' => 'value2',
            ]
        ];
    }
}
```

131
通过指定行为配置数组相应的键可以给行为关联一个名称。这种行为称为**命名行为**。上例中,有两个命名行为:`myBehavior2``myBehavior4` 。如果行为没有指定名称就是**匿名行为**
132

133
要动态附加行为,在对应组件里调用 [[yii\base\Component::attachBehavior()]] 方法即可,如:
134 135 136 137

```php
use app\components\MyBehavior;

138
// 附加行为对象
139 140
$component->attachBehavior('myBehavior1', new MyBehavior);

141
// 附加行为类
142 143
$component->attachBehavior('myBehavior2', MyBehavior::className());

144
// 附加配置数组
145 146 147 148 149 150 151
$component->attachBehavior('myBehavior3', [
    'class' => MyBehavior::className(),
    'prop1' => 'value1',
    'prop2' => 'value2',
]);
```

152
可以通过 [[yii\base\Component::attachBehaviors()]] 方法一次附加多个行为:
153 154 155

```php
$component->attachBehaviors([
156 157
    'myBehavior1' => new MyBehavior,  // 命名行为
    MyBehavior::className(),          // 匿名行为
158 159 160
]);
```

161
还可以通过[配置](concept-configurations.md)去附加行为:
162 163 164 165 166 167 168 169 170 171 172 173 174

```php
[
    'as myBehavior2' => MyBehavior::className(),

    'as myBehavior3' => [
        'class' => MyBehavior::className(),
        'prop1' => 'value1',
        'prop2' => 'value2',
    ],
]
```

175
详情请参考[配置](concept-configurations.md#configuration-format)章节。
176 177


178 179
使用行为
---------------
180

181
使用行为,必须像前文描述的一样先把它附加到 [[yii\base\Component|component]] 类或其子类。一旦行为附加到组件,就可以直接使用它。
182

183
行为附加到组件后,可以通过组件访问一个行为的**公共**成员变量或 getter 和 setter 方法定义的[属性](concept-properties.md)
184 185

```php
186 187 188
// "prop1" 是定义在行为类的属性
echo $component->prop1;
$component->prop1 = $value;
189 190
```

191
类似地也可以调用行为的**公共*方法:
192 193

```php
194 195 196
// bar() 是定义在行为类的公共方法
$component->bar();
```
197

198
如你所见,尽管 `$component` 未定义 `prop1``bar()` ,它们用起来也像组件自己定义的一样。
199

200
如果两个行为都定义了一样的属性或方法,并且它们都附加到同一个组件,那么**首先**附加上的行为在属性或方法被访问时有优先权。
201

202
附加行为到组件时的命名行为,可以使用这个名称来访问行为对象,如下所示:
203

204 205
```php
$behavior = $component->getBehavior('myBehavior');
206 207
```

208
也能获取附加到这个组件的所有行为:
209 210

```php
211 212
$behaviors = $component->getBehaviors();
```
213 214


215 216
移除行为
------------
217

218
要移除行为,可以调用 [[yii\base\Component::detachBehavior()]] 方法用行为相关联的名字实现:
219

220 221
```php
$component->detachBehavior('myBehavior1');
222 223
```

224
也可以移除**全部**行为:
225 226

```php
227
$component->detachBehaviors();
228 229 230
```


231 232
使用 `TimestampBehavior`
----------------------------
233

234
最后以 [[yii\behaviors\TimestampBehavior]] 的讲解来结尾,这个行为支持在 [[yii\db\ActiveRecord|Active Record]] 存储时自动更新它的时间戳属性。
235

236
首先,附加这个行为到计划使用该行为的 [[yii\db\ActiveRecord|Active Record]] 类:
237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262

```php
namespace app\models\User;

use yii\db\ActiveRecord;
use yii\behaviors\TimestampBehavior;

class User extends ActiveRecord
{
    // ...

    public function behaviors()
    {
        return [
            [
                'class' => TimestampBehavior::className(),
                'attributes' => [
                    ActiveRecord::EVENT_BEFORE_INSERT => ['created_at', 'updated_at'],
                    ActiveRecord::EVENT_BEFORE_UPDATE => ['updated_at'],
                ],
            ],
        ];
    }
}
```

263
以上指定的行为数组:
264

265 266
* 当记录插入时,行为将当前时间戳赋值给 `created_at``updated_at` 属性;
* 当记录更新时,行为将当前时间戳赋值给 `updated_at` 属性。
267 268


269 270 271
保存 `User` 对象,将会发现它的 `created_at``updated_at` 属性自动填充了当前时间戳:

``php
272 273 274
$user = new User;
$user->email = 'test@example.com';
$user->save();
275
echo $user->created_at;  // 显示当前时间戳
276 277
```

278
[[yii\behaviors\TimestampBehavior|TimestampBehavior]] 行为还提供了一个有用的方法 [[yii\behaviors\TimestampBehavior::touch()|touch()]],这个方法能将当前时间戳赋值给指定属性并保存到数据库:
279 280 281 282 283 284

```php
$user->touch('login_time');
```


285 286
与 PHP traits 的比较
-------------------
287

288
尽管行为在 "注入" 属性和方法到主类方面类似于 [traits](http://www.php.net/traits) ,它们在很多方面却不相同。如上所述,它们各有利弊。它们更像是互补的而不是相互替代。
289 290


291 292 293
### 行为的优势

行为类像普通类支持继承。另一方面,traits 可以视为 PHP 语言支持的复制粘贴功能,它不支持继承。
294

295
行为无须修改组件类就可动态附加到组件或移除。要使用 traits,必须修改使用它的类。
296

297
行为是可配置的而 traits 不能。
298

299
行为以响应事件来自定义组件的代码执行。
300

301
当不同行为附加到同一组件产生命名冲突时,这个冲突通过先附加行为的优先权自动解决。而由不同 traits 引发的命名冲突需要通过手工重命名冲突属性或方法来解决。
302 303


304
### traits 的优势
305

306
traits 比起行为更高效,因为行为是对象,消耗时间和内存。
307

308
IDE 对 traits 更友好,因为它们是语言结构。
309 310