HttpURLConnection → OkHttp + Kotlin

HttpURLConnection → OkHttp + Kotlin

老写法(Java + HttpURLConnection)

java 复制代码
new Thread(() -> {
    HttpURLConnection connection = null;
    try {
        URL url = new URL("https://api.example.com/data");
        connection = (HttpURLConnection) url.openConnection();
        connection.setRequestMethod("GET");
        connection.setConnectTimeout(10000);
        connection.setReadTimeout(10000);

        int code = connection.getResponseCode();
        if (code == 200) {
            InputStream is = connection.getInputStream();
            BufferedReader reader = new BufferedReader(new InputStreamReader(is));
            StringBuilder sb = new StringBuilder();
            String line;
            while ((line = reader.readLine()) != null) {
                sb.append(line);
            }
            final String result = sb.toString();

            runOnUiThread(() -> updateUI(result));
        }
    } catch (IOException e) {
        e.printStackTrace();
    } finally {
        if (connection != null) connection.disconnect();
    }
}).start();

问题在哪里

手写 Socket 级别的连接管理、超时、流读取,代码量爆炸。线程切换全靠手动 new Thread + runOnUiThread。请求重试、缓存、HTTPS 证书校验全要自己实现,没有拦截器机制。

新写法(OkHttp + Coroutines)

添加依赖:

groovy 复制代码
implementation "com.squareup.okhttp3:okhttp:4.12.0"
kotlin 复制代码
private val client = OkHttpClient.Builder()
        .connectTimeout(10, TimeUnit.SECONDS)
        .readTimeout(10, TimeUnit.SECONDS)
        .build()

viewModelScope.launch(Dispatchers.IO) {
    val request = Request.Builder()
            .url("https://api.example.com/data")
            .get()
            .build()

    try {
        val response = client.newCall(request).execute()
        if (response.isSuccessful) {
            val result = response.body?.string() ?: ""
            withContext(Dispatchers.Main) {
                updateUI(result)
            }
        }
    } catch (e: IOException) {
        withContext(Dispatchers.Main) {
            showError(e.message)
        }
    }
}

一句话注意

OkHttp 的 execute() 是同步方法,需要在后台线程调用。enqueue() 是异步回调,但用了协程就尽量用 execute() + 协程管理,避免回调嵌套。

response.body?.string() 只能调用一次,第二次返回空。如果需要反复读,先存到变量里。OkHttp 内部有自己的连接池和线程池,创建一次以单例使用即可,不要每次都 new。


Java Android 老项目迁移系列,持续更新中。

相关推荐
Flittly4 小时前
【AgentScope Java新手村系列】(16)从RAG到多路检索
java·spring boot·spring
小兔崽子去哪了4 小时前
Java 生成二维码解决方案
java·后端
石山岭4 小时前
自己动手写了一个 Android 虚拟定位 App:GPSSimulate 技术实
android·前端
杉氧6 小时前
副作用 (Side Effects) 全攻略:如何像大师一样掌控 Composable 的生命周期?
android·架构·android jetpack
人活一口气8 小时前
从JVM调优到MCP协议:Java全栈技术体系深度总结与企业级架构实践
java·spring boot
NE_STOP10 小时前
Vibe Coding -- 完整项目案例实操
java
唐青枫10 小时前
别再把 inline 当性能开关:Kotlin 内联、noinline、crossinline 与 reified 实战详解
kotlin
荣码10 小时前
GraphRAG:普通RAG只能回答"点"的问题,我踩了4个坑才搞懂
java·python
SimonKing10 小时前
Google第三方授权登录
java·后端·程序员
明月光81810 小时前
从一行 @Builder 说起:重新拾起 Java 的 Lombok、注解与 Builder 模式
java