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

testly · 2015年05月27日 · 最后由 daxian37 回复于 2018年04月11日 · 4451 次阅读
本帖已被设为精华帖!

流量篇

最近在研究 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;
   }
}

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

如果觉得我的文章对您有用,请随意打赏。您的支持将鼓励我继续创作!
共收到 42 条回复 时间 点赞
仅楼主可见
向阳 利用 anyproxy 做 app 网络流量测试 中提及了此贴 03月17日 14:15
AppetizerIO 优化 APP 网络流量 中提及了此贴 01月03日 15:04
AppetizerIO [该话题已被删除] 中提及了此贴 01月03日 14:51
luxee 回复

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

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

绵绵不绝 回复

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

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

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

testly #45 · 2017年07月12日 Author

、棒

找到相应的资料:

/** 
    * 获取手机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;  
   }  


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

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

missgong0 回复

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

testly 回复

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

testly #38 · 2017年03月24日 Author
missgong0 回复

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

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

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

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

第四张图标错喽哦

匿名 #35 · 2016年06月03日

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

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

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

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

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

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

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

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

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

testly #25 · 2015年06月08日 Author

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

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

testly #13 · 2015年05月27日 Author

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

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

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

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

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

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

等很多处

这个可用性不高。

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

要不要那么快

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

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

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

3楼 已删除
testly #13 · 2015年05月27日 Author

#1 楼 @seveniruby 还没睡么?

我表示连夜加精

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