深入理解Koin框架之koin-core

深入理解Koin

[toc]

koin是使用kotlin编写的一款轻量级依赖注入(DI)框架,是Android开发领域依赖注入框架的后起之秀,与目前主流的依赖注入框架Dagger2相比,它的原理更容易理解和掌握,大有取代Dagger2的趋势。本文将从实践经验出发,深入浅出的介绍koin的使用技巧和自己对koin源码的理解,希望可以为读者解决一些日常开发中的遇到问题和带来一些对koin框架原理的较为深入认识。由于是基于个人的认知和理解水平,如有疏漏和误解,恳请各位读者指正

一、Koin简介

koin是github上的一个开源项目,项目地址如下:https://github.com/InsertKoinIO/koin。本部分的内容主要来自该项目的描述。与Dagger编译时自动生成注入代码不一样的是,koin是不生成模版代码,而是使用一套DSL来定义对象的注入,因为,整个注入逻辑都可以通过源代码直观的反映出来,而Dagger2的注解方式隐含的逻辑的太多,学习成本比较搞,各种注解的作用,生成的代码的可读性等都是使用Dagger2的一些痛点,同时Dagger2不是kotlin编写,无法使用kotin的一些特性,在kotlin越来越流行的背景下,koin框架也越来越得到开发人员的认可。

由于服务端的依赖注入主要是Spring框架,所以Koin的主要应用领域还是Android平台,但从koin框架的架构来看,它也可以用于服务端和JAVA环境开发。以koin-core和koin-core-ext为基础库,koin以扩展的方式提供了android(androidx)和 ktor的支持。

核心库

  • koin-core
  • koin-core-ext

android

  • koin-android
  • koin-android-ext
  • koin-android-scope
  • koin-android-viewmodel

androidx

  • koin-androidx-ext
  • koin-androidx-fragment
  • koin-androidx-scope
  • koin-androidx-viewmodel

ktor

  • koin-ktor

koin整个工程的代码量不是很大,掌握起来不会有太大的难度。后续内容会详细介绍上述模块,同时会从实践的角度解读这些代码。

二、使用入门

koin作为一个应用框架,最重要的是了解如何使用,只有在逐步使用的过程中,随着遇到的问题一步一步的才产生对掌握原理的需求。因此,将使用入门放到正文的最前面,希望用简单的代码感性的认识一下koin是一个什么框架以及如何使用koin

2.1 依赖声明

目前koin的版本是2.1.5,以gradle的依赖声明方式来声明引入koin。引入核心包就可以开发的JAVA应用,如果是android,需要声明android扩展依赖,ktor开发同理

ext.koin_version = '2.1.5' 
//引入核心包
implementation "org.koin:koin-core:$koin_version"
implementation "org.koin:koin-core-ext:$koin_version"

//如果是android开发,引入下面的android扩展包
//implementation "org.koin:koin-android:$koin_version"
//implementation "org.koin:koin-android-scope:$koin_version"
//implementation "org.koin:koin-android-viewmodel:$koin_version"
//implementation "org.koin:koin-android-ext:$koin_version"

//引入androidx扩展
//implementation "org.koin:koin-androidx-scope:$koin_version"
//implementation "org.koin:koin-androidx-viewmodel:$koin_version"
//implementation "org.koin:koin-androidx-fragment:$koin_version"
//implementation "org.koin:koin-androidx-ext:$koin_version"

//如果是开发ktor应用,引入ktor扩展
//implementation "org.koin:koin-ktor:$koin_version"

2.2 例程

下面通过对源码中的例子coffee-maker来讲解一下koin的注入流程.

coffee-maker这个应用的逻辑很简单,有一家咖啡店(CoffeeApp)有一部咖啡机(CoffeeMaker),它可以煮咖啡(brew),同时咖啡机由加热器(Heater-接口),泵(Pump-接口)组成,在本例中加热器是一个电子加热器(ElectricHeater-实现Heater),泵是一个热虹吸式再沸器(Thermosiphon-实现Pump),因此抽象出以下基础模型:

interface Heater {
    fun on()
    fun off()
    fun isHot(): Boolean
}
class ElectricHeater : Heater {

    private var heating: Boolean = false

    override fun on() {
        println("~ ~ ~ heating ~ ~ ~")
        heating = true
    }

    override fun off() {
        heating = false
    }

    override fun isHot(): Boolean = heating
}
interface Pump {
    fun pump()
}
class Thermosiphon(private val heater: Heater) : Pump {
    override fun pump() {
        if (heater.isHot()) {
            println("=> => pumping => =>")
        }
    }
}
class CoffeeMaker(private val pump: Pump, private val heater: Heater) {

    fun brew() {
        heater.on()
        pump.pump()
        println(" [_]P coffee! [_]P ")
        heater.off()
    }
}

上面的三类模型是我们煮一杯咖啡需要用的对象,DI做目的是将类的对象生成和对象使用分离,因为类的对象生成通常是比较复杂且变动比较频繁。因此DI框架来负责对象生成,开发人员只需要告诉DI框架,我需要那种类型的对象即可,可以极大的减少开发人员的工作量。因此对于咖啡店(CoffeeApp)的来说,他只需要一台咖啡机。而咖啡机怎么组装的,有哪些零部件,他并不关心。

class CoffeeApp : KoinComponent {
    val maker: CoffeeMaker by inject()
}

这里的maker(咖啡机)不直接调用CoffeeMaker构造方法(因为调用构造方法还得先构造出Heater 和 Pump对象,太复杂了),而是使用by inject() 来向框架要一个CoffeeMaker 对象。拿到对象之后就可以调用CoffeeMaker的brew方法来煮咖啡了。

我们来看以下煮一杯咖啡的完整流程:

fun main() {
    startKoin {
        printLogger()
        modules(listOf(coffeeAppModule))
    }
    val coffeeShop = CoffeeApp()
    coffeeShop.maker.brew()
    stopKoin()
}

startKoin 函数是启动koin框架的入口,传入一个定义在KoinApplication上的一个扩展lamda类型KoinAppDeclaration来配置koin。
modules 是KoinApplication的一个成员函数,来配置这个koinApplication有哪些module,本例中传入listOf(coffeeAppModule)是配置依赖的地方。

val coffeeAppModule = module {
    single { CoffeeMaker(get(), get()) }
    single<Pump> { Thermosiphon(get()) }
    single<Heater> { ElectricHeater() }
}

module 是一个顶层函数,接受一个定义在Module上的扩展lamda类型ModuleDeclaration来配置这个module

single 是Module的一个成员方法,它接受一个定义在Scope上的扩展lamda类型Definition来配置在Scope(这里是rootScope)中生成一个具体的对象。single方法定义的对象在该scope中会缓存起来,多个地方依赖同一个对象

get是定义在Scope上的成员方法,自动进行类型推断。这里将是从rootScope中查找对应的对象定义,CoffeeMaker(get(), get())中,第一个体get会去查找Pump类的对象,第二个会去查找Heater类对象。这个两个对象正好是接下来的两行注入的。

stopKoin() 也是一个顶层函数,用于停止koin,清理koin中的对象

这个例程示范了koin的基本用法,也是koin DSL的基本语法

startKoin {//启动koin
    modules() //加载modules
}
  module  {//声明一个module
    single {//单例对象,多次依赖只产生一个对象
        Constructor()
    }
    factory {//工厂对象,每次依赖产生一个新对象
        Constructor()
    }
}
}
stopKoin() // 结束

三、 Koin进阶

相信看完例程后我们对koin有了一个比较感性的认识,它通过一个DSL来声明各个类如何生成对象(单例模式和工厂模式),如果一个类的构造方法需要依赖另外一个类的对象,则调用具有类型推断功能的get()从koin中查找这个对象即可。因此,这个module的DSL看起来非常的简洁。调用具有0个或者多个get()参数的构造方法即可。如另外一个例程中

fun perfModule400() = module {
    single { A1() } // new A1()
    single { B1(get()) }// new B1(A1)
    single { C1(get(), get()) }// new C1(A1,B1)
    single { D1(get(), get(), get()) } // new D1(A1,B1,C1)
    single { A2() } //new A2()
    single { B2(get()) } // new B2(A2)
    single { C2(get(), get()) } // new C2(A2,B2)
}

这里我们用的是最简单的注入场景,实际中可能会存在更加复杂的注入场景。下面来说明以下

3.1 Qualifer

上面的例程中,一个类只声明了一种创建对象的方法,因此koin查找的时候自然就找到这个对象,但是如果一个类需要生成多种类型的对象时,koin就无法确定使用哪一个对象。此时就需要Qualifer来标记不同的对象,相当于为创建这种对象的方式起一个别名,这与Dagger2是相似的,知识Koin并不是使用注解来实现的,而是定义了一个Qualifier接口,并提供了两个实现:

  • StringQualifier 以字符串为value的qualifier
  • TypeQualifier 以类为type,以类的全限定名为value的qualifier

同时提供了多个顶层函数来生成Qualifier,支持字符串,枚举和Class作为value,如下调用都可以生成一个qualifier,常用的是StringQualifier

name("qualifer_name")
named<T>(enum<T>)
qualifier("qualifer_name")
qualifier<T>(enum<T>)
named<T>()
qualifier<T>()

为了更好的理解和使用qualifier,我们修改以下煮咖啡场景。假设现ElectricHeater 有两种规格,高功率和低功率的,现在咖啡店需要两台咖啡机,高低功率各一台。

class SizedElectricHeater(val size:Int) : ElectricHeater() {//extend ElectricHeater
    override fun on() {
        println("~ ~ ~ heating ~ ~ ~ with ${size} W")
        heating = true
    }
}

注入配置如下:

val coffeeAppModule = module {
    single(name("high")) { CoffeeMaker(get(name("high")), get(name("high"))) }
    single(name("low")) { CoffeeMaker(get(name("low")), get(name("low"))) }
    single<Pump>(named("high")) { Thermosiphon(get(named("high"))) }
    single<Pump>(named("low")) { Thermosiphon(get(named("low"))) }
    single<Heater>(named("high")) { SizedElectricHeater(3000) }
    single<Heater>(named("low")) { SizedElectricHeater(1000) }
}
class CoffeeApp : KoinComponent {
    val lowMaker: CoffeeMaker by inject(named("low"))
    val highMaker: CoffeeMaker by inject(named("high"))
}

namedqualifer 是可以混用的,他们返回的都是StringQualifier或者TypeQualifier

3.2 Scope

注入对象都是有作用范围的,如果没有指定scope的话就是koin的一个rootScope,如果指定scope,注入时就会从该scope中去查找声明的对象
如上面的例子,由于Heater有2种功率的产品,所以在在module里面Pump,CoffeeMaker都声明了high和low,我们可以使用scope来简化这种场景

 val coffeeAppModule = module {
      scope(named("low")) {
        scoped {// this will be single
            SizedElectricHeater(1000)
        }
        scoped {
            Thermosiphon(get())
        }
        scoped {
            CoffeeMaker(get<Thermosiphon>(), get<SizedElectricHeater>())
        }
    }
    scope(named("high")) {
        scoped {// this will be single
            SizedElectricHeater(3000)
        }.bind(Heater::class)
        scoped {
            Thermosiphon(get())
        }.bind(Pump::class)
        scoped {
            CoffeeMaker(get(), get())
        }
    }
}

这里声明了两个scope, 他们的qualifier 是 name("low"),named("high"),这样配置之后,在使用这些对方的时候,可以按照下面的方式去获取

 val lowMaker by getKoin().getOrCreateScope("lowScope", named("low")).inject<CoffeeMaker>();
 val highMaker by getKoin().getOrCreateScope("highScope", named("high")).inject<CoffeeMaker>() ;

这里有几个新的函数需要说明一下

scope 是Module的成员函数,用于在一个模块内自定义scope,它接受一个qualifier来标志这个scope和一个定义在ScopeDSL上的一个lamda扩展函数,
scoped 是ScopeDSL的一个成员函数,用于定义一个single类型的实例
bind 是定义在BeanDefinition上的扩展函数,用于配置该对象的兼容的类

 scoped {
       Thermosiphon(get())
 }.bind(Pump::class)

由于ThermosiphonPump的子类,而查询对象时,查找的是Pump的对象,因此Thermosiphon这个对象定义需要bindPump::class才能找到这个对象。

3.3 Parameter

在上面的例子中,因为Heater增加了一个功率参数,导致注入配置发生一系列的变化,那如果功率的参数是在获取对象的时候传入的话,就更加灵活。Koin也是是支持在获取对象的时候传入构造参数的。Koin的构造参数是通过DefinitionParameters来描述的,它支持最多穿5个参数

scope(named("parameter2")) {
        scoped {
                (size:Int) -> SizedElectricHeater(size)
        }.bind(Heater::class)
        scoped {
            (size:Int) -> Thermosiphon(get<SizedElectricHeater> {
                parametersOf(size)
            })
        }.bind(Pump::class)
        scoped {
            (size:Int) -> CoffeeMaker(get { parametersOf(size)},get { parametersOf(size)})
        }
    }

val paramteredMaker2 by getKoin().getOrCreateScope("param2", named("parameter2")).inject<CoffeeMaker> {
        parametersOf(5500)
    };
 

在这个scope中,定义了一个带参数的SizedElectricHeaterThermosiphon构造的时候,通过通过
parametersOf传递构造参数。在最外层的app里面,才传递最终的参数5000.这里有两个语法。

  • (size:Int) -> SizedElectricHeater(size)这个表达式是定义在Scope上的lamda扩展函数,用于定义参数格式
  • parametersOf顶层函数通过一个vararg构造出一个DefinitionParameters类型的参数
3.4 Module

Module是一组关系较为密切的对象模块,它定义了一个模块内定义的对象的一些通用属性,主要有

  • createAtStart :对象是否是在startKoin的时候就创建
  • override: 当一个对象存在多个相同的定义时,后面的定义是否覆盖前面的定义
  • rootScope:当没有指定scope时,使用rootScope,当指定的scope中没有查到到定义时,会在rootScope中继续查找
  • otherScopes:自定义的scope列表,当调用module的成员方法scope的时候就可以创建一个自定义的scope

module提供两个成员方法,singlefactory来配置注入的对象

single 相当于是单例模式,多次注入为同一个对象

factory 相当于工厂模式,每次注入一个新的对象

3.5 KoinComponent

KoinComponent 是对象注入的场所,通常业务场景里需要实现这个接口,当然也可以不实现而直接使用koin,因为它只是封装了的koin的几个方法,使得应用程序可方便的获取到注入的对象。总的来说,它有三个重要方法

get 立即从koin中获取指定的对象

inject 使用延迟加载从koin中获取指定的对象

bind 获取一个同时绑定了主类和次类的对象,返回次类对象

 scope(named("high")) {
        scoped {
            SizedElectricHeater(3000)// 主类
        }.bind(Heater::class)//次类
 }
val highHeater = getKoin().getOrCreateScope("high", named("high"))
    .bind<Heater,SizedElectricHeater>()

返回一个Heater类型的对象,它实际是一个SizedElectricHeater(3000)对象

总的来说,koin是非常精巧的一个注入框架,得益于kotlin语言的扩展函数,建立了一套注入的DSL,成功的摆脱了代码生成和反射这两种比较重的实现方式。在理解它的DSL基础上,整个框架的学习成本相对来说比较低。上面讲解的Scope,Qualifier,Module,KoinComponent和Parameter这几个重要概念,其中很多也是借鉴了Dagger2的架构,因此如果有Dagger2的开发经验的话,迁移到koin也是比较容易的。从应用的角度看,熟练使用上述的几个模型,就能满足大部分的业务开发需要。而下面开始,将要深入的源码里去深入理解koin,所有模型和它的DSL

四、原理分析

本章节主要基于源码来分析koin的原理,将围绕koin的运行状态,模块加载,Scope管理,DSL中各个模型的运行机制和对象注入流程这几个核心内容来讲解。

4.1 koin的运行状态

与koin的运行状态相关的类和方法如下:

  • Koin 核心类,封装了内部关于module加载,scope管理,对象注入,属性注入等内部模块,对外暴露应用接口。
  • KoinContext(实现类GlobalContext),koin的上下文,管理一个koin对象,负责koin对象的注册和关闭,应用中可以通过KoinContext来获取koin对象
  • KoinApplication。负责配置一个koin,比如需要为koin配置哪些模块,输入哪些属性和日志平台配置,是koin的入口。
  • KoinContextHandler koinContext的持有者单例对象,可以方便的获取一个注册的koinContext。
  • ContextFunctions.kt 定义了启动koin的便捷顶层方法startKoin

koin的启动过程就是生成一个KoinApplication,它有一个koin成员变量,然后这个KoinAppliation会给koin装载一系列的module和属性,然后将这个koin交给一个KoinContext,最后将这个koinContext 存到KoinContextHandler,以便后续使用。

下面结合源码来讲解一下

  • startKoin
fun startKoin(koinContext: KoinContext = GlobalContext(), appDeclaration: KoinAppDeclaration): KoinApplication {
    KoinContextHandler.register(koinContext) //(1)
    val koinApplication = KoinApplication.init()//(2)
    KoinContextHandler.start(koinApplication)//(3)
    appDeclaration(koinApplication) //(4)
    koinApplication.createEagerInstances() //(5)
    return koinApplication
}

首先生成一个koinContext,默认值为GlobalContext(), appDeclaration 是KoinApplication上的一个扩展lamda表达式,用于配置KoinApplication

typealias KoinAppDeclaration = KoinApplication.() -> Unit

(1) 将koinContext注册到KoinContextHandler,一个应用只能注册一次,只有一个koinContext,注册之后就可以在任何地方通过KoinContextHandler.getContext()取到这个koinContext

fun register(koinContext: KoinContext) = synchronized(this) {
        if (_context != null) {
            error("A KoinContext is already started")
        }
        _context = koinContext
    }
    

(2)KoinApplication.init()是KoinApplication的一个静态工厂方法,返回一个初始化好的KoinApplication

companion object {

        /**
         * Create a new instance of KoinApplication
         */
        @JvmStatic
        fun init(): KoinApplication {
            val app = KoinApplication()
            app.init()
            return app
        }
    }

Koin是KoinApplication的成员变量,在初始化KoinApplication的时候,生成Koin的对象,

init 时创建koin的rootScope定义

    class KoinApplication private constructor() {

        val koin = Koin() // 生成koin对象

        internal fun  init() {
            koin._scopeRegistry.createRootScopeDefinition()// 创建rootScope定义
        }

(3)KoinContextHandler.start(koinApplication) 将koinApplication与koinContext绑定,

    fun start(koinApplication: KoinApplication) {
        getContext().setup(koinApplication)
    }

GlobalContext:

    override fun setup(koinApplication: KoinApplication) = synchronized(this) {
        if (_koin != null) {
            throw KoinAppAlreadyStartedException("A Koin Application has already been started")
        }
        _koin = koinApplication.koin
    }

(4)在koinApplication,Koin,KoinContext 都准备好之后,再回调appDeclaration(koinApplication)回到应用自己的配置逻辑,并将这个koinApplication传出去。下面就是对这个koinApplication的简单配置说明(printLogger 和modules 都是koinApplication的成员方法)

    startKoin {
        printLogger()// 使用System.out.print来输出日记
        modules(listOf(coffeeAppModule))// 加载modules
    }

(5)koinApplication.createEagerInstances(),应用配置按照自己的需要配置好koinapplication,接着又回到框架的逻辑,创建eager的对象(最终会进入到InstanceRegistry的createEagerInstances)


    internal fun createEagerInstances() {
        instances.values.filterIsInstance<SingleInstanceFactory<*>>()
                .filter { instance -> instance.beanDefinition.options.isCreatedAtStart }
                .forEach { instance ->
                    instance.get(
                            InstanceContext(_koin, _scope)
                    )
                }
    }
    

即将些配置了isCreateAtStart的对象,在startKoin的时候就提前创建好对象,相当于是预加载。到这里,koin的环境就完全搭建起来了,在需要注入对象的地方,可以直接通过koin的来获取。

4.2 模块加载

在上面KoinApplication的配置中说到,会调用koinApplition的modules来配置注入的模块,现在来看看加载这些模块的逻辑。

KoinApplication.modules

  fun modules(modules: List<Module>): KoinApplication {
        if (koin._logger.isAt(Level.INFO)) {
            val duration = measureDuration {
                loadModules(modules) 
            }
            val count = koin._scopeRegistry.size()
            koin._logger.info("loaded $count definitions - $duration ms")
        } else {
            loadModules(modules)//(1)
        }
        if (koin._logger.isAt(Level.INFO)) {
            val duration = measureDuration {
                koin.createRootScope()
            }
            koin._logger.info("create context - $duration ms")
        } else {
            koin.createRootScope() //(2)
        }
        return this
    }
    

modules传入一个List<Module>,然后会调用loadModuleskoin.createRootScope()

(1) KoinApplication.loadModules

    private fun loadModules(modules: List<Module>) {
        koin.loadModules(modules)
    }

进入Koin.loadModules

    fun loadModules(modules: List<Module>) = synchronized(this) {
        _modules.addAll(modules)
        _scopeRegistry.loadModules(modules)
    }

首先,将modules都存入到koin的_modules字段中,然后接着调用_scopeRegistry.loadModules

ScopeRegistry.loadModules

    internal fun loadModules(modules: Iterable<Module>) {
        modules.forEach { module ->
            if (!module.isLoaded) {
                loadModule(module)
                module.isLoaded = true
            } else {
                _koin._logger.error("module '$module' already loaded!")
            }
        }
    }

循环调用ScopeRegistry.loadModule

    private fun loadModule(module: Module) {
        declareScope(module.rootScope)
        declareScopes(module.otherScopes)
    }

每加载一个module,会将个module相关的的rootScope和otherScopes为参数调用到declareScope。module的rootScope和otherScopes都是ScopeDenifition类型的对象,经过declare之后,会生成正在的Scope对象

ScopeRegistry. declareScope

    private fun declareScope(scopeDefinition: ScopeDefinition) {
        declareDefinitions(scopeDefinition)
        declareInstances(scopeDefinition)
    }

ScopeRegistry.declareDefinitions

    private fun declareDefinitions(definition: ScopeDefinition) {
         if (scopeDefinitions.contains(definition.qualifier.value)) {
              mergeDefinitions(definition)
            } else {
                _scopeDefinitions[definition.qualifier.value] = definition.copy()
            }
     }
     

scopeDefinitions是scopeRegistry的属性,类型是Map<QualifierValue, ScopeDefinition>,如果已经存在一个qualifer相同的scopeDenifition,则进行合并,复杂将新的scopeDenifition存起来。mergeDefinitions的逻辑相对复杂。

ScopeRegistry.mergeDefinitions

    private fun mergeDefinitions(definition: ScopeDefinition) {
        val existing = scopeDefinitions[definition.qualifier.value]
            ?: error("Scope definition '$definition' not found in $_scopeDefinitions")
        definition.definitions.forEach {
            existing.save(it)
        }
    }

首先找到已经存在的那个scopeDenifition。scopeDenifition的主要结构是一个Set<BeanDenifition>,即这个scope中定义了哪些对象。合并的时候就是将传入的scopeDefinition中定义的对象save到已经存在的scopeDenifition中去.

BeanDenifition.save

    fun save(beanDefinition: BeanDefinition<*>, forceOverride: Boolean = false) {
        if (definitions.contains(beanDefinition)) {
            if (beanDefinition.options.override || forceOverride) {
                _definitions.remove(beanDefinition)
            } else {
                val current = definitions.firstOrNull { it == beanDefinition }
                throw DefinitionOverrideException("Definition '$beanDefinition' try to override existing definition. Please use override option or check for definition '$current'")
            }
        }
        _definitions.add(beanDefinition)
    }

如果之前已经存在一个相同的beanDefinition的话,需要新的beanDenition是不是强制override或者module是强制override.如果是的话,移除旧的,然后加入新的BeanDefinition,否则会报异常DefinitionOverrideException。因此override是一个比较重要的配置。我们回到前面,看看beanDefinition.options.overrideforceOverride 这两个的值是如何确定的。

(1) 首先是Module会进行全局的定义

    class Module(
        val createAtStart: Boolean,
        val override: Boolean
     )

例如:

    val overridedModule = module(override = true) {
         single { 
             ElectricHeater()
         }
    }

(2) 在BeanDenifion中,成员options中包含有override属性

    data class BeanDefinition<T>(
        val scopeDefinition: ScopeDefinition,
        val primaryType: KClass<*>,
        val qualifier: Qualifier? = null,
        val definition: Definition<T>,
        val kind: Kind,
        val secondaryTypes: List<KClass<*>> = listOf(),
        val options: Options = Options(),
        val properties: Properties = Properties(),
        val callbacks: Callbacks<T> = Callbacks()
) 
    data class Options(var isCreatedAtStart: Boolean = false, var override: Boolean = false)
    

而这个options是在mudule的single或者factory方法生成BeanDenifition的时候生成的

     inline fun <reified T> single(
            qualifier: Qualifier? = null,
            createdAtStart: Boolean = false,
            override: Boolean = false,
            noinline definition: Definition<T>
    ): BeanDefinition<T> {
        return Definitions.saveSingle(
                qualifier,
                definition,
                rootScope,
                makeOptions(override, createdAtStart)
        )
    }
    fun makeOptions(override: Boolean, createdAtStart: Boolean = false): Options =
            Options(this.createAtStart || createdAtStart, this.override || override)

因此可以看到,override的值的确定逻辑是由module和single/factory方法共同决定,任何一个为true。则BeanDefinion的override为true

上面介绍的是ScopeRegistry.declareDefinitions,是ScopeRegistry.declareScope中的一个主要分支,这个分支,将Module中定义的的BeanDefinition放到了正确的ScopeDefinition中。第二个分支是ScopeRegistry.declareInstances(scopeDefinition),这个方法将BeanDefinition转换成具体的生成对象的工厂,在注入对象的时候,将会使用这些工厂来生成具体的实例。

ScopeRegistry.declareInstances

     private fun declareInstances(scopeDefinition: ScopeDefinition) {
        _scopes.values.filter { it._scopeDefinition == scopeDefinition }.forEach { it.loadDefinitions(scopeDefinition) }
    }

_scopes是已经存在的Scope对象,初识时,scopes为空的,因此,在开始loadModules时,并不会出现scope中的BeanDefinition转换成工厂,但是如果已经存在这样的scope时,就会进入到Scope.loadDefinitions(scopeDefinition)

Scope.loadDefinitions:

    fun loadDefinitions(scopeDefinition: ScopeDefinition) {
            scopeDefinition.definitions.forEach {
             _instanceRegistry.createDefinition(it)
            }
     }

_instanceRegistry是Scope的一个InstanceRegistry类型的成员变量,这里遍历scopeDefinition中定义的BeanDefinitions(definitions),调用_instanceRegistry.createDefinition来转换成InstantFactory

InstanceRegistry. createDefinition

    internal fun createDefinition(definition: BeanDefinition<*>) {
        saveDefinition(definition, false)
    }

InstanceRegistry. saveDefinition

    fun saveDefinition(definition: BeanDefinition<*>, override: Boolean) {
        val defOverride = definition.options.override || override
        val instanceFactory = createInstanceFactory(_koin, definition) //(1)
        saveInstance(//(2)
                indexKey(definition.primaryType, definition.qualifier),
                instanceFactory,
                defOverride
        )
        definition.secondaryTypes.forEach { clazz ->//(3)
            if (defOverride) {
                saveInstance(
                        indexKey(clazz, definition.qualifier),
                        instanceFactory,
                        defOverride
                )
            } else {
                saveInstanceIfPossible(
                        indexKey(clazz, definition.qualifier),
                        instanceFactory
                )
            }
        }
    }

InstanceRegistry. createInstanceFactory

    private fun createInstanceFactory(
            _koin: Koin,
            definition: BeanDefinition<*>
    ): InstanceFactory<*> {
        return when (definition.kind) {
            Kind.Single -> SingleInstanceFactory(_koin, definition)
            Kind.Factory -> FactoryInstanceFactory(_koin, definition)
        }

(1)首先,根据single还是factory方法定义的生成不同的InstantaceFactory。

(2)将工厂关联到主类

(3) 将工厂关联到所有的次类

至此,module的加载逻辑全部结束。需要注意的是,BeanDefinition转InstanceFactory的逻辑并不是每次都会执行的。如果scope不存在的话,这部分逻辑是不执行的。还有几个疑问是,BeanDefinition的属性比较多,比如primaryType和secondaryTypes 这两个字段对BeanDefinition转InstanceFactory都有影响,那他们是什么意思,如何配置的,这个也是比较重要的内容。但不管怎么说,我们loadModule的主线逻辑我们已经分析完毕。后面会再次补充细节一些的分支逻辑。

(2)在KoinApplication.loadModules结束后,接着做的是koin.createRootScope(),创建rootScope对象

Koin.createRootScope:

  fun createRootScope() {
      _scopeRegistry.createRootScope()
  }

ScopeRegistry.createRootScope:

    internal fun createRootScope() {
        if (_rootScope == null) {
            _rootScope =
                createScope(ScopeDefinition.ROOT_SCOPE_ID, ScopeDefinition.ROOT_SCOPE_QUALIFIER, null)
        }
    }  
    

ScopeRegistry. createScope:

    fun createScope(scopeId: ScopeID, qualifier: Qualifier, source: Any? = null): Scope {
        if (scopes.contains(scopeId)) {
            throw ScopeAlreadyCreatedException("Scope with id '$scopeId' is already created")
        }

        val scopeDefinition = scopeDefinitions[qualifier.value]
        return if (scopeDefinition != null) {
            val createdScope: Scope = createScope(scopeId, scopeDefinition, source)
            _scopes[scopeId] = createdScope//(1)
            createdScope
        } else {
            throw NoScopeDefFoundException("No Scope Definition found for qualifer '${qualifier.value}'")
        }
    }
    

_scopes 是一个Map,新创建的Scope对象会存入到ScopeRegistry的_scopes里面

ScopeRegistry. createScope:

  private fun createScope(scopeId: ScopeID, scopeDefinition: ScopeDefinition, source: Any?): Scope {
        val scope = Scope(scopeId, scopeDefinition, _koin, source)
        val links = _rootScope?.let { listOf(it) } ?: emptyList()
        scope.create(links)
        return scope
    }

调用Scope的构造方法,然后将_rootScope作为links,调用scope的create方法

4.3 模块卸载

加载的module也可以卸载,卸载时,会将这个module从加载的module集合中移除,同时会将这个module中声明的所有的scope中的BeanDenifition,InstanceFactory都移除,并将module.isLoaded设置成false

fun unloadModules(modules: List<Module>) = synchronized(this) {
        _scopeRegistry.unloadModules(modules)
        _modules.removeAll(modules)
    }
    fun unloadModules(modules: List<Module>) = synchronized(this) {
        _scopeRegistry.unloadModules(modules)
        _modules.removeAll(modules)
    }
    fun unloadModules(modules: Iterable<Module>) {
        modules.forEach {
            unloadModules(it)
        }
    }
    
    fun unloadModules(module: Module) {
        val scopeDefinitions = module.otherScopes + module.rootScope
        scopeDefinitions.forEach {
            unloadDefinitions(it)
        }
        module.isLoaded = false
    }
    
    private fun unloadDefinitions(scopeDefinition: ScopeDefinition) {
        unloadInstances(scopeDefinition)
        _scopeDefinitions.values.firstOrNull { it == scopeDefinition }?.unloadDefinitions(scopeDefinition)
    }

    private fun unloadInstances(scopeDefinition: ScopeDefinition) {
        _scopes.values.filter { it._scopeDefinition == scopeDefinition }.forEach { it.dropInstances(scopeDefinition) }
    }
  

这里需要注意的是,卸载module时,module声明的scope对象和scopeDefinition对象仍然保留在ScopeRegistry中,只是这些scope中与这个module相关的BeanDefinition和InstanceFactory被清空了。

4.4 scope管理

关于Scope有两个概念,一个是ScopeDefinition,是关于scope的定义,这个是在module里面通过scope函数来定义的;另一是Scope,是根据ScopeDefinition生成的真正的Scope对象,koin 使用ScopeRegistry来管理Scope 和 ScopeDefinition,其中ScopeDefinition时在loadModule的时候注册的,在Module初始化时就会生成一个rootScope的ScopeDefinition,scope函数生成的会scopeDefinition放到module.otherScope.

Module初始化:

    class Module(
        val createAtStart: Boolean,
        val override: Boolean
    ) {
         val rootScope: ScopeDefinition = ScopeDefinition.rootDefinition()
         ......

Module.scope:

    fun scope(qualifier: Qualifier, scopeSet: ScopeDSL.() -> Unit) {
        val scopeDefinition = ScopeDefinition(qualifier)
        ScopeDSL(scopeDefinition).apply(scopeSet)
        otherScopes.add(scopeDefinition)
    }

而真正生成Scope的地方是在需要注入对象的地方。总的说来有两个地方。一个是上文说到的,在koinApplication.loadModules结束后,会创建好rootScope,另外一个是手动调用koin.createScope等方法。在讲createRootScope的时候,已经讲述了createScope函数,这里就不在重复。这里主要讲一下Scope本身属性

  • scopeId,String的别名,Scope的ID
  • _source,Any,如果在scope中没有找到对象,此时返回_soure
  • _linkedScope,关联的其他scope,主要是rootScope
  • _instanceRegistry,解析好的对象工厂管理器

Scope包含了koin的主要业务逻辑,koin的很多接口也是委托到scope来实现。ScopeRegsitry是注册Scope的组件。因此要创建或者获取一个scope都是通过ScopeRegsitry来实现的,我们来看一下scope的获取逻辑

Koin.getOrCreateScope:

    fun getOrCreateScope(scopeId: ScopeID, qualifier: Qualifier): Scope {
        return _scopeRegistry.getScopeOrNull(scopeId) ?: createScope(scopeId, qualifier)
    }

可以看到,标志一个scope需要两个维度,一个是scopeId,另外一个qualifier。首先以scopeId来查询是否有存在的scope
ScopeRegister.getScopeOrNull:

    fun getScopeOrNull(scopeId: ScopeID): Scope? {
        return scopes[scopeId]
    }

在ScopeRegistry中用一个Map来存放已经创建的scope,key为ScopeID。如果没有找到,就创建

ScopeRegister.createScope:

     fun createScope(scopeId: ScopeID, qualifier: Qualifier, source: Any? = null): Scope {
        if (scopes.contains(scopeId)) {
            throw ScopeAlreadyCreatedException("Scope with id '$scopeId' is already created")
        }

        val scopeDefinition = scopeDefinitions[qualifier.value]
        return if (scopeDefinition != null) {
            val createdScope: Scope = createScope(scopeId, scopeDefinition, source)
            _scopes[scopeId] = createdScope
            createdScope
        } else {
            throw NoScopeDefFoundException("No Scope Definition found for qualifer '${qualifier.value}'")
        }
    }

这段逻辑比较简单,如果已经存在相同的scopeId了,则抛出异常ScopeAlreadyCreatedException。否则以qualifer去查找scopeDefinition,如果存在这样的scopeDefinition,则用该scopeDefinition创建一个新的scope,然后记录下载并当作结果返回,否则报异常NoScopeDefFoundException,scopeId 对与scope本身来说是没有意义的。从这段代码我们可以看出,如果scopeId与第一个scope绑定后,如果这个scope删除后,是可以绑定到另外一个qualifiered的scope,即一个scopeId是有可能对应不同qualifer的scope的。使用时也需要注意,尽量不要出现这种情况。

scope也是可以被删除的

Koin.deleteScope:

    fun deleteScope(scopeId: ScopeID) {
         _scopeRegistry.deleteScope(scopeId)
    }   

ScopeRegister.deleteScope:

    fun deleteScope(scopeId: ScopeID) {
        _scopes.remove(scopeId)
    }
4.6 Module 和Scope的关系
  1. scope由module来定义。也就是ScopeDefinition由module来生成
  2. scope对象一旦生成,就与module没有太大关系,因为scope 和scopeDefintion对象生成后,就交给ScopeRegistry管理,scopeRegistry是koin的成员变量,因此可以不局限于定义的module内,是koin全局共享的
4.7 对象查找过程

对象的查找逻辑最终是在scope中实现

Scope.get:

    fun <T> get(
        clazz: KClass<*>,
        qualifier: Qualifier? = null,
        parameters: ParametersDefinition? = null
    ): T {
        return if (_koin._logger.isAt(Level.DEBUG)) {
            val qualifierString = qualifier?.let { " with qualifier '$qualifier'" } ?: ""
            _koin._logger.debug("+- '${clazz.getFullName()}'$qualifierString")
            val (instance: T, duration: Double) = measureDurationForResult {
                resolveInstance<T>(qualifier, clazz, parameters)
            }
            _koin._logger.debug("|- '${clazz.getFullName()}' in $duration ms")
            return instance
        } else {
            resolveInstance(qualifier, clazz, parameters)
        }
    }

入参三个:

  1. 请求对象的class,
  2. 对象的限定符qulifier,可选,默认为null
  3. 生成对象的参数,可选,默认为null

然后会进入到resolveInstance函数

    private fun <T> resolveInstance(
        qualifier: Qualifier?,
        clazz: KClass<*>,
        parameters: ParametersDefinition?
    ): T {
        if (_closed) {
            throw ClosedScopeException("Scope '$id' is closed")
        }
        //TODO Resolve in Root or link
        val indexKey = indexKey(clazz, qualifier)
        return _instanceRegistry.resolveInstance(indexKey, parameters)
            ?: findInOtherScope<T>(clazz, qualifier, parameters) ?: getFromSource(clazz)
            ?: throwDefinitionNotFound(qualifier, clazz)
    }   

查找是由三个层级:

  1. 首先在当前的scope中查找,如果没有查找,则
  2. 在linkedScope中查找,默认是rootScope,如若还没有找到,则
  3. 尝试使用source对象,如果source的类型不一致则
  4. 抛出异常。

因此需要要注意的是,一个module里声明的对象,不能依赖到另外modudle中自定义scope中声明的对象,但是可以依赖到其他module中的rootScope中声明的对象。

    val module1 = module {

    single {
        CoffeeMaker(get { parametersOf(7500) }, get { parametersOf(7500) })
    }
}   
val module3 = module(override = true) {
    single { (size: Int) ->
        SizedElectricHeater(size);
    }.bind(Heater::class)

    single { (size: Int) ->
        Thermosiphon(get<SizedElectricHeater> {
            parametersOf(size)
        })
    }.bind(Pump::class)
}

可以成功的获取到module1中定义的CoffeeMaker

    val paramteredModuel2Maker = getKoin().get<CoffeeMaker>()

但是如果是这样:

    val module1 = module {
     single {
            CoffeeMaker(get { parametersOf(7500) }, get { parametersOf(7500) })
        }
    }
        
    val module2 = module() {
       scope(named("module2")) {
         scoped { (size: Int) ->
                    SizedElectricHeater(size);
         }.bind(Heater::class)
        }
        scope(named("module2")) {
            scoped { (size: Int) ->
                    Thermosiphon(get<SizedElectricHeater> {
                    parametersOf(size)
            })
            }.bind(Pump::class)
        }
    }

则不能找到CoffeeMaker的定义。

作如下修改:

    val module1 = module {
     single(named('module2')){
            CoffeeMaker(get { parametersOf(7500) }, get { parametersOf(7500) })
        }
    }

就又可以通过如下的方式获取到CoffeeMaker对象

    val paramteredModuel2Maker = getKoin().get<CoffeeMaker>(named("module2"))

因为在module1 和module2中都声明了named("module2")这个scope,在这个scope里声明了依赖的三个对象

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