Model类,集中整个应用的数据和业务逻辑——场景、属性和标签:
1 /** 2 * Returns a list of scenarios and the corresponding active attributes. 3 * An active attribute is one that is subject to validation in the current scenario. 4 * 返回所有场景及与之对应的 active 属性的列表 5 * active 属性是指在默认场景中验证的 6 * The returned array should be in the following format: 7 * 返回的格式如下: 8 * ```php 9 * [ 10 * ‘scenario1‘ => [‘attribute11‘, ‘attribute12‘, ...], 11 * ‘scenario2‘ => [‘attribute21‘, ‘attribute22‘, ...], 12 * ... 13 * ] 14 * ``` 15 * 16 * By default, an active attribute is considered safe and can be massively assigned. 17 * 默认情况下,一个active属性被认为是安全的,可以被批量赋值 18 * If an attribute should NOT be massively assigned (thus considered unsafe), 19 * 如果不允许批量赋值,则在属性名称前面加“!” 20 * please prefix the attribute with an exclamation character (e.g. `‘!rank‘`). 21 * 22 * The default implementation of this method will return all scenarios found in the [[rules()]] 23 * declaration. A special scenario named [[SCENARIO_DEFAULT]] will contain all attributes 24 * found in the [[rules()]]. Each scenario will be associated with the attributes that 25 * are being validated by the validation rules that apply to the scenario. 26 * 默认情况下会返回[[rules()]]中定义的所有场景,其中[[SCENARIO_DEFAULT]] 默认场景包含所有的属性 27 * 每个场景对应的属性值将按指定的验证规则验证 28 * 29 * @return array a list of scenarios and the corresponding active attributes. 30 */ 31 public function scenarios() 32 { 33 // 默认情况下有default 的场景 34 $scenarios = [self::SCENARIO_DEFAULT => []]; 35 foreach ($this->getValidators() as $validator) { 36 // 遍历validator,取出所有提到的场景,包括 on 和 except 37 foreach ($validator->on as $scenario) { 38 $scenarios[$scenario] = []; 39 } 40 foreach ($validator->except as $scenario) { 41 $scenarios[$scenario] = []; 42 } 43 } 44 // 取出所有场景的名称 45 $names = array_keys($scenarios); 46 47 foreach ($this->getValidators() as $validator) { 48 if (empty($validator->on) && empty($validator->except)) { 49 // 如果 validator 即没有定义 on,也没有定义 except,就放到所有的场景中 50 foreach ($names as $name) { 51 // 循环 $validator 的所有属性 52 foreach ($validator->attributes as $attribute) { 53 $scenarios[$name][$attribute] = true; 54 } 55 } 56 } elseif (empty($validator->on)) { 57 // 如果没有定义 on 58 foreach ($names as $name) { 59 if (!in_array($name, $validator->except, true)) { 60 // 而且场景不在 except 中, 就将这个属性加入到相应的场景中 61 foreach ($validator->attributes as $attribute) { 62 $scenarios[$name][$attribute] = true; 63 } 64 } 65 } 66 } else { 67 // 如果定义了 on 68 foreach ($validator->on as $name) { 69 // 就将这个属性加入到 on 定义的场景中 70 foreach ($validator->attributes as $attribute) { 71 $scenarios[$name][$attribute] = true; 72 } 73 } 74 } 75 } 76 77 /** 78 * 将 $scenarios 从 79 * 80 * ~~~ 81 * [ 82 * ‘default‘ => [], 83 * ‘scenario1‘ => [‘attribute11‘ => true, ‘attribute12‘ => true, ...], 84 * ‘scenario2‘ => [‘attribute21‘ => true, ‘attribute22‘ => true, ...], 85 * ‘scenario3‘ => [], 86 * ... 87 * ] 88 * ~~~ 89 * 转化为 90 * ~~~ 91 * [ 92 * ‘default‘ => [], 93 * ‘scenario1‘ => [‘attribute11‘, ‘attribute12‘, ...], 94 * ‘scenario2‘ => [‘attribute21‘, ‘attribute22‘, ...], 95 * ... 96 * ] 97 * ~~~ 98 */ 99 foreach ($scenarios as $scenario => $attributes) { 100 // 去除掉没有属性值的场景 101 if (!empty($attributes)) { 102 // 取出场景中的属性名称 103 $scenarios[$scenario] = array_keys($attributes); 104 } 105 } 106 107 return $scenarios; 108 } 109 110 /** 111 * Returns the form name that this model class should use. 112 * 113 * 返回表单的名称,就是这个 model 的类名,类名即表名 114 * 115 * The form name is mainly used by [[\yii\widgets\ActiveForm]] to determine how to name 116 * the input fields for the attributes in a model. If the form name is "A" and an attribute 117 * name is "b", then the corresponding input name would be "A[b]". If the form name is 118 * an empty string, then the input name would be "b". 119 * 表单名称用于[[\yii\widgets\ActiveForm]]确定模型中的属性名对应的输入域的名称 120 * 如果表单名为A,属性名为b,对应的输入名为A[b],如果表单名为空,则输入名为b 121 * 122 * The purpose of the above naming schema is that for forms which contain multiple different models, 123 * the attributes of each model are grouped in sub-arrays of the POST-data and it is easier to 124 * differentiate between them. 125 * 上述命名模式的目的是,对于包含多个不同模型的表单,每个模型的属性分组在对应的表单名下的数组,能更好的区分它们 126 * 127 * By default, this method returns the model class name (without the namespace part) 128 * as the form name. You may override it when the model is used in different forms. 129 * 默认情况下该方法返回不包含命名空间的类名称,可以通过重写使用别的表名 130 * 131 * @return string the form name of this model class. 132 * @see load() 133 */ 134 public function formName() 135 { 136 // ReflectionClass 类,是php内置的反射类,用于获取类的信息,包含了一个类的有关信息 137 $reflector = new ReflectionClass($this); 138 // 获取类的短名,就是不含命名空间(namespace)的那一部分 139 return $reflector->getShortName(); 140 } 141 142 /** 143 * Returns the list of attribute names. 144 * 返回属性名的列表 145 * 146 * By default, this method returns all public non-static properties of the class. 147 * You may override this method to change the default behavior. 148 * 默认只会返回 public 且不是 static 的属性,可以重写该方法定义不同的返回类型 149 * 150 * @return array list of attribute names. 151 */ 152 public function attributes() 153 { 154 $class = new ReflectionClass($this); 155 $names = []; 156 /** 157 *ReflectionClass::getProperties — ReflectionClass反射类的方法,用于获取一组属性,有下列参数: 158 ReflectionProperty::IS_STATIC 表示返回 static 的属性。 159 ReflectionProperty::IS_PUBLIC 表示返回 public 的属性。 160 ReflectionProperty::IS_PROTECTED 表示返回 protected 的属性。 161 ReflectionProperty::IS_PRIVATE 表示返回 private 的属性。 162 */ 163 foreach ($class->getProperties(\ReflectionProperty::IS_PUBLIC) as $property) { 164 // 如果是public的属性,并且不是static的,就认为是它的attribute 165 if (!$property->isStatic()) { 166 // 获取该属性的名称 167 $names[] = $property->getName(); 168 } 169 } 170 171 return $names; 172 } 173 174 /** 175 * Returns the attribute labels. 176 * 返回属性的标签 177 * 178 * Attribute labels are mainly used for display purpose. For example, given an attribute 179 * `firstName`, we can declare a label `First Name` which is more user-friendly and can 180 * be displayed to end users. 181 * 属性标签主要用于显示目的,例如给属性‘firstName‘一个‘First Name’标签,能够更友好的显示给最终用户 182 * 183 * By default an attribute label is generated using [[generateAttributeLabel()]]. 184 * This method allows you to explicitly specify attribute labels. 185 * 默认调用[[generateAttributeLabel()]]方法生成标签 186 * Note, in order to inherit labels defined in the parent class, a child class needs to 187 * merge the parent labels with child labels using functions such as `array_merge()`. 188 * 注意,为了继承父类的标签,之类需要调用`array_merge()`合并父类的标签 189 * 190 * @return array attribute labels (name => label) 191 * @see generateAttributeLabel() 192 */ 193 public function attributeLabels() 194 { 195 return []; 196 }
时间: 2025-01-06 13:08:16