原文链接:http://blog.csdn.net/qq_30379689/article/details/53242953

在你的手机更多设置或者高级设置中,我们会发现有个无障碍的功能,很多人不知道这个功能具体是干嘛的,其实这个功能是为了增强用户界面以帮助残障人士,或者可能暂时无法与设备充分交互的人们

它的具体实现是通过AccessibilityService服务运行在后台中,通过AccessibilityEvent接收指定事件的回调。这样的事件表示用户在界面中的一些状态转换,例如:焦点改变了,一个按钮被点击,等等。这样的服务可以选择请求活动窗口的内容的能力。简单的说AccessibilityService就是一个后台监控服务,当你监控的内容发生改变时,就会调用后台服务的回调方法

AccessibilityService使用

1.1 创建服务类

编写自己的Service类,重写onServiceConnected()方法、onAccessibilityEvent()方法和onInterrupt()方法

public class QHBAccessibilityService extends AccessibilityService {/*** 当启动服务的时候就会被调用*/@Overrideprotected void onServiceConnected() {super.onServiceConnected();}/*** 监听窗口变化的回调*/@Overridepublic void onAccessibilityEvent(AccessibilityEvent event) {int eventType = event.getEventType();//根据事件回调类型进行处理}/*** 中断服务的回调*/@Overridepublic void onInterrupt() {}
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27

下面是对AccessibilityService中常用的方法的介绍

  • disableSelf():禁用当前服务,也就是在服务可以通过该方法停止运行
  • findFoucs(int falg):查找拥有特定焦点类型的控件
  • getRootInActiveWindow():如果配置能够获取窗口内容,则会返回当前活动窗口的根结点
  • getSeviceInfo():获取当前服务的配置信息
  • onAccessibilityEvent(AccessibilityEvent event):有关AccessibilityEvent事件的回调函数,系统通过sendAccessibiliyEvent()不断的发送AccessibilityEvent到此处
  • performGlobalAction(int action):执行全局操作,比如返回,回到主页,打开最近等操作
  • setServiceInfo(AccessibilityServiceInfo info):设置当前服务的配置信息
  • getSystemService(String name):获取系统服务
  • onKeyEvent(KeyEvent event):如果允许服务监听按键操作,该方法是按键事件的回调,需要注意,这个过程发生了系统处理按键事件之前
  • onServiceConnected():系统成功绑定该服务时被触发,也就是当你在设置中开启相应的服务,系统成功的绑定了该服务时会触发,通常我们可以在这里做一些初始化操作
  • onInterrupt():服务中断时的回调

1.2 声明服务

既然是个后台服务,那么就需要我们在manifests中配置该服务信息

<service
    android:name=".AccessibilityService.QHBAccessibilityService"android:enabled="true"android:exported="true"android:label="@string/label"android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE"><intent-filter><action android:name="android.accessibilityservice.AccessibilityService" /></intent-filter>
</service>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10

我们必须注意:任何一个信息配置错误,都会使该服务无反应

  • android:label:在无障碍列表中显示该服务的名字 
  • android:permission:需要指定BIND_ACCESSIBILITY_SERVICE权限,这是4.0以上的系统要求的
  • intent-filter:这个name是固定不变的

1.3 配置服务参数

配置服务参数是指:配置用来接受指定类型的事件,监听指定package,检索窗口内容,获取事件类型的时间等等。其配置服务参数有两种方法:

1.3.1 方法一

在原先的manifests中增加meta-data标签指定xml文件

<service
    android:name=".AccessibilityService.QHBAccessibilityService"android:enabled="true"android:exported="true"android:label="@string/label"android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE"><intent-filter><action android:name="android.accessibilityservice.AccessibilityService" /></intent-filter><meta-data
        android:name="android.accessibilityservice"android:resource="@xml/accessibility_service_config" />
</service>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14

接下来是accessibility_service_config文件的配置

<?xml version="1.0" encoding="utf-8"?>
<accessibility-service xmlns:android="http://schemas.android.com/apk/res/android"android:accessibilityEventTypes="typeNotificationStateChanged|typeWindowStateChanged|typeWindowContentChanged|typeWindowsChanged"android:accessibilityFeedbackType="feedbackGeneric"android:accessibilityFlags="flagDefault"android:canRetrieveWindowContent="true"android:description="@string/description"android:notificationTimeout="100"android:packageNames="com.tencent.mm" />
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9

下面是对xml参数的介绍

  • accessibilityEventTypes:表示该服务对界面中的哪些变化感兴趣,即哪些事件通知,比如窗口打开,滑动,焦点变化,长按等。具体的值可以在AccessibilityEvent类中查到,如typeAllMask表示接受所有的事件通知
  • accessibilityFeedbackType:表示反馈方式,比如是语音播放,还是震动
  • canRetrieveWindowContent:表示该服务能否访问活动窗口中的内容。也就是如果你希望在服务中获取窗体内容,则需要设置其值为true
  • description:对该无障碍功能的描述,具体体现在下图 
  • notificationTimeout:接受事件的时间间隔,通常将其设置为100即可
  • packageNames:表示对该服务是用来监听哪个包的产生的事件,这里以微信的包名为例

1.3.2 方法二

通过代码为我们的AccessibilityService配置AccessibilityServiceInfo信息,这里我们可以抽取成一个方法进行设置

private void settingAccessibilityInfo() {String[] packageNames = {"com.tencent.mm"};AccessibilityServiceInfo mAccessibilityServiceInfo = new AccessibilityServiceInfo();// 响应事件的类型,这里是全部的响应事件(长按,单击,滑动等)mAccessibilityServiceInfo.eventTypes = AccessibilityEvent.TYPES_ALL_MASK;// 反馈给用户的类型,这里是语音提示mAccessibilityServiceInfo.feedbackType = AccessibilityServiceInfo.FEEDBACK_SPOKEN;// 过滤的包名mAccessibilityServiceInfo.packageNames = packageNames;setServiceInfo(mAccessibilityServiceInfo);
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11

在这里涉及到了AccessibilityServiceInfo类,AccessibilityServiceInfo类被用于配置AccessibilityService信息,该类中包含了大量用于配置的常量字段及用来xml属性,常见的有:accessibilityEventTypes,canRequestFilterKeyEvents,packageNames等等

1.4 启动服务

这里我们需要在无障碍功能里面手动打开该项功能,否则无法继续进行,通过下面代码可以打开系统的无障碍功能列表。此处若不理解,也可以先跳过或先下载下来代码对照理解。

Intent intent = new Intent(Settings.ACTION_ACCESSIBILITY_SETTINGS);
startActivity(intent);
  • 1
  • 2
  • 1
  • 2

1.5 处理事件信息

由于我们监听了事件的通知栏和界面等信息,当我们指定packageNames的通知栏或者界面发生变化时,会通过onAccessibilityEvent回调我们的事件,接着进行事件的处理

@Override
public void onAccessibilityEvent(AccessibilityEvent event) {int eventType = event.getEventType();//根据事件回调类型进行处理switch (eventType) {//当通知栏发生改变时case AccessibilityEvent.TYPE_NOTIFICATION_STATE_CHANGED:break;//当窗口的状态发生改变时case AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED:break;}
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15

当我们微信收到通知时,状态栏会有一条推送信息到达,这个时候就会被TYPE_NOTIFICATION_STATE_CHANGED监听,执行里面的内容,当我们切换微信界面时,或者使用微信时,这个时候就会被TYPE_WINDOW_STATE_CHANGED监听,执行里面的内容

AccessibilityEvent的方法

  • getEventType():事件类型
  • getSource():获取事件源对应的结点信息
  • getClassName():获取事件源对应类的类型,比如点击事件是有某个Button产生的,那么此时获取的就是Button的完整类名
  • getText():获取事件源的文本信息,比如事件是有TextView发出的,此时获取的就是TextView的text属性。如果该事件源是树结构,那么此时获取的是这个树上所有具有text属性的值的集合
  • isEnabled():事件源(对应的界面控件)是否处在可用状态
  • getItemCount():如果事件源是树结构,将返回该树根节点下子节点的数量

1.6 获取节点信息

获取了界面窗口变化后,这个时候就要获取控件的节点。整个窗口的节点本质是个树结构,通过以下操作节点信息

1.6.1 获取窗口节点(根节点)

AccessibilityNodeInfo nodeInfo = getRootInActiveWindow();
  • 1
  • 1

1.6.2 获取指定子节点(控件节点)

//通过文本找到对应的节点集合
List<AccessibilityNodeInfo> list = nodeInfo.findAccessibilityNodeInfosByText(text);
//通过控件ID找到对应的节点集合,如com.tencent.mm:id/gd
List<AccessibilityNodeInfo> list = nodeInfo.findAccessibilityNodeInfosByViewId(clickId);
  • 1
  • 2
  • 3
  • 4
  • 1
  • 2
  • 3
  • 4

1.7 模拟节点点击

当我们获取了节点信息之后,对控件节点进行模拟点击、长按等操作,AccessibilityNodeInfo类提供了performAction()方法让我们执行模拟操作,具体操作可看官方文档介绍,这里列举常用的操作

//模拟点击
accessibilityNodeInfo.performAction(AccessibilityNodeInfo.ACTION_CLICK);
//模拟长按
accessibilityNodeInfo.performAction(AccessibilityNodeInfo.ACTION_LONG_CLICK);
//模拟获取焦点
accessibilityNodeInfo.performAction(AccessibilityNodeInfo.ACTION_FOCUS);
//模拟粘贴
accessibilityNodeInfo.performAction(AccessibilityNodeInfo.ACTION_PASTE);
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8

抢红包插件实现

2.1 原理分析

  1. 收到微信红包的推送信息,在推送信息中判断是否出现”[微信红包]”的消息提示,如果出现则点击进入聊天界面
  2. 通过遍历窗口树节点,发现带有”领取红包”字样的节点,则点击进入,即红包,弹出抢红包界面
  3. 在抢红包界面,通过ID获取”开”按钮的节点,则打开红包
  4. 在红包详情页面,通过ID获取返回键按钮的节点,点击并返回微信聊天界面

2.2 注意事项

  1. 由于微信每个版本的按钮ID都是不一样的,在我们的程序中是需要去修改按钮ID,以达到版本的适配
  2. 在获取控件ID的时候,注意其布局是否可点击,否则获取不可点击的控件,会使程序无反应

2.3 获取控件ID

当我们手机接入USB线时,在Android Device Monitor中的选择设备并开启Dump View Hierarchy for UI Automator工具,通过它可以获取控件信息

http://www.cnblogs.com/xdindex/p/4619675.html

获取”开”按钮ID和返回按钮ID

2.4 代码实现

注意:这里使用的是微信最新6.3.30版本的控件ID,如果是其他版本的请自行适配

public class QHBAccessibilityService extends AccessibilityService {private List<AccessibilityNodeInfo> parents;/*** 当启动服务的时候就会被调用*/@Overrideprotected void onServiceConnected() {super.onServiceConnected();parents = new ArrayList<>();}/*** 监听窗口变化的回调*/@Overridepublic void onAccessibilityEvent(AccessibilityEvent event) {int eventType = event.getEventType();switch (eventType) {//当通知栏发生改变时case AccessibilityEvent.TYPE_NOTIFICATION_STATE_CHANGED:List<CharSequence> texts = event.getText();if (!texts.isEmpty()) {for (CharSequence text : texts) {String content = text.toString();if (content.contains("[微信红包]")) {//模拟打开通知栏消息,即打开微信if (event.getParcelableData() != null &&event.getParcelableData() instanceof Notification) {Notification notification = (Notification) event.getParcelableData();PendingIntent pendingIntent = notification.contentIntent;try {pendingIntent.send();Log.e("demo","进入微信");} catch (Exception e) {e.printStackTrace();}}}}}break;//当窗口的状态发生改变时case AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED:String className = event.getClassName().toString();if (className.equals("com.tencent.mm.ui.LauncherUI")) {//点击最后一个红包Log.e("demo","点击红包");getLastPacket();} else if (className.equals("com.tencent.mm.plugin.luckymoney.ui.LuckyMoneyReceiveUI")) {//开红包Log.e("demo","开红包");inputClick("com.tencent.mm:id/bg7");} else if (className.equals("com.tencent.mm.plugin.luckymoney.ui.LuckyMoneyDetailUI")) {//退出红包Log.e("demo","退出红包");inputClick("com.tencent.mm:id/gd");}break;}}/*** 通过ID获取控件,并进行模拟点击* @param clickId*/@TargetApi(Build.VERSION_CODES.JELLY_BEAN_MR2)private void inputClick(String clickId) {AccessibilityNodeInfo nodeInfo = getRootInActiveWindow();if (nodeInfo != null) {List<AccessibilityNodeInfo> list = nodeInfo.findAccessibilityNodeInfosByViewId(clickId);for (AccessibilityNodeInfo item : list) {item.performAction(AccessibilityNodeInfo.ACTION_CLICK);}}}/*** 获取List中最后一个红包,并进行模拟点击*/private void getLastPacket() {AccessibilityNodeInfo rootNode = getRootInActiveWindow();recycle(rootNode);if(parents.size()>0){parents.get(parents.size() - 1).performAction(AccessibilityNodeInfo.ACTION_CLICK);}}/*** 回归函数遍历每一个节点,并将含有"领取红包"存进List中** @param info*/public void recycle(AccessibilityNodeInfo info) {if (info.getChildCount() == 0) {if (info.getText() != null) {if ("领取红包".equals(info.getText().toString())) {if (info.isClickable()) {info.performAction(AccessibilityNodeInfo.ACTION_CLICK);}AccessibilityNodeInfo parent = info.getParent();while (parent != null) {if (parent.isClickable()) {parents.add(parent);break;}parent = parent.getParent();}}}} else {for (int i = 0; i < info.getChildCount(); i++) {if (info.getChild(i) != null) {recycle(info.getChild(i));}}}}/*** 中断服务的回调*/@Overridepublic void onInterrupt() {}
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
  • 79
  • 80
  • 81
  • 82
  • 83
  • 84
  • 85
  • 86
  • 87
  • 88
  • 89
  • 90
  • 91
  • 92
  • 93
  • 94
  • 95
  • 96
  • 97
  • 98
  • 99
  • 100
  • 101
  • 102
  • 103
  • 104
  • 105
  • 106
  • 107
  • 108
  • 109
  • 110
  • 111
  • 112
  • 113
  • 114
  • 115
  • 116
  • 117
  • 118
  • 119
  • 120
  • 121
  • 122
  • 123
  • 124
  • 125
  • 126
  • 127
  • 128
  • 129
  • 130
  • 131
  • 132
  • 133
  • 134
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
  • 79
  • 80
  • 81
  • 82
  • 83
  • 84
  • 85
  • 86
  • 87
  • 88
  • 89
  • 90
  • 91
  • 92
  • 93
  • 94
  • 95
  • 96
  • 97
  • 98
  • 99
  • 100
  • 101
  • 102
  • 103
  • 104
  • 105
  • 106
  • 107
  • 108
  • 109
  • 110
  • 111
  • 112
  • 113
  • 114
  • 115
  • 116
  • 117
  • 118
  • 119
  • 120
  • 121
  • 122
  • 123
  • 124
  • 125
  • 126
  • 127
  • 128
  • 129
  • 130
  • 131
  • 132
  • 133
  • 134

当收到红包发送的时候,Log的打印信息

11-21 13:53:06.275 2909-2909/com.handsome.boke2 E/demo: 进入微信
11-21 13:53:06.921 2909-2909/com.handsome.boke2 E/demo: 点击红包
11-21 13:53:07.883 2909-2909/com.handsome.boke2 E/demo: 开红包
11-21 13:53:08.732 2909-2909/com.handsome.boke2 E/demo: 退出红包
  • 1
  • 2
  • 3
  • 4
  • 1
  • 2
  • 3
  • 4

你可能会想到做一些窃取信息的软件,比如获取QQ密码、支付宝密码等等,哈哈,凡是EditText中设置inputType为password类型的,都无法获取其输入值

2.5 源码下载

学习AccessibilityService实现微信抢红包插件相关推荐

  1. android黑科技系列——微信抢红包插件原理解析和开发实现

    一.前言 自从几年前微信添加抢红包的功能,微信的电商之旅算是正式开始正式火爆起来.但是作为Android开发者来说,我们在抢红包的同时意识到了很多问题,就是手动去抢红包的速度慢了,当然这些有很多原因导 ...

  2. Android微信抢红包插件原理和实现 适配微信6.6.1版本

    Android微信抢红包插件原理和实现 适配微信6.6.1版本 一.前言 年关将至,小伙伴们又聊起了春节长假的旅行安排和各自家乡的年味习俗,不一而足.其中在各个微信群聊中抢红包也是为大家所津津乐道的. ...

  3. AccessibilityService之微信抢红包辅助功能实现

    这次我们来使用AccessibilityService实现一个微信抢红包的辅助工具. 先来一个简单的效果图,最后会有一个完整的效果图: 首先,先回想一下微信红包的领取流程. 1.通知栏出现微信红包的消 ...

  4. Android 微信抢红包插件

    转载开源工程:https://github.com/geeeeeeeeek/WeChatLuckyMoney 这个Android插件可以帮助你在微信群聊抢红包时战无不胜.当检测到红包时,插件会自动点击 ...

  5. Android微信抢红包插件源码解析

    这个Android插件可以帮助你在微信群聊抢红包时战无不胜.当检测到红包时,插件会自动点击屏幕,人工点击的速度无法比拟. 你正在查看的是dev分支,这个分支包含大量实验性的修改,不再更新.如果你希望有 ...

  6. Android 辅助功能(无障碍)自定义开发类似微信抢红包插件

    背景: 通常在无人值守的Android智能设备或不能触摸交互的设备中,当系统运行异常时, 系统有可能会弹出应用程序无响应(ANR)对话框,提示是否终止当前进程或等待,或应用程序异常退出时系统偶尔也会有 ...

  7. 微信抢红包插件 android 8.0,重磅!微信8.0专属红包全量上线!

    "本文由我原创,来源 

  8. 【安卓学习之微信抢红包】 微信抢红包 1 - 知识点归纳

    █ [安卓学习之微信抢红包] 微信抢红包 1 - 知识点归纳 █ 相关文章: ● [安卓学习之微信抢红包] 微信抢红包 1 - 知识点归纳 ● [安卓学习之微信抢红包] 微信抢红包 2 - 通知服务( ...

  9. APP逆向分析之钉钉抢红包插件的实现-iOS篇

    花费了很多天的原创文章,转载请注明出处https://yohunl.com/ding-ding-qiang-hong-bao-cha-jian-iospian/ ,谢谢! 网络上关于微信红包的分析文章 ...

  10. android微信的抢红包插件

    前言   之前看到了微信抢红包的插件,觉得这个功能实在强大了,这和之前我想实现的模拟点击事件基本相似,可以完美的触发一个view的点击事件,当然静默安装的原理也和抢红包的原理是一样的.   小米有开源 ...

最新文章

  1. 单片机彩灯移动实验_用S7-1200 PLC实现循环彩灯的控制,含源程序
  2. python中forward的参数_如何将关键字参数传递给preforward钩子使用的forward?
  3. 前 Google 工程师总结的算法面试指南
  4. Android双向滑动菜单完全解析,教你如何一分钟实现双向滑动特效
  5. Jedis操作Redis数据库
  6. linux上logbok实时日志_日志lombok插件安装及配置
  7. (摘)TMS320 DSP混合编程的方法研究
  8. 扫描到服务器的文件在哪个文件夹,云服务器的文件在哪个文件夹
  9. ubuntu ifconfig只有lo没有ens33的问题
  10. 我的邮箱又收到了一封信,这一次,关乎爱情
  11. ASP.NET MVC在IIS6下部署的小技巧
  12. java 正则 工具类_正则表达式工具类,正则表达式封装,Java正则表达式
  13. Java基础梳理(一)
  14. fastjson对Date的处理
  15. 互联网公司起名,其实都有套路的!
  16. 【教程:利用Axure快速原型工具制作电影信息网站】——小白也会,傻瓜式做法,简易快速!!!可应对(毕业)课程设计喔!!
  17. zynq-7000系列基于zynq-zed的vivado初步设计之linux下控制PL扩展的GPIO
  18. 按键精灵实现验证码识别
  19. 【2012年,当我回首来时路】
  20. 树上分治算法 + 路径剖分

热门文章

  1. 进度计划中的时间相关术语
  2. 泛娱乐出海走出水土不服,元宇宙社交诞生新机会!
  3. Tomcat安装配置
  4. 群晖 6.1 重置 mysql_群晖 “还原”/系统可恢复”无限循环终极解决
  5. Linux解决忘记密码的方法
  6. urlencode编码基本概念_URLencode编码简析
  7. java除法运算有小数_Java 整数间的除法运算如何保留所有小数位?
  8. NB-IoT无线通信模块与Lora无线通信协议技术分析与前景展望
  9. windows10华硕安装杜比音效
  10. 程序设计语言的特性——心理特性、工程特性、技术特性