Android 获取实时网速实现详解

 更新时间:2022年11月27日 16:08:22   作者:ChenYhong  
这篇文章主要为大家介绍了Android 获取实时网速实现详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

正文

最近接到个需求,需要计算WebView加载网页时的网速。查询了一下,Android没有提供直接获取网速的Api,但是提供了获取流量的类TrafficStats。本文介绍如何使用Trafficstats来实现获取网速功能。

TrafficStats简介

TrafficStats提供了一些获取设备从本次开机到目前为止传输/接收的流量的接口,如下:

方法参数说明
getTotalTxBytes-获取设备本次开机到目前为止,WI-FI、流量下传输的字节总数。
getTotalRxBytes-获取设备本次开机到目前为止,WI-FI、流量下接收的字节总数。
getMobileTxBytes-获取设备本次开机到目前为止,流量下传输的字节总数。
getMobileRxBytes-获取设备本次开机到目前为止,流量下接收的字节总数。
getUidTxBytesuid获取应用从本次开机到目前为止,WI-FI、流量下传输的字节总数。
getUidRxBytesuid获取应用从本次开机到目前为止,WI-FI、流量下接收的字节总数。

上述接口可以满足实现计算网速的需求,TrafficStats类其他接口可以查看官方文档

实现获取网速

可以通过一段时间内传输的流量除去时间计算出上行网速,通过一段时间内接收的流量除去时间计算出下行网速。

TrafficStats类的接口获取的网速是从开机时就开始计算的,因此,要计算一段时间内的流量需要在开始时获取一次流量数据,结束时获取一次流量数据,相减得出一段时间的实际流量。

实时网速

本文用getUidTxBytesgetUidRxBytes来演示,其他方法也是类似的,如下:

object NetSpeedUtils {
    var netSpeedCallback: NetSpeedCallback? = null
    private var timer: Timer? = null
    private var timerTask: TimerTask? = null
    private var lastTotalReceiveBytes: Long = 0
    private var lastTotalTransferBytes: Long = 0
    /**
     * 根据应用uid获取设备启动以来,该应用接收到的总字节数
     *
     * @param uid 应用的uid
     */
    fun getTotalReceiveBytes(): Long {
        var receiveBytes: Long = TrafficStats.UNSUPPORTED.toLong()
        ExampleApplication.exampleContext?.run {
            receiveBytes = TrafficStats.getUidRxBytes(applicationInfo.uid)
        }
        // 当获取不到时,会返回TrafficStats.UNSUPPORTED
        return if (receiveBytes == TrafficStats.UNSUPPORTED.toLong()) 0 else receiveBytes / 1024
    }
    /**
     * 根据应用uid获取设备启动以来,该应用传输的总字节数
     *
     * @param uid 应用的uid
     */
    fun getTotalTransferBytes(): Long {
        var transferBytes: Long = TrafficStats.UNSUPPORTED.toLong()
        ExampleApplication.exampleContext?.run {
            transferBytes = TrafficStats.getUidTxBytes(applicationInfo.uid)
        }
        // 当获取不到时,会返回TrafficStats.UNSUPPORTED
        return if (transferBytes == TrafficStats.UNSUPPORTED.toLong()) 0 else transferBytes / 1024
    }
    // 通过Timer每隔1秒计算网速
    private fun calculateNetSpeed() {
        ExampleApplication.exampleContext?.run {
            val nowTotalReceiveBytes = getTotalReceiveBytes()
            val nowTotalTransferBytes = getTotalTransferBytes()
            val downloadSpeed = nowTotalReceiveBytes - lastTotalReceiveBytes
            val uploadSpeed = nowTotalTransferBytes - lastTotalTransferBytes
            lastTotalReceiveBytes = nowTotalReceiveBytes
            lastTotalTransferBytes = nowTotalTransferBytes
            netSpeedCallback?.onNetSpeedChange("$downloadSpeed kb/s", "$uploadSpeed kb/s")
        }
    }
    fun startMeasuringNetSpeed() {
        if (timer == null && timerTask == null) {
            timer = Timer()
            timerTask = object : TimerTask() {
                override fun run() {
                    calculateNetSpeed()
                }
            }
            timer?.run { timerTask?.let { schedule(it, 0L, 1000L) } }
        }
    }
    fun stopMeasuringNetSpeed() {
        timerTask?.cancel()
        timerTask = null
        timer?.cancel()
        timer = null
    }
    interface NetSpeedCallback {
        fun onNetSpeedChange(downloadSpeed: String, uploadSpeed: String)
    }
}
// 示例类
class TrafficStatsActivity : BaseGestureDetectorActivity() {
    private lateinit var binding: LayoutTrafficStatsActivityBinding
    @SuppressLint("SetTextI18n")
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        binding = DataBindingUtil.setContentView(this, R.layout.layout_traffic_stats_activity)
        binding.includeTitle.tvTitle.text = "TrafficStatsExample"
        NetSpeedUtils.netSpeedCallback = object : NetSpeedUtils.NetSpeedCallback {
            override fun onNetSpeedChange(downloadSpeed: String, uploadSpeed: String) {
                binding.tvNetSpeed.run { post { text = "downloadSpeed:$downloadSpeed , uploadSpeed:$uploadSpeed" } }
            }
        }
        binding.btnStartMeasureNetSpeed.setOnClickListener {
            NetSpeedUtils.startMeasuringNetSpeed()
        }
        binding.btnStopMeasureNetSpeed.setOnClickListener {
            NetSpeedUtils.stopMeasuringNetSpeed()
        }
        initWebViewSetting(binding.webView)
        binding.webView.loadUrl("https://go.minigame.vip/")
    }
    @SuppressLint("JavascriptInterface", "SetJavaScriptEnabled")
    private fun initWebViewSetting(webView: WebView?) {
        webView?.run {
            settings.cacheMode = WebSettings.LOAD_DEFAULT
            settings.domStorageEnabled = true
            settings.allowContentAccess = true
            settings.allowFileAccess = true
            settings.useWideViewPort = true
            settings.loadWithOverviewMode = true
            settings.mixedContentMode = WebSettings.MIXED_CONTENT_ALWAYS_ALLOW
            settings.javaScriptEnabled = true
            settings.javaScriptCanOpenWindowsAutomatically = true
            settings.setSupportMultipleWindows(true)
        }
    }
    override fun onDestroy() {
        super.onDestroy()
        binding.webView.clearHistory()
        binding.webView.loadDataWithBaseURL(null, "", "text/html", "utf-8", null)
        binding.root.run {
            if (this is ViewGroup) {
                this.removeView(binding.webView)
            }
        }
        binding.webView.destroy()
    }
}

效果如图:

以上就是Android 获取实时网速实现详解的详细内容,更多关于Android 获取实时网速的资料请关注脚本之家其它相关文章!

相关文章

  • 详解Android WebView加载html片段

    详解Android WebView加载html片段

    本篇文章主要介绍了详解Android WebView加载html片段,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-08-08
  • Android开发中遇到端口号占用问题解决方法

    Android开发中遇到端口号占用问题解决方法

    这篇文章主要介绍了Android开发中遇到端口号占用问题解决方法,本文给出了一个简洁实用的方法来解决这个烦人的问题,需要的朋友可以参考下
    2015-06-06
  • 详解Android中实现Redux方法

    详解Android中实现Redux方法

    本篇文章给大家通过代码实例教学Android中实现Redux的方法,有需要的朋友跟着参考下吧。
    2018-01-01
  • 解决Android Studio 3.0 butterknife:7.0.1配置的问题

    解决Android Studio 3.0 butterknife:7.0.1配置的问题

    下面小编就为大家分享一篇解决Android Studio 3.0 butterknife:7.0.1配置的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2017-12-12
  • Android传感器使用实例介绍

    Android传感器使用实例介绍

    这篇文章主要为大家详细介绍了Android传感器的简单使用方法,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-12-12
  • Android实现可收缩和扩展的TextView

    Android实现可收缩和扩展的TextView

    这篇文章主要为大家详细介绍了Android实现可收缩和扩展的TextView,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-03-03
  • Android 中自定义Dialog样式的Activity点击空白处隐藏软键盘功能(dialog不消失)

    Android 中自定义Dialog样式的Activity点击空白处隐藏软键盘功能(dialog不消失)

    项目中需要开发带有EditText的Dialog显示,要求在编辑完EditText时,点击Dilog的空白处隐藏软键盘。但是Dialog不会消失。下面通过实例代码给大家分享实现方法,需要的的朋友参考下吧
    2017-04-04
  • Android Studio Gradle依赖冲突解决方法

    Android Studio Gradle依赖冲突解决方法

    这篇文章主要给大家介绍了关于Android Studio Gradle依赖冲突解决的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用Android Studio具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-04-04
  • Android自定义View实现QQ运动积分转盘抽奖功能

    Android自定义View实现QQ运动积分转盘抽奖功能

    这篇文章主要为大家详细介绍了Android自定义View实现QQ运动积分转盘抽奖功能,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-10-10
  • Flutter开发之Widget自定义总结

    Flutter开发之Widget自定义总结

    这篇文章主要给大家介绍了关于Flutter开发中Widget自定义的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用Flutter具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-04-04

最新评论