Android流量统计

摘要:
发现有两种主要方法可以使用系统统计类TrafficStats通过系统文件解析获取和读取TrafficStatsstatlonggetMobileRxBytes()//获取通过移动连接接收的总字节数。

项目中需要对Android设备进行流量统计来进行资费结算,所以对Android设备流量统计进行了一些调研。发现流量统计主流上有两种方式

  • 使用系统统计类TrafficStats获取
  • 通过系统文件解析读取

TrafficStats

  • static long getMobileRxBytes() //获取通过Mobile连接收到的字节总数,不包含WiFi
  • static long getMobileRxPackets() //获取Mobile连接收到的数据包总数
  • static long getMobileTxBytes() //Mobile发送的总字节数
  • static long getMobileTxPackets() //Mobile发送的总数据包数
  • static long getTotalRxBytes() //获取总的接受字节数,包含Mobile和WiFi等
  • static long getTotalRxPackets() //总的接受数据包数,包含Mobile和WiFi等
  • static long getTotalTxBytes() //总的发送字节数,包含Mobile和WiFi等
  • static long getTotalTxPackets() //发送的总数据包数,包含Mobile和WiFi等
  • static long getUidRxBytes(int uid) //获取某个网络UID的接受字节数
  • static long getUidTxBytes(int uid) //获取某个网络UID的发送字节数

这些方法 获取到的流量数据都是从手机上次开机到当前的流量使用情况,这个没关系,我们可以监听手机的开机和关机。关机的时候(或者每间隔一段时间) 将数据都存储起来,下次开机后通过以上方法获取到的数据在叠加上.

但是这些方法里面,只能我发现在这些方法里,缺少我最需要的那个方法,我想获取某个app接受和发送的2G/3G流量,不包含wifi,这些API无法满足该需求。

于是乎,便从系统文件的解析上着手吧

/proc/net/xt_qtaguid/stats格式

/proc/net/xt_qtaguid/stats内包含着各个app使用流量的情况,先来简单看一下这个文件的内容

Android流量统计第1张

这输出内容整体是一个表结构,首先我们需要看一下表中各列字段代表什么

  • idx : 序号
  • iface : 代表流量类型(rmnet表示2G/3G, wlan表示Wifi流量,lo表示本地流量)
  • acct_tag_hex :线程标记(用于区分单个应用内不同模块/线程的流量)
  • uid_tag_int : 应用uid,据此判断是否是某应用统计的流量数据
  • cnt_set : 应用前后标志位:1:前台, 0:后台
  • rx_btyes : receive bytes 接受到的字节数
  • rx_packets : 接收到的任务包数
  • tx_bytes : transmit bytes 发送的总字节数
  • tx_packets : 发送的总包数
  • rx_tcp_types : 接收到的tcp字节数
  • rx_tcp_packets : 接收到的tcp包数
  • rx_udp_bytes : 接收到的udp字节数
  • rx_udp_packets : 接收到的udp包数
  • rx_other_bytes : 接收到的其他类型字节数
  • rx_other_packets : 接收到的其他类型包数
  • tx_tcp_bytes : 发送的tcp字节数
  • tx_tcp_packets : 发送的tcp包数
  • tx_udp_bytes : 发送的udp字节数
  • tx_udp_packets : 发送的udp包数
  • tx_other_bytes : 发送的其他类型字节数
  • tx_other_packets : 发送的其他类型包数

可以看到,用文件解析的方式获取流量可以获得比直接使用acct_tag_hex要丰富的多的信息,包括区分WIFI和移动网络,区分应用内不同模块功能流量,前后台流量等。

使用shell命令来查看进程流量

接下来说明下uid_tag_int这个字段

pid:Processid(进程id)
获取pid的shell命令:

grep com.car.tencent

Android流量统计第2张

  • uid:每个应用程序一个uid(如果一个app拥有两个进程,则有两个不同pid,但两个pid对应的uid是相同的)
    根据pid获取uid
adb shell cat /proc/1643/status
  • 根据uid获取流量
adb shell cat /proc/net/xt_qtaguid/stats | grep uid

如何使用代码解析/proc/net/xt_qtaguid/stats文件

static const char *QTAGUID_UID_STATS = "/proc/net/xt_qtaguid/stats";

struct Stats {
    uint64_t rxBytes;
    uint64_t rxPackets;
    uint64_t txBytes;
    uint64_t txPackets;
    uint64_t tcpRxPackets;
    uint64_t tcpTxPackets;
};

static int parseUidStats(const uint32_t uid, struct Stats *stats) {
    FILE *fp = fopen(QTAGUID_UID_STATS, "r");
    if (fp == NULL) {
        return -1;
    }

    char buffer[384];
    char iface[32];
    uint32_t idx, cur_uid, set;
    uint64_t tag, rxBytes, rxPackets, txBytes, txPackets;

    while (fgets(buffer, sizeof(buffer), fp) != NULL) {
        if (sscanf(buffer,
                   "%" SCNu32 " %31s 0x%" SCNx64 " %u %u %" SCNu64 " %" SCNu64
                           " %" SCNu64 " %" SCNu64 "",
                   &idx, iface, &tag, &cur_uid, &set, &rxBytes, &rxPackets,
                   &txBytes, &txPackets) == 9) {
            //绕过本地回环地址
            if (strcmp(iface, "lo") && uid == cur_uid && tag == 0L) {
                stats->rxBytes += rxBytes;
                stats->rxPackets += rxPackets;
                stats->txBytes += txBytes;
                stats->txPackets += txPackets;
            }
        }
    }

    if (fclose(fp) != 0) {
        return -1;
    }
    return 0;
}

static jlong getUidStat(JNIEnv *env, jclass clazz, jint uid, jint type) {
    struct Stats stats;
    memset(&stats, 0, sizeof(Stats));
    if (parseUidStats(uid, &stats) == 0) {
        return getStatsType(&stats, (StatsType) type);
    } else {
        return UNKNOWN;
    }
}

这段代码里面,需要注意的点是如何忽略本地回环流量

if (strcmp(iface, "lo") && uid == cur_uid && tag == 0L) {

因为iface是网络类型,一共只有三个类型,rmnet表示2G/3G, wlan表示Wifi流量,lo表示本地流量。所以这里直接使用了字符串比对的方式。

扩展

  • /proc/net/xt_qtaguid/iface_stat_fmt:该文件中已经按照类型统计出了不同类型网络数据的总数,因此获取当前设备的总接受字节数等设备级统计信息,直接分析该文件,能最快获取结果
  • /proc/net/xt_qtaguid/stats:该文件中则是按照uid进行了分类,适合于做更细力度(应用级/线程级)的分析

附录:

参考资料:

免责声明:文章转载自《Android流量统计》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇nuxtjs服务端部署流程js对flv提取h264、aac音视频流下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

SQL Server Profiler 常见问题总结

1、跟踪指定数据库   SELECT DB_ID('数据名称')   原文:https://jingyan.baidu.com/article/647f0115be128a7f2048a87d.html 2. 列筛选器找不到DatabaseId   先勾选 显示所有列 ,再进入 列筛选器 即可看到.          原文的第四点:https://jing...

C# Sql 触发器

触发器是一种特殊类型的存储过程,它不同于之前的我们介绍的存储过程。触发器主要是通过事件进行触发被自动调用执行的。而存储过程可以通过存储过程的名称被调用。 Ø 什么是触发器     触发器对表进行插入、更新、删除的时候会自动执行的特殊存储过程。触发器一般用在check约束更加复杂的约束上面。触发器和普通的存储过程的区别是:触发器是当对某一个表进行操作。诸如:...

二、Metrics指标类型

Prometheus 的客户端库中提供了四种核心的指标类型。但这些类型只是在客户端库(客户端可以根据不同的数据类型调用不同的 API 接口)和在线协议中,实际在 Prometheus server 中并不对指标类型进行区分,而是简单地把这些指标统一视为无类型的时间序列 2.1、Counter (计数器) ​ Counter 类型代表一种样本数据单调递增的指...

关于redshift数据库当中的STL_LOAD_ERRORS问题的解决

今天写了Python脚本准备将s3上面的数据迁移到redshift上面去,突然发现在数据load的时候出现了STL_LOAD_ERRORS,刚接触到redshift也没有报错根本不知道怎么解决。  这里显示报错了,具体的日志要去STL_LOAD_ERRORS当中查看,这个其实是一张表。在这个表当中有报错的相信原因。怎么去查询原因那,查询的语句如下: S...

交换机的三种转发模式

下面是交换机6大工作原理: 1、基于源MAC地址学习 2、基于目标MAC地址转发  3、同一接口可以学习到多个MAC地址  4、同一个MAC地址被多个接口学习到,选择后学习到的接口 5、收到广/组播帧, 向本VLAN的其他所有接口转发  6、对于没有目标MAC地址表项的帧,向本VLAN的其他所有接口转发   交换机的三种转发模式: 1、直通式转发:  是指...

Mysql基础01-语法

数据库 数据的存储:将数据放到表中,表再放到库中。 一个数据库中可以有多个表,每个表都有一个名字,用来标识自己。表名具有唯一性。 表由列组成,我们也称为字段。每个字段描述了它所含有的数据的意义表由列组成,我们也称为字段。每个字段描述了它所含有的数据的意义 表中的数据是按行存储的,一行即为一条记录。 MySQL下载安装略过 MySQL设置命令 net sta...