前幾天我們游戲在一個同事的Android手機(jī)上啟動時無法正常進(jìn)入,經(jīng)查發(fā)現(xiàn)Application.temporaryCachePath和Application.persistentDataPath返回空字符串。便花時間認(rèn)真研究了一下Unity3D的路徑問題。我們常用的是以下四個路徑:
Application.dataPath
Application.streamingAssetsPath
Application.persistentDataPath
Application.temporaryCachePath
根據(jù)測試,詳細(xì)情況如下:
iOS:
Application.dataPath????????????/var/containers/Bundle/Application/app sandbox/xxx.app/Data
Application.streamingAssetsPath /var/containers/Bundle/Application/app sandbox/test.app/Data/Raw
Application.temporaryCachePath /var/mobile/Containers/Data/Application/app sandbox/Library/Caches
Application.persistentDataPath??/var/mobile/Containers/Data/Application/app sandbox/Documents
iOS和Mac OS X不同于Windows,app都是在一個沙盒空間中運(yùn)行,每個app也有一個獨(dú)立的數(shù)據(jù)存儲空間,各app彼此不能互相訪問、打擾。
dataPath是app程序包安裝路徑,app本身就在這里,此目錄是只讀的。streamingAssetsPath是dataPath下的Raw目錄。
app的獨(dú)立數(shù)據(jù)存儲目錄下有三個文件夾:Documents,Library和tmp。
Documents目錄,這個目錄用于存儲需要長期保存的數(shù)據(jù),比如我們的熱更新內(nèi)容就寫在這里。需要注意的是,iCloud會自動備份此目錄,如果此目錄下寫入的內(nèi)容較多,審核的可能會被蘋果拒掉。
Library目錄,這個目錄下有兩個子目錄,Caches和Preferences。
????Caches是一個相對臨時的目錄,適合存放下載緩存的臨時文件,空間不足時可能會被系統(tǒng)清除,Application.temporaryCachePath返回的就是此路徑。我把熱更新的臨時文件寫在這里,等一個版本的所有內(nèi)容更新完全后,再把內(nèi)容轉(zhuǎn)移到Documents目錄。
????Preferences用于應(yīng)用存儲偏好設(shè)置,用NSUserDefaults讀取或設(shè)置。
tmp目錄,臨時目錄,存放應(yīng)用運(yùn)行時臨時使用的數(shù)據(jù)。
需要注意的是,以上無論臨時、緩存或者普通目錄,如果不需要的數(shù)據(jù),都請刪除。不要占用用戶的存儲空間,像微信就是壞榜樣。
下面是各路徑對應(yīng)的OC訪問方法
app安裝路徑: [[NSBundle mainBundle] resourcePath]
app數(shù)據(jù)沙盒存儲根目錄: NSHomeDirectory()
Documents: NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)
Library:?????NSSearchPathForDirectoriesInDomains(NSLibraryDirectory, NSUserDomainMask, YES)
Caches:?????NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES)
tmp:????????NSTemporaryDirectory()
Android:
Application.dataPath??????????? /data/app/package name-1/base.apk
Application.streamingAssetsPath jar:file:///data/app/package name-1/base.apk!/assets
Application.temporaryCachePath /storage/emulated/0/Android/data/package name/cache
Application.persistentDataPath?? /storage/emulated/0/Android/data/package name/files
看Android上的路徑,跟iOS有點(diǎn)類似,簡單說一下。Android的幾個目錄是apk程序包、內(nèi)存存儲(InternalStorage)和外部存儲(ExternalStorage)目錄。
apk程序包目錄: apk的安裝路徑,/data/app/package name-n/base.apk,dataPath就是返回此目錄。
內(nèi)部存儲目錄: /data/data/package name-n/,用戶自己或其它app都不能訪問該目錄。打開會發(fā)現(xiàn)里面有4個目錄(需要root)
????cache 緩存目錄,類似于iOS的Cache目錄
????databases 數(shù)據(jù)庫文件目錄
????files 類似于iOS的Documents目錄
????shared_prefs 類似于iOS的Preferences目錄,用于存放常用設(shè)置,比如Unity3D的PlayerPrefs就存放于此
外部存儲目錄: 在內(nèi)置或外插的sd上,用戶或其它app都可以訪問,外部存儲目錄又分私有和公有目錄。
????公有目錄是像DCIM、Music、Movies、Download這樣系統(tǒng)創(chuàng)建的公共目錄,當(dāng)然你也可以像微信那樣直接在sd卡根目錄創(chuàng)建一個文件夾。好處嘛,就是卸載app數(shù)據(jù)依舊存在。
????私有目錄在/storage/emulated/n/Android/data/package name/,打開可以看到里面有兩個文件夾cache和files。為什么跟內(nèi)部存儲目錄重復(fù)了?這是為了更大的存儲空間,以防內(nèi)存存儲空間較小。推薦把不需要隱私的、較大的數(shù)據(jù)存在這里,而需要隱私的或較小的數(shù)據(jù)存在內(nèi)部存儲空間。
下面是各路徑對應(yīng)的Java訪問方法:
apk包內(nèi): AssetManager.open(String filename)
內(nèi)部存儲: context.getFilesDir().getPath() or context.getCacheDir().getPath()
外部存儲: context.getExternalFilesDir(null).getPath() or context.getExternalCacheDir().getPath()
理解了Android存儲的原理,最后來說說開頭提到的bug,Application.temporaryCachePath/persistentDataPath返回空字符串。這其實(shí)因?yàn)闄?quán)限的原因,app沒有聲明訪問外部存儲空間的權(quán)限,但是Application.temporaryCachePath/ ApplicationpersistentDataPath卻想返回外部存儲的路徑。這是Unity3D的bug,沒有權(quán)限本應(yīng)該拋出一個異常或者錯誤,讓開發(fā)者知道原因。
經(jīng)反復(fù)測試發(fā)現(xiàn),有【外置SD卡】的設(shè)備上,如果聲明讀/寫外部存儲設(shè)備的權(quán)限,會返回外部存儲路徑,不聲明則會返回內(nèi)部存儲路徑,這樣不會有問題。而在【無外置SD卡】的設(shè)備上,不管是否聲明讀/寫外部存儲設(shè)備的權(quán)限,Application.temporaryCachePath/persistentDataPath都返回外部存儲路徑,但是又沒有權(quán)限,就可能會導(dǎo)致返回null了,之所以說可能是因?yàn)檫@個bug不是必現(xiàn),如果出現(xiàn)了設(shè)備重啟之后就好了,懷疑是linux設(shè)備mount問題。但是出了問題,我們不能跟用戶說你重啟一下手機(jī)就好了。
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
Windows:
Application.dataPath:????????????應(yīng)用的appname_Data/
Application.streamingAssetsPath: 應(yīng)用的appname_Data/StreamingAssets
Application.temporaryCachePath: C:\Users\username\AppData\Local\Temp\company name\product name
Application.persistentDataPath:?? C:\Users\username\AppData\LocalLow\company name\product name
參考:
- iOS Data Storage Guidelines
- Android API: Storage Options
- 徹底理解Android中的內(nèi)部存儲與外部存儲
|