12

我的应用程序正在尝试计算通过 WiFi/LAN 和移动数据连接发送和接收的字节数。为此,我TrafficStats在某个时间点获取计数器的值,并在下次检查时从其值中减去该值。

// get current values of counters
long currentMobileTxBytes = TrafficStats.getMobileTxBytes();
long currentMobileRxBytes = TrafficStats.getMobileRxBytes();
long totalTxBytes = TrafficStats.getTotalTxBytes();
long totalRxBytes = TrafficStats.getTotalRxBytes();

// to get mobile data count, subtract old from current
long currentMobileSent = currentMobileTxBytes - oldMobileTxBytes;
long currentMobileReceived = currentMobileRxBytes - oldMobileRxBytes;

// to get WiFi/LAN data count, subtract total from mobile
long currentNetworkSent = totalTxBytes - currentMobileTxBytes;
long currentNetworkReceived = totalRxBytes - currentMobileRxBytes;

我觉得上述算法是合理的,但是,我不确定如何检查这些计数器的准确性。例如,当我尝试通过 WiFi 将 2.7MB 的文件上传到 Dropbox 时,currentMobileSent我得到的值约为 10MB。即使在下一次检查之前没有上网,我得到的非零值表明我在等待期间确实收到了一些字节的数据。

有没有办法让我检查TrafficStats这些数字是如何得出的?我知道除了我的浏览器之外,可能还有其他应用程序在后台运行连接到互联网,但是 2.7MB 到 10MB 似乎是一个巨大的跳跃——我什至没有做任何事情就“收到”了一次 90MB。还是我计算发送和接收字节的方式有问题?

4

1 回答 1

15

来自技术共和国

  1. 在 Eclipse 中创建一个新的 Android 项目。请记住使用 TrafficStats 类,您必须针对 Android 2.2 (Froyo) 或更高版本的 API。

  2. 在该/res/layout文件夹中,我们将创建一个 activity_main.xml 资源。对于这个项目,我们只是在垂直堆叠的线性布局中使用一系列文本视图。

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
   android:layout_width="fill_parent"
   android:layout_height="fill_parent"
   android:orientation="vertical">

   <TextView
       android:layout_width="fill_parent"
       android:layout_height="wrap_content"
       android:gravity="center"
       android:paddingBottom="20dip"
       android:text="Traffic Stats Demo"
       android:textSize="16sp"
       android:textStyle="bold" />

   <TextView
       android:layout_width="fill_parent"
       android:layout_height="wrap_content"
       android:gravity="center"
       android:text="Transmit Bytes"
       android:textColor="#00ff00"
       android:textSize="14sp" />

   <TextView
       android:id="@+id/TX"
       android:layout_width="fill_parent"
       android:layout_height="wrap_content"
       android:gravity="center"
       android:text="0"
       android:textSize="14sp" />

   <TextView
       android:layout_width="fill_parent"
       android:layout_height="wrap_content"
       android:gravity="center"
       android:text="Receive Bytes"
       android:textColor="#ff0000"
       android:textSize="14sp" />

   <TextView
       android:id="@+id/RX"
       android:layout_width="fill_parent"
       android:layout_height="wrap_content"
       android:gravity="center"
       android:text="0"
       android:textSize="14sp" />
</LinearLayout>

布局到位后,我们可以转到 /src 文件夹。通过扩展 Activity/AppCompatActivity 类创建 MainActivity.java。让我们继续声明三个私有类变量。

MainActivity.java

package com.authorwjf;

import android.app.Activity;
import android.app.AlertDialog;
import android.net.TrafficStats;
import android.os.Bundle;
import android.os.Handler;
import android.widget.TextView;

public class Main extends Activity {
    private Handler mHandler = new Handler();
    private long mStartRX = 0;
    private long mStartTX = 0;
}

我们将使用 on create override 来初始化我们的私有变量,并在 UI 线程上安排回调。记下对枚举 TrafficStats.UNSUPPORTED 的检查。虽然我对 TrafficStats 类的体验一帆风顺,但 Google 官方文档指出,某些设备可能不支持这种类型的报告,在这种情况下,调用会返回上述值。出于这个原因,防御性地编写代码是一个好主意,正如我在这里演示的那样。

MainActivity.java

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    mStartRX = TrafficStats.getTotalRxBytes();
    mStartTX = TrafficStats.getTotalTxBytes();

    if (mStartRX == TrafficStats.UNSUPPORTED || mStartTX == TrafficStats.UNSUPPORTED) {
        AlertDialog.Builder alert = new AlertDialog.Builder(this);
        alert.setTitle("Uh Oh!");
        alert.setMessage("Your device does not support traffic stat monitoring.");
        alert.show();
    } else {
        mHandler.postDelayed(mRunnable, 1000);
    }
}

最后但并非最不重要的一点是,我们需要更新我们的显示并重新安排可运行文件。

MainActivity.java

private final Runnable mRunnable = new Runnable() {
    public void run() {
        TextView RX = (TextView) findViewById(R.id.RX);
        TextView TX = (TextView) findViewById(R.id.TX);
        long rxBytes = TrafficStats.getTotalRxBytes() - mStartRX;
        RX.setText(Long.toString(rxBytes));
        long txBytes = TrafficStats.getTotalTxBytes() - mStartTX;
        TX.setText(Long.toString(txBytes));
        mHandler.postDelayed(mRunnable, 1000);
    }
};
于 2012-10-07T04:15:24.880 回答