Kotlin Dagger2

首先:进行运行环境的配置
step1:
导入build.grable闭包配置
在需要使用Dagger2模块的build.grable文件内添加(运行环境和jar包)
implementation "com.google.dagger:dagger:2.20"
kapt "com.google.dagger:dagger-compiler:2.20"
step2:
在使用模块的build.gradle 内添加
apply plugin: 'kotlin-kapt'
主要内容
依赖注入:
在当前类运行时当前对象与某个对象发生依赖关系,把这种依赖在一个合适的时候注入“运行时”,也就是依赖注入,在没调用时不与其他对象发生依赖关系,在运行调用时就与其他对象发生依赖关系。

 @inject 和 @component : 注解 和 组件
 @Module 和 @Provides :在module 内有一个 Provides
 @Scope 和 @Singleton :作用域(使用范围) 和 使用单例模式
 @Qualifier 和 @Named :限定符
使用Dagger2 依赖注入

@Inject:
也就是注入想要使用的对象实例,注意这里注入的时实例,所以使用注入不需要实例化对象。
使用 @Inject 注解,可以不需要实例化带有依赖注入注解 @Inject 的对象。使用时需要加入lateinit关键字,这样可以在不调用时不占用多余的内存空间。
@Inject时对应出现的,被依赖对象要使用@Inject 依赖对象也要使用@Inject。
@Component:注入器,连接目标类和依赖类的桥梁
@Component注解的必须时接口或者抽象类。Component依赖关系是通过dependencies属性添加。
App必须有一个Component用来管理全局实例
简单说就是,可以通过Component访问到Module中提供的依赖注入对象。假设,如果有两个Module,AModule、BModule,如果Component只注册了AModule,而没有注册BModule,那么BModule中提供的对象,无法进行依赖注入!

@Module:
@Module用来解决的问题
1.使用第三方库的时候,如果第三方库没有使用dagger2 的@Inject并且第三方库无法修改其中的内容,那么也就无法使用@Inject注解了
2.接口不能实例化,只能通过实现类实例化。
@Module的使用
Module是一个简单工厂,是用来床创建类实例的方法,比如第三方库中的类无法修改,但是还是要使用@Inject注解的化,那么就可以把想要注解的对象的实例化过程提取到Module工厂里,这样就不怕无法使用@Inject标记实例化而无法使用@Inject注解
@Component 可以通过modules属性加入多个module,这样就可以得到多个实例化工厂中的对象。
@provides:
 @provides是在Module中使用的,Module是一个提供具体实例化对象的工厂,那么@provides就是Module中用来标注创建实例的方法。

实例化流程
 Component搜索@Inject注解的属性
 Component查找Module中以@Provides注解的对应方法,创建实例
Inject 和 Module 维度
 (一种是直接使用@Inject标记被实现对象和实现对象来实例化)
 (一种是使用Module在Module里使用@Provides注解来实例化具体的对象)
注意:
如果@Inject 和 Module 两种创建方式都存在的化,那么Module的优先级会高@Inject
 比如先去找Module中有没有实现这个对象,如果没有的话再去@Inject中找对象

具体使用:
第一种:直接使用@Inject注解标注需要实现的属性对象
第一步:在引用属性加上 @Inject注解
如:这是属性 mPresenter:T 加上@Inject注解
open class BaseMvpActivity<T: BasePresenter<out BaseView>> :BaseActivity(),BaseView {
override fun showLoading() {
}

override fun hideLoading() {
}

override fun onError() {
}
// 创建一个BasePresenter的引用
@Inject
lateinit var mPresenter: T

}
第二步:被@Inject标注的具体类型对象需要在构造函数加上@Inject标注
因为类的引用属性有被@Inject标注,既然标注了就需要实现当前引用属性的具体类,使用就需要在具体类的构造方法加上@Inject标注,如果当前的引用用来实例化多个类,那么那几个有被实现的类都需要在构造方法前加上@Inject (如:@Inject constructor())
如:RegisterPresenter是一个具体的实现类,它实现了@Inject标注的引用属性的类
class RegisterPresenter @Inject constructor(): BasePresenter<RegisterView>(){
// 这里由于使用了Dagger2 的@Inject注解所以当前类可以使用@Inject注解引用的实例对象
@Inject
lateinit var userService: UserService
fun register(mobile: String, verifyCode: String, pwd: String){
/*
业务逻辑
如:请求网络,做注册,将最后的结果返回给Activity
*/
// val userService = UserServiceImpl()
// register 就是一个被被监听者
userService.register(mobile,verifyCode,pwd)
.execute(object : BaseResoureSubscriber<Boolean>(){
override fun onNext(t: Boolean) {
mView.onRegisterResult(t)
}
})
}

fun login(mobile: String,pwd: String){
    /*
        业务逻辑
        如:请求网络,做注册,将最后的结果返回给Activity
     */
    val userService = UserServiceImpl()
    // register 就是一个被被监听者
    userService.register(mobile,"",pwd)
            .execute(object : BaseResoureSubscriber<Boolean>(){
                override fun onNext(t: Boolean) {
                    mView.onRegisterResult(t)
                }
            })
}

}

第二种,使用Module来实现接口的实现类
第一步,实现Module工厂类
@Module
class UserModule {
/**
* @param service 通过@Provides 标注之后,Dagger2就会使用递归的形式
* 去查找当前被标注方法的参数列表的具体类是否有被@Inject标注
* 如果参数列表中的具体实现类有被@Inject标注,
* 那么就将当前参数名称的属性实例化使属性带有具体的实现,
* 也就是实例化的对象。
*
* @return 它的返回类型是当前参数列表中具体类实现的接口类型
*/
@Provides
fun providesUserService(service: UserServiceImpl): UserService{
return service
}

}
第二步,使用@Inject标注Module工厂中想要使用@Provides实现的接口实现类
如:UsrServiceImpl是一个实现了UsrService接口的类
class UserServiceImpl @Inject constructor() : UserService {
// 在具体的类 UserRepository的构造方法加上标注 @Inject,
// 那么这里只要使用@Inject标注一个延迟实例化的属性,
// 只要当前被@Inject标注的属性有被调用到,那么@Inject
// 就会自动找到当前标注的属性的具体类,通过具体类去实现当前的
// 属性对象。
@Inject
lateinit var repository: UserRepository
override fun register(mobile: String, verifyCode: String, pwd: String): Flowable<Boolean> {
// val repository = UserRepository()
return Flowable.create({ it.onNext(true) }, BackpressureStrategy.BUFFER)

}

}
第三步,使用@Inject lateinit 延迟加载在调用时通过Module工厂自动实例化类
@Inject
lateinit var userService: UserService

第四步,使用Component标注把Module中得到的实例注入到具体的类中,这样被注入的实例才可以使用Module中的实例。
// 要注入的Module
@Component(modules = arrayOf(UserModule::class))
interface UserComponent {
// component 需要注入的地方,也就是主调用的地方
fun inject(activity:RegisterActivity)
}

第五步,使用内部创建的Compoent对象注入依赖,具体看以下代码。

与当前对象依赖的有Register
class RegisterActivity : BaseMvpActivity<RegisterPresenter>(),RegisterView {
override fun onRegisterResult(result: Boolean) {
toast("注册成功").show()
}

override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)
    setContentView(R.layout.activity_register)

// mPresenter = RegisterPresenter()
// mPresenter.mView = this
initInjection()
mRegisterBtn.setOnClickListener {
mPresenter.register(mMobileEt.text.toString(),mVerifyCodeEt.text.toString(),mPwdEt.text.toString())
}
}

private fun initInjection() {
    /*
            注入Module,通过builder得到Builder的引用,userModule用来检查UserModule时否为空,
        通过build得到UserModule的对象,最后通过inject将当前位置的对象注入到Component,创建于当前对象有依赖的所有对象。

比如以上使用mPresenter.register调用了UserService的实现方法,就等于register
方法得到了UserService接口的注入。
*/
DaggerUserComponent.builder().userModule(UserModule()).build().inject(this)
mPresenter.mView = this
}
}

@Scope
 @Conpe主要用于Component的组织方式,(也就是用来标注Component的作用域级别)
 管理COmponent和Module之间的匹配关系
 提高可读性,见名知意
@Singleton
 @Scope是一个接口,@Singleton是实现@Scope的一个子类
 @Singleton 是@Scope的一种默认实现
 @Singleton并没有实现单例的能力
 ApplicationComponent单例是由代码控制实现的

@Scope的使用
以Component组织方式自定义Scope

自定义Scope与Dagger2的配置流程总结:
说明:@Singleton 的具体作用是在创建 Component进行module注入的时候,
如果@Component是被@Singleton标注的,
// 比如,AppComponent可以注入的是由被@Singleton标注的module工厂
@Singleton
// @component是被@Singleton标注的,所以只能注入module中被@Singleton标注的工厂方法的实例
@Component(modules = arrayOf(AppModule::class))
那么@Component(module = arrayof(module:: class)),
这里的module工厂的工厂方法也必须是被@Singleton标注的
@Module
class AppModule(private val context: BaseApplication) {
// @Singleton 标注的后就可以使后台知道Component可以传入的是哪个module
@Singleton
// module 创建所有与BaseApplication由所依赖并且由加@Inject标注的实例对象
@Provides
fun providesActivity(): Context {
return context
}
}
自定义Scope 如果想要由多个@Singleton 来标注多个作用域,也就是多个模块那么就可以自己自定义一个@Singleton
一个ActivityScope
@Scope
@Documented
@Retention(RUNTIME)
annotation class ActivityScope
用自定义@Scope来标注module工厂方法
@Module
class ActivityModule(private val activity: Activity) {
// module 创建所有与BaseApplication由所依赖并且由加@Inject标注的实例对象
@Provides
fun providesContext(): Activity {
return activity
}
}

注意:@Singleton在标注module时,必须标注在工厂方法,不可标注在整个module类,并且如果@Singleton标注的时某个高层父类,那么其他module中的工厂方法如果有实现了这个高层父类,就不可用其他标注标注这个工厂方法
总结:dagger2 在查找创建对象的过程
step1: 创建一个module
找module中是否有创建该对象的工厂(这也就是为什么module用来处理第三方框架不能使用dagger2 的原因了,因为它可以让我们自己添加创建对象的工厂而不是
通过@Inject标注具体对象的构造方法后在其他地方使用@Injec标注有被@Inject标注的实例就可以使用dagger2创建实例而不用我们自己去创建实例)
step2:创建一个component注入
component注入是用于那些可以方便修改本地代码的场景,而第三方无法修改代码就只能用module
compenent的使用是通过将具体的类文件注入到使用的地方,那些有被@inject标的对象或者module持有对象工厂的对象都将得到创建,在使用这些对象的时候将不需要显示的创建对象。
step3:在被注入的地方或者想要使用注入对象的地方进行操作
被注入的地方使用代码
DaggerUserComponent.builder().userModule(UserModule()).build().inject(this)
;这段代码用来创建传入当前实例到module进行创建实例
;具体的执行流程
dagger2默认生成 DaggerUserCompent这里的UserCompent就是使用@Component标注进行注入对象的文件模板。
再用builder() 来得到Builder()的引用,Builder()内就存在一个userModule用来检查传入的UserModule() 对象是否为空,不为空则使用build()来得到DaggerUserComponent() 对象, 使用对象中的inject()接口方法传入当前对象到 UserComponent进行注入对象
最后,在当前类内部就可以使用module工厂内的对象

@Qualifier
注解迷失:同一个接口有多个实现鳄类,编译报错,分不清使用哪一个实现类
使用限定符分
@Named
Qualifier的一种实现方式
以名称区分使用的那种注解实现
自定义Qualifier
@Qualifier
@Retention(RetentionPolicy.RUNTIME)
annotation.class.ActivityQualifier

使用:比如一个module里有两个返回同样接口类型的工厂方法,那么在使用@Inject注解调用实现对象的时候,后台机制将不知道如何选择调用哪个工厂方法来实例化类,
这里使用了@Named() 括号内标注的就相当于一个用来区分工厂方法的标识符,第一个工厂方法在括号内使用的表示名不和第二个标识符的一样,在使用@Inject来标注本地实现的引用对象时,要想知道具体时实例化哪一个对象,就得加上@Named(),括号内的区分字符串来区分实例化的对象

定义
/**

  • @创建者 xia
  • @创建时间 2018/12/27 21:11
  • @描述 一个用于创建接口实现类实例对象的工厂
    /
    @Module
    class UserModule {
    /
    *
    • @param service 通过@Provides 标注之后,Dagger2就会使用递归的形式
    • 去查找当前被标注方法的参数列表的具体类是否有被@Inject标注
    • 如果参数列表中的具体实现类有被@Inject标注,
    • 那么就将当前参数名称的属性实例化使属性带有具体的实现,
    • 也就是实例化的对象。
    • @return 它的返回类型是当前参数列表中具体类实现的接口类型
      */
      @Named("service")
      @Provides
      fun providesUserService(service: UserServiceImpl): UserService{
      return service
      }
      @Named("service2")
      @Provides
      fun providesUserService2(service: UserServiceImpl2): UserService{
      return service
      }
      }

使用
class RegisterPresenter @Inject constructor(): BasePresenter<RegisterView>(){
/*
使用@Inject标注属性让Module工厂来实现具体的接口实现类,
因为接口中有使用@Provider标注实现了 UserService接口的类
/
@Inject
// 域变量
@field:[Named("service")]
lateinit var userService: UserService
@Inject
@field:[Named("service2")]
lateinit var userService2: UserService
fun register(mobile: String, verifyCode: String, pwd: String){
/

业务逻辑
如:请求网络,做注册,将最后的结果返回给Activity
*/
// val userService = UserServiceImpl()
// register 就是一个被被监听者
userService.register(mobile,verifyCode,pwd)
.execute(object : BaseResoureSubscriber<Boolean>(){
override fun onNext(t: Boolean) {
mView.onRegisterResult(t)
}
})
}

fun register2(mobile: String, verifyCode: String, pwd: String){
    /*
        业务逻辑
        如:请求网络,做注册,将最后的结果返回给Activity
     */

// val userService = UserServiceImpl()
// register 就是一个被被监听者
userService2.register(mobile,verifyCode,pwd)
.execute(object : BaseResoureSubscriber<Boolean>(){
override fun onNext(t: Boolean) {
mView.onRegisterResult(t)
}
})
}

fun login(mobile: String,pwd: String){
    /*
        业务逻辑
        如:请求网络,做注册,将最后的结果返回给Activity
     */
    val userService = UserServiceImpl()
    // register 就是一个被被监听者
    userService.register(mobile,"",pwd)
            .execute(object : BaseResoureSubscriber<Boolean>(){
                override fun onNext(t: Boolean) {
                    mView.onRegisterResult(t)
                }
            })
}

}

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,463评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,868评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,213评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,666评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,759评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,725评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,716评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,484评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,928评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,233评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,393评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,073评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,718评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,308评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,538评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,338评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,260评论 2 352

推荐阅读更多精彩内容