parent
a702542e72
commit
d28d4abc51
@ -0,0 +1,415 @@ |
|||||||
|
--- |
||||||
|
Android 数据持久化之 SharedPreferences |
||||||
|
--- |
||||||
|
|
||||||
|
#### 目录 |
||||||
|
|
||||||
|
1. 思维导图 |
||||||
|
|
||||||
|
2. SharedPreference |
||||||
|
- 常见问题 |
||||||
|
|
||||||
|
- 基本使用以及适用范围 |
||||||
|
- 核心原理以及源码分析 |
||||||
|
- 注意事项以及优化建议 |
||||||
|
|
||||||
|
3. 参考 |
||||||
|
|
||||||
|
#### 思维导图 |
||||||
|
|
||||||
|
![](https://github.com/Omooo/Android-Notes/blob/master/images/SharedPreferences.png?raw=true) |
||||||
|
|
||||||
|
#### SharedPreference |
||||||
|
|
||||||
|
##### 常见问题 |
||||||
|
|
||||||
|
1. SharedPreferences 是如何初始化的,它会阻塞线程嘛?如果会,是什么原因。而且每次获取 SP 对象真的会很慢吗? |
||||||
|
2. commit 和 apply 的区别,commit 一定会在主线程操作嘛? |
||||||
|
3. SP 在使用时需要注意哪些问题,以及有什么优化点呢? |
||||||
|
|
||||||
|
##### 基本使用以及适用范围 |
||||||
|
|
||||||
|
基本使用: |
||||||
|
|
||||||
|
``` |
||||||
|
SharedPreferences sharedPreferences = this.getSharedPreferences(getLocalClassName(), MODE_PRIVATE); |
||||||
|
SharedPreferences.Editor editor = sharedPreferences.edit(); |
||||||
|
editor.putString("key", "value"); |
||||||
|
editor.apply(); |
||||||
|
``` |
||||||
|
|
||||||
|
SharedPreferences 本身是一个接口,程序无法直接创建 SharedPreferences 实例,只能通过 Context 提供的 getSharedPreferences(String name, int mode) 方法来获取 SharedPreferences 实例,name 表示要存储的 xml 文件名,第二个参数直接写 Context.MODE_PRIVAT,表示该 SharedPreferences 数据只能被本应用读写。当然还有 MODE_WORLD_READABLE 等,但是已经被废弃了,因为 SharedPreference 在多进程下表现并不稳定。 |
||||||
|
|
||||||
|
适用范围: |
||||||
|
|
||||||
|
保存少量的数据,且这些数据的格式简单,适用保存应用的配置参数,但不建议使用 SP 来存储大规模数据,可能会降低性能。 |
||||||
|
|
||||||
|
##### 核心原理以及源码分析 |
||||||
|
|
||||||
|
核心原理: |
||||||
|
|
||||||
|
保存基于 XML 文件存储的 key-value 键值对数据,在 /data/data/\<package name>/shared_prefs 目录下。 |
||||||
|
|
||||||
|
SharedPreferences 本身只能获取数据而不支持存储和修改,存储修改是通过 SharedPreferences.Editor 来实现的,它们两个都只是接口,真正的实现在 SharedPreferencesImpl 和 EditorImpl 。 |
||||||
|
|
||||||
|
源码分析: |
||||||
|
|
||||||
|
在此之前,我们先看 ContextImpl 中的一个静态成员变量: |
||||||
|
|
||||||
|
```java |
||||||
|
/* |
||||||
|
* String:包名,File:SP 文件,SharedPreferencesImpl:SP 实例对象 |
||||||
|
*/ |
||||||
|
private static ArrayMap<String, ArrayMap<File, SharedPreferencesImpl>> sSharedPrefsCache; |
||||||
|
``` |
||||||
|
|
||||||
|
因为一个进程只会存在一个 ContextImpl.class 对象,所以同一个进程内的所有 SharedPreferences 都保存在这个静态列表中。我们在看它的初始化: |
||||||
|
|
||||||
|
```java |
||||||
|
private ArrayMap<File, SharedPreferencesImpl> getSharedPreferencesCacheLocked() { |
||||||
|
if (sSharedPrefsCache == null) { |
||||||
|
sSharedPrefsCache = new ArrayMap<>(); |
||||||
|
} |
||||||
|
|
||||||
|
final String packageName = getPackageName(); |
||||||
|
ArrayMap<File, SharedPreferencesImpl> packagePrefs = sSharedPrefsCache.get(packageName); |
||||||
|
if (packagePrefs == null) { |
||||||
|
packagePrefs = new ArrayMap<>(); |
||||||
|
sSharedPrefsCache.put(packageName, packagePrefs); |
||||||
|
} |
||||||
|
|
||||||
|
return packagePrefs; |
||||||
|
} |
||||||
|
``` |
||||||
|
|
||||||
|
sSharedPrefsCache 是一个 ArrayMap,它存储的是包名和 packagePrefs 的映射关系,而 packagePrefs 存储的是 SharedPreferences 文件与 SharedPreferences 实例对象之间的映射关系。 |
||||||
|
|
||||||
|
这里,可以稍微总结一下,sSharedPrefsCache 会保存加载到内存中的 SharedPreferences 对象,当用户需要获取 SP 对象的时候,首先会在 sSharedPrefsCache 中查找,如果没找到,就创建一个新的 SP 对象添加到 sSharedPrefsCache 中,并且以当前应用的包名为 key。 |
||||||
|
|
||||||
|
除此之外,需要注意的是,ContextImpl 类中并没有定义将 SharedPreferences 对象移除 sSharedPrefsCache 的方法,所以一旦加载到内存中,就会存在直至进程销毁。相对的,也就是说,SP 对象一旦加载到内存,后面任何时间使用,都是从内存中获取,不会再出现读取磁盘的情况。 |
||||||
|
|
||||||
|
然后在看 ContextImpl#getSharedPrefenerces 方法: |
||||||
|
|
||||||
|
```java |
||||||
|
public SharedPreferences getSharedPreferences(String name, int mode) { |
||||||
|
//... |
||||||
|
File file; |
||||||
|
synchronized (ContextImpl.class) { |
||||||
|
if (mSharedPrefsPaths == null) { |
||||||
|
mSharedPrefsPaths = new ArrayMap<>(); |
||||||
|
} |
||||||
|
file = mSharedPrefsPaths.get(name); |
||||||
|
if (file == null) { |
||||||
|
file = getSharedPreferencesPath(name); |
||||||
|
mSharedPrefsPaths.put(name, file); |
||||||
|
} |
||||||
|
} |
||||||
|
return getSharedPreferences(file, mode); |
||||||
|
} |
||||||
|
``` |
||||||
|
|
||||||
|
mSharedPrefsPath 记录了所有的 SP 文件,以文件名为 key,具体文件为 value 的 map 结构。根据传入的 name 查找是否有这个文件,若没有就创建并添加到 mSharedPrefsPaths 中,若存在就跳到下面这个方法了: |
||||||
|
|
||||||
|
ContextImpl#getSharedPreferences: |
||||||
|
|
||||||
|
```java |
||||||
|
public SharedPreferences getSharedPreferences(File file, int mode) { |
||||||
|
SharedPreferencesImpl sp; |
||||||
|
synchronized (ContextImpl.class) { |
||||||
|
final ArrayMap<File, SharedPreferencesImpl> cache = getSharedPreferencesCacheLocked(); |
||||||
|
sp = cache.get(file); |
||||||
|
if (sp == null) { |
||||||
|
checkMode(mode); |
||||||
|
if (getApplicationInfo().targetSdkVersion >= android.os.Build.VERSION_CODES.O) { |
||||||
|
if (isCredentialProtectedStorage() |
||||||
|
&& !getSystemService(UserManager.class) |
||||||
|
.isUserUnlockingOrUnlocked(UserHandle.myUserId())) { |
||||||
|
throw new IllegalStateException("SharedPreferences in credential encrypted " |
||||||
|
+ "storage are not available until after user is unlocked"); |
||||||
|
} |
||||||
|
} |
||||||
|
sp = new SharedPreferencesImpl(file, mode); |
||||||
|
cache.put(file, sp); |
||||||
|
return sp; |
||||||
|
} |
||||||
|
} |
||||||
|
if ((mode & Context.MODE_MULTI_PROCESS) != 0 || |
||||||
|
getApplicationInfo().targetSdkVersion < android.os.Build.VERSION_CODES.HONEYCOMB) { |
||||||
|
sp.startReloadIfChangedUnexpectedly(); |
||||||
|
} |
||||||
|
return sp; |
||||||
|
} |
||||||
|
``` |
||||||
|
|
||||||
|
原来是根据传入的 file 从 ArrayMap<File, SharedPreferencesImpl> 拿到 SharedPreferences(SharedPreferencesImpl) 实例。关键代码其实并不多,但是我还是把所有代码都贴上了,因为这里我们能看到一个兼容性问题以及多进程问题,兼容性问题是指如果在 Android O 及更高版本中,通过传入的 file 拿到的 SharedPreferences 实例为空,说明该文件目录是用户无权限访问的,会直接抛出一个异常。多进程问题是指在 Context.MODE_MULTI_PROCESS 下,可能存在记录丢失的情况。 |
||||||
|
|
||||||
|
拿到了 SharedPreferencesImpl 实例之后,看一下其构造方法: |
||||||
|
|
||||||
|
```java |
||||||
|
SharedPreferencesImpl(File file, int mode) { |
||||||
|
mFile = file; |
||||||
|
mBackupFile = makeBackupFile(file); |
||||||
|
mMode = mode; |
||||||
|
mLoaded = false; |
||||||
|
mMap = null; |
||||||
|
mThrowable = null; |
||||||
|
startLoadFromDisk(); |
||||||
|
} |
||||||
|
|
||||||
|
private void startLoadFromDisk() { |
||||||
|
synchronized (mLock) { |
||||||
|
mLoaded = false; |
||||||
|
} |
||||||
|
//直接创建一个线程来读取磁盘文件 |
||||||
|
new Thread("SharedPreferencesImpl-load") { |
||||||
|
public void run() { |
||||||
|
loadFromDisk(); |
||||||
|
} |
||||||
|
}.start(); |
||||||
|
} |
||||||
|
|
||||||
|
private void loadFromDisk() { |
||||||
|
synchronized (mLock) { |
||||||
|
if (mLoaded) { |
||||||
|
return; |
||||||
|
} |
||||||
|
if (mBackupFile.exists()) { |
||||||
|
mFile.delete(); |
||||||
|
mBackupFile.renameTo(mFile); |
||||||
|
} |
||||||
|
} |
||||||
|
//... |
||||||
|
synchronized (mLock) { |
||||||
|
mLoaded = true; |
||||||
|
mThrowable = thrown; |
||||||
|
//... |
||||||
|
} |
||||||
|
} |
||||||
|
``` |
||||||
|
|
||||||
|
果然,它是在子线程读取的磁盘文件,所以说 SP 对象初始化过程本身的确不会造成主线程的阻塞。但是真的不会阻塞嘛?这里需要注意,在读取完磁盘文件后,把 mLoaded 置为 true,继续往下看。 |
||||||
|
|
||||||
|
我们知道,写 SP 只能通过 SP.Editor,源码如下: |
||||||
|
|
||||||
|
SharedPreferencesImpl#edit: |
||||||
|
|
||||||
|
```java |
||||||
|
@Override |
||||||
|
public Editor edit() { |
||||||
|
synchronized (mLock) { |
||||||
|
awaitLoadedLocked(); |
||||||
|
} |
||||||
|
return new EditorImpl(); |
||||||
|
} |
||||||
|
|
||||||
|
private void awaitLoadedLocked() { |
||||||
|
//... |
||||||
|
while (!mLoaded) { |
||||||
|
try { |
||||||
|
mLock.wait(); |
||||||
|
} catch (InterruptedException unused) { |
||||||
|
} |
||||||
|
} |
||||||
|
//... |
||||||
|
} |
||||||
|
``` |
||||||
|
|
||||||
|
从上面代码可知,只有子线程从磁盘加载完数据之后,mLoaded 才会被置为 true,所以说虽然从磁盘读取数据是在子线程中进行并不会阻塞主线程,但是如果文件在读取之前获取某个 SharedPreferences 的值,那么主线程就可能被阻塞住,直到子线程加载完文件为止,所以说保存的 SP 文件不宜太大。 |
||||||
|
|
||||||
|
EditorImpl 就是 Editor 真正的实现类,在这里面我们能看到我们经常使用的 putXxx 方法: |
||||||
|
|
||||||
|
```java |
||||||
|
//... |
||||||
|
private final Map<String, Object> mModified = new HashMap<>(); |
||||||
|
private boolean mClear = false; |
||||||
|
|
||||||
|
@Override |
||||||
|
public Editor putString(String key, @Nullable String value) { |
||||||
|
synchronized (mEditorLock) { |
||||||
|
mModified.put(key, value); |
||||||
|
return this; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public Editor putInt(String key, int value) { |
||||||
|
synchronized (mEditorLock) { |
||||||
|
mModified.put(key, value); |
||||||
|
return this; |
||||||
|
} |
||||||
|
} |
||||||
|
``` |
||||||
|
|
||||||
|
然后就是执行提交操作了,分两种,一种是 commit,一种是 apply,这里我把两个方法放在一块展示,便于查看区别: |
||||||
|
|
||||||
|
EditorImpl#commit / apply: |
||||||
|
|
||||||
|
```java |
||||||
|
public boolean commit() { |
||||||
|
//1. 提交修改到内存中 |
||||||
|
MemoryCommitResult mcr = commitToMemory(); |
||||||
|
//2. 调用 enqueueDiskWrite 方法,注意第二个参数为 null |
||||||
|
SharedPreferencesImpl.this.enqueueDiskWrite( |
||||||
|
mcr, null /* sync write on this thread okay */); |
||||||
|
try { |
||||||
|
mcr.writtenToDiskLatch.await(); |
||||||
|
} catch (InterruptedException e) { |
||||||
|
return false; |
||||||
|
} finally { |
||||||
|
//... |
||||||
|
} |
||||||
|
notifyListeners(mcr); |
||||||
|
return mcr.writeToDiskResult; |
||||||
|
} |
||||||
|
|
||||||
|
public void apply() { |
||||||
|
//1. 提交修改到内存中 |
||||||
|
final MemoryCommitResult mcr = commitToMemory(); |
||||||
|
final Runnable awaitCommit = new Runnable() { |
||||||
|
@Override |
||||||
|
public void run() { |
||||||
|
try { |
||||||
|
mcr.writtenToDiskLatch.await(); |
||||||
|
} catch (InterruptedException ignored) { |
||||||
|
} |
||||||
|
} |
||||||
|
}; |
||||||
|
|
||||||
|
QueuedWork.addFinisher(awaitCommit); |
||||||
|
|
||||||
|
Runnable postWriteRunnable = new Runnable() { |
||||||
|
@Override |
||||||
|
public void run() { |
||||||
|
awaitCommit.run(); |
||||||
|
QueuedWork.removeFinisher(awaitCommit); |
||||||
|
} |
||||||
|
}; |
||||||
|
//2. 调用 enqueueDiskWrite 方法,注意第二个参数为 postWriteRunnable |
||||||
|
SharedPreferencesImpl.this.enqueueDiskWrite(mcr, postWriteRunnable); |
||||||
|
notifyListeners(mcr); |
||||||
|
} |
||||||
|
``` |
||||||
|
|
||||||
|
可以看到 commit 和 apply 都会先把修改提交到内存中,然后在在通过 enqueueDiskWrite 将要写入磁盘的任务进行排队, commitToMemory 方法源码就不贴了,其实就是将数据插入到 mMap 中,这是对内存中的数据进行更新,着重看一下 enqueueDiskWrite 方法: |
||||||
|
|
||||||
|
SharedPreferencesImpl#enqueueDiskWrite 方法: |
||||||
|
|
||||||
|
```java |
||||||
|
/** |
||||||
|
* Enqueue an already-committed-to-memory result to be written |
||||||
|
* to disk. |
||||||
|
* |
||||||
|
* They will be written to disk one-at-a-time in the order |
||||||
|
* that they're enqueued. |
||||||
|
* |
||||||
|
* @param postWriteRunnable if non-null, we're being called |
||||||
|
* from apply() and this is the runnable to run after |
||||||
|
* the write proceeds. if null (from a regular commit()), |
||||||
|
* then we're allowed to do this disk write on the main |
||||||
|
* thread (which in addition to reducing allocations and |
||||||
|
* creating a background thread, this has the advantage that |
||||||
|
* we catch them in userdebug StrictMode reports to convert |
||||||
|
* them where possible to apply() ...) |
||||||
|
*/ |
||||||
|
private void enqueueDiskWrite(final MemoryCommitResult mcr, |
||||||
|
final Runnable postWriteRunnable) { |
||||||
|
final boolean isFromSyncCommit = (postWriteRunnable == null); |
||||||
|
|
||||||
|
final Runnable writeToDiskRunnable = new Runnable() { |
||||||
|
@Override |
||||||
|
public void run() { |
||||||
|
synchronized (mWritingToDiskLock) { |
||||||
|
writeToFile(mcr, isFromSyncCommit); |
||||||
|
} |
||||||
|
synchronized (mLock) { |
||||||
|
mDiskWritesInFlight--; |
||||||
|
} |
||||||
|
if (postWriteRunnable != null) { |
||||||
|
postWriteRunnable.run(); |
||||||
|
} |
||||||
|
} |
||||||
|
}; |
||||||
|
|
||||||
|
// Typical #commit() path with fewer allocations, doing a write on |
||||||
|
// the current thread. |
||||||
|
if (isFromSyncCommit) { |
||||||
|
boolean wasEmpty = false; |
||||||
|
synchronized (mLock) { |
||||||
|
wasEmpty = mDiskWritesInFlight == 1; |
||||||
|
} |
||||||
|
//直接在当前执行 run 是有两个条件的,即来自 commit 并且 wasEmpty 为 true |
||||||
|
if (wasEmpty) { |
||||||
|
writeToDiskRunnable.run(); |
||||||
|
return; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
QueuedWork.queue(writeToDiskRunnable, !isFromSyncCommit); |
||||||
|
} |
||||||
|
``` |
||||||
|
|
||||||
|
敲黑板,这是重点了。我保留了注释,从注释可以看出,首先第二个参数如果为 null 就说明来自 commit,如果非空就说明来自 apply。然后通过构造一个 writeToDiskRunnable ,那么什么时候它会 run 呢?那就是当方法调用来自 commit 并且 mDiskWritesInFlight == 1,这个 mDiskWritesInFlight 是在哪赋值的呢? |
||||||
|
|
||||||
|
```java |
||||||
|
private int mDiskWritesInFlight = 0; |
||||||
|
private MemoryCommitResult commitToMemory() { |
||||||
|
//... |
||||||
|
mDiskWritesInFlight++; |
||||||
|
//... |
||||||
|
} |
||||||
|
``` |
||||||
|
|
||||||
|
嘿,这就清楚了,前面我们说过不管是 commit 还是 apply 都会先把修改提交到内存中,然后 mDiskWritesInFlight++,然后在每次构造 writeToDiskRunnable 的时候又会 mDiskWritesInFlight--,当为 1 的时候就说明前面的提交到内存的修改都已经提交的磁盘上了。那么来自 commit 的写磁盘任务就直接在当前线程即 UI 线程里执行了,如果前面还有写磁盘任务没完成,就和 apply 一样添加到 QueueWork 里,其实就是异步执行了。 |
||||||
|
|
||||||
|
所以对于 commit 操作来说,并不是绝对的就一定在 UI 线程执行,那这样有什么好处呢? |
||||||
|
|
||||||
|
其实很好理解,如果先 apply 在紧接着 commit,那么如果不放在同一个线程中执行,就有可能导致 apply 的数据在 commit 之后被写入到磁盘中,磁盘中的数据是错误的,而且和内存中的数据不一致。 |
||||||
|
|
||||||
|
对于 apply 和 commit ,它是如何保证同步的呢?在这两个方法里都有 mcr.writtenToDiskLatch.await(),它其实是一个 CountDownLatch。 |
||||||
|
|
||||||
|
关于它,我这里直接引用了网上的对它的介绍: |
||||||
|
|
||||||
|
> CountDownLatch 是一个同步工具类,它允许一个或多个线程一直等待,直到其他线程的操作执行完成后在执行。 |
||||||
|
|
||||||
|
OK,不要多说了。 |
||||||
|
|
||||||
|
关于添加和修改就完了,剩下就是取数据操作了: |
||||||
|
|
||||||
|
SharedPreferencesImpl#getXxx : |
||||||
|
|
||||||
|
```java |
||||||
|
@Override |
||||||
|
@Nullable |
||||||
|
public String getString(String key, @Nullable String defValue) { |
||||||
|
synchronized (mLock) { |
||||||
|
awaitLoadedLocked(); |
||||||
|
//直接从内存中取数据 |
||||||
|
String v = (String)mMap.get(key); |
||||||
|
return v != null ? v : defValue; |
||||||
|
} |
||||||
|
} |
||||||
|
``` |
||||||
|
|
||||||
|
**总结:** |
||||||
|
|
||||||
|
1. sSharedPrefsCache 是一个 ArrayMap\<String,ArrayMap<File,SharedPreferencesImpl>>,它会保存加载到内存中的 SharedPreferences 对象,ContextImpl 类中并没有定义将 SharedPreferences 对象移除 sSharedPrefsCache 的方法,所以一旦加载到内存中,就会存在直至进程销毁。相对的,也就是说,SP 对象一旦加载到内存,后面任何时间使用,都是从内存中获取,不会再出现读取磁盘的情况 |
||||||
|
2. SharedPreferences 和 Editor 都只是接口,真正的实现在 SharedPreferencesImpl 和 EditorImpl ,SharedPreferences 只能读数据,它是在内存中进行的,Editor 则负责存数据和修改数据,分为内存操作和磁盘操作 |
||||||
|
3. 获取 SP 只能通过 ContextImpl#getSharedPerferences 来获取,它里面首先通过 mSharedPrefsPaths 根据传入的 name 拿到 File ,然后根据 File 从 ArrayMap<File, SharedPreferencesImpl> cache 里取出对应的 SharedPrederenceImpl 实例 |
||||||
|
4. SharedPreferencesImpl 实例化的时候会启动子线程来读取磁盘文件,但是在此之前如果通过 SharedPreferencesImpl#getXxx 或者 SharedPreferences.Editor 会阻塞 UI 线程,因为在从 SP 文件中读取数据或者往 SP 文件中写入数据的时候必须等待 SP 文件加载完 |
||||||
|
5. 在 EditorImpl 中 putXxx 的时候,是通过 HashMap 来存储数据,提交的时候分为 commit 和 apply,它们都会把修改先提交到内存中,然后在写入磁盘中。只不过 apply 是异步写磁盘,而 commit 可能是同步写磁盘也可能是异步写磁盘,在于前面是否还有写磁盘任务。对于 apply 和 commit 的同步,是通过 CountDownLatch 来实现的,它是一个同步工具类,它允许一个线程或多个线程一致等待,直到其他线程的操作执行完之后才执行 |
||||||
|
6. SP 的读写操作是线程安全的,它对 mMap 的读写操作用的是同一把锁,考虑到 SP 对象的生命周期与进程一致,一旦加载到内存中就不会再去读取磁盘文件,所以只要保证内存中的状态是一致的,就可以保证读写的一致性 |
||||||
|
|
||||||
|
##### 注意事项以及优化建议 |
||||||
|
|
||||||
|
1. 强烈建议不要在 SP 里面存储特别大的 key/value ,有助于减少卡顿 / ANR |
||||||
|
2. 请不要高频的使用 apply,尽可能的批量提交;commit 直接在主线程操作,更要注意了 |
||||||
|
3. 不要使用 MODE_MULTI_PROCESS |
||||||
|
4. 高频写操作的 key 与高频读操作的 key 可以适当的拆分文件,以减少同步锁竞争 |
||||||
|
5. 不要连续多次 edit,每次 edit 就是打开一次文件,应该获取一次 edit,然后多次执行 putXxx,减少内存波动,所以在封装方法的时候要注意了 |
||||||
|
|
||||||
|
#### 参考 |
||||||
|
|
||||||
|
[全面剖析SharedPreferences](http://gityuan.com/2017/06/18/SharedPreferences/) |
||||||
|
|
||||||
|
[浅析SharedPreferences](https://juejin.im/post/5bcbd780f265da0ad948056a) |
||||||
|
|
@ -0,0 +1,23 @@ |
|||||||
|
--- |
||||||
|
单例模式 |
||||||
|
--- |
||||||
|
|
||||||
|
#### 目录 |
||||||
|
|
||||||
|
1. 思维导图 |
||||||
|
2. 特点定义以及使用场景 |
||||||
|
3. 五种实现方式 |
||||||
|
4. 注意事项 |
||||||
|
|
||||||
|
#### 思维导图 |
||||||
|
|
||||||
|
#### 特点定义以及使用场景 |
||||||
|
|
||||||
|
单例即确保该类只有一个实例,避免产生多个对象消耗过多的资源。在一些工具类中用的非常多。它主要有以下几个特点: |
||||||
|
|
||||||
|
1. 私有的构造函数 |
||||||
|
2. 只能通过静态方法或者枚举返回单例类对象 |
||||||
|
3. 确保单例类在反序列化时不会重新构建对象 |
||||||
|
|
||||||
|
#### 五种实现方式 |
||||||
|
|
After Width: | Height: | Size: 191 KiB |
Loading…
Reference in new issue