SlideShare a Scribd company logo
Android Clean
Architecture ことはじめ
Mercari, Inc.
Tomoaki Imai
08/08/2015 Android All Stars @Shibuya dots.
Tomoaki Imai
Mercari, Inc.
twitter: @tomoaki_imai
github: tomoima525
今日はみなさんに質問が
あります
胸に手を当てて振り返って
みてください
キレイな設計、できてますか?
Androidにおける設計の難しさ
• Life Cycle(画面再生成 etc)
• 数多く存在するApi
• OSごとのバージョン管理
みんな似たようなことで
悩んでる
ライフサイクル
考えるのめんどくさい
画面回転が入ると面倒
Activity, Fragmentが
どんどん肥大化する
ディレクトリ構成が
プロジェクトでまちまち
Unit/UI Testが難しい
“Android Clean Architecture”
という考え方
The Clean Architecture by Uncle Bob
http://blog.8thlight.com/uncle-bob/2012/08/13/the-clean-architecture.html
“Android Clean Architecture”
という考え方
The Clean Architecture
(Enterprise Application)
(Web Service)
+ =
Android
Clean Architecture
“Android Clean Architecture”
という考え方
http://www.slideshare.net/shinnosukekugimiya/ss-50705888
https://github.com/android10/Android-CleanArchitecture
今日みなさんに持って返って
もらいたいもの
• Android Clean Architectureの思想をさっくり
理解する
• Android Clean Architectureの実際的な実装方
法を理解する
• 既存の設計と比べてどういうメリットがある
か理解する
標準的なWeb設計の指針
“MVC Architecture”
MVC Architecture
Model
アプリケーションデータ
ビジネスロジック
View Modelをレンダリングする
Controller ユーザーイベントをハンドルする
View
Model
Controller
User
usessees
Life Cycle
Data source
Frameworks
Android Frameworks
Activity
Life Cycle
MVC Architecture on Android
Model
アプリケーションデータ
(http, DB, memory色々)
ビジネスロジック
View
Modelをレンダリングする
(端末の状態を意識しながら)
Controller
ユーザーイベントをハンドルする
(端末状態をハンドルする)
(色んなViewもハンドルする)
View
Model
Controller
User
usessees
MVCでアプリ
Sample App
• GithubからFollowerを取って
くるアプリ
• Followerのrepository見れる
• 選択したFollowerはメモリに
キャッシュし、2回目以降は
トーストで通知する
https://github.com/tomoima525/CleanArchitectureSample
/app配下 -> MVC で実装
/cleanarchitecture配下 -> Clean Architecture で実装
private GithubApi mApi;
private UserMemoryCache mUserMemoryCache;
…
mApi = createApi();
mUserMemoryCache = UserMemoryCache.getInstance();
…
public void HogeTask(…){
//ビジネスロジック…
mApi.listFollowersAsync(…);
mUserMemoryCache.put(…);

}
Models
• Modelのデータ元は多様
• ModelとControllerのインタラクション(ビジネスロジック)がテストしにく
い
public class MainActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
// create views...
}
@Override
protected void onResume(){
// resume/update views when system state is resumed
}
@Override
protected void onPause(){
// pause views when system state is paused
}
}
Views
• Viewの状態管理が必要
public class MainActivity {
// User events
public void onClick(){}
public void afterTextChanged(){}
// System events
public void onSaveInstanceState(Bundle outState){}
public void onRestoreInstanceState(Bundle inState){}
// View events
public void gotoUserDetailActivity(Context context){}
}
Controllers
• ViewとControllerがActivity上に混在している
• 画面再生成などのシステムの状態もハンドルする必要がある
MVC Architecture on Android
における課題
View(の状態管理)とControllerがActivity上に混在している
ModelとController間のインタラクション(ビジネスロジック)が
システム状態(LifeCycle)に影響を受ける/テストしにくい
Modelのデータ・ソースが多様
各層の依存関係が強い
Clean Way…?
MVC Architecture on Android
における課題
View(の状態管理)とControllerがActivity上に混在している
ModelとController間のインタラクション(ビジネスロジック)が
システム状態(LifeCycle)に影響を受ける/テストしにくい
Modelのデータ・ソースが多様
各層の依存関係が強い
MVP Architecture
MVP Architecture
• Presenterを介してModelを制御する設計
View
Presenterからの情報を
レンダリングする
-> 端末の状態は配慮不要
になる
Presenter
- Activityから分割
- ユーザーイベントを
ハンドルする
- Modelの結果をViewに
返すAndroid Frameworks
Activity
View
Model
Presenter
User
usessees
public interface ShowUserListView {
// UI として表示したい処理を定義する
void showLoading();
void hideLoading();
void showNoResultCase();
void hideNoResultCase();
void showResult(Collection<User> usersCollection);
}
Views
public abstract class Presenter {
// Activity LifeCycle
public abstract void initialize();
public abstract void resume();
public abstract void pause();
public abstract void destroy();
}
Presenters
public class ShowUserListPresenter extends Presenter {
public void setShowUserListView(ShowUserListView view){
mShowUserListView = view; //Viewを監視できるようにする
}
public void getFollowerList(String user){
mShowUserListView.showLoading(); // Activity側にCallbackする
}
}
Activity
public class MainActivity implements ShowUserListView{
@Override
public void onCreate(){
mShowUserListPresenter = new ShowUserListPresenter(…);
mShowUserListPresenter.setShowUserListView(this);
}
@Override
public void showLoading() {
//実際のUI処理はActivityで実行できる!
mListView.setVisibility(View.GONE);
mProgress.setVisibility(View.VISIBLE);
}
@Override
public void showResult(Collection<User> usersCollection) {
mListView.setVisibility(View.VISIBLE);
mUserAdapter.refresh(usersCollection);
}
}
MVC Architecture on Android
における課題
MVP Architecture
ModelとController間のインタラクション(ビジネスロジック)が
システム状態(LifeCycle)に影響を受ける/テストしにくい
Modelのデータ・ソースが多様
各層の依存関係が強い
MVC Architecture on Android
における課題
MVP Architecture
ModelとController間のインタラクション(ビジネスロジック)が
システム状態(LifeCycle)に影響を受ける/テストしにくい
Modelのデータ・ソースが多様
各層の依存関係が強い
Domain Layer
• Domain Layerにビジネスロジック
(interactorまたはuse casesと呼ぶ)を
集約する
• Modelとの処理はDomain経由で行う
• 処理は非同期で実行する
• Pure Java
Domain Layer
Domain
Android Frameworks
Activity
View
Model
Presenter
User
usessees
Domain Layer
UseCase
<<abstract>>
ベースとなるスレッド処理を実装
ConcreteUseCase
UseCaseを継承
ロジック処理の実行、Presenterへのコールバック
PostExecutionThread
(UIThread)
スレッド実行後の処理
• Domain層は大まかにUseCase, Threadにより成立
• 設計方法は多くの手法がある
public abstract class UseCase<T> {
// スレッドをキュー処理できるようにする
private ExecutorService mExecutorService
= Executors.newSingleThreadExecutor();
public void start(final T params) {
mExecutorService.submit(new Runnable() {
@Override
public void run() {
call(params);
}
});
}
//スレッド内でcallされるメソッド
abstract protected void call(T params);
}
UseCase
https://github.com/kgmyshin/Android-archを参考
public class FollowerListUseCase extends UseCase<String> {
@Override
public void execute(String user, FollowerListUseCaseCallback callback){
mCallback = callback;
this.start(user); // (1) ExecutorServiceのスレッド経由でcall()が呼ばれる
}
@Override
protected void call(String user) {
// (2) Modelからデータ取得、バリデーション等のロジック処理。
// …省略
// (3) 処理後はUIThread( PostExecutionThread )にCallbackを返す
mPostExecutionThread.post(() -> {mCallback.onUserListLoaded(usersCollection);});
}
// UIThreadに返すCallbackのinterface
interface FollowerListUseCaseCallback {
onUserListLoaded(final Collection<User> usersCollection);
onError();
}
}
ConcreteUseCase
Presenter
public class ShowUserListPresenter extends Presenter
implements FollowerListUseCaseCallback {
private FollowerListUseCase mFollowerListUseCase;
public void getFollowerList(String user){
// (1) usecaseを実行 -> 別スレッドへ
mFollowerListUseCase.execute(user, this);
}
// (2) FollowerListUseCaseCallback経由で処理結果を受け取る
@Override
onUserListLoaded(final Collection<User> usersCollection){
// Viewへ結果を渡す
}
@Override
onError(){
// error handling
}
}
MVC Architecture on Android
における課題
MVP Architecture
Domain Layer
Modelのデータ・ソースが多様
各層の依存関係が強い
MVC Architecture on Android
における課題
MVP Architecture
Domain Layer
Modelのデータ・ソースが多様
各層の依存関係が強い
Data Layer
Android Frameworks
Data Layer
• データをEntity(メソッドやデー
タ構造体のかたまり)として扱う
• Domain Layerにデータ元を意識
させないためにRepository パ
ターンで実装する
Domain
Android Frameworks
Activity
View
Data
Presenter
User
usessees
Repository Pattern
• ドメイン駆動開発で利用されるデザインパターン
• UseCaseで必要なデータソースを集約した
Repositoryクラスを実装する
Data
UserRepository DocsRepository XXRepository
public class UserRepository {
private GithubApi mApi; // Github API用のインスタンス
private UserMemoryCache mUserMemoryCache; //メモリキャッシュ用インスタンス
public UserRepository(){
mUserMemoryCache = UserMemoryCache.getInstance();
mApi = createGithubApi();
}
public void getFollowers(String user,UserListCallback callback) {
mApi.listFollowersAsync( ... // Networkから取得
}
public void putUser(User user){
mUserMemoryCache.put(… // メモリにキャッシュ
}
}
Repository
MVC Architecture on Android
における課題
MVP Architecture
Domain Layer
Data Layer(Repository Pattern)
各層の依存関係が強い
MVC Architecture on Android
における課題
MVP Architecture
Domain Layer
Data Layer(Repository Pattern)
各層の依存関係が強い
Dependency Inversion
Principle
Dependency Inversion
Principle
• 依存関係逆転の原則
• 上位モジュールは下位モジュー
ルに依存しない
例) PresenterはDomainの実装が
どのようになってても影響を受
けない
• interfaceを各層の間に置く
Android Frameworks
Domain
Android Frameworks
Activity
View
Data
Presenter
User
usessees
interface
interface
public interface UserRepository {
// Domain層(UseCase)側はこの interfaceの仕様だけ知っていればOK
void getFollowers(String userId, UserRepositoryCallback callback);
void getUser(String userId, UserRepositoryCallback callback);
void putUser(User user);
}
Dependency Inversion
//実際の実装は interfaceを継承する
public class UserRepositoryImpl implements UserRepository {
@Override
public void getFollowers(String userId, UserRepositoryCallback callback){
//データ取得 …
}
@Override
public void putUser(User user) {
}
//…
}
MVC Architecture on Android
における課題
MVP Architecture
Domain Layer
Data Layer(Repository Pattern)
Dependency Inversion Principle
どう変わった?
こう変わった!
Android Frameworks
Domain
Android Frameworks
Activity
View
Data
Presenter
User
usessees
interface
interface
UIがビジネスロジックから分離して
見通しが良くなった
ビジネスロジックがAndroidのライフサイ
クルから分離され、テストが容易に
Modelの仕様変更に柔軟に
対応可能になった
モジュールの置き換えが
容易になった
なんか難しい…?
少しずつ手をつけると
よいかも
どこから手を付けるべきか
手法 実現難易度
MVP Architecture ☆
Domain Layer ☆☆☆
Data Layer(Repository Pattern) ☆☆
Dependency Inversion Principle ☆☆
Next step…
• Dependency Injection
• 簡潔に書ける
• モジュールの載せ替えが容易
Next step…
• Implementing Observer Pattern
• Domain Layerの煩雑なスレッド処理, Callback地獄か
らの開放
参考例:
https://github.com/android10/Android-CleanArchitecture
– Clean Architecture, Uncle Bob
Architecture is about Intents,
not Frameworks
– Clean Architecture, Uncle Bob
設計とは(良いコードが書きたいっていう)
意志である!
フレームワークなんて関係ねぇ!(意訳)

More Related Content

PDF
Androidアプリ開発にクリーンアーキテクチャを取り入れよう
PDF
Jenkinsを用いたAndroidアプリビルド作業効率化
PPTX
今から始める、Windows 10&新.NETへの移行戦略
PDF
Try_to_writecode_practicaltest #atest_hack
PPTX
iOSにおけるUIテスト@potetotips
PPTX
新しい Visual Studio & .NET と新時代のアーキテクチャ
PDF
Session11 - LabVIEW NXG Training Course
PDF
.NET 最新ロードマップと今押さえておきたい技術要素
Androidアプリ開発にクリーンアーキテクチャを取り入れよう
Jenkinsを用いたAndroidアプリビルド作業効率化
今から始める、Windows 10&新.NETへの移行戦略
Try_to_writecode_practicaltest #atest_hack
iOSにおけるUIテスト@potetotips
新しい Visual Studio & .NET と新時代のアーキテクチャ
Session11 - LabVIEW NXG Training Course
.NET 最新ロードマップと今押さえておきたい技術要素

What's hot (20)

PDF
【BS4】時は来たれり。今こそ .NET 6 へ移行する時。
PDF
AndroidStudioはじめました ~Gradle編~
PDF
Kifの紹介
PDF
【BS2】.NET 6 最新アップデート
PPTX
C#の書き方
PDF
Android アプリ開発における Gradle ビルドシステム
PDF
Azure と Visual Studio で実践するモダナイゼーションとクラウド ネイティブ アプリケーション開発
PDF
受託開発でテストファーストしたらXXXを早期発見できてハイアジリティになったはなし
PDF
【BS15】.NET アップグレード アシスタントで簡単にできます! .NET Framework アプリの .NET 6 へのマイグレーション
PDF
【BS13】チーム開発がこんなにも快適に!コーディングもデバッグも GitHub 上で。 GitHub Codespaces で叶えられるシームレスな開発
PDF
Session5 - LabVIEW NXG Training Course
PDF
【BS3】Visual Studio 2022 と .NET 6 での Windows アプリ開発技術の紹介
PDF
[DevSummit2013S]Android_Multi-Version_Multi-Device
PDF
"今" 押さえておきたい! Web アプリ開発の技術トレンドとツールの進化
PPTX
GBDC 勉強会 #2 Android Studio 実践レポート
PDF
Moq & Fakes Framework を使った実践的ユニットテスト - BuildInsider
PDF
Session1 - LabVIEW NXG Training Course
PDF
第5回業開中心会議
PDF
Visual Studio 2013 と HTML5 で実現するマルチデバイス/マルチプラットフォーム アプリの開発
PDF
AndroidJUnitRunner で JUnit 4 形式のテストに移行
【BS4】時は来たれり。今こそ .NET 6 へ移行する時。
AndroidStudioはじめました ~Gradle編~
Kifの紹介
【BS2】.NET 6 最新アップデート
C#の書き方
Android アプリ開発における Gradle ビルドシステム
Azure と Visual Studio で実践するモダナイゼーションとクラウド ネイティブ アプリケーション開発
受託開発でテストファーストしたらXXXを早期発見できてハイアジリティになったはなし
【BS15】.NET アップグレード アシスタントで簡単にできます! .NET Framework アプリの .NET 6 へのマイグレーション
【BS13】チーム開発がこんなにも快適に!コーディングもデバッグも GitHub 上で。 GitHub Codespaces で叶えられるシームレスな開発
Session5 - LabVIEW NXG Training Course
【BS3】Visual Studio 2022 と .NET 6 での Windows アプリ開発技術の紹介
[DevSummit2013S]Android_Multi-Version_Multi-Device
"今" 押さえておきたい! Web アプリ開発の技術トレンドとツールの進化
GBDC 勉強会 #2 Android Studio 実践レポート
Moq & Fakes Framework を使った実践的ユニットテスト - BuildInsider
Session1 - LabVIEW NXG Training Course
第5回業開中心会議
Visual Studio 2013 と HTML5 で実現するマルチデバイス/マルチプラットフォーム アプリの開発
AndroidJUnitRunner で JUnit 4 形式のテストに移行
Ad

Similar to Android cleanarchitecture (20)

PPTX
GoF デザインパターン 2009
PDF
Next2Dで始めるゲーム開発 - Game Development Starting with Next2D
PDF
Building asp.net core blazor and elasticsearch elasticsearch using visual stu...
PDF
Android Studioの魅力
PDF
Azure IoT Edge で Custom Vision
PDF
システムのモダナイズ 落ちても良いアプリの作り方
PDF
Rx java x retrofit
PPTX
エンタープライズ分野での実践AngularJS
PDF
Build Windows ラップアップ
PDF
大きな泥のカタマリを相手にするためのアジャイルと努力と苦労 by Joe Yoder (XP祭り2014)
PPTX
Microsoft Tunnel 概要
PDF
TypeScript ファースト ステップ (v.0.9 対応版) ~ Any browser. Any host. Any OS. Open Sourc...
PDF
ソフトウェアエンジニアリングの全体とIoT時代のモデリングおよび関連する品質
PDF
運用に自動化を求めるのは間違っているだろうか
PDF
【BS7】GitHubをフル活用した開発
PPTX
13016 n分で作るtype scriptでnodejs
PDF
C#勉強会 ~ C#9の新機能 ~
PDF
cocos2d-xとネイティブ間の連携
PDF
ソーシャルアプリ勉強会(第一回資料)配布用
PDF
Tokyo GTUG Bootcamp2010
GoF デザインパターン 2009
Next2Dで始めるゲーム開発 - Game Development Starting with Next2D
Building asp.net core blazor and elasticsearch elasticsearch using visual stu...
Android Studioの魅力
Azure IoT Edge で Custom Vision
システムのモダナイズ 落ちても良いアプリの作り方
Rx java x retrofit
エンタープライズ分野での実践AngularJS
Build Windows ラップアップ
大きな泥のカタマリを相手にするためのアジャイルと努力と苦労 by Joe Yoder (XP祭り2014)
Microsoft Tunnel 概要
TypeScript ファースト ステップ (v.0.9 対応版) ~ Any browser. Any host. Any OS. Open Sourc...
ソフトウェアエンジニアリングの全体とIoT時代のモデリングおよび関連する品質
運用に自動化を求めるのは間違っているだろうか
【BS7】GitHubをフル活用した開発
13016 n分で作るtype scriptでnodejs
C#勉強会 ~ C#9の新機能 ~
cocos2d-xとネイティブ間の連携
ソーシャルアプリ勉強会(第一回資料)配布用
Tokyo GTUG Bootcamp2010
Ad

More from Tomoaki Imai (9)

PDF
Android development at mercari 2015
PDF
Tips for better CI on Android
PDF
Debot android debugging library
PDF
What I learned about communication in Sanfrancisco
PDF
Developing better debug_components
PDF
Development at Mercari
PDF
ユーザーを待たせないためにできること
PDF
US進出でのAndroid開発inメルカリ Mercari US App Development
PDF
ログ管理でウキウキAndroid Life (Log Management in Android)
Android development at mercari 2015
Tips for better CI on Android
Debot android debugging library
What I learned about communication in Sanfrancisco
Developing better debug_components
Development at Mercari
ユーザーを待たせないためにできること
US進出でのAndroid開発inメルカリ Mercari US App Development
ログ管理でウキウキAndroid Life (Log Management in Android)

Android cleanarchitecture