Kotlin -> object声明和object表达式

核心区别总结

特性 object声明(具名单例) object表达式(匿名对象)
实例数量 ✅ 单例(全局唯一) ❌ 每次创建新实例
声明位置 顶层或类内部) 表达式位置
生命周期 应用级 局部作用域
使用场景 全局配置、工具类、策略常量 临时实现、回调、局部定制

详细对比代码

1. 具名单例对象 - 全局声明

kotlin 复制代码
object ImmediateStart : StartStrategy {
    override operator fun invoke(block: suspend () -> Unit) {
        println("ImmediateStart: ${this.hashCode()}")
    }
}

2. 匿名对象表达式 - 局部创建

kotlin 复制代码
class TestClass {
    fun testDifference() {
        val anonymous1 = object : StartStrategy {
            override operator fun invoke(block: suspend () -> Unit) {
                println("Anonymous1: ${this.hashCode()}")
            }
        }
        
        val anonymous2 = object : StartStrategy {
            override operator fun invoke(block: suspend () -> Unit) {
                println("Anonymous2: ${this.hashCode()}")
            }
        }
        
        // 验证实例数量
        val singleton1 = ImmediateStart
        val singleton2 = ImmediateStart
        
        println("=== 单例对象测试 ===")
        println("singleton1 === singleton2: ${singleton1 === singleton2}")  // true
        singleton1 { }  // 输出相同hashCode
        singleton2 { }  // 输出相同hashCode
        
        println("\n=== 匿名对象测试 ===")
        println("anonymous1 === anonymous2: ${anonymous1 === anonymous2}")  // false
        anonymous1 { }  // 输出不同hashCode
        anonymous2 { }  // 输出不同hashCode
    }
}

字节码层面的区别

object声明编译后

kotlin 复制代码
public final class ImmediateStart implements StartStrategy {
    public static final ImmediateStart INSTANCE = new ImmediateStart();
    private ImmediateStart() {}  // 私有构造函数,确保单例
    // ...
}

object表达式编译后

java 复制代码
// 每次调用都会生成类似这样的代码:
new StartStrategy() {  // 每次new一个新实例
    @Override
    public Object invoke(Function0 block) {
        // ...
    }
}

实际应用场景对比

kotlin 复制代码
class NetworkManager {
    
    // 场景1: 全局策略配置 - 使用object声明
    companion object {
        val DEFAULT_STRATEGY = ImmediateStart  // 全局共享同一个实例
    }
    
    // 场景2: 临时策略定制 - 使用object表达式
    fun createCustomStrategy(delay: Long): StartStrategy {
        return object : StartStrategy {  // 每次调用创建新实例
            override operator fun invoke(block: suspend () -> Unit) {
                GlobalScope.launch {
                    delay(delay)
                    block()
                }
            }
        }
    }
    
    fun demonstrateUsage() {
        // 单例使用
        val strategy1 = DEFAULT_STRATEGY
        val strategy2 = DEFAULT_STRATEGY
        println("Same instance: ${strategy1 === strategy2}")  // true
        
        // 匿名对象使用
        val custom1 = createCustomStrategy(1000)
        val custom2 = createCustomStrategy(1000)
        println("Different instances: ${custom1 === custom2}")  // false
    }
}

内存和性能影响

kotlin 复制代码
class PerformanceTest {
    
    fun testMemoryUsage() {
        // 单例对象 - 内存友好
        repeat(1000) {
            val strategy = ImmediateStart  // 始终引用同一个实例
            // 没有额外内存分配
        }
        
        // 匿名对象 - 可能造成内存压力
        repeat(1000) {
            val strategy = object : StartStrategy {  // 每次创建新对象
                override operator fun invoke(block: suspend () -> Unit) {}
            }
            // 创建了1000个不同的对象实例
        }
    }
}
相关推荐
算法如诗1 分钟前
**MATLAB R2025a** 环境下,基于 **双向时间卷积网络(BITCN)+ 双向长短期记忆网络(BiLSTM)** 的多特征分类预测完整实现
开发语言·网络·matlab
k09333 分钟前
在组件外(.js文件)中使用pinia的方法2--在http.js中使用pinia
开发语言·javascript·http
我是好小孩4 分钟前
【Android】RecyclerView的高度问题、VH复用概念、多子项的实现;
android·java·网络
张彦峰ZYF5 分钟前
高并发优惠权益聚合接口的优雅实现(含超时控制 + 来源标识 + Fallback 降级)
java·后端·面试
4Forsee7 分钟前
【Android】模板化解决复杂场景的滑动冲突问题
android·java·rpc
彭同学学习日志11 分钟前
解决 Android Navigation 组件导航栏配置崩溃:从错误到实现的完整指南
android·kotlin
若水不如远方12 分钟前
深入 Dubbo 服务暴露机制:从注解到网络的完整链路剖析
java·dubbo
tanxinji13 分钟前
Netty编写Echo服务器
java·netty
法的空间16 分钟前
让 Flutter 资源管理更智能
android·flutter·ios
二川bro17 分钟前
第44节:物理引擎进阶:Bullet.js集成与高级物理模拟
开发语言·javascript·ecmascript