はじめに
みなさん、こんにちは、この記事は私のドラフトに1年間入っています。 時間があれば、後で延期しましたが、yii2の差し迫ったリリースに関連して、私はそれを完成させて読者のレビューに載せることにしました。
3年間、私はメガフラワーの1つの非常に大きなプロジェクトに取り組んできました。 そして、開発のある時点で、クラスが多すぎて、それらの名前が
ContentDiscount 、
ItemDiscountになった
ItemDiscount 、私はそれについて何かをする必要があることに気づき、プロジェクトに名前空間を導入することにしました。 まあ、彼らが言うように、もしあなたがその
ような散歩に出たら、それからどこでもすぐに、そしてそこに少しだけではなく、他の場所ではありません。
それでは、アプリケーションのクラスの主要なタイプを「調理」する方法を見てみましょう。
基本
どこでも名前空間を使用することにしたので、
appルートをルート名前空間として選択しました(
applicationが長すぎます)。 ただし、yiiはそれを理解しないため、configで定義する必要がありました(index.phpでも可能)が、configはパスに沿って接続されていたため、初期化時に
Yii::setPathOfAlias使用できません
Yii::setPathOfAlias (状況が変わった可能性がありますか? )、index.phpを変更する必要がありました。
これが私のindex.phpの見え方です $yii=dirname(__FILE__).'/yii/framework/yii.php'; $config=dirname(__FILE__).'/protected/config/main.php';
コントローラー
ここではすべてがシンプルであるように思われます-configで示されているcontrollerNamespace、たとえば、エイリアス
appなどはすべて正常に機能します。 そしていや! とりあえず、しばらくの間、これは、コントローラーが何らかの種類のフォルダー(たとえば、testおよび
app\test名前空間)にある場合に当てはまります。 Yiiは、テストフォルダーで
app名前空間で検索します。 作業が必要であり、バグレポートを作成してプールリクエストを行う時間はありませんでした(ただし、それを行うことはできます)ので、決定を書くことにしました。 これを行うには、
CWepApplicationから継承し、
createControllerメソッドを再定義しました。 多くのコードを複製しなければならなかったので、あまり美しくありませんでしたが、プロジェクトの内部問題を解決するためにこの方法をブロックしなければなりませんでした。
ウェプアプリケーション class WebApplication extends CWebApplication {
そして、これが私たちのindex.phpが見え始めた方法です: モジュール
コントローラーは既にありますが、さらに記述できますが、モジュールに入れたい場合はどうでしょうか? モジュールは
Yii::createComponent設定によって定義されます。つまり、クラス名を手動で指定することで使用できます。
config array( 'modules'=>array( 'front'=>array( 'class'=>'front\FrontModule' ) ) )
yiiは
frontエイリアスについて何も知らないため、このメソッドは機能しません。
appエイリアスと同じ原理で設定に登録することは可能
appが、コードの冗長性を考慮してこの方法はあまり好きではありませんでした(モジュールの名前のみを書きたい)ので、簡単に変更して子孫の
CWebApplicationを変更し
CWebApplication 。
Webアプリケーション class WebApplication extends CWebApplication {
解決策は完全ではなく、バグレポートがコンパイルされます(モジュールクラス名を指定すると、yiiはそれを見つけることができませんか?
app.modules.ModuleClassの形式で
app.modules.ModuleClassする
app.modules.ModuleClass )。 今、私はこれをすべて変更し、
CWebApplicationより
CWebApplication 、たとえば、モジュールのあるフォルダー内の構成にエイリアスのインストールを置き、それをメイン構成に接続すると思います。
モジュールを把握しましたが、サブモジュールに関しては、すぐに同じ問題に直面します。 はい。また、モジュール内のコントローラーを正しく動作させるには、各モジュールの
controllerNamespaceを手動で指定する必要があります。 これを修正するには、すべてのモジュールの基本クラスを定義します。
基本モジュール class BaseModule extends \CWebModule { protected function init() { parent::init();
コードの一部は特性に入れることができますが、それはあなたにお任せします。
コンソールコマンド
初めて「名前空間」コマンドを起動できなかったとき、「CConsoleApplication」または「CConsoleCommandRunner」のいずれかに
commandNampespace似たものが
commandNampespaceませんでした(機能の作成リクエストがありますか?)。 私は
commandMapに向かって掘り始めましたが、ここでも失望が待っていました。
コードは、
ImportComクラスが見つからなかったという事実を
ImportCom 。
しかし、試行錯誤によって、実用的なソリューションが見つかりました。
この方法のマイナス面のうち、構成内のすべてのチームの絶対名を指定する必要があることに注意する必要があります。
今日、これが問題の唯一の解決策であり、他のものを見つけることができませんでした。
モデル
それでモデルに着きました。 モデルはとにかく名前空間で使用できるため、すべてがここにあるはずですが、リレーションメソッドがどのように見えるようになったかを見たとき、私はそれを修正することにしました。 最初に、各モデルでクラス名を使用して定数を定義しました
const CLASS_NAME=__CLASS_NAME__; 。
次に、ベースモデルを定義することで簡単に実行することにしました(ソリューションはyii2で調査されています)。
ベースモデルNamespaceRecord class NamespaceRecord extends CActiveRecord { public static function className() { return get_called_class(); } }
これらのアクションの後、モデルはよりシンプルで美しくなりました。
それは: public function relations(){ return array( 'country'=>'app\location\Country', ) }
次のようになりました: public function relations(){ return array( 'country'=>Country::className(), ) }
フォームにはまだ問題がありましたが、yiiはすでにこれを修正しました。
ウィジェット
長い間、この
$this->widget(' \ ')を自分のビューに書いていましたが、yii2のリリースで、ウィジェットをyii2のようにしました。 これを行うために、すべてのウィジェットの基本クラスを定義しました。
NSWidgetベースウィジェット class NSWidget extends \CWidget{ public static function begin($options=array()) { return \Yii::app()->controller->beginWidget(get_called_class(),$options); } public static function end() { return \Yii::app()->controller->endWidget(); } public static function runWidget($options=array()) { return \Yii::app()->controller->widget(get_called_class(),$options,true); } }
すべて、今ではビューに書き込むことができます echo MyWidgetNS\MyWidget::begin($options); echo MyWidgetNS\MyWidget::end();
記事に対するコメントや提案があれば、それだけです-書いて、修正します。
UPD。 本日、サブフォルダー内のコントローラーに関するバグを修正しました。 CWebApplicationを再定義する必要はもうありません。 エイリアスの場合、設定でオプションを使用することもできます(アプリケーションが修正されているため、すでに必要です)。
config 'aliases'=>array( 'app'=>'application' ),
同じオプションを使用して、モジュールの「クランチ」を捨てることができます