๐ฑ dreaming DiNO
[Kotlin] ์๋๋ก์ด๋ Q ์ดํ (API 28) ์ธ๋ถ ์ ์ฅ์ ํ์ผ ์ฝ๊ธฐ ํ ์คํธ ๋ณธ๋ฌธ
Android/Android Studio
[Kotlin] ์๋๋ก์ด๋ Q ์ดํ (API 28) ์ธ๋ถ ์ ์ฅ์ ํ์ผ ์ฝ๊ธฐ ํ ์คํธ
MK_____ 2023. 5. 25. 17:03์๋๋ก์ด๋ Q ์ดํ, ์๋๋ฐ์ค ๊ตฌ์กฐ์ ์ธ๋ถ ์ ์ฅ์์์ ํ์ผ์ ์ฝ์ด ๋ณด์
์๋๋ก์ด๋ Q์ Scoped Storage
TODO:: ํ๋ก์ ํธ์ ํฌํจ๋ ํ ์คํธ ํ์ผ์, ์ธ๋ถ ์ ์ฅ์ com.app.a ๋ก ๋ณต์ฌํ์ฌ, ํ์ํ ๋๋ง๋ค ์ฑ์ ์ธ๋ถ ์ ์ฅ์์ ์ ๊ทผํ์ฌ ๋ด์ฉ์ ์ฝ์ด ์ฌ ๊ฒ!
1. ํ๋ก์ ํธ ๋ด, assets/datasets/aaa.txt
2. ์ธ๋ถ ์ ์ฅ์ ์ฝ๊ธฐ/์ฐ๊ธฐ ๊ฐ๋ฅ ์ฌ๋ถ ํ์ธ
// TODO:: ์ ์ฅ์ ํ
์คํธ
VCDebugLog.d(logTag, "isReadable: ${isExternalStorageReadable()}") // true
VCDebugLog.d(logTag, "isWritable: ${isExternalStorageWritable()}") // true
testCopy()
// Checks if a volume containing external storage is available
// for read and write.
fun isExternalStorageWritable(): Boolean {
return Environment.getExternalStorageState() == Environment.MEDIA_MOUNTED
}
// Checks if a volume containing external storage is available to at least read.
fun isExternalStorageReadable(): Boolean {
return Environment.getExternalStorageState() in
setOf(Environment.MEDIA_MOUNTED, Environment.MEDIA_MOUNTED_READ_ONLY)
}
3. ์ธ๋ถ ์ฑ ์ ์ฅ์๋ก ํ์ผ ๋ณต์ฌ
private fun testCopy() {
VCDebugLog.i(logTag, "testCopy-()")
val assetManager = resources.assets
var files: Array<String>? = null
try {
files = assetManager.list("datasets/") // asset ํด๋์ ๊ฒฝ๋ก
} catch (e: IOException) {
VCDebugLog.e("Failed to get asset file list. e => $e")
}
if (files != null) for (filename in files) {
var `in`: InputStream? = null
var out: OutputStream? = null
if (filename == "aaa.txt") {
VCDebugLog.d(logTag, "filename: $filename") // asset ํด๋์ ์๋ ํ์ผ ์ฐพ๊ธฐ
try {
// ์ธ๋ถ ์ฑ ์ ์ฅ์๋ก ํ์ผ ๋ณต์ฌ
val externalStorageVolumes: Array<out File> =
ContextCompat.getExternalFilesDirs(applicationContext, null)
val primaryExternalStorage = externalStorageVolumes[0]
`in` = assetManager.open("datasets/$filename") // asset ํด๋ ๊ฒฝ๋ก์ ์๋ ํ์ผ ์ฝ๊ธฐ
out = FileOutputStream("$primaryExternalStorage/$filename") // ์ธ๋ถ ์ฑ ์ ์ฅ์์ ๊ฒฝ๋ก
copyFile(`in`, out)
VCDebugLog.d(logTag, "Success copy file")
checkIsExistFile()
} catch (e: IOException) {
VCDebugLog.e("Failed to copy asset file. filename => $filename, e => $e")
} finally {
if (`in` != null) {
try {
`in`.close()
//`in` = null
} catch (e: IOException) {
}
}
if (out != null) {
try {
out.flush()
out.close()
//out = null
} catch (e: IOException) {
// nothing
}
}
}
}
}
}
Logcat ๋ก๊ทธํ์ธ
5. ์ธ๋ถ ์ฑ ์ ์ฅ์์ ํ์ผ์ด ์ ๋ณต์ฌ ๋์ด์๋์ง ํ์ธ
private fun checkIsExistFile() { // ์ธ๋ถ ์ฑ ์ ์ฅ์์ ํ์ผ ์๋์ง ํ์ธ
VCDebugLog.i(logTag, "checkIsExistFile-()")
val appSpecificExternalDir = File(applicationContext.getExternalFilesDir(null), "aaa.txt")
VCDebugLog.d(logTag, "appSpecificExternalDir.name: ${appSpecificExternalDir.name}")
VCDebugLog.d(logTag, "appSpecificExternalDir.path: ${appSpecificExternalDir.path}")
VCDebugLog.d(logTag, "appSpecificExternalDir.absolutePath: ${appSpecificExternalDir.absolutePath}")
}
Logcat ๋ก๊ทธํ์ธ
์ ์ฅ ํ์ธ!
์ด์ , ํ์ํ ๊ณณ์์ ์ธ๋ถ ์ฑ ์ ์ฅ์์ ์๋ ํ ์คํธ ํ์ผ์ ์ ๊ทผํด, content ๋ฅผ ๋ฝ์๋ณด์.
Fragment ์์ ํ์ธํด๋ดค๋ค.
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
...
// TODO:: ์ธ๋ถ ์ฑ ์ ์ฅ์์ ์ ์ฅ๋ ํ์ผ ์ฝ๊ธฐ
readTextInExStorage()
}
private fun readTextInExStorage() {
VCDebugLog.i(logTag, "readTextInExStorage")
try {
var line: String? = null // ํ์ค์ฉ ์ฝ๊ธฐ
val buf = BufferedReader(FileReader(File(context?.getExternalFilesDir(null), "aaa.txt")))
while (buf.readLine().also { line = it } != null) {
VCDebugLog.d(logTag, "line: $line") // line: Hi, this is mia
}
buf.close()
} catch (e: FileNotFoundException) {
e.printStackTrace()
} catch (e: IOException) {
e.printStackTrace()
}
}