现在的位置: 首页 > 综合 > 正文

android流量监控类TrafficStats讲解

2013年09月02日 ⁄ 综合 ⁄ 共 2971字 ⁄ 字号 评论关闭

最近无聊看了关于android流量监控方面的知识,发觉android在这方面做得很是全面,google已经将对于流量监控类TrafficStats都封装好了,非常容易上手,对于学习android开发的人来说,无疑是极大的利器。

先此声明:以下的例子大部分来自于网络(就是我学流量监控的时候,在网上找的例子),但个人觉得,例子本身写得非常棒,所以自己稍作修改拿来摆上架~~~呵呵呵呵呵。

好了,废话不多说,转入正题。

在看代码前,先说明一下android怎样统计手机内的流量的使用情况的呢?其实,

Android是基于linux的一个操作系统。

在Android中,你用Root Explorer去查看系统文件的话,与流量监控相关的会有这么几个文件/proc/net/dev这个文件中具体记录的暂时不是非常清楚,可能是整个系统的一个流量情况。

/proc/uid_stat/%d" 和"/proc/uid_stat/%d" %d为进程的UID。这个文件里只有两项数据tcp_rcv和tcp_snd。

看命名大家应该就能看出代表什么,一个代表总的接受字节数,一个代表总的发送字节数。

这两个文件为非标准linux内核文件,由android内核层/kernel/net/Socket.c 的__sock_sendmsg函数负责写入, 用户层套接字通信在内核层最终会调用此函数 (包括本地套接字和网络套接字)。

而Android在2.3之前是没有封装响应的流量监控API的。在2.3之后呢,把数据流量监控封装到了Android.net.TrafficStats类中。其原理就是读取上文提到的那几处文件。其中有的方法也是读取的别的文件。

先让我们来看一下配置文件,main.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:orientation="vertical" >
    <Button 
        android:id="@+id/showList"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:text="显示当前程序流量"/>
</LinearLayout>

配置文件非常的简单,只是定义了一个Button组件,其id为showList

好了,再来看一下activity内的代码

public class TrafficStatsActivity extends Activity {
    /** Called when the activity is first created. */
	private Button showBtn=null;
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        showBtn=(Button) findViewById(R.id.showList);
        showBtn.setOnClickListener(new ButtonListener());
    }
    
    public void getAppTrafficList(){
    	//获取所有的安装在手机上的应用软件的信息,并且获取这些软件里面的权限信息
    	PackageManager pm=getPackageManager();//获取系统应用包管理
    	//获取每个包内的androidmanifest.xml信息,它的权限等等
    	List<PackageInfo> pinfos=pm.getInstalledPackages
    			(PackageManager.GET_UNINSTALLED_PACKAGES | PackageManager.GET_PERMISSIONS);
    	//遍历每个应用包信息
    	for(PackageInfo info:pinfos){
    		//请求每个程序包对应的androidManifest.xml里面的权限
    		String[] premissions=info.requestedPermissions;
    		if(premissions!=null && premissions.length>0){
    			//找出需要网络服务的应用程序
    			for(String premission : premissions){
    				if("android.permission.INTERNET".equals(premission)){
    					//获取每个应用程序在操作系统内的进程id
    					int uId=info.applicationInfo.uid;
    					//如果返回-1,代表不支持使用该方法,注意必须是2.2以上的
    					long rx=TrafficStats.getUidRxBytes(uId);
    					//如果返回-1,代表不支持使用该方法,注意必须是2.2以上的
    					long tx=TrafficStats.getUidTxBytes(uId);
    					if(rx<0 || tx<0){
    						continue;
    					}else{
    						Toast.makeText(this, info.applicationInfo.loadLabel(pm)+"消耗的流量--"+
    					Formatter.formatFileSize(this, rx+tx), Toast.LENGTH_SHORT);
    					}
    				}
    			}
    		}
    	}
    }
    private class ButtonListener implements OnClickListener{

		@Override
		public void onClick(View v) {
			// TODO Auto-generated method stub
			getAppTrafficList();
		}
    	
    }
}

这里要注意的是这三句代码:

//获取每个应用程序在操作系统内的进程id
int uId=info.applicationInfo.uid;
//如果返回-1,代表不支持使用该方法,注意必须是2.2以上的
long rx=TrafficStats.getUidRxBytes(uId);
//如果返回-1,代表不支持使用该方法,注意必须是2.2以上的
long tx=TrafficStats.getUidTxBytes(uId);

头一句代码主要的作用是获取声明到的网络权限的application的进程系统id,就类似与我们window下的进程PID。
而第二句的作用毫无疑问的就是根据该进程的id获取到改应用程序开机到现在从网络上接收到的字节(以KB为单位),最后一句则是根据该进程的id获取到改应用程序开机到现在向网络上发送的字节(以KB为单位)。好了其他的也就没什么了。一些代码我也子啊程序内注有注释了。最后说一句,TrafficStats这个类只能在android2.2以上的版本才能够使用,还有,测试的时候最好用真机来测试。因为模拟器的话,可能会造成rx和tx这两个参数总为-1.
^ _^

抱歉!评论已关闭.