此文章为网上诸多文章总结所得,主要借鉴别人的思想,并在其中添加了自己的一些思考,使实现更加明确。
在没有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中添加权限声明
<user-permission android:name="android.permission.READ_PHONE_STATE"/>
<user-permission android:name="android.permission.PACKAGE_USAGE_STATS" tools:ignore="ProtectedPermissions"/>
<user-permission android:name="android.permission.READ_NETWORK_USAGE_HISTORY" tools:ignore="ProtectedPermissions"/>
并在<mainfest xmlns:android="http://schemas.android.com/apk/res/android"后添加
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());