Android应用流量统计调研

参考:http://m.mamicode.com/info-detail-1802432.html?from=singlemessage&isappinstalled=1

Android应用流量统计在6.0之前一直没有太好的办法,官方虽然提供了TrafficStats,但其主要功能是设备启动以来流量的统计信息,和时间信息无法很好的配合。最近再看TrafficStats类时,发现说明中提到,为获取更具健壮性的网络历史数据,建议使用NetworkStatsManager。

本文首先简单对比下TrafficStats和NetworkStatsManager各自的限制和优缺点,然后详细说明NetworkStatsManager的用法,并给出主要代码。

TrafficStats(android 6.0之前使用)

Android API8提供了android.net.TrafficStats类。
通过此类能获取设备重启以来网络信息,部分函数如下所示:

static long  getMobileRxBytes()  //获取通过移动数据网络收到的字节总数
static long  getMobileTxBytes()  //通过移动数据网发送的总字节数 
static long  getTotalRxBytes()  //获取设备总的接收字节数
static long  getTotalTxBytes()  //获取设备总的发送字节数
static long  getUidRxBytes(int uid)  //获取指定uid的接收字节数 
static long  getUidTxBytes(int uid) //获取 指定uid的发送字节数 

通过文档及上述函数可以知道,TrafficStats能够获取设备的数据流量和总的网络流量消耗(一般情况下也就得到Wi-Fi下的流量信息);可以查询uid对应的流量信息,而uid可以通过应用的包名查询到,因此能够查询某个应用的流量统计信息(不考虑shareuid)。非常方便的是,它的使用不需要特别的权限。另一方面它也一些限制:
(1)无法获取应用的数据流量消耗
从文档中仅能获取到指定uid的流量,但无法区分不同网络类型下的消耗
间接方法是通过监听网络切换,做好流量记录(但是要保证你的应用一直存活,且一定准确接收到网络切换信息),基本不可用。
(2)无法获取某个时间段内的流量消耗
从API文档中看,函数参数没有与时间相关的信息。而且重要的一点是,TrafficStats类中记录的是设备重启以来的流量统计信息。因为TrafficStats 类,底层还是读取/proc/net/xt_qtaguid/stats 对内容进行解析,将得到对应的结果返回上层。
使用方法:直接调用
如获取开机以来设备总的网络使用统计(单位:字节)
TrafficStats. getTotalRxBytes()+TrafficStats. getTotalTxBytes()

NetworkStatsManager(android6.0之后)

在Android 6.0(API23)中新增加的类,提供网络使用历史统计信息,同时特别强调了可查询指定时间间隔内的统计信息。看看部分函数(非静态):

//查询指定网络类型在某时间间隔内的总的流量统计信息
NetworkStats.Bucket querySummaryForDevice(int networkType, String subscriberId, long startTime, long endTime)

//查询总的网络使用统计信息
NetworkStats querySummary(int networkType, String subcriberId, long startTime, long endTime)

 //查询某uid在指定网络类型和时间间隔内的流量统计信息
NetworkStats queryDetailsForUid(int networkType, String subscriberId, long startTime, long endTime, int uid) 

//查询指定网络类型在某时间间隔内的详细的流量统计信息(包括每个uid)
NetworkStats queryDetails(int networkType, String subscriberId, long startTime, long endTime) 

从上述函数和文档看,NetworkStatsManager类克服了TrafficStats的查询限制,而且统计信息也不再是设备重启以来的数据。但它也有自己的限制和缺点。
(1)权限限制
NetworkStatsManager的使用需要额外的权限,”android.permission.PACKAGE_USAGE_STATS”是系统权限,需要主动引导用户开启应用的“有权查看使用情况的应用”(使用记录访问权限)权限;另外需要动态申请“ android.permission.READ_PHONE_STATE ”危险权限。
(2)文档不完善
不好说是文档不全,还是我没找对。首先文档中没有给出类的实例对象的构造方法,一开始还是反射获取的,后来才发现可以通过获取系统服务方式得到。另外queryDetailsForUid函数中设置的时间间隔不太有用,没能及时的获取流量统计信息,而是有两个小时的时间间隔。还好可以在querySummary函数中获得。

代码示例

下面说说具体的使用和代码,使用前必须明确的是这里的统计信息都是在网络层以上的数据。
1.权限设置
(1)AndroidManifest中添加权限声明

android :name= "android.permission.READ_PHONE_STATE" />
android :name= "android.permission.PACKAGE_USAGE_STATS"  tools :ignore= "ProtectedPermissions" />

2.查看设备和某应用的流量统计
(1)获取NetworkStatsManager示例对象
networkStatsManager = (NetworkStatsManager) getSystemService( NETWORK_STATS_SERVICE) ;

(2)查询某应用(uid)的数据流量统计信息

TelephonyManager tm = ( TelephonyManager) getSystemService( TELEPHONY_SERVICE) ;
String subId = tm.getSubscriberId() ;
NetworkStats summaryStats ;
long summaryRx =  0 ;
long summaryTx =  0 ;
NetworkStats.Bucket summaryBucket =  new NetworkStats.Bucket() ;
long summaryTotal =  0 ;
summaryStats =  networkStatsManager.querySummary(ConnectivityManager. TYPE_MOBILE subId getMillsTime() System. currentTimeMillis()) ;
do {
    summaryStats.getNextBucket(summaryBucket) ;
    int summaryUid = summaryBucket.getUid() ;
    int uid  =  getUidByPackageName( this, "com.qiyi.video") ;
    if (uid == summaryUid) {
        Log. e(MainActivity. class.getSimpleName() "uid:" + summaryBucket.getUid() +  " rx:" + summaryBucket.getRxBytes() +
                " tx:" + summaryBucket.getTxBytes()) ;
        summaryRx += summaryBucket.getRxBytes() ;
        summaryTx += summaryBucket.getTxBytes() ;
    }
while (summaryStats.hasNextBucket()) ;
Log. e(MainActivity. class.getSimpleName() " 爱奇艺当前时间段内移动流量使用情况:: "+ "uid:" getUidByPackageName( this, "com.mobilewise.mobileware") +  " rx:" + summaryRx +  " tx:" + summaryTx) ;

3.附赠实用函数
(1)应用包名查uid

public static int  getUidByPackageName(Context context String packageName) {
    int uid = - 1 ;
    PackageManager packageManager = context.getPackageManager() ;
    try {
        PackageInfo packageInfo = packageManager.getPackageInfo(packageName PackageManager. GET_META_DATA) ;
        uid = packageInfo. applicationInfo. uid ;
   catch (PackageManager.NameNotFoundException e) {
    }
    return uid ;
}

(2)获得具体日期的时间(毫秒)
// 1970-01-01 00:00:00.000 到具体日期的毫秒数
public long  getMillsTime()  throws ParseException {
    // 将字符串转为日期
    SimpleDateFormat sdf= new SimpleDateFormat( "yyyy-MM-dd HH:mm:ss") ;
    String dstr= "2017-09-20 00:00:00 " ;
    java.util.Date date=sdf.parse(dstr) ;
    long  s1=date.getTime() ; // 将时间转为毫秒
    return s1 ;
}

统计测试
(1)测试设备
华为Mate8 Android 6.0设备上实际测试
(2)流量差距
当天系统统计爱奇艺使用移动流量47.96M,代码统计的爱奇艺移动流量为47.40M

你可能感兴趣的:(学习笔记,Android基础)