【网络优化】Android 如何监听系统网络连接成功

思路一句话:

"把『系统网络连接成功』当成一个系统广播事件,注册一个 ConnectivityManager.NetworkCallback(或者 legacy 的 BroadcastReceiver),一旦收到 『已连接且可上网』的回调,就启动协程/线程去调你的 API。"


  1. 权限(AndroidManifest.xml)
xml 复制代码
<!-- 普通权限,安装即授予 -->
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<!-- 需要真正上网,别忘 INTERNET -->
<uses-permission android:name="android.permission.INTERNET" />

  1. 生命周期感知型 NetworkCallback(推荐)
kotlin 复制代码
class MyApp : Application() {

    override fun onCreate() {
        super.onCreate()
        NetworkProbe.init(this)
    }
}

object NetworkProbe {

    private val cm by lazy { app.getSystemService(Context.CONNECTIVITY_SERVICE) as ConnectivityManager }

    private lateinit var app: Application

    fun init(app: Application) {
        this.app = app
        // 兼容 O 以下
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
            registerNetworkCallback()
        } else {
            registerLegacyReceiver()
        }
    }

    @TargetApi(Build.VERSION_CODES.N)
    private fun registerNetworkCallback() {
        val request = NetworkRequest.Builder()
            .addCapability(NetworkCapabilities.NET_CAPABILITY_INTERNET)
            .addCapability(NetworkCapabilities.NET_CAPABILITY_VALIDATED) // 已验证可上网
            .build()

        cm.registerNetworkCallback(
            request,
            object : ConnectivityManager.NetworkCallback() {

                // 第一次连接或网络恢复都会走到这里
                override fun onAvailable(network: Network) {
                    super.onAvailable(network)
                    callMyApi()
                }

                // 如果只想『从断网到联网』触发一次,可再覆写 onLost 做状态机
            }
        )
    }

    // 兼容低版本
    private fun registerLegacyReceiver() {
        val filter = IntentFilter(ConnectivityManager.CONNECTIVITY_ACTION)
        app.registerReceiver(object : BroadcastReceiver() {
            override fun onReceive(c: Context, i: Intent) {
                val info = cm.activeNetworkInfo
                if (info?.isConnected == true) {
                    callMyApi()
                }
            }
        }, filter)
    }

    // 真正去调接口
    private fun callMyApi() {
        // 网络回调里不能做重 IO,切到 IO 线程
        CoroutineScope(Dispatchers.IO).launch {
            try {
                val rsp = RetrofitClient.api.ping() // 你的 Retrofit 接口
                Log.d("NetworkProbe", "API 调用成功:$rsp")
                // 如果只想成功一次,可在这里 unregister 或者置标志位
            } catch (e: Exception) {
                Log.e("NetworkProbe", "API 失败", e)
            }
        }
    }
}

  1. 只想在『前台 Activity』里监听?
    LifecycleObserverregisterNetworkCallback 搬进 onResumeonPauseunregister 即可,避免后台重复触发。

  1. 常见坑提示
  • 不要把 onAvailable 当成"Wi-Fi 连上了",它只代表系统选中的默认网络变更;
    加上 NET_CAPABILITY_VALIDATED 可确保已经通过 204/443 探测,真正可上网。
  • 7.0+ 静态注册 CONNECTIVITY_ACTION 无效,必须动态注册或改用 NetworkCallback
  • 回调线程是 系统 Binder 线程,不能直接做网络 IO,务必切协程/线程池。
  • 如果用户处于 ** captive portal **(需要网页认证),VALIDATED 不会满足,API 也调不通,符合预期。

一句话总结

"注册 ConnectivityManager.NetworkCallback → 在 onAvailable()NET_CAPABILITY_VALIDATED 时切 IO 线程调你的接口" 。

相关推荐
minji...28 分钟前
Linux 线程同步与互斥(三) 生产者消费者模型,基于阻塞队列的生产者消费者模型的代码实现
linux·运维·服务器·开发语言·网络·c++·算法
冬奇Lab1 小时前
MediaPlayer 播放器架构:NuPlayer 的 Source/Decoder/Renderer 三驾马车
android·音视频开发·源码阅读
运维行者_1 小时前
OpManager MSP NetFlow Analyzer集成解决方案,应对多客户端网络流量监控挑战
大数据·运维·服务器·网络·数据库·自动化·运维开发
dashizhi20153 小时前
共享文件禁止拖动本地磁盘、共享文件禁止另存为、禁止打印共享文件、禁止复制共享文件的方法
运维·服务器·网络·安全·电脑
网教盟人才服务平台3 小时前
AI 全面重塑网络攻防生态,智能安全进入深度对抗时代
网络·人工智能·安全
炸炸鱼.3 小时前
Python 操作 MySQL 数据库
android·数据库·python·adb
用户41659673693554 小时前
nextlib 项目架构与深度技术指南 (Architecture & Technical Master Guide)
android
aq55356005 小时前
Laravel10.x重磅升级,新特性一览
android·java·开发语言
Trouvaille ~5 小时前
【MySQL篇】数据类型:存储数据的基础
android·数据库·mysql·adb·字符集·数据类型·基础入门
头铁的伦6 小时前
QNX 网络模型
linux·网络·车载系统