0%

单例什么时候会销毁

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
📌 1. 普通 Java 单例 (应用级)
public class MySingleton {
private static MySingleton instance;
private MySingleton() {}
public static MySingleton getInstance() {
if (instance == null) instance = new MySingleton();
return instance;
}
}


instance 静态变量挂在 类加载器 上,只要这个类还在内存里,就不会销毁。

在 Android 应用进程退出时(系统杀掉应用进程),单例实例才会被回收。

如果你手动写 instance = null;,垃圾回收器(GC)可能在之后清理。

👉 结论:这种单例和应用进程同生共死。

📌 2. 带 Context 的单例
public class MyManager {
private static MyManager instance;
private Context context;
private MyManager(Context context) {
this.context = context.getApplicationContext(); // 必须用 ApplicationContext!
}
public static synchronized MyManager getInstance(Context context) {
if (instance == null) instance = new MyManager(context);
return instance;
}
}


如果保存了 Activity Context,那么 Activity 销毁时无法释放,可能导致 内存泄漏。

如果保存了 Application Context,那就和进程同生共死,进程退出才销毁。

📌 3. Kotlin object 单例
object MySingleton {
fun doSomething() {}
}


这种实现其实就是 静态类,生命周期 = 应用进程生命周期。

进程被杀时销毁。

📌 4. 依赖注入框架中的单例 (如 Dagger/Hilt)

“单例”受 容器管理,并不一定等于应用级单例。

如果是 @Singleton,一般和 Application 同生命周期。

如果是作用域内的(比如 ActivityScope),在 Activity 销毁时就会释放。

✅ 总结:

如果是自己写的 静态单例,通常 只有在应用进程被杀死时才会销毁。

如果单例里引用了 短生命周期对象(Activity、Service),那会导致泄漏,除非手动释放。

想要提前销毁,可以手动 instance = null;,然后等待 GC 回收。


单例(Singleton)生命周期等于整个进程,如果单例里直接引用了 短生命周期对象(如 Activity、Service、Context 等),在这些对象销毁后单例仍然持有引用,就会造成 内存泄漏。
✅ 预防方法
1. 不要在单例中保存短生命周期对象
2. 使用 WeakReference
如果确实要持有短生命周期对象,用 WeakReference 包装,避免强引用阻止回收。
3. 用回调时解绑
单例常做事件分发或回调管理,这种情况下要记得 在 Activity/Service 销毁时解除注册。