从类中调用<reified T>内联函数

问题描述

简而言之:我想创建一个类型为T的泛型类,该类应调用类型为T的经过修饰的内联函数,以便能够更通用地使用Gson

详细信息:
你好。我最近开始为Android开发人员使用kotlin,并且试图创建一个通用的网络客户端类。该类将从服务器获取数据,将响应转换为通用类的类型提供的数据类型,然后返回响应,如下所示:

class GenericNetworkService<RESPONSE_TYPE> {

    fun getDataFromServer(url: String): RESPONSE_TYPE {
        val request = Request.Builder().url(url).build()

        val response = OkHttpClient().newCall(request).execute()

        val convertor = Gson()

        val result: RESPONSE_TYPE = convertor.fromJson(response.body!!.string(),RESPONSE_TYPE)

        return result

    }
}

//calling
//fun main() {

//    val listofPersons = GenericNetworkService<List<Person>>().getDataFromServer(url1)
//    println(listofPersons)

//    val listofTeacher = GenericNetworkService<List<Teacher>>().getDataFromServer(url2)
//    println(listofTeacher)
// }

在这里,只有gson.fromJson()函数是实际需要类的细节的函数,最终输出必须在该函数中进行转换。 但是它没有将泛型类型带入第二个参数,因此上述程序无法正常工作。搜索了一些互联网文章之后,我遇到了这个Typetoken Api,它在一定程度上有助于实现一般行为:


inline fun <reified T> fromJson(json: String): T {
   return Gson().fromJson(json,object: Typetoken<T>(){}.type)
}

//calling : 
// val data : List<Teacher> = fromJson(response)

使用内联修饰函数,我能够实现我最初想要的一切:


private inline fun <reified RESPONSE_TYPE> getGenericRespSync(url: String): RESPONSE_TYPE {
    val okHttpClient = generateNetworkClient()
    val request = Request.Builder().url(url).build()
    val response: Response = okHttpClient.newCall(request).execute()

    val body: Reader = response.body!!.charStream()
    val finalResponse: RESPONSE_TYPE =  getConvertedListForResponse(body) //this is the same reified inline function as above with some non signinficant additions
    return finalResponse
}


但是我希望我的其他类通过某个类实例调用函数。即类似GenericNetworkService<List<Person>>().getGenericRespSync(url1)的东西应该返回List而不是直接调用getGenericRespSync(url1)。我如何实现这种功能

解决方法

好吧,你可以定义

inline fun <reified RESPONSE_TYPE> GenericNetworkService<RESPONSE_TYPE>.getGenericRespSync1(url: String): RESPONSE_TYPE = getGenericRespSync<RESPONSE_TYPE>(url)

(我以不同的方式命名了该函数,因此它可以调用您先前定义的getGenericRespSync)。当然,它只使用被调用的实例来获取其类型参数,否则将忽略它。

,

不幸的是,类构造函数不能使用与内联函数相同的reified机制。但是,您可以将类型定义为常规构造函数参数,并使用顶级内联函数创建类的新实例

class GenericNetworkService<RESPONSE_TYPE>(private val responseTypeClass: Class<RESPONSE_TYPE>) {

    fun getDataFromServer(url: String): RESPONSE_TYPE {
        val request = Request.Builder().url(url).build()

        val response = OkHttpClient().newCall(request).execute()

        val convertor = Gson()

        val result: RESPONSE_TYPE = convertor.fromJson(response.body!!.string(),responseTypeClass)

        return result

    }
}

inline fun <reified RESPONSE_TYPE> createService() = GenericNetworkService(RESPONSE_TYPE::class.java)

当然,您也可以将createService重命名为GenericNetworkService,使其看起来像构造函数。这在kotlin中变得越来越普遍,尤其是如果您查看协程的核心代码,因此我将不再将其视为反模式。