对于某些模型,我们在MySQL中使用valid
布尔值实现了软删除。
在类中,scopes方法定义如下:
public function scopes() {
return array(
'valid'=>array(
'condition'=>"t.valid=1",
)
);
}
这样,当我们加载模型时,我们可以调用范围,使其仅包含有效(未删除)的模型以及其他查找条件,或者它恰好是什么。
这不是很干,我想知道是否有另一种方法可以实现相同的东西,也许可以应用于接口,所有模型派生的抽象Model类,或者,如果使用5.4,特质。
答案 0 :(得分:3)
Yii有一个名为Behaviors的功能 这与php 5.4特性类似,但也适用于早期版本。
SoftDeleteBehavior.php:
class SoftDeleteBehavior extends CActiveRecordBehavior {
public $deleteAttribute = 'valid';
public $deletedValue = 0;
public function beforeDelete($event) {
if ($this->deleteAttribute !== null) {
$this->getOwner()->{$this->deleteAttribute} = $this->deletedValue;
$this->getOwner()->update(array($this->deleteAttribute));
// prevent real deletion of record from database
$event->isValid = false;
}
}
/**
* Default scope to be applied to active record's default scope.
* ActiveRecord must call this from our own default scope.
* @return array the scope to be applied to default scope
*/
public function defaultScope() {
return array(
'condition' => $this->getOwner()->getTableAlias(false,false).'.'.$this->deleteAttribute
. ' <> '.var_export($this->deletedValue, true),
);
}
}
然后我有这个类从行为中应用deafultscope: ActiveRecord.php(我在这个类中有更多的方法,缺点是你需要调用父方法,如果你需要扩展方法):
class ActiveRecord extends CActiveRecord {
public function defaultScope() {
$scope = new CDbCriteria();
foreach ($this->behaviors() as $name => $value) {
$behavior = $this->asa($name);
if ($behavior->enabled && method_exists($behavior,'defaultScope')) {
$scope->mergeWith($behavior->defaultScope());
}
}
return $scope;
}
}
然后你在模特中使用它:
class MyModel extends ActiveRecord {
public function behaviors() {
return array(
'SoftDeleteBehavior' => array(
'class' => 'application.components.behaviors.SoftDeleteBehavior',
),
);
}
}
PROTIP :使用gii生成模型时可以指定自己的ActiveRecord类