此文章为网上诸多文章总结所得,主要借鉴别人的思想,并在其中添加了自己的一些思考,使实现更加明确。
在没有Root的情况下,Android应用流量统计在6.0之前一直没有太好的办法,官方虽然提供了TrafficStats,但其主要功能是设备启动以来流量的统计信息,和时间信息无法很好的配合。最近再看TrafficStats类时,发现说明中提到,为获取更具鲁棒性的网络历史数据,建议使用NetworkStatsManager。
在Android 6.0(API23)中新增加的类,提供网络使用历史统计信息,同时特别强调了可查询指定时间间隔内的统计信息。看看部分函数(非静态):
//查询指定网络类型在某时间间隔内的总的流量统计信息 NetworkStats.BucketquerySummaryForDevice(intnetworkType,StringsubscriberId,longstartTime,longendTime)
//查询某uid在指定网络类型和时间间隔内的流量统计信息NetworkStatsqueryDetailsForUid(intnetworkType,StringsubscriberId,longstartTime,longendTime,intuid)
//查询指定网络类型在某时间间隔内的详细的流量统计信息(包括每个uid)NetworkStatsqueryDetails(intnetworkType,StringsubscriberId,longstartTime,longendTime)
从上述函数和文档看,NetworkStatsManager类克服了TrafficStats的查询限制,而且统计信息也不再是设备重启以来的数据。但它也有自己的限制和缺点。
(1)权限限制
NetworkStatsManager的使用需要额外的权限,”android.permission.PACKAGE_USAGE_STATS”是系统权限,需要主动引导用户开启应用的“有权查看使用情况的应用”(使用记录访问权限)权限,后面会有代码示例。
(2)文档不完善
不好说是文档不全,还是我没找对。首先文档中没有给出类的实例对象的构造方法,一开始还是反射获取的,后来才发现可以通过获取系统服务方式得到。另外queryDetailsForUid函数中设置的时间间隔不太有用,没能及时的获取流量统计信息,而是有两个小时的时间间隔,且获得的连续性不好。不过可以在querySummary函数中获得。实现方法为通过querySummary获得所有uid的流量,在检索出需要查找的uid的流量;由于获取流量时会产生通讯流量,因此,会出现相同的uid,实现时将两个uid的流量相加获得总流量(获取时,没有subid也成功了,不明白怎么回事)。
代码示例
下面说说具体的使用和代码,使用前必须明确的是这里的统计信息都是在网络层以上的数据。
1.权限设置
(1)AndroidManifest中添加权限声明
并在 xmlns:tools="http://schemas.android.com/tools" (2)代码中主动引导用户开启权限 READ_PHONE_STATE需要主动获取,此外还要获取历史历史记录的权限: private boolean hasPermissionToReadNetworkStats() { int permissionCheck = ContextCompat.checkSelfPermission(this, Manifest.permission.READ_PHONE_STATE); if (permissionCheck != PackageManager.PERMISSION_GRANTED) { ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.READ_PHONE_STATE}, REQUEST_READ_PHONE_STATE); } else { //TODO } if(Build.VERSION.SDK_INT returntrue; } finalAppOpsManagerappOps=(AppOpsManager)getSystemService(Context.APP_OPS_SERVICE); intmode=appOps.checkOpNoThrow(AppOpsManager.OPSTR_GET_USAGE_STATS, android.os.Process.myUid(),getPackageName()); if(mode==AppOpsManager.MODE_ALLOWED){ returntrue; } requestReadNetworkStats(); returnfalse; } //打开“有权查看使用情况的应用”页面 privatevoidrequestReadNetworkStats(){ Intentintent=newIntent(Settings.ACTION_USAGE_ACCESS_SETTINGS); startActivity(intent); } 2.查看设备和某应用的流量统计 (1)获取NetworkStatsManager示例对象 NetworkStatsManagernetworkStatsManager=(NetworkStatsManager)getSystemService(NETWORK_STATS_SERVICE); (2)查询设备总的流量统计信息 NetworkStats.Bucketbucket=null;//获取到目前为止设备的Wi-Fi流量统计 bucket=networkStatsManager.querySummaryForDevice(ConnectivityManager.TYPE_WIFI,"",0,System.currentTimeMillis()); Log.i("Info","Total: "+(bucket.getRxBytes()+bucket.getTxBytes())); (3)查询某应用(uid)的数据流量统计信息 //获取subscriberId TelephonyManager tm=TelephonyManager)getSystemService(TELEPHONY_SERVICE); StringsubId=tm.getSubscriberId(); NetworkStatssummaryStats; longsummaryRx=0; longsummaryTx=0; longsummaryRx =0; longStartTx=0; NetworkStats.BucketsummaryBucket=newNetworkStats.Bucket(); longsummaryTotal=0; summaryStats=networkStatsManager.querySummary(ConnectivityManager.TYPE_MOBILE,subId,getTimesMonthmorning(),System.currentTimeMillis()); do{ summaryStats.getNextBucket(summaryBucket); intsummaryUid=summaryBucket.getUid(); if(uid==summaryUid){ summaryRx+=summaryBucket.getRxBytes(); Rx+=summaryRx; summaryTx+=summaryBucket.getTxBytes(); Tx+=summaryTx; } }while(summaryStats.hasNextBucket());