Notice
Recent Posts
Recent Comments
Link
ยซ   2025/04   ยป
์ผ ์›” ํ™” ์ˆ˜ ๋ชฉ ๊ธˆ ํ† 
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
Tags
more
Archives
Today
Total
๊ด€๋ฆฌ ๋ฉ”๋‰ด

๐ŸŒฑ 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()
    }

}