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();
 
}

p=2010">本文參考资料

Android研究之监听自身应用被卸载代码实现的更多相关文章

  1. Android应用如何监听自己是否被卸载及卸载反馈功能的实现

    一个应用被用户卸载肯定是有理由的,而开发者却未必能得知这一重要的理由,毕竟用户很少会主动反馈建议,多半就是用得不爽就卸,如果能在被卸载后获取到用户的一些反馈,那对开发者进一步改进应用是非常有利的.目前 ...

  2. Android 短信监听及用途分析

    监听系统短信这个只能作为一个技术点来研究下,读者可能在工作中可能不会哦涉及到,一般的应用软件也不会有这个需求 但是作为程序员呢,多了解一下也是好的. Android 监听系统短信有什么用? 1.对系统 ...

  3. Android 手势水平监听判断

    package com.zihao.ui; import com.zihao.R; import android.os.Bundle; import android.app.Activity; imp ...

  4. Android中如何监听GPS开启和关闭

    转自 chenming 原文 Android中如何监听GPS开启和关闭   摘要: 本文简单总结了如何监听GPS开关的小技巧 有时需要监听GPS的开关(这种需求并不多见).实现的思路是监听代表 GPS ...

  5. Android手机上监听短信的两种方式

    Android手机上监听短信有两种方式: 1. 接受系统的短信广播,操作短信内容. 优点:操作方便,适合简单的短信应用. 缺点:来信会在状态栏显示通知信息. AndroidManifest.xml: ...

  6. 【Android】Android 监听apk安装替换卸载广播

    [Android]Android 监听apk安装替换卸载广播 首先是要获取应用的安装状态,通过广播的形式 以下是和应用程序相关的Broadcast Action ACTION_PACKAGE_ADDE ...

  7. android的电话监听

    android的电话监听 新建一个项目,结构图如下: PhoneService: package com.demo.tingdianhua; import android.app.Service; i ...

  8. Android零基础入门第34节:Android中基于监听的事件处理

    原文:Android零基础入门第34节:Android中基于监听的事件处理 上一期我们学习了Android中的事件处理,也详细学习了Android中基于监听的事件处理,同时学会了匿名内部类形式,那么本 ...

  9. Android实现网络监听

    一.Android Wifi常用广播 网络开发中主体会使用到的action: ConnectivityManager.CONNECTIVITY_ACTION WifiManager.WIFI_STAT ...

随机推荐

  1. 私有成员 ECMAScript6 weakmap

    把变量放到构造函数里进行私有化,每一个实例都会有一个getName方法,这是低效率的.function Person(name) { this.getName = function() { retur ...

  2. 浅谈postgresql的GIN索引(通用倒排索引)

    1.倒排索引原理 倒排索引来源于搜索引擎的技术,可以说是搜索引擎的基石.正是有了倒排索引技术,搜索引擎才能有效率的进行数据库查找.删除等操作.在详细说明倒排索引之前,我们说一下与之相关的正排索引并与之 ...

  3. 学习C++ -> 构造函数与析构函数

    学习C++ -> 构造函数与析构函数 一.构造函数的介绍    1. 构造函数的作用        构造函数主要用来在创建对象时完成对对象属性的一些初始化等操作, 当创建对象时, 对象会自动调用 ...

  4. HBase流量限制和表负载均衡剖析

    1.概述 在HBase-1.1.0之前,HBase集群中资源都是全量的.用户.表这些都是没有限制的,看似完美实则隐患较大.今天,笔者就给大家剖析一下HBase的流量限制和表的负载均衡. 2.内容 也许 ...

  5. RobotFramework自动化测试环境配置

    现在工作是做自动化测试平台维护的,平台用的C#做的,主要是用来测试CMBRun项目,它是c/s结构的项目,而b/s结构的项目主要使用RF+Python来做.做这块之前听过自动化测试,身边的朋友也有做这 ...

  6. VMware系统克隆

    第1章 搭建VMware实战环境 1.1 vmware主机配置-网络配置 1.1.1 虚拟主机添加网卡信息(5) a.右键虚拟主机→设置→添加虚拟网卡硬件设备 b.设置网络适配器类型→完成添加 1.1 ...

  7. PyCharm汉化、破解教程

    汉化 1.将 C:\Program Files (x86)\JetBrains\PyCharm 2017\lib(路径是你的安装路径)目录下的resources_en.jar文件复制出来之后删除,以备 ...

  8. Spring IOC容器分析(3) -- DefaultListableBeanFactory

    上一节介绍了封装bean对象的BeanDefinition接口.从前面小结对BeanFactory的介绍中,我们知道bean对象是存储在map中,通过调用getBean方法可以得到bean对象.在接口 ...

  9. 2017阿里云双11-云服务器ECS优惠活动最强解读和购买指南

    本站之前介绍了<爆款云服务器,限时2折起>,这其实是阿里云双11之前的预热活动:四款低配的机型,二折给用户(每个用户限购一台),非常的实惠,有很多阅读了本站文章的用户都一次性购买了三年的. ...

  10. Java 核心内容相关面试题【3】

    目录 面向对象编程(OOP) 常见的Java问题 Java线程 Java集合类 垃圾收集器 异常处理 Java小应用程序(Applet) Swing JDBC 远程方法调用(RMI) Servlet ...