两种途径
使用到的类:
TrafficStats(设备启动以来流量的统计信息)
NetworkStatsManager(网络历史数据)
1、TrafficStats类
是读取Linux提供的文件对象系统类型的文本进行解析。
android.net.TrafficStats类中,提供了多种静态方法,可以直接调用获取,返回类型均为 long,如果返回等于-1代表 UNSUPPORTED ,当前设备不支持统计。
TrafficStats能够获取设备的数据流量和总的网络流量消耗(一般情况下也就得到Wi-Fi下的流量信息);
可以查询uid对应的流量信息,而uid可以通过应用的包名查询到,因此能够查询某个应用的流量统计信息(不考虑shareuid)。
非常方便的是,它的使用不需要特别的权限。另一方面它也一些限制:
(1)无法获取应用的数据流量消耗
从文档中仅能获取到指定uid的流量,但无法区分不同网络类型下的消耗
间接方法是通过监听网络切换,做好流量记录(但是要保证你的应用一直存活,且一定准确接收到网络切换信息),基本不可用。
(2)无法获取某个时间段内的流量消耗
从API文档中看,函数参数没有与时间相关的信息。而且重要的一点是,TrafficStats类中记录的是设备重启以来的流量统计信息。因为TrafficStats 类,底层是读取/proc/net/xt_qtaguid/stats 对内容进行解析,将得到的结果返回上层。
Android系统中封装了一套流量数据API,这些API可以很好的管理Android系统流量使用情况。我们可以基于这些Android API来实现管理手机流量的功能。
这些API很好的封装在了android.net包下的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的发送字节数 */
2、NetworkStatsManager
NetworkStatsManager类是在Android 6.0(API23)中新增加的类,提供网络使用历史统计信息,同时特别强调了可查询指定时间间隔内的统计信息。
NetworkStatsManager类克服了TrafficStats的查询限制,而且统计信息也不再是设备重启以来的数据。但它也有自己的限制和缺点。
(1)权限限制
NetworkStatsManager的使用需要额外的权限,”android.permission.PACKAGE_USAGE_STATS”是系统权限,需要引导用户开启应用的“有权查看使用情况的应用”(使用记录访问权限)权限
看看部分函数(非静态):
//查询指定网络类型在某时间间隔内的总的流量统计信息
NetworkStats.BucketquerySummaryForDevice(intnetworkType,StringsubscriberId,longstartTime,longendTime)
//查询某uid在指定网络类型和时间间隔内的流量统计信息
NetworkStats.queryDetailsForUid(intnetworkType,StringsubscriberId,longstartTime,longendTime,intuid)
//查询指定网络类型在某时间间隔内的详细的流量统计信息(包括每个uid)
NetworkStats.queryDetails(intnetworkType,StringsubscriberId,longstartTime,longendTime)
申请权限
<uses-permission android:name="android.permission.PACKAGE_USAGE_STATS" tools:ignore="ProtectedPermissions"/> <uses-permission android:name="android.permission.READ_PHONE_STATE" />
用户手动授权
private boolean hasPermissionToReadNetworkStats() { if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M) { return true; } final AppOpsManager appOps = (AppOpsManager) getSystemService(Context.APP_OPS_SERVICE); int mode = 0; if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.Q) { mode = appOps.unsafeCheckOpRaw(AppOpsManager.OPSTR_GET_USAGE_STATS, android.os.Process.myUid(), getPackageName()); } if (mode == AppOpsManager.MODE_ALLOWED) { return true; } requestReadNetworkStats(); return false; } // 打开“有权查看使用情况的应用”页面 private void requestReadNetworkStats() { Intent intent = new Intent(Settings.ACTION_USAGE_ACCESS_SETTINGS); startActivity(intent); }
统计使用情况
NetworkStatsManager networkStatsManager = (NetworkStatsManager) getSystemService(NETWORK_STATS_SERVICE); NetworkStats.Bucket bucket = null; // 获取到目前为止设备的Wi-Fi流量统计 bucket = networkStatsManager.querySummaryForDevice(ConnectivityManager.TYPE_WIFI, "", 0, System.currentTimeMillis());
遇到的问题:NetworkStats.Bucket bucket 为空
原因 :我收集的是以太网的使用情况,SDK没有支持这个功能,所以获取到的bucket为空,源码如下 :
public Bucket querySummaryForDevice(int networkType, String subscriberId, long startTime, long endTime) throws SecurityException, RemoteException { NetworkTemplate template; try { template = createTemplate(networkType, subscriberId); } catch (IllegalArgumentException e) { if (DBG) Log.e(TAG, "Cannot create template", e); return null; } return querySummaryForDevice(template, startTime, endTime); }
private static NetworkTemplate createTemplate(int networkType, String subscriberId) { final NetworkTemplate template; switch (networkType) { case ConnectivityManager.TYPE_MOBILE: template = subscriberId == null ? NetworkTemplate.buildTemplateMobileWildcard() : NetworkTemplate.buildTemplateMobileAll(subscriberId); break; case ConnectivityManager.TYPE_WIFI: template = NetworkTemplate.buildTemplateWifiWildcard(); break; default: throw new IllegalArgumentException("Cannot create template for network type " + networkType + ", subscriberId '" + NetworkIdentity.scrubSubscriberId(subscriberId) + "'."); } return template; }
SDK只支持对 WiFi和移动数据流量的统计
扩展方法:
frameworks\base\core\java\android\app\usage\NetworkStatsManager.java
增加case:
case ConnectivityManager.TYPE_ETHERNET: template = NetworkTemplate.buildTemplateEthernet(); break;
frameworks\base\core\java\android\net\NetworkTemplate.java
java 获取当天(今日)零点零分零秒
两种方法 一种得到的是时间戳,一种得到是日期格式:
1.日期格式的
Calendar calendar = Calendar.getInstance(); calendar.setTime(new Date()); calendar.set(Calendar.HOUR_OF_DAY, 0); calendar.set(Calendar.MINUTE, 0); calendar.set(Calendar.SECOND, 0); Date zero = calendar.getTime();
2.时间戳
long current = System.currentTimeMillis(); long zero = current/(1000*3600*24)*(1000*3600*24) - TimeZone.getDefault().getRawOffset(); public static void main(String[] args) { long current=System.currentTimeMillis();//当前时间毫秒数 long zero=current/(1000*3600*24)*(1000*3600*24)-TimeZone.getDefault().getRawOffset();//今天零点零分零秒的毫秒数 long twelve=zero+24*60*60*1000-1;//今天23点59分59秒的毫秒数 long yesterday=System.currentTimeMillis()-24*60*60*1000;//昨天这一时间的毫秒数 System.out.println(new Timestamp(current));//当前时间 System.out.println(new Timestamp(yesterday));//昨天这一时间点 System.out.println(new Timestamp(zero));//今天零点零分零秒 System.out.println(new Timestamp(twelve));//今天23点59分59秒 }