2024.12.10
“放置系”なのにサイバー攻撃を監視・検知、「統合ログ管理ツール」とは 最先端のログ管理体制を実現する方法
提供:LINE株式会社
リンクをコピー
記事をブックマーク
大石将邦氏:では、次のトピックに移りたいと思います。次はプロジェクトのマルチモジュール化についてです。そもそも、なぜマルチモジュール化が必要なんでしょうか?
プロジェクトをモジュールに分割することによって、ソースコードが機能ごとに分割されます。そうすると変更による影響範囲が把握しやすくなります。そうすることによって、例えば拡張性を高めることができるわけですね。拡張性というのは数値化するのが難しい指標かもしれませんが、一方でもっとわかりやすい指標としてはビルドスピードの向上というのが挙げられます。
LINE Androidアプリは、テストコードを抜いたプロダクションのコードだけでも160万行を超えています。これだけ大きいとビルドに掛かる時間はかなり長くなるので、モジュール化することによってビルド時間を短縮させることは我々のチーム内でも非常に重要なトピックの1つです。
さらに、比較的新しい機能であるDynamic Feature Moduleとon-demand deliveryという機能を利用することによって、アプリの機能をユーザの端末に分割してダウンロードさせることができるようになります。そうするとインストール時のアプリサイズを大きく削減することができるということにもなります。
ということで、モジュール化は基本的に良いことなんですけど、実際にやろうとすると依存性の問題とかいろいろありまして実際にやるのは意外と難しいです。そこで、我々がどのような方針でモジュール化を行っているか、その手法についてお話しようと思います。
ではまず、図で説明しましょう。モノリシックな1つの大きなプロジェクトがあり、そのクラスの関係がこの図のようになっているとします。このうち右下のFoo featureと書かれた部分をモジュールとして切り出すことを考えましょう。
まず、機能の入り口となるクラスをFacadeクラスとして抽出します。これはデザインパターンの1つであるFacadeパターンのことです。具体的な例を挙げると、例えばLINEのようなメッセージアプリにおいて自撮りカメラの部分を独立した機能として切り出す場合、そのカメラ機能を立ち上げるメソッドをFacadeクラスに用意する感じですね。
このように、ある機能を呼び出すメソッドをFacadeクラスに定義していきます。そして、その機能の外からその機能を呼び出す場合、必ずそのFacadeクラスに対して呼び出しを行うようにします。
先ほどの図に戻ると、Foo featureのコードは合計3ヶ所から呼び出されています。なのでこの3つの呼び出しをFacadeクラスを経由するように変えます。このとき、Foo featureから「外向き」の呼び出しについては今はとくに触る必要はありません。あくまでfeatureへの呼び出しをFacadeクラスとして切り出すことが重要です。
次に、先ほどのFacadeクラスをインターフェースとインプリメンテーションクラスに分割します。
そうすると、Foo featureの「外」のクラスはFooFacadeのインターフェースだけを知っていればよい状態になります。
ここまでくれば、Foo featureをモジュールとして分割することができます。app moduleはFooFacadeインターフェースだけを見ていてFoo module本体には依存しておらず、一方でFoo module本体はapp moduleに依存している状態になります。「これでモジュール化が完了です!」と言いたいところなんですが、1つ問題が残っています。
それは実際のコードでFacadeのメソッドを呼び出すときに、そのインスタンスをどうやって手に入れるのかという問題です。ここで勘の良い方は当然「DI使ったらどう?」と思っているでしょう。
ここでもう一度お尋ねします。今、Android開発者の方でDagger2を使っているという方は手を挙げていただけますか?
(会場挙手)
やっぱり多いですね。ほぼデファクトのDagger2、もしくは最近流行りのKoinなど、Android開発で使われるDIツールはいろいろありますが、マルチモジュールプロジェクトにおける今のような問題を解決するのはとても面倒くさかったりするんです。なので、我々はこの問題を解決するのに独自のライブラリを作ることにしました。
この独自ライブラリを使うと、マルチモジュールプロジェクトの依存関係が簡単に解決できます。
まず、Facadeインターフェースの方にcompanion objectとして数行のコードを書き加えます。一方で、インプリメンテーションクラスの方にはAutoServiceというアノテーションを付けます。たったこれだけで先ほどの依存関係の問題は解決します。
実際に使う場面では、このコードのように”by component(FooFacade)”と書くだけです。これでFooFacadeImplのインスタンスがどこからでも簡単に取得することができるようになります。
これで本当にFoo featureのモジュール化が完了です。このようなことをプロジェクト全体でやっていきます。
引き続き、この図の右上の部分と左下の部分をそれぞれモジュール化してみましょう。やることは同じです。それぞれについてFacadeクラスを作り、それをインターフェースとインプリメンテーションに分割します。すると最終的にこうなります。
もはやapp moduleは、それぞれのFacadeインターフェースにしかアクセスしません。また、各feature moduleがお互いを呼び出す場合も、相手のFacadeインターフェースだけを知っていて、それに対してアクセスしています。こうやって分割していくことによって、大きなモノリシックなプロジェクトをfeature moduleに分割できるのです。
実際に我々のLINE Androidプロジェクトも、このような方針で現在もモジュール化を行っています。
モジュール化の話についてまとめましょう。まず、ある機能の入り口をFacadeとして抽出し、他の機能からはそのFacadeに対してのみアクセスするようにします。そして、そのFacadeをインターフェースとインプリメンテーションの2つに分割します。最後に、我々の作ったLichという名前のライブラリによって、Facadeの依存関係を解決して完了です。
ここまで何度か話題にしましたが、我々LINE Androidプロジェクトの開発において、いくつかの基礎的なライブラリを作ってきました。これらのライブラリをまとめて、Lichという名前のオープンソースプロダクトとしてGitHub上に公開しています。
今日紹介しましたCoroutines、およびマルチモジュールの依存性解決に関するものの他にもViewModelやokhttpなどに関するライブラリたちをまとめてLichという名前でこのURLにて公開しています。ぜひ、このライブラリをチェックアウトして、みなさんの開発にも役立てていただけると幸いです。
そして気に入っていただけたら、GitHubのスターを付けていただけるとありがたいです。
最後にこのライブラリを紹介させていただいたところで、私の発表は以上となります。ありがとうございました。
(会場拍手)
LINE株式会社
2024.12.10
メールのラリー回数でわかる「評価されない人」の特徴 職場での評価を下げる行動5選
2024.12.09
10点満点中7点の部下に言うべきこと 部下を育成できない上司の特徴トップ5
2024.12.09
国内の有名ホテルでは、マグロ丼がなんと1杯「24,000円」 「良いものをより安く」を追いすぎた日本にとって値上げが重要な理由
2024.12.12
会議で発言しやすくなる「心理的安全性」を高めるには ファシリテーションがうまい人の3つの条件
2023.03.21
民間宇宙開発で高まる「飛行機とロケットの衝突」の危機...どうやって回避する?
2024.12.10
職場であえて「不機嫌」を出したほうがいいタイプ NOと言えない人のための人間関係をラクにするヒント
2024.12.12
今までとこれからで、エンジニアに求められる「スキル」の違い AI時代のエンジニアの未来と生存戦略のカギとは
PR | 2024.11.26
なぜ電話営業はなくならない?その要因は「属人化」 通話内容をデータ化するZoomのクラウドサービス活用術
PR | 2024.11.22
「闇雲なAI導入」から脱却せよ Zoom・パーソル・THE GUILD幹部が語る、従業員と顧客体験を高めるAI戦略の要諦
2024.12.11
大企業への転職前に感じた、「なんか違うかも」の違和感の正体 「親が喜ぶ」「モテそう」ではない、自分の判断基準を持つカギ