Android项目实战--手机卫士34--流量管理
来源:互联网 发布:深圳西乡淘宝培训 编辑:程序博客网 时间:2024/04/28 16:22
最新实战教程,让你了解Android自动化刷量、作弊与防作弊的那些事,案例:刷友盟统计、批量注册苹果帐号
之前一直忙着找工作,所有都没有时间更新这个系列了,不过其实也差不多要完成的了,后面的一些都是一些比较零散的东西了
好了,不多说,现在就进入我们的正题,今天我们就讲一下那个流量管理的功能,其实流量管理并不难,Android里面已经有自己的api的了
但由于它里面有一些东西,我也弄不明白,所以今天这个只是一个简单的示例,大家回去之后想继续完善的话,就要自己花时间研究一下了
好,我们先来看一下是我们要实现的界面
我们要实现的界面就是这样的了,我们使用一个抽屉的控件
首先,我们先来建一个model类,用来存放我们要显示的信息
com.xiaobin.security.domain.TrafficInfo
package com.xiaobin.security.domain;import android.graphics.drawable.Drawable;public class TrafficInfo{private String name;private Drawable icon;private int uid;public String getName(){return name;}public void setName(String name){this.name = name;}public Drawable getIcon(){return icon;}public void setIcon(Drawable icon){this.icon = icon;}public int getUid(){return uid;}public void setUid(int uid){this.uid = uid;}}
上面的那个uid就是我们的应用在android里面的uid啦,它从一装进手机里面开始,就不会变化的啦
写完model类之后,我们就先来写一下界面吧,我们这次,用到了一个控件,叫抽屉(SlidingDrawer),下面我把我们的布局文件粘出来
traffic_manager.xml
<?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" > <LinearLayout android:layout_width="match_parent" android:layout_height="40dip" android:background="@drawable/title_background" android:gravity="center_vertical|center_horizontal" android:orientation="horizontal" > <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="@string/traffic_manager" android:textColor="@android:color/white" android:textSize="22sp" /> </LinearLayout> <LinearLayout android:layout_width="match_parent" android:layout_height="wrap_content" android:orientation="horizontal" > <TextView android:id="@+id/tv_traffic_2g_3g" android:layout_width="0dip" android:layout_height="wrap_content" android:layout_weight="1" android:paddingLeft="8dip" android:text="@string/traffic" /> <TextView android:id="@+id/tv_traffic_wifi" android:layout_width="0dip" android:layout_height="wrap_content" android:layout_weight="1" android:paddingLeft="8dip" android:text="@string/traffic" /> </LinearLayout> <SlidingDrawer android:layout_width="match_parent" android:layout_height="match_parent" android:content="@+id/lv_traffic_content" android:handle="@+id/iv_traffic_handle" android:orientation="vertical" > <ImageView android:id="@id/iv_traffic_handle" android:layout_width="wrap_content" android:layout_height="wrap_content" android:contentDescription="@string/hello_world" android:src="@drawable/handle" /> <ListView android:id="@id/lv_traffic_content" android:layout_width="match_parent" android:layout_height="match_parent" android:cacheColorHint="@android:color/white" /> </SlidingDrawer></LinearLayout>
在SlidingDrawer这个节点里面,大家要注意了,我们必须要给它指定一个handle和content的,不然,它是会报错的
好啦,节点写好之后,我们就可以写我们的逻辑啦
首先,我们就要先拿到我们总的流量啦
private void setTotalTraffic(){//拿到2G和3G的总共接收到的数据大小long total_2g_3g_received = TrafficStats.getMobileRxBytes();//拿到2G和3G的总共发送出去的数据大小long total_2g_3g_transmitted = TrafficStats.getMobileTxBytes();//拿到2G和3G的总数据大小long total_2g_3g = total_2g_3g_received + total_2g_3g_transmitted;tv_traffic_2g_3g.setText("2G/3g 总流量:" + TextFormater.dataSizeFormat(total_2g_3g));//拿到总共接收到的数据大小long total_received = TrafficStats.getTotalRxBytes();//拿到总共发送的数据大小long total_transmitted = TrafficStats.getTotalTxBytes();//拿到总数据大小long total = total_received + total_transmitted;////拿到wifi的总数据大小long total_wifi = total - total_2g_3g;tv_traffic_wifi.setText("wifi 总流量:" + TextFormater.dataSizeFormat(total_wifi));}
其实Android有一个类就是用来获取这些流量的信息的,TrafficStats
就这样,我们就可以拿到了wifi和2g/3g的总流量的了
注意,这个流量是会随着你每一次关机,都会设置为0的,也就是说,当你关机一次之后,这个值就是变成0的啦,所有要做一个流量管理的,就要把以前的也记录下来,然后再进行统计了
好啦,拿到了,总的流量之后,我们就要来拿到对应的一个个应用的流量了
首先,我们会先拿到一个会首先流量信息的应用列表先的
//拿到所有会产生流量的应用信息private void initResolveInfos(){trafficInfos.clear();//拿到一个包管理器PackageManager packageManager = this.getPackageManager();Intent intent = new Intent();//android.intent.action.MAIN这个action代表的就是应用的入口intent.setAction("android.intent.action.MAIN");//android.intent.category.LAUNCHER代表的就是在桌面创建一个图标intent.addCategory("android.intent.category.LAUNCHER");//这个方法就是根据对应的条件,intent指定条件,然后查询出相应的activity//那么根据我们上面设置的intent,我们就可以知道,我们要查询的是应用的入口activity而且是桌面上有图标的activity//因为这样的应用,才会有可能产生流量的List<ResolveInfo> resolveInfos = packageManager.queryIntentActivities(intent, 0);for(ResolveInfo resolveInfo : resolveInfos){//得到应用的名字String name = resolveInfo.loadLabel(packageManager).toString();//得到应用的图标Drawable icon = resolveInfo.loadIcon(packageManager);//得到应用的包名String packageName = resolveInfo.activityInfo.packageName;int uid = 0;;try{//得到应用的packageInfo对象PackageInfo packageInfo = packageManager.getPackageInfo(packageName, 0);//得到这个应用对应的uiduid = packageInfo.applicationInfo.uid;//根据uid得到这个应用的接收数据大小long received = TrafficStats.getUidRxBytes(uid);//根据uid得到这个应用的发送数据大小long transmitted = TrafficStats.getUidTxBytes(uid);//有些应用不会产生流量信息的,拿到的值就会是-1//不产生流量的,我们就不把它加入到list里面if(received == -1 && transmitted == -1){continue;}}catch (NameNotFoundException e){e.printStackTrace();}TrafficInfo trafficInfo = new TrafficInfo();trafficInfo.setName(name);trafficInfo.setIcon(icon);trafficInfo.setUid(uid);trafficInfos.add(trafficInfo);}}
上面这个方法的注释已经很详细了,我就不多说啦,有什么不明白的,可以提出来
有了这个应用的列表之后,不用想的了,我们肯定会写一个adapter的了
private class TrafficAdapter extends BaseAdapter{@Overridepublic int getCount(){return trafficInfos.size();}@Overridepublic Object getItem(int position){return trafficInfos.get(position);}@Overridepublic long getItemId(int position){return position;}@Overridepublic View getView(int position, View convertView, ViewGroup parent){View view;ViewHolder holder;TrafficInfo info = trafficInfos.get(position);if(convertView == null){view = View.inflate(TrafficManagerActivity.this, R.layout.traffic_manager_item, null);holder = new ViewHolder();holder.iv_traffic_icon = (ImageView) view.findViewById(R.id.iv_traffic_icon);holder.tv_traffic_name = (TextView) view.findViewById(R.id.tv_traffic_name);holder.tv_traffic_received = (TextView) view.findViewById(R.id.tv_traffic_received);holder.tv_traffic_transmitted = (TextView) view.findViewById(R.id.tv_traffic_transmitted);view.setTag(holder);}else{view = convertView;holder = (ViewHolder) view.getTag();}holder.iv_traffic_icon.setImageDrawable(info.getIcon());holder.tv_traffic_name.setText(info.getName());//根据uid得到这个应用的接收数据大小long received = TrafficStats.getUidRxBytes(info.getUid());//根据uid得到这个应用的发送数据大小long transmitted = TrafficStats.getUidTxBytes(info.getUid());holder.tv_traffic_received.setText(TextFormater.dataSizeFormat(received));holder.tv_traffic_transmitted.setText(TextFormater.dataSizeFormat(transmitted));return view;}}private class ViewHolder{ImageView iv_traffic_icon;TextView tv_traffic_name;TextView tv_traffic_received;TextView tv_traffic_transmitted;}
好啦,写到这里,我们的流量获取以及显示的操作,就基本上是完成的啦,但是还是有一点小问题要处理的,我们的流量是会变化的,但是我们这里根本就没有刷新机制,
所以,接下来,我们就用一个计时器来进行刷新的
@Overrideprotected void onStart(){timer = new Timer();timerTask = new TimerTask(){@Overridepublic void run(){Message msg = Message.obtain();handler.sendMessage(msg);}};timer.schedule(timerTask, 1000, 3000);super.onStart();}
@Overrideprotected void onStop(){if(timer != null){timer.cancel();timer = null;timerTask = null;}super.onStop();}
我在onStart和onStop方法里面,进行了对一个计时器的初始化和取消,这样子,我们的流量列表就可以刷新的了,通过handler来刷新
@SuppressLint("HandlerLeak")private Handler handler = new Handler(){public void handleMessage(Message msg) {adapter.notifyDataSetChanged();}};
好啦,这样子,流量管理的功能就读完的了,下面把完整的activity粘出来
com.xiaobin.security.ui.TrafficManagerActivity
package com.xiaobin.security.ui;import java.util.ArrayList;import java.util.List;import java.util.Timer;import java.util.TimerTask;import android.annotation.SuppressLint;import android.app.Activity;import android.content.Intent;import android.content.pm.PackageInfo;import android.content.pm.PackageManager;import android.content.pm.PackageManager.NameNotFoundException;import android.content.pm.ResolveInfo;import android.graphics.drawable.Drawable;import android.net.TrafficStats;import android.os.Bundle;import android.os.Handler;import android.os.Message;import android.view.View;import android.view.ViewGroup;import android.view.Window;import android.widget.BaseAdapter;import android.widget.ImageView;import android.widget.ListView;import android.widget.TextView;import com.xiaobin.security.R;import com.xiaobin.security.domain.TrafficInfo;import com.xiaobin.security.utils.TextFormater;public class TrafficManagerActivity extends Activity{private TextView tv_traffic_2g_3g;private TextView tv_traffic_wifi;private ListView lv_traffic_content;private TrafficAdapter adapter;private List<TrafficInfo> trafficInfos;private Timer timer;private TimerTask timerTask;@SuppressLint("HandlerLeak")private Handler handler = new Handler(){public void handleMessage(Message msg) {adapter.notifyDataSetChanged();}};@Overrideprotected void onCreate(Bundle savedInstanceState){super.onCreate(savedInstanceState);requestWindowFeature(Window.FEATURE_NO_TITLE);setContentView(R.layout.traffic_manager);tv_traffic_2g_3g = (TextView) findViewById(R.id.tv_traffic_2g_3g);tv_traffic_wifi = (TextView) findViewById(R.id.tv_traffic_wifi);setTotalTraffic();trafficInfos = new ArrayList<TrafficInfo>();initResolveInfos();lv_traffic_content = (ListView) findViewById(R.id.lv_traffic_content);adapter = new TrafficAdapter();lv_traffic_content.setAdapter(adapter);}@Overrideprotected void onStart(){timer = new Timer();timerTask = new TimerTask(){@Overridepublic void run(){Message msg = Message.obtain();handler.sendMessage(msg);}};timer.schedule(timerTask, 1000, 3000);super.onStart();}@Overrideprotected void onStop(){if(timer != null){timer.cancel();timer = null;timerTask = null;}super.onStop();}private void setTotalTraffic(){//拿到2G和3G的总共接收到的数据大小long total_2g_3g_received = TrafficStats.getMobileRxBytes();//拿到2G和3G的总共发送出去的数据大小long total_2g_3g_transmitted = TrafficStats.getMobileTxBytes();//拿到2G和3G的总数据大小long total_2g_3g = total_2g_3g_received + total_2g_3g_transmitted;tv_traffic_2g_3g.setText("2G/3g 总流量:" + TextFormater.dataSizeFormat(total_2g_3g));//拿到总共接收到的数据大小long total_received = TrafficStats.getTotalRxBytes();//拿到总共发送的数据大小long total_transmitted = TrafficStats.getTotalTxBytes();//拿到总数据大小long total = total_received + total_transmitted;////拿到wifi的总数据大小long total_wifi = total - total_2g_3g;tv_traffic_wifi.setText("wifi 总流量:" + TextFormater.dataSizeFormat(total_wifi));}//拿到所有会产生流量的应用信息private void initResolveInfos(){trafficInfos.clear();//拿到一个包管理器PackageManager packageManager = this.getPackageManager();Intent intent = new Intent();//android.intent.action.MAIN这个action代表的就是应用的入口intent.setAction("android.intent.action.MAIN");//android.intent.category.LAUNCHER代表的就是在桌面创建一个图标intent.addCategory("android.intent.category.LAUNCHER");//这个方法就是根据对应的条件,intent指定条件,然后查询出相应的activity//那么根据我们上面设置的intent,我们就可以知道,我们要查询的是应用的入口activity而且是桌面上有图标的activity//因为这样的应用,才会有可能产生流量的List<ResolveInfo> resolveInfos = packageManager.queryIntentActivities(intent, 0);for(ResolveInfo resolveInfo : resolveInfos){//得到应用的名字String name = resolveInfo.loadLabel(packageManager).toString();//得到应用的图标Drawable icon = resolveInfo.loadIcon(packageManager);//得到应用的包名String packageName = resolveInfo.activityInfo.packageName;int uid = 0;;try{//得到应用的packageInfo对象PackageInfo packageInfo = packageManager.getPackageInfo(packageName, 0);//得到这个应用对应的uiduid = packageInfo.applicationInfo.uid;//根据uid得到这个应用的接收数据大小long received = TrafficStats.getUidRxBytes(uid);//根据uid得到这个应用的发送数据大小long transmitted = TrafficStats.getUidTxBytes(uid);//有些应用不会产生流量信息的,拿到的值就会是-1//不产生流量的,我们就不把它加入到list里面if(received == -1 && transmitted == -1){continue;}}catch (NameNotFoundException e){e.printStackTrace();}TrafficInfo trafficInfo = new TrafficInfo();trafficInfo.setName(name);trafficInfo.setIcon(icon);trafficInfo.setUid(uid);trafficInfos.add(trafficInfo);}}//============================================================================================private class TrafficAdapter extends BaseAdapter{@Overridepublic int getCount(){return trafficInfos.size();}@Overridepublic Object getItem(int position){return trafficInfos.get(position);}@Overridepublic long getItemId(int position){return position;}@Overridepublic View getView(int position, View convertView, ViewGroup parent){View view;ViewHolder holder;TrafficInfo info = trafficInfos.get(position);if(convertView == null){view = View.inflate(TrafficManagerActivity.this, R.layout.traffic_manager_item, null);holder = new ViewHolder();holder.iv_traffic_icon = (ImageView) view.findViewById(R.id.iv_traffic_icon);holder.tv_traffic_name = (TextView) view.findViewById(R.id.tv_traffic_name);holder.tv_traffic_received = (TextView) view.findViewById(R.id.tv_traffic_received);holder.tv_traffic_transmitted = (TextView) view.findViewById(R.id.tv_traffic_transmitted);view.setTag(holder);}else{view = convertView;holder = (ViewHolder) view.getTag();}holder.iv_traffic_icon.setImageDrawable(info.getIcon());holder.tv_traffic_name.setText(info.getName());//根据uid得到这个应用的接收数据大小long received = TrafficStats.getUidRxBytes(info.getUid());//根据uid得到这个应用的发送数据大小long transmitted = TrafficStats.getUidTxBytes(info.getUid());holder.tv_traffic_received.setText(TextFormater.dataSizeFormat(received));holder.tv_traffic_transmitted.setText(TextFormater.dataSizeFormat(transmitted));return view;}}private class ViewHolder{ImageView iv_traffic_icon;TextView tv_traffic_name;TextView tv_traffic_received;TextView tv_traffic_transmitted;}}
好啦,今天就到这里,如果有什么不明白的,可以留言
注意,用模拟器来测试这个流量管理是会有点问题的,模拟器不支持一些功能的,所以最好用真机来测试
最后,和大家说一下
为了方便大家的交流,我创建了一个群,这样子大家有什么疑问也可以在群上交流
群号是298440981
今天源码下载
- Android项目实战--手机卫士34--流量管理
- Android项目实战--手机卫士
- Android项目实战--手机卫士--结束
- Android实战--手机卫士
- Android项目实战--手机卫士08--获取手机联系人
- android项目-手机卫士
- Android项目实战--手机卫士01--启动界面
- Android项目实战--手机卫士02--与服务器交互
- Android项目实战--手机卫士04--自定义图片
- Android项目实战--手机卫士07--设置向导
- Android项目实战--手机卫士19--短信的恢复
- Android项目实战--手机卫士23--程序锁界面
- Android项目实战--手机卫士29--杀死后台进程
- Android项目实战--手机卫士01--启动界面 -
- Android项目实战--手机卫士33--ExpandableListView的使用
- Android项目实战--手机卫士35--清除程序缓存
- Android项目实战--手机卫士开发系列教程
- Android项目实战--手机卫士01--启动界面
- 算法导论 第10章 基本数据结构
- 站长而不可以点击自己网站中的广告?
- 利用DB2进行分页
- <c:forEach>
- 有“接”有“纳” 户籍制度改革瞄准破除城乡壁垒
- Android项目实战--手机卫士34--流量管理
- 求和最大的子数组
- 预处理命令 # 和 ##
- VC屏幕截图,保存为Bmp文件
- 网线回路的制作方法
- STL Bitsets ---应用
- java注解解析
- v8 hack ---给js添加新的关键字
- 内存设备描述表