Upgrade to Pro
— share decks privately, control downloads, hide ads and more …
Speaker Deck
Features
Speaker Deck
PRO
Sign in
Sign up for free
Search
Search
Dagger2: Практический ликбез по работе с кинжалами
Search
Dmitry Polishuk
September 25, 2015
Programming
0
120
Dagger2: Практический ликбез по работе с кинжалами
Краткое введение по работе с Dagger2
Dmitry Polishuk
September 25, 2015
Tweet
Share
Other Decks in Programming
See All in Programming
Ruby GitHub Packages
bkuhlmann
0
630
障害対応を起点としたもっといい開発と運用のサイクル作りのためにできること / Hatena Enginner Seminar #29
polamjag
0
180
はてなにおける CSS Modules、及び CSS Modules に足りないもの / CSS Modules in Hatena, and CSS Modules missing parts
mizdra
7
930
Apache Hive 4 on Treasure Data
ryukobayashi
0
330
⼤規模⾔語モデルの拡張(RAG)が 終わったかも知れない件について
nearme_tech
23
15k
検証も兼ねて個人開発でHonoとかと向き合った話
hanetsuki
1
1k
Fragment Composition of GraphQL
quramy
7
1k
try! Swift Tokyo 初参加報告LT
hinakko2
0
220
dbtのドメイン分割による データ基盤の改善とDigdagとの連携
sakama
0
350
Snowflakeで眠ったデータを起こそう!
estie
0
120
1BRC--Nerd Sniping the Java Community
gunnarmorling
0
340
VS Code をプロダクトにどう取り込むか
onomax
1
360
Featured
See All Featured
Put a Button on it: Removing Barriers to Going Fast.
kastner
58
3.1k
Code Reviewing Like a Champion
maltzj
514
39k
Why You Should Never Use an ORM
jnunemaker
PRO
51
8.6k
The Invisible Side of Design
smashingmag
294
49k
Designing with Data
zakiwarfel
96
4.8k
The Art of Programming - Codeland 2020
erikaheidi
42
12k
A Tale of Four Properties
chriscoyier
151
22k
YesSQL, Process and Tooling at Scale
rocio
164
13k
Creatively Recalculating Your Daily Design Routine
revolveconf
210
11k
Web Components: a chance to create the future
zenorocha
305
41k
What’s in a name? Adding method to the madness
productmarketing
PRO
16
2.6k
XXLCSS - How to scale CSS and keep your sanity
sugarenia
241
1.2M
Transcript
Dagger2: Практический ликбез по работе с кинжалами Дмитрий Полищук
Что такое сильно связный код?
Начнем издалека public class NetworkClient { } private static final
class NetworkClientHolder { static NetworkClient INSTANCE = new NetworkClient(); } public FutureTask<Response> execute(Request request) { ... } public NetworkClient() { this.client = new OkHttpClient(); } public static NetworkClient getInstance() { return NetworkClientHolder.INSTANCE; }
Продолжаем public class MyFragment extends Fragment { } @Override public
void onResume() { super.onResume(); updateFeed(); } private void updateFeed() { NetworkClient client = NetworkClient.getInstance(); FutureTask<Response> response = client.execute(request); ... }
К. Киммель “Снежный ком” Разработчики Менеджеры Тестировщики
〉 Весь проект начинает зависеть именно от этих классов-синглтонов 〉
Проект становится монолитным 〉 И как следствие - его очень тяжело поддерживать Проблемы синглтонов:
НЕ ТВОЙ БРО MyActivity NetworkClient OkHttpClient MyActivity OkHttpClient NetworkClient ТВОЙ
БРО
Внедрение зависимости через конструктор public class NetworkClient { } public
FutureTask<Response> execute(Request request) { ... } public NetworkClient(Client client) { this.client = client; }
Внедрение зависимости через конструктор public class MyFragment extends Fragment {
@Override public void onResume() { super.onResume(); updateFeed(); } } private void updateFeed() { NetworkClient client = NetworkClient.getInstance(); FutureTask<Response> response = client.execute(request); ... } private void updateFeed() { Client client = new OkHttpClient(); NetworkClient client = new NetworkClient(client); FutureTask<Response> response = client.execute(request); ... }
Внедрение зависимостей c Dagger2
Создаем модуль @Module public class NetworkModule { } @Provides @Singleton
Client provideOkHttpClient() { return new OkHttpClient(); }
Модуль -‐ это конфиг
NetworkClient и constructor injection public class NetworkClient { } public
FutureTask<Response> execute(Request request) { ... } public NetworkClient(Client client) { this.client = client; } @Inject @Singleton private final Client client;
NetworkClient и setter injection public class NetworkClient { } public
FutureTask<Response> execute(Request request) { ... } public void setClient(Client client) { this.client = client; } @Inject @Singleton public NetworkClient() { } @Inject private final Client client; private Client client;
Внедрение NetworkClient с Dagger2 public class MyFragment extends Fragment {
} private void updateFeed() { Client client = new OkHttpClient(); NetworkClient client = new NetworkClient(client); FutureTask<Response> response = client.execute(request); ... } private void updateFeed() { FutureTask<Response> response = client.execute(request); ... } @Inject NetworkClient client; @Override public void onActivityCreated(Bundle state) { super.onActivityCreated(state); getComponent().inject(this); } Injection в поле
Создаем @Component @Component(modules = { NetworkModule.class }) public interface AppComponent
{ } @Singleton
Как это работает? @Generated("dagger.internal.codegen.ComponentProcessor") public final class MyFragment_MembersInjector implements MembersInjector<MyFragment>
{ private final MembersInjector<BaseFragment> supertypeInjector; private final Provider<NetworkClient> clientProvider; public MyFragment_MembersInjector(MembersInjector<BaseFragment> supertypeInjector, Provider<NetworkClient> clientProvider) { assert supertypeInjector != null; this.supertypeInjector = supertypeInjector; assert clientProvider != null; this.clientProvider = clientProvider; } @Override public void injectMembers(MyFragment instance) { if (instance == null) { throw new NullPointerException("Cannot inject members into a null reference"); } supertypeInjector.injectMembers(instance); instance.client = clientProvider.get(); } public static MembersInjector<MyFragment> create(MembersInjector<BaseFragment> supertypeInjector, Provider<NetworkClient> cli return new MyFragment_MembersInjector(supertypeInjector, clientProvider); } }
Как это работает? AppComponent component = DaggerAppComponent .builder() .networkModule(new NetworkModule())
.build(); component.inject(this);
@Component vs @Subcomponent @PerActivity @Subcomponent(modules = ActivityModule.class) public interface ActivitySubcomponent
{ } @Component(modules = { NetworkModule.class }) public interface AppComponent { ActivitySubcomponent plus(ActivityModule module); } Определяем @Subcomponent Добавляем plus()
@Qualifier @Module public class NetworkModule { @Provides @Singleton NetworkClient taxiClient(Client
client) { return new NetworkClient(client, "https://taxi.yandex.ru/api"); } @Provides @Singleton NetworkClient mapsClient(Client client) { return new NetworkClient(client, "https://maps.yandex.ru/api"); } } @Named("taxi") @Named("maps")
@Qualifier @Inject @Named("taxi") NetworkClient taxiClient; @Inject @Named("maps") NetworkClient mapsClient;
Dagger IntelliJ Plugin https://github.com/square/dagger-intellij-plugin
В чем прикол? 〉Сгенерированный код как будто бы это было
написано человеком 〉Обфускация через Proguard без единой настройки 〉Полное отсутствие рефлексии
Ресурсы 〉Dagger1: http://square.github.io/dagger/ 〉Dagger2: http://google.github.io/dagger 〉Пример миграции (U2020 на Dagger2):
https://clck.ru/9aSsV 〉Пример про тестирование: https://clck.ru/9aVu4 〉DAGGER 2 - A New Type of dependency injection: http://www.youtube.com/watch?v=oK_XtfXPkqw
Спасибо за внимание! Дмитрий Полищук @dpolishuk deepol@yandex-‐team.ru