@JvmField注解,例如告诉编译器不要生成getter和setter,而是生成Java中成员。在伴生对象的作用域内使用该注解标记某个成员,它产生的副作用是标记这个成员不在伴生对象内部作用域,而是作为一个Java最外层类的静态成员存在。从Kotlin的角度来看,这没有什么太大区别,但是如果你看一下反编译的字节代码,你就会注意到伴生对象以及他的成员都声明和最外层类的静态成员处于同一级别。
另一个有用的注解 @JvmStatic.这个注解允许你调用伴生对象中声明的方法就像是调用外层的类的静态方法一样。但是需要注意的是:在这种情况下,方法不会和上面的成员一样移出伴生对象的内部作用域。因为编译器只是向外层类中添加一个额外的静态方法,然后在该方法内部又委托给伴生对象。
一起来看一下这个简单的Kotlin类例子:
class MyClass { companion object { @JvmStatic fun aStaticFunction() {} } }
这是相应编译后的Java简化版代码:
public class MyClass { public static final MyClass.Companion Companion = new MyClass.Companion(); fun aStaticFunction() {//外层类中添加一个额外的静态方法 Companion.aStaticFunction();//方法内部又委托给伴生对象的aStaticFunction方法 } public static final class Companion { public final void aStaticFunction() {} } }
这里存在一个非常细微的差别,但在某些特殊的情况下可能会出问题。例如,考虑一下Dagger中的module(模块)。当定义一个Dagger模块时,你可以使用静态方法去提升性能,但是如果你选择这样做,如果您的模块包含静态方法以外的任何内容,则编译将失败。由于Kotlin在类中既包含静态方法,也保留了静态伴生对象,因此无法以这种方式编写仅仅包含静态方法的Kotlin类。
但是不要那么快放弃! 这并不意味着你不能这样做,只是它需要一个稍微不同的处理方式:在这种特殊的情况下,你可以使用Kotlin单例(使用object对象表达式而不是class类)替换含有静态方法的Java类并在每个方法上使用@JvmStatic注解。如下例所示:在这种情况下,生成的字节代码不再显示任何伴生对象,静态方法会附加到类中。
@Module object MyModule {
@Provides @Singleton @JvmStatic fun provideSomething(anObject: MyObject): MyInterface { return myObject } }
这又让你再一次明白了伴生对象仅仅是单例对象的一个特例。但它至少表明与许多人的认知是相反的,你不一定需要一个伴生对象来维护静态方法或静态变量。你甚至根本不需要一个对象来维护,只要考虑顶层函数或常量:它们将作为静态成员被包含在一个自动生成的类中(默认情况下,例如MyFileKt会作为MyFile.kt文件生成的类名,一般生成类名以Kt为后缀结尾)
我们有点偏离这篇文章的主题了,所以让我们继续回到伴生对象上来。现在你已经了解了伴生对象实质就是对象,也应该意识到它开放了更多的可能性,例如继承和多态。
这意味着你的伴生对象并不是没有类型或父类的匿名对象。它不仅可以拥有父类,而且它甚至可以实现接口以及含有对象名。它不需要被称为companion。这就是为什么你可以这样写一个Parcelable类:
class ParcelableClass() : Parcelable {
constructor(parcel: Parcel) : this()
override fun writeToParcel(parcel: Parcel, flags: Int) {}
override fun describeContents() = 0
companion object CREATOR : Parcelable.Creator { override fun createFromParcel(parcel: Parcel): ParcelableClass = ParcelableClass(parcel)
override fun newArray(size: Int): Array
这里, 伴生对象名为CREATOR,它实现了Android中的Parcelable.Creator接口,允许遵守Parcelable约定,同时保持比使用@JvmField注释在伴随对象内添加Creator对象更直观。Kotlin中引入了@Parcelize注解,以便于可以获得所有样板代码,但是在这不是重点…
为了使它变得更简洁,如果你的伴生对象可以实现接口,它甚至可以使用Kotlin中的代理来执行此操作:
class MyObject { companion object : Runnable by MyRunnable() }
这将允许您同时向多个对象中添加静态方法!请注意,伴生对象在这种情况下甚至不需要作用域体,因为它是由代理提供的。
最后但同样重要的是,你可以为伴生对象定义扩展函数! 这就意味着你可以在现有的类中添加静态方法或静态属性,如下例所示:
class MyObject {
companion object
fun useCompanionExtension() { someExtension() }
}
fun MyObject.Companion.someExtension() {}//定义扩展函数
这样做有什么意义?我真的不知道。虽然Marcin Moskala建议使用此操作将静态工厂方法以Companion的扩展函数的形式添加到类中。
总而言之,伴生对象不仅仅是为了给缺少static修饰符的使用场景提供解决方案:
它们是真正的Kotlin对象,包括名称和类型,以及一些额外的功能。他们甚至可以不用于仅仅为了提供静态成员或方法场景。可以有更多其他选择,比如他们可以用作单例对象或替代顶层函数的功能。
与大多数场景一样,Kotlin意味着在你设计过程需要有一点点转变,但与Java相比,它并没有真正限制你的选择。如果有的话,也会通过提供一些新的、更简洁的方式让你去使用它。
欢迎关注Kotlin开发者联盟,这里有最新Kotlin技术文章,每周会不定期翻译一篇Kotlin国外技术文章。如果你也喜欢Kotlin,欢迎加入我们~~~
Kotlin系列文章,欢迎查看:
Kotlin邂逅设计模式系列:
当Kotlin完美邂逅设计模式之单例模式(一)
数据结构与算法系列:
每周一算法之二分查找(Kotlin描述)
Kotlin 原创系列:
教你如何完全解析Kotlin中的类型系统如何让你的回调更具Kotlin风味Jetbrains开发者日见闻(三)之Kotlin1.3新特性(inline class篇)JetBrains开发者日见闻(二)之Kotlin1.3的新特性(Contract契约与协程篇)JetBrains开发者日见闻(一)之Kotlin/Native 尝鲜篇教你如何攻克Kotlin中泛型型变的难点(实践篇)教你如何攻克Kotlin中泛型型变的难点(下篇)教你如何攻克Kotlin中泛型型变的难点(上篇)Kotlin的独门秘籍Reified实化类型参数(下篇)有关Kotlin属性代理你需要知道的一切浅谈Kotlin中的Sequences源码解析浅谈Kotlin中集合和函数式API完全解析-上篇浅谈Kotlin语法篇之lambda编译成字节码过程完全解析浅谈Kotlin语法篇之Lambda表达式完全解析浅谈Kotlin语法篇之扩展函数浅谈Kotlin语法篇之顶层函数、中缀调用、解构声明浅谈Kotlin语法篇之如何让函数更好地调用浅谈Kotlin语法篇之变量和常量浅谈Kotlin语法篇之基础语法
Effective Kotlin翻译系列
[译]Effective Kotlin系列之考虑使用原始类型的数组优化性能(五) 自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)
最后
在这里我和身边一些朋友特意整理了一份快速进阶为Android高级工程师的系统且全面的学习资料。涵盖了Android初级——Android高级架构师进阶必备的一些学习技能。
附上:我们之前因为秋招收集的二十套一二线互联网公司Android面试真题(含BAT、小米、华为、美团、滴滴)和我自己整理Android复习笔记(包含Android基础知识点、Android扩展知识点、Android源码解析、设计模式汇总、Gradle知识点、常见算法题汇总。)
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!
oid源码解析、设计模式汇总、Gradle知识点、常见算法题汇总。)
[外链图片转存中…(img-xgCxh8ac-1712304850434)]
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!
参考阅读
发表评论