很多安全卫士类软件都实现了网速监测功能,也算是一个比较实用的功能。Android下,TrafficStats类实现了对流量的统计。

[java] view plain copy  

  1. static long getMobileRxBytes()//获取通过Mobile连接收到的字节总数,但不包含WiFi
  2. static long getMobileRxPackets()//获取Mobile连接收到的数据包总数
  3. static long getMobileTxBytes()//Mobile发送的总字节数
  4. static long getMobileTxPackets()//Mobile发送的总数据包数
  5. static long getTotalRxBytes()//获取总的接受字节数,包含Mobile和WiFi等
  6. static long getTotalRxPackets()//总的接受数据包数,包含Mobile和WiFi等
  7. static long getTotalTxBytes()//总的发送字节数,包含Mobile和WiFi等
  8. static long getTotalTxPackets()//发送的总数据包数,包含Mobile和WiFi等
  9. static long getUidRxBytes(int uid)//获取某个网络UID的接受字节数
  10. static long getUidTxBytes(intuid) //获取某个网络UID的发送字节数

这些就是TrafficStats提供的一些接口。那么,我们首先要创建一个桌面悬浮窗,用于显示网速。然后再Service里面,显示悬浮窗。

[java] view plain copy  

  1. import android.app.Service;
  2. import android.content.Intent;
  3. import android.graphics.PixelFormat;
  4. import android.os.Binder;
  5. import android.os.IBinder;
  6. import android.util.Log;
  7. import android.view.Gravity;
  8. import android.view.LayoutInflater;
  9. import android.view.MotionEvent;
  10. import android.view.View;
  11. import android.view.View.OnClickListener;
  12. import android.view.View.OnTouchListener;
  13. import android.view.WindowManager;
  14. import android.view.WindowManager.LayoutParams;
  15. import android.widget.LinearLayout;
  16. import android.widget.TextView;
  17. import com.yyh.utils.WidgetUtils;
  18. public class ManagerService extends Service {
  19. private static final String TAG = "ManagerService";
  20. public LinearLayout mFloatLayout;
  21. public WindowManager.LayoutParams wmParams;
  22. public WindowManager mWindowManager;
  23. public TextView mFloatView;
  24. private ServiceBinder binder = new ServiceBinder();
  25. @Override
  26. public void onCreate() {
  27. super.onCreate();
  28. createFloatView();
  29. }
  30. @Override
  31. public int onStartCommand(Intent intent, int flags, int startId) {
  32. return START_STICKY_COMPATIBILITY;
  33. }
  34. @Override
  35. public IBinder onBind(Intent intent) {
  36. return binder;
  37. }
  38. @Override
  39. public void onTrimMemory(int level) {
  40. Log.i("test", " onTrimMemory...");
  41. }
  42. /** 创建悬浮窗 */
  43. private void createFloatView() {
  44. wmParams = new WindowManager.LayoutParams();
  45. mWindowManager = (WindowManager) getApplication().getSystemService(getApplication().WINDOW_SERVICE);
  46. wmParams.type = LayoutParams.TYPE_SYSTEM_ALERT;// 设置window
  47. // type为TYPE_SYSTEM_ALERT
  48. wmParams.format = PixelFormat.RGBA_8888;// 设置图片格式,效果为背景透明
  49. wmParams.flags = LayoutParams.FLAG_NOT_FOCUSABLE;// 设置浮动窗口不可聚焦(实现操作除浮动窗口外的其他可见窗口的操作)
  50. wmParams.gravity = Gravity.LEFT | Gravity.TOP;// 默认位置:左上角
  51. wmParams.width = WidgetUtils.dpToPx(getApplicationContext(), 65);
  52. wmParams.height = WindowManager.LayoutParams.WRAP_CONTENT;
  53. wmParams.x = (WidgetUtils.getScreenWidth(getApplicationContext()) - wmParams.width) / 2;// 设置x、y初始值,相对于gravity
  54. wmParams.y = 10;
  55. // 获取浮动窗口视图所在布局
  56. LayoutInflater inflater = LayoutInflater.from(getApplication());
  57. mFloatLayout = (LinearLayout) inflater.inflate(R.layout.float_layout, null);
  58. mWindowManager.addView(mFloatLayout, wmParams);// 添加mFloatLayout
  59. mFloatView = (TextView) mFloatLayout.findViewById(R.id.speed);
  60. mFloatLayout.measure(View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED), View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED));
  61. // 设置监听浮动窗口的触摸移动
  62. mFloatView.setOnTouchListener(new OnTouchListener() {
  63. @Override
  64. public boolean onTouch(View v, MotionEvent event) {
  65. // getRawX是触摸位置相对于屏幕的坐标,getX是相对于按钮的坐标
  66. wmParams.x = (int) event.getRawX() - mFloatView.getMeasuredWidth() / 2;
  67. Log.i(TAG, "RawX" + event.getRawX());
  68. Log.i(TAG, "X" + event.getX());
  69. wmParams.y = (int) event.getRawY() - mFloatView.getMeasuredHeight() / 2 - 25;// 减25为状态栏的高度
  70. Log.i(TAG, "RawY" + event.getRawY());
  71. Log.i(TAG, "Y" + event.getY());
  72. mWindowManager.updateViewLayout(mFloatLayout, wmParams);// 刷新
  73. return false; // 此处必须返回false,否则OnClickListener获取不到监听
  74. }
  75. });
  76. mFloatView.setOnClickListener(new OnClickListener() {
  77. @Override
  78. public void onClick(View arg0) {
  79. // do something... 跳转到应用
  80. }
  81. });
  82. }
  83. public void setSpeed(String str) {
  84. mFloatView.setText(str.toString());
  85. }
  86. @Override
  87. public void onDestroy() {
  88. super.onDestroy();
  89. if (mFloatLayout != null && mWindowManager != null) {
  90. mWindowManager.removeView(mFloatLayout);// 移除悬浮窗口
  91. }
  92. startService(new Intent(this, ManagerService.class));
  93. }
  94. class ServiceBinder extends Binder {
  95. public ManagerService getService() {
  96. return ManagerService.this;
  97. }
  98. }
  99. }

然后构造一个网速监测的工具类,用于获取流量变化信息。

[java] view plain copy  

  1. import java.io.FileNotFoundException;
  2. import java.io.IOException;
  3. import java.io.RandomAccessFile;
  4. import java.math.BigDecimal;
  5. import java.util.Timer;
  6. import java.util.TimerTask;
  7. import android.content.Context;
  8. import android.content.pm.ApplicationInfo;
  9. import android.content.pm.PackageManager;
  10. import android.content.pm.PackageManager.NameNotFoundException;
  11. import android.net.TrafficStats;
  12. import android.os.Handler;
  13. import android.os.Message;
  14. import android.util.Log;
  15. /**
  16. * 应用的流量信息
  17. * @author yuyuhang
  18. */
  19. public class TrafficInfo {
  20. private static final int UNSUPPORTED = -1;
  21. private static final String LOG_TAG = "test";
  22. private static TrafficInfo instance;
  23. static int uid;
  24. private long preRxBytes = 0;
  25. private Timer mTimer = null;
  26. private Context mContext;
  27. private Handler mHandler;
  28. /** 更新频率(每几秒更新一次,至少1秒) */
  29. private final int UPDATE_FREQUENCY = 1;
  30. private int times = 1;
  31. public TrafficInfo(Context mContext, Handler mHandler, int uid) {
  32. this.mContext = mContext;
  33. this.mHandler = mHandler;
  34. this.uid = uid;
  35. }
  36. public TrafficInfo(Context mContext, Handler mHandler) {
  37. this.mContext = mContext;
  38. this.mHandler = mHandler;
  39. }
  40. public static TrafficInfo getInstant(Context mContext, Handler mHandler) {
  41. if (instance == null) {
  42. instance = new TrafficInfo(mContext, mHandler);
  43. }
  44. return instance;
  45. }
  46. /**
  47. * 获取总流量
  48. */
  49. public long getTrafficInfo() {
  50. long rcvTraffic = UNSUPPORTED; // 下载流量
  51. long sndTraffic = UNSUPPORTED; // 上传流量
  52. rcvTraffic = getRcvTraffic();
  53. sndTraffic = getSndTraffic();
  54. if (rcvTraffic == UNSUPPORTED || sndTraffic == UNSUPPORTED)
  55. return UNSUPPORTED;
  56. else
  57. return rcvTraffic + sndTraffic;
  58. }
  59. /**
  60. * 获取下载流量 某个应用的网络流量数据保存在系统的/proc/uid_stat/$UID/tcp_rcv | tcp_snd文件中
  61. */
  62. public long getRcvTraffic() {
  63. long rcvTraffic = UNSUPPORTED; // 下载流量
  64. rcvTraffic = TrafficStats.getUidRxBytes(uid);
  65. if (rcvTraffic == UNSUPPORTED) { // 不支持的查询
  66. return UNSUPPORTED;
  67. }
  68. Log.i("test", rcvTraffic + "--1");
  69. RandomAccessFile rafRcv = null, rafSnd = null; // 用于访问数据记录文件
  70. String rcvPath = "/proc/uid_stat/" + uid + "/tcp_rcv";
  71. try {
  72. rafRcv = new RandomAccessFile(rcvPath, "r");
  73. rcvTraffic = Long.parseLong(rafRcv.readLine()); // 读取流量统计
  74. catch (FileNotFoundException e) {
  75. Log.e(LOG_TAG, "FileNotFoundException: " + e.getMessage());
  76. rcvTraffic = UNSUPPORTED;
  77. catch (IOException e) {
  78. Log.e(LOG_TAG, "IOException: " + e.getMessage());
  79. e.printStackTrace();
  80. finally {
  81. try {
  82. if (rafRcv != null)
  83. rafRcv.close();
  84. if (rafSnd != null)
  85. rafSnd.close();
  86. catch (IOException e) {
  87. Log.w(LOG_TAG, "Close RandomAccessFile exception: " + e.getMessage());
  88. }
  89. }
  90. Log.i("test", rcvTraffic + "--2");
  91. return rcvTraffic;
  92. }
  93. /**
  94. * 获取上传流量
  95. */
  96. public long getSndTraffic() {
  97. long sndTraffic = UNSUPPORTED; // 上传流量
  98. sndTraffic = TrafficStats.getUidTxBytes(uid);
  99. if (sndTraffic == UNSUPPORTED) { // 不支持的查询
  100. return UNSUPPORTED;
  101. }
  102. RandomAccessFile rafRcv = null, rafSnd = null; // 用于访问数据记录文件
  103. String sndPath = "/proc/uid_stat/" + uid + "/tcp_snd";
  104. try {
  105. rafSnd = new RandomAccessFile(sndPath, "r");
  106. sndTraffic = Long.parseLong(rafSnd.readLine());
  107. catch (FileNotFoundException e) {
  108. Log.e(LOG_TAG, "FileNotFoundException: " + e.getMessage());
  109. sndTraffic = UNSUPPORTED;
  110. catch (IOException e) {
  111. Log.e(LOG_TAG, "IOException: " + e.getMessage());
  112. e.printStackTrace();
  113. finally {
  114. try {
  115. if (rafRcv != null)
  116. rafRcv.close();
  117. if (rafSnd != null)
  118. rafSnd.close();
  119. catch (IOException e) {
  120. Log.w(LOG_TAG, "Close RandomAccessFile exception: " + e.getMessage());
  121. }
  122. }
  123. return sndTraffic;
  124. }
  125. /**
  126. * 获取当前下载流量总和
  127. */
  128. public static long getNetworkRxBytes() {
  129. return TrafficStats.getTotalRxBytes();
  130. }
  131. /**
  132. * 获取当前上传流量总和
  133. */
  134. public static long getNetworkTxBytes() {
  135. return TrafficStats.getTotalTxBytes();
  136. }
  137. /**
  138. * 获取当前网速,小数点保留一位
  139. */
  140. public double getNetSpeed() {
  141. long curRxBytes = getNetworkRxBytes();
  142. if (preRxBytes == 0)
  143. preRxBytes = curRxBytes;
  144. long bytes = curRxBytes - preRxBytes;
  145. preRxBytes = curRxBytes;
  146. //int kb = (int) Math.floor(bytes / 1024 + 0.5);
  147. double kb = (double)bytes / (double)1024;
  148. BigDecimal bd = new BigDecimal(kb);
  149. return bd.setScale(1, BigDecimal.ROUND_HALF_UP).doubleValue();
  150. }
  151. /**
  152. * 开启流量监控
  153. */
  154. public void startCalculateNetSpeed() {
  155. preRxBytes = getNetworkRxBytes();
  156. if (mTimer != null) {
  157. mTimer.cancel();
  158. mTimer = null;
  159. }
  160. if (mTimer == null) {
  161. mTimer = new Timer();
  162. mTimer.schedule(new TimerTask() {
  163. @Override
  164. public void run() {
  165. if(times == UPDATE_FREQUENCY){
  166. Message msg = new Message();
  167. msg.what = 1;
  168. //msg.arg1 = getNetSpeed();
  169. msg.obj = getNetSpeed();
  170. mHandler.sendMessage(msg);
  171. times = 1;
  172. else {
  173. times++;
  174. }
  175. }
  176. }, 1000, 1000); // 每秒更新一次
  177. }
  178. }
  179. public void stopCalculateNetSpeed() {
  180. if (mTimer != null) {
  181. mTimer.cancel();
  182. mTimer = null;
  183. }
  184. }
  185. /**
  186. * 获取当前应用uid
  187. */
  188. public int getUid() {
  189. try {
  190. PackageManager pm = mContext.getPackageManager();
  191. ApplicationInfo ai = pm.getApplicationInfo(mContext.getPackageName(), PackageManager.GET_ACTIVITIES);
  192. return ai.uid;
  193. catch (NameNotFoundException e) {
  194. e.printStackTrace();
  195. }
  196. return -1;
  197. }
  198. }

然后再Activity里面,就需要去启动Service,以显示悬浮窗,然后TrafficInfo通过Handler,把当前网速发给Activity,Activity调用Service的方法了来更新悬浮窗的View。

[java] view plain copy  

  1. import java.util.List;
  2. import android.app.ActivityManager;
  3. import android.app.ActivityManager.RunningAppProcessInfo;
  4. import android.content.ComponentName;
  5. import android.content.Context;
  6. import android.content.Intent;
  7. import android.content.ServiceConnection;
  8. import android.os.Bundle;
  9. import android.os.Handler;
  10. import android.os.IBinder;
  11. import android.os.Message;
  12. import android.support.v7.app.ActionBarActivity;
  13. import android.util.Log;
  14. import android.widget.TextView;
  15. import com.yyh.utils.TrafficInfo;
  16. public class MainActivity extends ActionBarActivity {
  17. private ActivityManager mActivityManager;
  18. private TextView mTextView;
  19. Handler mHandler;
  20. TrafficInfo speed;
  21. ManagerService service;
  22. @Override
  23. protected void onCreate(Bundle savedInstanceState) {
  24. super.onCreate(savedInstanceState);
  25. setContentView(R.layout.activity_main);
  26. mActivityManager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
  27. mTextView = (TextView) findViewById(R.id.tv);
  28. try {
  29. mHandler = new Handler() {
  30. @Override
  31. public void handleMessage(Message msg) {
  32. if (msg.what == 1) {
  33. mTextView.setText(msg.obj + "kb/s");
  34. if(service != null)
  35. service.setSpeed(msg.obj+"kb/s"); // 设置网速
  36. }
  37. super.handleMessage(msg);
  38. }
  39. };
  40. speed = new TrafficInfo(this,mHandler,TrafficInfo.getUid());
  41. speed.startCalculateNetSpeed(); // 开启网速监测
  42. catch (Exception e) {
  43. e.printStackTrace();
  44. }
  45. Log.i("test","总流量="+speed.getTrafficInfo());
  46. Intent intent = new Intent(MainActivity.this, ManagerService.class);
  47. bindService(intent, conn, Context.BIND_AUTO_CREATE);
  48. }
  49. private ServiceConnection conn = new ServiceConnection() {
  50. public void onServiceConnected(ComponentName name, IBinder binder) {
  51. service = ((ManagerService.ServiceBinder) binder).getService();
  52. }
  53. public void onServiceDisconnected(ComponentName name) {
  54. service = null;
  55. }
  56. };
  57. @Override
  58. protected void onDestroy() {
  59. super.onDestroy();
  60. speed.stopCalculateNetSpeed();
  61. unbindService(conn);
  62. }
  63. }

这样,就能够显示悬浮窗。如果还需要后台运行,那Service就需要常驻不被杀死,这部分请参考我另一篇博文:Android 通过JNI实现守护进程,保证Service服务不被杀死

附上Demo源码:Android 流量与网速监测(悬浮窗) 源码

原文地址:http://blog.csdn.net/yyh352091626/article/details/50599621

Android实现流量统计和网速监控悬浮窗相关推荐

  1. Windows网络通信流量和网速监控设计(一)

    第二篇文章连接 文章目录 三种数据通信传输模式 一.单工通信 二.半双工通信 三.全双工通信 相关基础知识 1.__stdcall 2.GetIfTable()函数 3.MIB_IFTABLE结构体 ...

  2. android中检测网速,Android 流量与网速监测(悬浮窗) 源码

    [实例简介] 对应博文链接:http://blog.csdn.net/yyh352091626/article/details/50599621 Android 实现流量与网速监测(悬浮窗) 源码. ...

  3. 网络知识:内网、外网、宽带、带宽、流量、网速之间的联系?

    相信很多人都不知道内网.外网.宽带.带宽.流量.网速之间的区别与联系,今天小编给大家介绍一下,希望对大家能有所帮助! 一.带宽与宽带的区别是什么? 带宽是量词,指的是网速的大小,比如1Mbps的意思是 ...

  4. 物联网卡和流量卡网速对比,看看谁更强?

    物联网卡和流量卡网速对比,看看谁更强? 物联网卡是运营商面向物联网企业提供的一种智能设备数据接入业务,物联网卡作为智能设备信息传递的桥梁,来实现物与物.物与人的交互结合,可以说物联网卡也是一种流量卡, ...

  5. 了解内网、外网、宽带、带宽、流量、网速

    了解内网.外网.宽带.带宽.流量.网速. 一.带宽与宽带的区别是什么? 带宽是指在单位时间内(1秒)内能传输的数据量.网络和高速公路类似,带宽越大,就类似高速公路的车道越多,其通行能力越强.把城市的道 ...

  6. Android聊天室源码开发,悬浮窗的实现及封装思路

    为了实现语音聊天室源码的多场景应用,悬浮小窗口在语音聊天室源码开发中还是比较常见的需求,主要是指用户在语音连麦过程中打开其它界面或者退到桌面时语音连麦不会中断且会有一个悬浮小窗口位于最上层,接下来我们 ...

  7. EveryDay-Shell之ifconfig输出详解“——netspeed.sh网速监控脚本

    对应文件中,比如p1p1设备对应目录: $cd /sys/class/net/p1p1/statistics $ls -l -r--r--r-- 1 root root 4096  2月  4 14: ...

  8. 99%的人都不知道内网、外网、宽带、带宽、流量、网速之间的区别与联系

    大纲结构 一.带宽与宽带的区别是什么? 带宽是量词,指的是网速的大小,比如1Mbps的意思是一兆比特每秒,这个数值就是指带宽. 宽带是名词,说明网络的传输速率很高 .宽带的标准各不相同,最初认为128 ...

  9. 【计算机网络】了解内网、外网、宽带、带宽、流量、网速

    文章目录 一.带宽与宽带的区别是什么? 1.宽带 2.带宽 3.流量 二.带宽.网速和流量之间的关系 三.上行带宽和下行带宽是什么意思?各有什么作用? 四.服务器的上行和下行带宽理解 五.内网ip和外 ...

  10. 内网、外网、宽带、带宽、流量、网速之间的区别与联系

    目录 一.带宽与宽带的区别是什么? 二.带宽.网速和流量之间的关系 三.上行带宽和下行带宽是什么意思?各有什么作用? 四.服务器的上行和下行带宽理解 五.内网ip和外网ip区别 一.带宽与宽带的区别是 ...

最新文章

  1. 手把手教你如何做建模竞赛(baseline代码讲解)
  2. 【RecyclerView】 十二、RecyclerView 数据更新 ( 修改单条数据 | 批量修改数据 )
  3. Win10 UWP开发系列:解决Win10不同版本的Style差异导致的兼容性问题
  4. Java 集合系列17之 TreeSet详细介绍(源码解析)和使用示例
  5. linux ping策略打开_linux ping策略打开_如何在Linux服务器禁止和开启ping包 互联网技术圈 互联网技术圈......
  6. 后台服务系统之搭建ZooKeeper注册中心
  7. 如何理解VMware内存资源管理
  8. 杭电1262--寻找素数对(打表)
  9. 141. 环形链表 golang
  10. Struts2基础知识(二)
  11. 苹果推出App Store搜索建议功能
  12. Python使用K-means聚类算法进行分类案例一则
  13. RISC-V架构能否有效挑战ARM和英特尔?
  14. Django简介与基本命令
  15. 39. 确保判别式是纯函数
  16. 阶段3 1.Mybatis_07.Mybatis的连接池及事务_3 mybatis连接池的分类
  17. 如何在网页中播放FLV文件的代码
  18. 最全中华古诗词数据库,收录30多万诗词
  19. 全国计算机二级证书NO00000,计算机二级成绩查询
  20. Android Studio打开之后class显示灰色 代码没有颜色区分 输入没了提示的解决方法

热门文章

  1. uniapp 日期时间 计算
  2. Linux中修改HTTP默认主页
  3. Kruskal算法 数据结构 浙江大学 陈越、何钦铭
  4. QGIS编译---QGIS3.10.6 + Qt5.11.2 + VS2015 ---32位版本
  5. moss2007 是什么_MOSS 2007站点权限
  6. 如何关掉visual studio 2012 实时调试器
  7. C语言实现程序的暂停
  8. 牛客 NC24858 [USACO 2009 Nov S]Job Hunt
  9. 2019辽宁公务员考试行测常识大全:公务员常识40000问(五十九)(2)
  10. css mix-blend,CSS混合模式mix-blend-mode/background-blend-mode简介