将字节数组转换为PDF文件并保存

3

我正在使用Kotlin Android应用程序调用API端点。 在此API调用中,我返回了一个字节数组。 我想知道是否有一种将字节数组转换为pdf文件并将其保存在手机下载文件夹中的方法。

    private suspend fun getIDCard(sessionID: String?, carriermemid: String?): ByteArray? {
        var results: String = ""

        val postData = "{\"sessionid\": \" $sessionID\"}"
        val outputStreamWriter: OutputStreamWriter

        var byteArray: ByteArray? = null
        val url: URL = URL(idURL + carriermemid)
        val conn: HttpURLConnection = url.openConnection() as HttpURLConnection

        try {
            conn.setRequestProperty(apiConfig.typeKey, apiConfig.typeValueJSON)
            conn.setRequestProperty("Accept", "application/json")
            conn.setRequestProperty("sessionid", sessionID)
            conn.requestMethod = apiConfig.methodGet


            val responseCode: Int = conn.responseCode
            println("Response Code :: $responseCode")
            //returning 404
            return if (responseCode == HttpURLConnection.HTTP_OK) {// connection ok
                var out: ByteArrayOutputStream? = ByteArrayOutputStream()
                val `in`: InputStream = conn.inputStream
                var bytesRead: Int
                val buffer = ByteArray(1024)
                while (`in`.read(buffer).also { bytesRead = it } > 0) {
                    out!!.write(buffer, 0, bytesRead)
                }
                out!!.close()
                byteArray = out.toByteArray()
                return byteArray

            } else {
                return byteArray
            }

        } catch (ex: Exception) {
            ex.printStackTrace()
        } finally {
            conn.disconnect()
        }
        return byteArray
    }

4
将字节数组转换为PDF文件并保存在手机下载文件夹中。字节数组中包含什么可以转换为PDF?我认为字节数组将包含PDF文件,您所要做的就是将数组中的字节保存到文件中即可完成。 - blackapps
@blackapps 字节数组确实包含PDF。至于我是如何寻找示例的,这让我感到尴尬。 - yams
1
所以你的问题基本上是:如何将文件存储在全局下载文件夹中? - tynn
@tynn 将字节数组转换为文件以存储在全局下载中。 - yams
3个回答

11

这将获取Android下载目录,并将字节数组写入PDF文件(假设字节数组包含PDF)。将File.createTempFile更改为任何您喜欢的文件(您不需要创建临时文件):

fun writeBytesAsPdf(bytes : ByteArray) {
   val path = requireContext().getExternalFilesDir(Environment.DIRECTORY_DOWNLOADS)
    var file = File.createTempFile("my_file",".pdf", path)
    var os = FileOutputStream(file);
    os.write(bytes);
    os.close();
}

您还需要在清单文件中添加android.permission.WRITE_EXTERNAL_STORAGE权限。


为了避免在viewModel中需要上下文,您可以在fragment/activity中获取路径并将其保存在viewModel中,对于特定用户来说应该是恒定的。 - user2199860
你也不需要使用createTemporaryFile,在下载文件夹中可能会起作用,也可能不起作用。你可以直接使用(或类似):File(path,“my_file.pdf”) - user2199860
是的,清单已更新,让我尝试摆脱创建临时文件。 - yams
那也不行。可能是权限或字节数组的问题,我稍后会查看一下。 - yams
那并不是解决问题的方法。请查看被接受的答案。 - yams

1

查看使用Retrofit和Kotlin协程下载PDF文件的方法,您可以使用:

private const val BUFFER_SIZE = 4 * 1024

private fun copyStreamToFile(inputStream: InputStream, outputFile: File) {
    inputStream.use { input ->
        val outputStream = FileOutputStream(outputFile)
        outputStream.use { output ->
            val buffer = ByteArray(BUFFER_SIZE)
            while (true) {
                val byteCount = input.read(buffer)
                if (byteCount < 0) break
                output.write(buffer, 0, byteCount)
            }
            output.flush()
        }
    }
}

或者

private fun InputStream.saveToFile(file: String) = use { input ->
    File(file).outputStream().use { output ->
        input.copyTo(output)
    }
}

同时你应该创建文件。

private fun createFile(context: Context, name: String): File? {
    val storageDir = context.getExternalFilesDir(Environment.DIRECTORY_DOWNLOADS)?.path
    var file = File("$storageDir/$name.pdf")
    return storageDir?.let { file }
}

-1

所以这就是最终解决问题的方法。差点忘了发布这个。我通过resolveinfo列表和授权权限的方式成功地解决了问题,以下是具体步骤:

                    try {
                        val pdfbyte = viewModel.getPDFImage()
                        val path =
                            requireContext().getExternalFilesDir(Environment.DIRECTORY_DOWNLOADS)
                                .toString() + File.separator
                        val fileName = "my_idcard.pdf"
                        val storageDir =
                            requireContext().getExternalFilesDir(Environment.DIRECTORY_DOWNLOADS)?.path
                        var file = File("$storageDir/$fileName")
                        var os = FileOutputStream(file);
                        os.write(pdfbyte);
                        os.close();
                        val intent = Intent(Intent.ACTION_VIEW)
                        val uri: Uri = FileProvider.getUriForFile(
                            requireContext(),
                            BuildConfig.APPLICATION_ID + ".provider",
                            file
                        )
                        intent.setDataAndType(uri, "application/pdf")
                        intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION)
                        intent.flags = Intent.FLAG_ACTIVITY_NO_HISTORY
                        val resInfoList: List<ResolveInfo> = requireActivity().getPackageManager()
                            .queryIntentActivities(intent, PackageManager.MATCH_DEFAULT_ONLY)

                        for (resolveInfo in resInfoList) {
                            val packageName = resolveInfo.activityInfo.packageName
                            requireActivity().grantUriPermission(
                                packageName,
                                uri,
                                Intent.FLAG_GRANT_WRITE_URI_PERMISSION or Intent.FLAG_GRANT_READ_URI_PERMISSION
                            )
                        }
                        startActivity(intent)
                    } catch (e: ActivityNotFoundException) {
                        println("*************************NO PDF**************************")
                    }

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接