1.通过jni实现函数
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
|
//LOG宏定义
#define LOG_INFO(tag, msg) __android_log_write(ANDROID_LOG_INFO, tag, msg)
#define LOG_DEBUG(tag, msg) __android_log_write(ANDROID_LOG_DEBUG, tag, msg)
#define LOG_WARN(tag, msg) __android_log_write(ANDROID_LOG_WARN, tag, msg)
#define LOG_ERROR(tag, msg) __android_log_write(ANDROID_LOG_ERROR, tag, msg)
/* 内全局变量begin */
static char
c_TAG[] = "onEvent"; static
jboolean b_IS_COPY = JNI_TRUE; jstring
Java_com_example_uninstallself_Observer_register(JNIEnv* env,
jobject thiz,
jstring path, jstring url, jint version) { jstring
tag = (*env)->NewStringUTF(env, c_TAG); //初始化log
LOG_DEBUG((*env)->GetStringUTFChars(env,
tag, &b_IS_COPY), (*env)->GetStringUTFChars(env,
(*env)->NewStringUTF(env, "init OK"),
&b_IS_COPY));
//fork子进程。以运行轮询任务
pid_t
pid = fork();
if
(pid < 0) { //出错log
LOG_ERROR((*env)->GetStringUTFChars(env,
tag, &b_IS_COPY), (*env)->GetStringUTFChars(env,
(*env)->NewStringUTF(env,
"fork failed !!!"), &b_IS_COPY));
}
else if (pid == 0) { //子进程注冊文件夹监听器
int
fileDescriptor = inotify_init(); if
(fileDescriptor < 0) {
LOG_DEBUG((*env)->GetStringUTFChars(env,
tag, &b_IS_COPY), (*env)->GetStringUTFChars(env,
(*env)->NewStringUTF(env,
"inotify_init failed !!!"),
&b_IS_COPY)); exit(1);
}
int
watchDescriptor;
watchDescriptor
= inotify_add_watch(fileDescriptor, (*env)->GetStringUTFChars(env,
path, NULL), IN_DELETE);
if
(watchDescriptor < 0) { LOG_DEBUG((*env)->GetStringUTFChars(env,
tag, &b_IS_COPY),
(*env)->GetStringUTFChars(env,
(*env)->NewStringUTF(env,
"inotify_add_watch failed !!!"),
&b_IS_COPY));
exit(1);
}
//分配缓存。以便读取event。缓存大小=一个struct inotify_event的大小。这样一次处理一个event
void
*p_buf = malloc(sizeof(struct inotify_event));
if
(p_buf == NULL) { LOG_DEBUG((*env)->GetStringUTFChars(env,
tag, &b_IS_COPY),
(*env)->GetStringUTFChars(env,
(*env)->NewStringUTF(env,
"malloc failed !!!"),
&b_IS_COPY));
exit(1);
}
//開始监听
LOG_DEBUG((*env)->GetStringUTFChars(env,
tag, &b_IS_COPY),
(*env)->GetStringUTFChars(env,
(*env)->NewStringUTF(env,
"start observer"),
&b_IS_COPY));
//read会堵塞进程,
size_t readBytes
= read(fileDescriptor, p_buf, sizeof(struct
inotify_event)); //走到这里说明收到文件夹被删除的事件。注销监听器
free(p_buf);
inotify_rm_watch(fileDescriptor,
IN_DELETE); //文件夹不存在log
LOG_DEBUG((*env)->GetStringUTFChars(env,
tag, &b_IS_COPY), (*env)->GetStringUTFChars(env,
(*env)->NewStringUTF(env,
"uninstalled"), &b_IS_COPY));
if
(version >= 17) { //4.2以上的系统因为用户权限管理更严格,须要加上 --user 0
execlp("am",
"am", "start", "--user", "0", "-a", "android.intent.action.VIEW",
"-d",
(*env)->GetStringUTFChars(env,
url, NULL), (char *) NULL); }
else {
execlp("am",
"am", "start", "-a", "android.intent.action.VIEW", "-d",
(*env)->GetStringUTFChars(env, url, NULL),
(char
*) NULL); }
//扩展:能够运行其它shell命令,am(即activity manager),能够打开某程序、服务,broadcast intent,等等
}
else { //父进程直接退出,使子进程被init进程领养,以避免子进程僵死
}
return
(*env)->NewStringUTF(env, "Hello from JNI !"); }
|
2.定义UninstallObserver
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
public
class UninstallObserver { static{
System.loadLibrary("observer");
}
/***
*
* @param path 须要监听的文件路径。可用 getApplicationContext().getFilesDir().getPath()
* @param url 卸载调转http
* @param version android.os.Build.VERSION.SDK_INT
* @return
*/
public
static native String register(String path, String url, int version);
}
|
3.简单使用
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
@Override
protected void
onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Toast.makeText(getApplicationContext(),
getApplicationContext().getFilesDir().getPath()
+ "," + Build.VERSION.SDK_INT, 1).show();
long
a = System.currentTimeMillis(); String
str = UninstallObserver.register(getApplicationContext().getFilesDir().getPath(), "http://www.baidu.com",
android.os.Build.VERSION.SDK_INT);
long
b = System.currentTimeMillis();
Toast.makeText(getApplicationContext(),
str+","+(b-a), 1).show();
}
|
Android研究之监听自身应用被卸载代码实现的更多相关文章
-
Android应用如何监听自己是否被卸载及卸载反馈功能的实现
一个应用被用户卸载肯定是有理由的,而开发者却未必能得知这一重要的理由,毕竟用户很少会主动反馈建议,多半就是用得不爽就卸,如果能在被卸载后获取到用户的一些反馈,那对开发者进一步改进应用是非常有利的.目前 ...
-
Android 短信监听及用途分析
监听系统短信这个只能作为一个技术点来研究下,读者可能在工作中可能不会哦涉及到,一般的应用软件也不会有这个需求 但是作为程序员呢,多了解一下也是好的. Android 监听系统短信有什么用? 1.对系统 ...
-
Android 手势水平监听判断
package com.zihao.ui; import com.zihao.R; import android.os.Bundle; import android.app.Activity; imp ...
-
Android中如何监听GPS开启和关闭
转自 chenming 原文 Android中如何监听GPS开启和关闭 摘要: 本文简单总结了如何监听GPS开关的小技巧 有时需要监听GPS的开关(这种需求并不多见).实现的思路是监听代表 GPS ...
-
Android手机上监听短信的两种方式
Android手机上监听短信有两种方式: 1. 接受系统的短信广播,操作短信内容. 优点:操作方便,适合简单的短信应用. 缺点:来信会在状态栏显示通知信息. AndroidManifest.xml: ...
-
【Android】Android 监听apk安装替换卸载广播
[Android]Android 监听apk安装替换卸载广播 首先是要获取应用的安装状态,通过广播的形式 以下是和应用程序相关的Broadcast Action ACTION_PACKAGE_ADDE ...
-
android的电话监听
android的电话监听 新建一个项目,结构图如下: PhoneService: package com.demo.tingdianhua; import android.app.Service; i ...
-
Android零基础入门第34节:Android中基于监听的事件处理
原文:Android零基础入门第34节:Android中基于监听的事件处理 上一期我们学习了Android中的事件处理,也详细学习了Android中基于监听的事件处理,同时学会了匿名内部类形式,那么本 ...
-
Android实现网络监听
一.Android Wifi常用广播 网络开发中主体会使用到的action: ConnectivityManager.CONNECTIVITY_ACTION WifiManager.WIFI_STAT ...
随机推荐
-
tp 多语言支持
tp支持多语言 通过get来改变语言的 http://localhost/tp/index.php/Admin/User/add/hl/zh-cn http://localhost/tp/index. ...
-
win7下搭建opengles2.0编程环境
原帖地址:http://sixgod.org/archives/72 1.下载AMD的OpenGL ES2.0的模拟器,地址: http://www.opengles-book.com/ESEmu ...
-
JMeter对Oracle数据库进行压力测试
步骤 (1)复制ORACLE的JDBC驱动JAR包文件(ojdbc14.jar)到JMeter的lib目录下. (2)运行jmeter.bat (3)建立线程组:右键测试计划->添加->T ...
-
【基础】Oracle 表空间和数据文件
多个表空间的优势:1.能够将数据字典与用户数据分离出来,避免由于字典对象和用户对象保存在同一个数据文件中而产生的I/O冲突2.能够将回退数据与用户数据分离出来,避免由于硬盘损坏而导致永久性的数据丢失3 ...
-
[非技术参考]C#基础:使用Thread创建线程(1)
Thread类可以创建和控制线程,Thread类的构造函数重载为接受ThreadStart和ParameterizedThreadStart类型的委托参数.下面我们用一个例子来解释怎样用Thread类 ...
-
IntelliJ IDEA 17和Maven构建javaWeb项目
前言 电脑又断电了,眼看着写好的东西就没有了,这是第二次犯这个错误了.很难受呀!还是回到正题吧,我们来使用IDEA和Maven构建一个JavaWeb项目 软件环境: IDEA:2017.2.1 JDK ...
-
Pygame常用方法
'''import pygame# 初始化pygame库,让计算机硬件准备pygame.init()# ----------窗口相关操作-----------# 创建窗口window = pygame ...
-
echart 图例
说明:stack相同,两个bar合并但是不会重叠 如果需要重叠 用barGap: '-100%', 根据不同的需求来使用两者. <template> <div> echart ...
-
struts2 标签使用注意
版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/qilixiang012/article/details/31954501 通常是用html标签.而不 ...
-
break与continue语句
1.break:立即退出循环 <!DOCTYPE html> <html lang="en"> <head> <meta charset= ...