Yii 2 コア・フレームワーク・コード・スタイル
下記のコード・スタイルが Yii 2.x コアと公式エクステンションの開発に用いられています。 コアに対してコードをプル・リクエストをしたいときは、これを使用することを考慮してください。 私たちは、あなたが自分のアプリケーションにこのコード・スタイルを使うことを強制するものではありません。あなたにとってより良いコード・スタイルを自由に選んでください。
なお、CodeSniffer のための設定をここで入手できます: https://github.com/yiisoft/yii2-coding-standards
全体として、私たちは PSR-2 互換のスタイルを使っていますので、 PSR-2 に適用されることは、 すべて私たちのコード・スタイルにも適用されます。
- ファイルは
<?phpまたは<?=のタグを使わなければならない。 - ファイルの末尾には一個の改行があるべきである。
- ファイルは PHP コードに対して BOM 無しの UTF-8 だけを使わなければならない。
- コードはインデントに、タブではなく、4個の空白を使わなければならない。
- クラス名は
StudlyCapsで宣言されなくてはならない。 - クラス内の定数はアンダースコアで区切られた大文字だけで宣言されなければならない。
- メソッド名は
camelCaseで宣言されなければならない。 - プロパティ名は
camelCaseで宣言されなければならない。 - プロパティ名は private である場合はアンダースコアで始まらなければならない。
else ifではなく常にelseifを使用すること。
2. ファイル
Section titled “2. ファイル”2.1. PHP タグ
Section titled “2.1. PHP タグ”- PHP コードは
<?php ?>または<?=タグを使わなければなりません。他のタグの変種、例えば<?を使ってはなりません。 - ファイルが PHP コードのみを含む場合は、末尾の
?>を含むべきではありません。 - 各行の末尾に空白を追加してはいけません。
- PHP コードを含む全てのファイルの名前は
.phpという拡張子で終るべきです。
2.2. 文字エンコーディング
Section titled “2.2. 文字エンコーディング”PHP コードは BOM 無しの UTF-8 のみを使わなければなりません。
3. クラス名
Section titled “3. クラス名”クラス名は StudlyCaps で宣言されなければなりません。例えば、Controller、Model。
4. クラス
Section titled “4. クラス”ここで “クラス” という用語はあらゆるクラスとインタフェイスを指すものとします。
- クラスは
CamelCaseで命名されなければなりません。 - 中括弧は常にクラス名の下の行に書かれるべきです。
- 全てのクラスは PHPDoc に従ったドキュメント・ブロックを持たなければなりません。
- クラス内のすべてのコードは4個の空白によってインデントされなければなりません。
- 一つの PHP ファイルにはクラスが一つだけあるべきです。
- 全てのクラスは名前空間に属すべきです。
- クラス名はファイル名と合致すべきです。クラスの名前空間はディレクトリ構造と合致すべきです。
/** * Documentation */class MyClass extends \yii\base\BaseObject implements MyInterface{ // コード}4.1. 定数
Section titled “4.1. 定数”クラスの定数はアンダースコアで区切られた大文字だけで宣言されなければなりません。 例えば、
<?phpclass Foo{ const VERSION = '1.0'; const DATE_APPROVED = '2012-06-01';}4.2. プロパティ
Section titled “4.2. プロパティ”- Public なクラス・メンバを宣言するときは
publicキーワードを明示的に指定します。 - Public および protected な変数はクラスの冒頭で、すべてのメソッドの宣言に先立って宣言されるべきです。 Private な変数もまたクラスの冒頭で宣言されるべきですが、 変数がクラスのメソッドのごく一部分にのみ関係する場合は、変数を扱う一群のメソッドの直前に追加しても構いません。
- クラスにおけるプロパティの宣言の順序は、その可視性に基づいて、 public から始まり、protected、private と続くべきです。
- 同じ可視性を持つプロパティの順序については、厳格な規則はありません。
- より読みやすいように、プロパティの宣言は空行を挟まずに続け、プロパティ宣言とメソッド宣言のブロック間には2行の空行を挟むべきです。 また、異なる可視性のグループの間に、1行の空行を追加するべきです。
- Private 変数は
$_varNameのように名付けるべきです。 - Public なクラス・メンバとスタンドアロンな変数は、
先頭を小文字にした
$camelCaseで名付けるべきです。 - 説明的な名前を使うこと。
$iや$jのような変数は使わないようにしましょう。
例えば、
<?phpclass Foo{ public $publicProp1; public $publicProp2;
protected $protectedProp;
private $_privateProp;
public function someMethod() { // ... }}4.3. メソッド
Section titled “4.3. メソッド”- 関数およびメソッドは、先頭を小文字にした
camelCaseで名付けるべきです。 - 名前は、関数の目的を示す自己説明的なものであるべきです。
- クラスのメソッドは常に修飾子
private、protectedまたはpublicを使って、可視性を宣言すべきです。varは許可されません。 - 関数の開始の中括弧は関数宣言の次の行に置くべきです。
/** * Documentation */class Foo{ /** * Documentation */ public function bar() { // コード return $value; }}4.4 PHPDoc ブロック
Section titled “4.4 PHPDoc ブロック”@param、@var、@propertyおよび@returnはbool、int、string、arrayまたはnullとして型を宣言しなければなりません。ModelまたはActiveRecordのようなクラス名を使うことも出来ます。- 型付きの配列に対しては
ClassName[]を使います。 - PHPDoc の最初の行には、メソッドの目的を記述しなければなりません。
- メソッドが何かをチェックする (たとえば、
isActive,hasClassなど) ものである場合は、最初の行はChecks whetherで始まらなければなりません。 @returnは、厳密に何が返されるのかを明示的に記述しなければなりません。
/** * Checks whether the IP is in subnet range * * @param string $ip an IPv4 or IPv6 address * @param int $cidr the CIDR lendth * @param string $range subnet in CIDR format e.g. `10.0.0.0/8` or `2001:af::/64` * @return bool whether the IP is in subnet range */ private function inRange($ip, $cidr, $range) { // ... }4.5 コンストラクタ
Section titled “4.5 コンストラクタ”- PHP 4 スタイルのコンストラクタの代りに、
__constructが使われるべきです。
- PHP の全ての型と値には小文字を使うべきです。このことは、
true、false、nullおよびarrayにも当てはまります。
既存の変数の型を変えることは悪いプラクティスであると見なされています。本当に必要でない限り、そのようなコードを書かないように努めましょう。
public function save(Transaction $transaction, $argument2 = 100){ $transaction = new Connection; // 駄目 $argument2 = 200; // 良い}5.2 文字列
Section titled “5.2 文字列”- 文字列が変数および一重引用符を含まない場合は、一重引用符を使います。
$str = 'こんな具合に。';- 文字列が一重引用符を含む場合は、余計なエスケープを避けるために二重引用符を使ってもかまいません。
$str1 = "Hello $username!";$str2 = "Hello {$username}!";下記は許可されません。
$str3 = "Hello ${username}!";文字列を連結するときは、ドットの前後に空白を追加します。
$name = 'Yii' . ' Framework';文字列が長い場合、書式は以下のようにします。
$sql = "SELECT *" . "FROM `post` " . "WHERE `id` = 121 ";5.3 配列
Section titled “5.3 配列”配列には、私たちは PHP 5.4 の短縮構文を使用しています。
- 負の数を配列のインデックスに使わないこと。
配列を宣言するときは、下記の書式を使います。
$arr = [3, 14, 15, 'Yii', 'Framework'];一つの行には多過ぎるほど要素がたくさんある場合は、
$arr = [ 3, 14, 15, 92, 6, $test, 'Yii', 'Framework',];連想配列には下記の書式を使います。
$config = [ 'name' => 'Yii', 'options' => ['usePHP' => true],];5.4 制御文
Section titled “5.4 制御文”- 制御文の条件は括弧の前と後に一つの空白を持たなければなりません。
- 括弧の中の演算子は空白によって区切られるべきです。
- 開始の中括弧は同じ行に置きます。
- 終了の中括弧は新しい行に置きます。
- 単一行の文に対しても、常に中括弧を使用します。
if ($event === null) { return new Event();}if ($event instanceof CoolEvent) { return $event->instance();}return null;
// 下記は許容されませんif (!$model && null === $event) throw new Exception('test');そうしても意味が通じる場合は、return の後の else は避けてください。
ガード条件 を使用しましょう。
$result = $this->getResult();if (empty($result)) { return true;} else { // $result を処理}これは、次の方が良いです。
$result = $this->getResult();if (empty($result)) { return true;}
// $result を処理switch
Section titled “switch”switch には下記の書式を使用します。
switch ($this->phpType) { case 'string': $a = (string) $value; break; case 'integer': case 'int': $a = (int) $value; break; case 'boolean': $a = (bool) $value; break; default: $a = null;}5.5 関数呼び出し
Section titled “5.5 関数呼び出し”doIt(2, 3);
doIt(['a' => 'b']);
doIt('a', [ 'a' => 'b', 'c' => 'd',]);5.6 無名関数 (lambda) の宣言
Section titled “5.6 無名関数 (lambda) の宣言”function/use トークンと開始括弧の間の空白に注意してください。
// 良い$n = 100;$sum = array_reduce($numbers, function ($r, $x) use ($n) { $this->doMagic(); $r += $x * $n; return $r;});
// 悪い$n = 100;$mul = array_reduce($numbers, function($r, $x) use($n) { $this->doMagic(); $r *= $x * $n; return $r;});ドキュメント
Section titled “ドキュメント”-
ドキュメントの文法については phpDoc を参照してください。
-
ドキュメントの無いコードは許容されません。
-
全てのクラス・ファイルは、ファイル・レベルの doc ブロックを各ファイルの先頭に持ち、 クラス・レベルの doc ブロックを各クラスの直前に持たなければなりません。
-
メソッドが実際に何も返さないときは
@returnを使う必要はありません。 -
yii\base\BaseObjectから派生するクラスのすべての仮想プロパティは、クラスの doc ブロックで@propertyタグでドキュメントされます。 これらの注釈は、buildディレクトリで./build php-docコマンドを走らせることにより、 対応する getter や setter の@returnや@paramタグから自動的に生成されます。 getter や setter に@propertyタグを追加することによって、これらのメソッドによって導入されるプロパティに対して ドキュメントのメッセージを明示的に与えることが出来ます。 これは@returnで記述されているのとは違う説明を与えたい場合に有用です。 下記が一例です。<?php/*** Returns the errors for all attribute or a single attribute.* @param string $attribute attribute name. Use null to retrieve errors for all attributes.* @property array An array of errors for all attributes. Empty array is returned if no error.* The result is a two-dimensional array. See [[getErrors()]] for detailed description.* @return array errors for all attributes or the specified attribute. Empty array is returned if no error.* Note that when returning errors for all attributes, the result is a two-dimensional array, like the following:* ...*/public function getErrors($attribute = null)
<?php/** * @link https://www.yiiframework.com/ * @copyright Copyright (c) 2008 Yii Software LLC * @license https://www.yiiframework.com/license/ *//** * Component is the base class that provides the *property*, *event* and *behavior* features. * * @include @yii/docs/base-Component.md * * @author Qiang Xue <qiang.xue@gmail.com> * @since 2.0 */class Component extends \yii\base\BaseObject関数 / メソッド
Section titled “関数 / メソッド”/** * Returns the list of attached event handlers for an event. * You may manipulate the returned [[Vector]] object by adding or removing handlers. * For example, * * ``` * $component->getEventHandlers($eventName)->insertAt(0, $eventHandler); * ``` * * @param string $name the event name * @return Vector list of attached event handlers for the event * @throws Exception if the event is not defined */public function getEventHandlers($name){ if (!isset($this->_e[$name])) { $this->_e[$name] = new Vector; } $this->ensureBehaviors(); return $this->_e[$name];}Markdown
Section titled “Markdown”上記の例に見られるように、phpDoc コメントの書式設定には markdown を使います。
ドキュメントの中でクラス、メソッド、プロパティをクロス・リンクするために使える追加の文法があります。
[[canSetProperty]]は、同じクラス内のcanSetPropertyメソッドまたはプロパティへのクロス・リンクを生成します。[[Component::canSetProperty]]は、同じ名前空間内のComponentクラスのcanSetPropertyメソッドへのクロス・リンクを生成します。[[yii\base\Component::canSetProperty]]は、yii\base名前空間のComponentクラスのcanSetPropertyメソッドへのクロス・リンクを生成します。[[Component]]は、同じ名前空間内のComponentクラスへのクロス・リンクを生成します。ここでも、クラス名に名前空間を追加することが可能です。
上記のリンクにクラス名やメソッド名以外のラベルを付けるためには、次の例で示されている文法を使うことが出来ます。
... as displayed in the [[header|header cell]].| の前の部分がメソッド、プロパティ、クラスへの参照であり、| の後ろの部分がリンクのラベルです。
下記の文法を使ってガイドにリンクすることも可能です。
[link to guide](guide:file-name.md)[link to guide](guide:file-name.md#subsection)- 一行コメントは
//で開始されるべきです。#は使いません。 - 一行コメントはそれ自体で一行を占めるべきです。
=== [] 対 empty()
Section titled “=== [] 対 empty()”可能な場合は empty() を使います。
複数の return ポイント
Section titled “複数の return ポイント”条件の入れ子が込み入ってきた場合は、早期の return を使います。メソッドが短いものである場合は、特に問題にしません。
self 対 static
Section titled “self 対 static”以下の場合を除いて、常に static を使います。
- 定数へのアクセスには
selfを使わなければなりません:self::MY_CONSTANT - private な静的プロパティへのアクセスには
selfを使わなければなりません:self::$_events - 再帰呼出しにおいて、拡張クラスの実装ではなく、現在のクラスの実装を再び呼び出したいときなど、合理的な理由がある場合には、
selfを使うことが許可されます。
「何かをするな」を示す値
Section titled “「何かをするな」を示す値”コンポーネントに対して「何かをしない」という設定を許可するプロパティは false の値を受け入れるべきです。null、''、または [] がそういう値であると見なされるべきではありません。
ディレクトリ/名前空間の名前
Section titled “ディレクトリ/名前空間の名前”- 小文字を使います。
- オブジェクトを表すものには複数形の名詞を使います (例えば、validators)。
- 機能や特徴を表す名前には単数形を使います (例えば、web)。
- 出来れば単一の語の名前空間にします。
- 単一の語が適切でない場合は、camelCase を使います。