WorkManager 在后台抛出 NetworkOnMainThreadException

WorkManager throws NetworkOnMainThreadException when in background

提问人:Viktor Pesic 提问时间:1/3/2022 最后编辑:Viktor Pesic 更新时间:1/3/2022 访问量:544

问:

我希望我的应用程序在使用 workmanager 时在后台阅读电子邮件并发送通知,我的工作管理器在后台任务上抛出 NetworkOnMainThreadException 几次执行后停止工作: 我想制作一个可靠的应用程序来为新收到的电子邮件发送通知,这意味着我需要它每 15 分钟永久执行一次。 你知道我怎样才能避免这个问题吗?

WorkManager 类:

class WorkerMan(private val mContext: Context, workerParameters: WorkerParameters) :
    Worker(mContext, workerParameters) {


    @SuppressLint("RestrictedApi", "CheckResult")
    val email = inputData.getString("email")
    val password = inputData.getString("password")

    @SuppressLint("RestrictedApi")
    override fun doWork(): Result {
        println("WorkManager: Work called")
        println("WorkMananager time: " + LocalTime.now())
        try {



       // Thread.sleep(5000)


        val session = Session.getDefaultInstance(Properties())
        val store = session.getStore("imaps")
        store.connect(
            "mail.metropolitan.ac.rs",
            993,
            email,
            password
        )
        val inbox = store.getFolder("INBOX")
        inbox.open(Folder.READ_ONLY)


        val messages = inbox.search(
            FlagTerm(Flags(Flags.Flag.SEEN), false)
        )


        Arrays.sort(
            messages
        ) { m1: Message, m2: Message ->
            try {
                return@sort m2.sentDate.compareTo(m1.sentDate)
            } catch (e: MessagingException) {
                throw RuntimeException(e)
            }
        }





        Thread.sleep(1000)
        println("- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - ")
        println("WorkManager Started")
        println("WorkMananager email: " + email)
        val current = LocalTime.now()
        println("WorkMananager time: " + current)
        println("Messages amount: " + messages.size)
        println("- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - ")

        for (message in messages) {


//            if (message.receivedDate.toInstant() >= Instant.now().minusMillis(1000 * 33 * 60)) {
                if (true) {

               // Thread.sleep(3000)
                println("=====================================================")
                println("NOTIFIKACIJA")

                var title = ""
                for (element in message.from) {
                    title += element.toString().substringAfter("<").substringBefore(">")
                    title += " "
                }
                println("Title :" + title)
                println("Subject :" + message.subject)
                println("Datum i vreme : " + message.receivedDate)

                title.replace("[", "")
                title.replace("]", "")



                FirebaseMessaging.getInstance().token.addOnSuccessListener { token: String ->
                    if (!TextUtils.isEmpty(token)) {
                        Log.d("TAG", "retrieve token successful : $token")
                        try {
                            send(token, message.subject, title)
                        } catch (e: MessagingException) {
                            e.printStackTrace()
                        }
                    } else {
                        Log.w("TAG", "token should not be null...")
                    }
                }.addOnFailureListener { e: Exception? -> }.addOnCanceledListener {}
                    .addOnCompleteListener { task: Task<String> ->
                        Log.v(
                            "TAG",
                            "This is the token : " + task.result
                        )
                    }


//
            }


        }
        println("=====================================================")
        Log.d("WorkManager", "Job finished")
        }catch (e : Exception){
            Log.d("WorkManager error", "doWork not executed")
            Log.d("WorkManager error", "error: ")
            Log.d("WorkManager error", e.printStackTrace().toString())
        }
        return Result.Success();
    }

}


fun send(to: String?, body: String?, title: String?): String? {
    try {
        val apiKey =
            "AAAA_Xfga4Q:APA91bH1cASekbIF7zSiqVkOtEJdQX2-qLO6yJp_iiTzYetdy6pRBl-uq28a27sdzDUvAUI51XO7IQaiKk_eccW0fTeFj8-4z7236mzoFaniTPMYR4Xmhzn5RYkAh-ON3tXKnELu7IEC"
        val url = URL("https://fcm.googleapis.com/fcm/send")
        val conn = url.openConnection() as HttpURLConnection
        conn.doOutput = true
        conn.requestMethod = "POST"
        conn.setRequestProperty("Content-Type", "application/json")
        conn.setRequestProperty("Authorization", "key=$apiKey")
        conn.doOutput = true
        val message = JSONObject()
        message.put("to", to)
        message.put("priority", "high")
        val notification = JSONObject()
        notification.put("title", title)
        notification.put("body", body)
        message.put("notification", notification)
        val os = conn.outputStream
        os.write(message.toString().toByteArray())
        os.flush()
        os.close()
        val responseCode = conn.responseCode
        println("\nSending 'POST' request to URL : $url")
        println("Post parameters : $message")
        println("Response Code : $responseCode")
        println("Response Code : " + conn.responseMessage)
        val `in` = BufferedReader(InputStreamReader(conn.inputStream))
        var inputLine: String?
        val response = StringBuffer()
        while (`in`.readLine().also { inputLine = it } != null) {
            response.append(inputLine)
        }
        `in`.close()

        // print result
        println(response.toString())
        return response.toString()
    } catch (e: Exception) {
        Log.d("WorkManager error", "send not executed")
        Log.d("WorkManager error", "error: ")
        Log.d("WorkManager error", e.printStackTrace().toString())
    }
    return "error"
}

调度 workmanager 的函数:

 @SuppressLint("RestrictedApi")
    private fun startWorker(email: String, password: String) {

        if (enableMailNotifications == 1) {


//            val constraints = Constraints.Builder()
//                .setRequiredNetworkType(NetworkType.CONNECTED).build()

            val data = Data.Builder()
            data.put("email", email)
            data.put("password", password)


            val build: PeriodicWorkRequest = PeriodicWorkRequest.Builder(
                WorkerMan::class.java,
                15,
                TimeUnit.MINUTES,
                15,
                TimeUnit.MINUTES,

                )

                .addTag("WorkManager")
                .setInputData(data.build())
                //  .setConstraints(constraints)
                .build()

            WorkManager.getInstance(this)
                .enqueueUniquePeriodicWork("WorkManager", ExistingPeriodicWorkPolicy.REPLACE, build)
        }else{
            println("WorkManager not called: Mail Notifications are turned off")
        }
    }

编辑:Full StackTraces:

pastebin.com/BVxyA4uF pastebin.com/ujYtNu60

android kotlin 后台进程 android-workmanager networkonmainthread

评论

0赞 CommonsWare 1/3/2022
我建议您编辑问题并提供与崩溃相关的完整堆栈跟踪。
0赞 Viktor Pesic 1/3/2022
好的,我添加了堆栈跟踪。
0赞 Selvin 1/3/2022
显然是因为你在里面打电话......使用另一个工人来做。sendonSuccess
0赞 CommonsWare 1/3/2022
您的函数正在后台线程上调用。您从中调用的回调函数正在主应用程序线程上调用。您需要找到一些替代 API,让您留在后台线程上,或者您需要自己安排在后台线程上进行调用。doWork()onSuccess()send()FirebaseMessagingsend()
0赞 Viktor Pesic 1/3/2022
有没有关于如何在后台线程上安排 send() 的文档?

答:

1赞 Viktor Pesic 1/3/2022 #1

我通过调用解决了一个问题

FirebaseMessaging.getInstance().token.addOnSuccessListener

在 MainActivity 中,并将令牌作为 WorkManager 的 inputData 传递。

data.put("token", token)

并通过以下方式在 WorkManager 类中检索它:

val token = inputData.getString("token")

评论

0赞 Community 1/3/2022
您的答案可以通过额外的支持信息得到改进。请编辑以添加更多详细信息,例如引文或文档,以便其他人可以确认您的答案是正确的。您可以在帮助中心找到有关如何写出好答案的更多信息。