移动性能测试 Android 性能测试实践 (四) 流量

testly · 发布于 2015年05月27日 · 最后由 18576470925 回复于 2017年11月05日 · 4762 次阅读
本帖已被设为精华帖!

流量篇

最近在研究IOS 的性能测试,时间太紧没来得及发帖,加通宵挤出时间给大家分享一点东西,希望对大家有所帮助!

Android 2.2之前
对于Android2.2 的流量 版本以前的系统的流量信息都存放在 proc/net/dev(或者 proc/self/net/dev)文件下,读取文件然后对其进行解析就行了。读取某一个应用的流量,则读取proc/uid_stat/uid /tcp_rcv 文件进行解析(注:模拟器下不存在这个目录)。如需查看某个应用的流量信息,可以通过以下命令来实现:

adb devices                         列出所有设备
adb -s 设备名称 shell                进入对应的设备
cd proc                             进入设备的属性目录
cd uid_stat                         进入 user id 状态目录,每个应用程序在安装的时候系统会为每个应用分配一个对应的 uid
ls                                  列出 uid_stat 目录下所有应用对应的 user id 目录
cd uid                              进入对应应用的 uid 目录
ls                                  查看对应 uid 目录下的 tcp_rcv  tcp_snd 目录
cat tcp_rcv                         查看该应用接收的数据信息
cat tcp_snd                         查看该应用发送的数据信息

Android 2.2之后

我这里有两种办法:

第一种
通过PID下面的net/dev
先找到应用的PID

adb shell ps

这边拿到PID:21896 然后在去/proc目录下的PID/net/dev面可以看到:

adb shell cat /proc/"+Pid+"/net/dev"

这边的wlan0代表wifi 上传下载量标识! 上传下载量单位是字节可以/1024换算成KB
这里可以看到下载的字节数 、数据包 和 发送的字节数 、数据包

小技巧:wlan0这些值如何初始化0 很简单 你打开手机飞行模式再关掉就清0了

第二种

通过proc/net/xt_qtaguid/stats

在说第二种获取流量方法之前先给这边先给大家说下uid

uid的获取可以在对应的PID下面去查看status,里面会查到uid

adb shell cat /proc/<pid>/status

下面这个方法是通过PackageManager去取:


try {
            PackageManager pm = getPackageManager();
            ApplicationInfo ai = pm.getApplicationInfo("PackageName", PackageManager.GET_ACTIVITIES);
            Log.d("!!", "!!" + ai.uid);
        } catch (NameNotFoundException e) {
            e.printStackTrace();
        }

拿到UID后呢继续:

adb shell cat /proc/net/xt_qtaguid/stats | grep uid

其中第6和8列为 rx_bytes(接收数据)和tx_bytes(传输数据)包含tcp,udp等所有网络流量传输的统计。
一个uid可能对应多个 进程,所以这有两行流量是累加的就求和就行。

用java去获取打印
我这边是用先获取PID然后调用!你可以把获取PID作为一个变量传到GetFlow里面来!
我这边只获取下载流量,你可以把上传下载的流量都获取出来!





//获取PID
public static String PID(String PackageName) throws IOException {

      String PID=null;
      Runtime runtime = Runtime.getRuntime();
      Process proc = runtime.exec("adb shell ps |grep "+PackageName);
      try {
          if (proc.waitFor() != 0) {
              System.err.println("exit value = " + proc.exitValue());
          }
          BufferedReader in = new BufferedReader(new InputStreamReader(
                  proc.getInputStream()));
          StringBuffer stringBuffer = new StringBuffer();
          String line = null;
          while ((line = in.readLine()) != null) {
              stringBuffer.append(line+" ");                                  
          }
      String str1=stringBuffer.toString();
      String str2=str1.substring(str1.indexOf(" "+PackageName)-46,str1.indexOf(" "+PackageName));
      String str3 =str2.substring(0,7);
      str3 = str3.trim();
      PID=str3;  
      } catch (InterruptedException e) {
          System.err.println(e);
      }finally{
          try {
              proc.destroy();
          } catch (Exception e2) {
          }
      }

      return PID;
}     






      //获取下载流量
public static double GetFlow(String PackageName) throws IOException {
      double FlowSize=0;
      String Pid=PID(PackageName);
  try{
      Runtime runtime = Runtime.getRuntime();
      Process proc = runtime.exec("adb shell cat /proc/"+Pid+"/net/dev");
      try {
          if (proc.waitFor() != 0) {
              System.err.println("exit value = " + proc.exitValue());
          }
          BufferedReader in = new BufferedReader(new InputStreamReader(
                  proc.getInputStream()));
          StringBuffer stringBuffer = new StringBuffer();
          String line = null;
          while ((line = in.readLine()) != null) {
              stringBuffer.append(line+" ");


          }
      String str1=stringBuffer.toString();
      String str2=str1.substring(str1.indexOf("wlan0:"),str1.indexOf("wlan0:")+90);
      String str4=str2.substring(7,16);
      str4 = str4.trim();
      int Flow=Integer.parseInt(str4);
      FlowSize=Flow/1024;

      } catch (InterruptedException e) {
          System.err.println(e);
      }finally{
          try {
              proc.destroy();
          } catch (Exception e2) {
          }
      }
  }
      catch (Exception StringIndexOutOfBoundsException)
      {

      }

      return FlowSize;
     }

获取每秒下载流量:

public static double Flow(String PackageName) throws IOException, InterruptedException
    {

        double Flow1=GetFlow(PackageName);
        Thread.sleep(1000);
        double Flow=GetFlow(PackageName)-Flow1;
        //System.out.println(GetFlow()-Flow1);
        return Flow ;

    }

场景设计

拿到流量值后在步骤前 将流量打印,再步骤完成后再打印一遍,再用步骤完成的流量值减去之前的流量值 得到这个步骤所消耗的流量!

场景案例:

拓展

下面的方法都是集成在Android 内部的方法:(仅供参考)

Android的TrafficStats类
前四个读取的/proc/net/dev里面的数据


   Android架构对流量的统计通过一个TrafficStats类可以直接获取
   获取总接受流量TrafficStats.getTotalRxBytes()
   获取总发送流量TrafficStats.getTotalTxBytes());
   获取不包含WIFI的手机GPRS接收量TrafficStats.getMobileRxBytes());
   获取不包含Wifi的手机GPRS发送量TrafficStats.getMobileTxBytes());
   统计某一个进程的总接收量TrafficStats.getUidRxBytes(Uid));
   统计某一个进程的总发送量TrafficStats.getUidTxBytes(Uid));




package cn.sunzn.trafficmanger;
import android.app.Activity;
import android.net.TrafficStats;
import android.os.Bundle;
import android.view.Menu;
public class MainActivity extends Activity {
   public void onCreate(Bundle savedInstanceState) {
       super.onCreate(savedInstanceState);
       setContentView(R.layout.activity_main);
       /** 获取手机通过 2G/3G 接收的字节流量总数 */
       TrafficStats.getMobileRxBytes();
       /** 获取手机通过 2G/3G 接收的数据包总数 */
       TrafficStats.getMobileRxPackets();
       /** 获取手机通过 2G/3G 发出的字节流量总数 */
       TrafficStats.getMobileTxBytes();
       /** 获取手机通过 2G/3G 发出的数据包总数 */
       TrafficStats.getMobileTxPackets();
       /** 获取手机通过所有网络方式接收的字节流量总数(包括 wifi) */
       TrafficStats.getTotalRxBytes();
       /** 获取手机通过所有网络方式接收的数据包总数(包括 wifi) */
       TrafficStats.getTotalRxPackets();
       /** 获取手机通过所有网络方式发送的字节流量总数(包括 wifi) */
       TrafficStats.getTotalTxBytes();
       /** 获取手机通过所有网络方式发送的数据包总数(包括 wifi) */
       TrafficStats.getTotalTxPackets();
       /** 获取手机指定 UID 对应的应程序用通过所有网络方式接收的字节流量总数(包括 wifi) */
       TrafficStats.getUidRxBytes(uid);
       /** 获取手机指定 UID 对应的应用程序通过所有网络方式发送的字节流量总数(包括 wifi) */
       TrafficStats.getUidTxBytes(uid);
   }
   public boolean onCreateOptionsMenu(Menu menu) {
       getMenuInflater().inflate(R.menu.activity_main, menu);
       return true;
   }
}

欢迎一起交流,一起进步 可以关注我的微信公众号:“测试开发进阶” - 点我关注

如果觉得我的文章对您有用,请随意打赏。您的支持将鼓励我继续创作!
共收到 41 条回复
104

我表示连夜加精

2113
testly · #2 · 2015年05月27日 作者

#1楼 @seveniruby 还没睡么?

3楼 已删除
Ee76af

其实还可以用tcpdump+wireshark可以实现 而且很精准_^

118

#4楼 @kasi iOS做不了这个。。。。

3059

#5楼 @monkey
iOS也可以tcpdump +wireshark,我一直这么测流量的

Ee76af

要不要那么快

118

#6楼 @wxpokay 嗯iOS可以用wireshark,我是说iOS做不了类似的比较好的自动化

118

话说。。我吐槽下。。你又hard code了。。

String str4=str2.substring(7,16);
 str4 = str4.trim();

等很多处

这个可用性不高。

Ee76af

monkey tcpdump+wireshark本身就可以实现自动化的

118

#10楼 @kasi 自动化是获取的自动化,但要结合场景在iOS上面不是很好的能够去做。另外,app本身关心的是详细数据与case场景的对应,这点就不是单纯抓包的自动化啦。获取数据肯定是ok的

Ee76af

这个我就不慎了解了,场景结合在android上面应该问题不大

2113
testly · #13 · 2015年05月27日 作者

#4楼 @kasi 你说的我也用过!

96

adb shell cat /proc/"+Pid+"/net/dev" ,无论pid是多少,输出都是一样的。
因此,这样是不能统计进程的流量的。

2113
testly · #16 · 2015年06月08日 作者

#15楼 @luxee 这个是总上行下行量

110

#4楼 @kasi 卡斯分享一个呗。
@testly 为啥不直接用 trafficstats 这个类呢?

96

我通过UID读取到的数据有好多条,但是我的应用只有一个进程啊,这些都相加得到网络流量吗?

96

感谢

1686

#16楼 @testly
我不知道楼主有没有用过一个叫做 安测试的工具 安装在手机里启动后可以测试指定应用的流量 内存 电量 CPU

96

很感谢你的文章,帮助很多,但是想吐槽一下,你的代码水平实在太...渣了

96

#15楼 @luxee 我擦,貌似真的一样啊

8502

#4楼 @kasi tcpdump+wireshark 怎么区分这个流量是自己的app产生的还是其他app产生的呢?

2373

#24楼 @x44562975 可以通过目标ip筛选,但是有些流量很难查出来,友盟这种数据统计的

96

#24楼 @x44562975 你是不是用tcpdump抓包,然后用wireshark分析?

960b2a

#10楼 @kasi monkey tcpdump+wireshark,这个怎么实现继续集成自动化呢?管家如何抓去手机里面的app的。

8772

第四张图标错喽哦

6853

tcp和udp都有一个header,这部分到底需不需要计算到流量里面去呢?

4024

String str2=str1.substring(str1.indexOf(" "+PackageName)-46,str1.indexOf(" "+PackageName)); 这句不太对哦,为啥一定是减去46呢,不一定吧

String regex = "\s+";
String[] arr=str1.split(regex);
PID=arr[1];
改成这样就完美了

2113
testly · #31 · 2017年03月24日 作者
4024missgong0 回复

谢谢。具体自己根据情况而定哈

4024
2113testly 回复

不过你这写的不错,已用😀 我就不用再写了。

9388
4024missgong0 回复

String regex = "\s+";
正则是两个反斜线,小问题记录下~

4845

上面三种获取流量的方式,所获取到的流量数值都是不一样的。这怎么破?

6504

请问下,如果想统计非wifi下的流量是哪个字段?rmnet字段?

6504
6504lose 回复

找到相应的资料:

/** 
    * 获取手机GPRS的下载流量 
    *  
    * @return 
    * */  
   public static long getMobileRxBytes() {  
       long ReturnLong = 0; // 查询到的结果  
       try {  
           File file = new File("/proc/net/dev");  
           FileInputStream inStream = new FileInputStream(file);  
           String a = readInStream(inStream);  
           int startPos = a.indexOf("rmnet0:");  
           a = a.substring(startPos);  
           Pattern p = Pattern.compile(" \\d+ ");  
           Matcher m = p.matcher(a);  
           while (m.find()) {  
               ReturnLong = Long.parseLong(m.group().trim());  
               break;  
           }  

       } catch (FileNotFoundException e1) {  
           e1.printStackTrace();  
       }  
       return ReturnLong;  
   }  

   /** 
    * 获取手机GPRS的上传流量 
    *  
    * @return 
    * */  
   public static long getMobileTxBytes() {  
       long ReturnLong = 0; // 查询到的结果  
       try {  
           int count = 0; // 返回结果时的计数器  
           File file = new File("/proc/net/dev");  
           FileInputStream inStream = new FileInputStream(file);  
           String a = readInStream(inStream);  
           int startPos = a.indexOf("rmnet0:");  
           a = a.substring(startPos);  
           Pattern p = Pattern.compile(" \\d+ ");  
           Matcher m = p.matcher(a);  
           while (m.find()) {  
               if (count == 8) {  
                   ReturnLong = Long.parseLong(m.group().trim());  
                   break;  
               }  
               count++;  

           }  

       } catch (FileNotFoundException e1) {  
           e1.printStackTrace();  
       }  
       return ReturnLong;  
   }  


2113
testly · #37 · 2017年07月12日 作者
6504lose 回复

、棒

51f154

赞一个!
有点遗憾的是,貌似python必须要pid才能获取到流量,如果app被杀掉,就没有pid了。假设我想记录一个app启动时消耗的流量,不好统计啊。

51f154

@seveniruby 大佬,能不能改用户名啊?这个显示电话号码贼尴尬

104
51f154zhizhizhi2017 回复

可以的. 我给你调整下. 你的id是什么?

51f154
104seveniruby 回复

不知道ID啊
新给一个ID吗?
zhizhizhi2017

51f154
32luxee 回复

是的,还以为自己脚本哪里出问题,骗死人啊,既然都一样,为何还分pid去查询。

需要 登录 后方可回复, 如果你还没有账号请点击这里 注册