Как добавить тайм-аут для AsyncTask в Kotlin - PullRequest
3 голосов
/ 30 марта 2019

Я новичок в Котлине. Я использую AsyncTask для выполнения данных JSON из API. Я хочу добавить тайм-аут через определенный промежуток времени, если у пользователя очень медленное или нестабильное подключение к данным, а затем показать пользователю диалоговое окно с предупреждением: «Извините, у вас нет правильного подключения к Интернету» с помощью кнопки, когда нажал, закрывает приложение.

Это мой AsyncTask код:

 inner class Arr : AsyncTask<String, String, String>(){



        }

        //        for build connection
        override fun doInBackground(vararg url: String?): String{

            var text : String
            val connection = URL(url[0]).openConnection() as HttpURLConnection

            try {
                connection.connect()
                text = connection.inputStream.use { it.reader().use{reader -> reader.readText()} }


            } finally{

                connection.disconnect()

            }
            return text
        }

        override fun onPostExecute(result: String?) {

            super.onPostExecute(result)
            handleJson(result)


        }

        override fun onProgressUpdate(vararg text: String?) {


        }

1 Ответ

2 голосов
/ 30 марта 2019

Есть несколько способов достичь этой цели.Ниже приведены два примера:

  1. Добавление таймаута с помощью HttpURLConnection:

    try {
        connection.connectTimeout = 5000 // We all timeout here
        connection.connect()
        text = connection.inputStream.use { it.reader().use{reader -> reader.readText()} }
    } finally{
        connection.disconnect()
    }
    
  2. Отключение вручную с помощью Handler & Runnable (Мы можем добиться того же самого, используя CountDownTimer или любой другой материал) :

    try {
        connection.connect()
        text = connection.inputStream.use { it.reader().use{reader -> reader.readText()} }
        // We all timeout here using Handler
        Handler().postDelayed(
            {
                connection.disconnect() // We disconnect manually
            },
            5000 // Timeout value
        )
    } finally{
        connection.disconnect()
    }
    

Редактировать для OP:

Используйте приведенный ниже класс для совершения вызова API и отображения предупреждения для пользователя , если время соединения истекло .

//We pass context to Activity/Fragment to display alert dialog
inner class TestApiCall(private val context: Context?) : AsyncTask<String, String, String?>() {

    //        for build connection
    override fun doInBackground(vararg url: String?): String? {
        var text: String? = null
        val connection = URL(url[0]).openConnection() as HttpURLConnection

        try {
            connection.connect()
            text = connection.inputStream.use { it.reader().use { reader -> reader.readText() } }
            handleTimeout { timedOut ->
                if (timedOut) {
                    text = null
                    connection.disconnect()
                    print("Timeout Executed")
                }
            }
        } finally {
            connection.disconnect()
        }
        return text
    }

    private fun handleTimeout(delay: Long = 5000, timeout: (Boolean) -> Unit) {
        Handler(Looper.getMainLooper()).postDelayed({
            timeout(true)
        }, delay)
    }

    override fun onPostExecute(result: String?) {
        super.onPostExecute(result)
        if (result != null) {
            //Handle result here
            print("Result --> $result")
        } else {
            //Result is null meaning it can be timed out
            context?.let { ctx ->
                val alertDialog = AlertDialog.Builder(ctx)
                alertDialog.setTitle("Some title here")
                alertDialog.setMessage("Notifying user about API error")
                alertDialog.create().show()
            }
        }
    }

    override fun onProgressUpdate(vararg text: String?) {
        //Update progress from here
    }
}

Вызовите его с Activity/Fragment, передав context и "URL вашего API" :

TestApiCall(context).execute("https://jsonplaceholder.typicode.com/todos/1")
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...