我们知道很多应用被卸载后,都会打开浏览器,请你反馈一下为何卸载他们,那他们怎么知道呢?
我知道很多人会想到监听卸载广播android.intent.action.PACKAGE_REMOVED,但是你在被卸载的时候进程被kill掉,这个广播你是来不及接受,也没办法处理!
另外一种去监听刚开始卸载的log,但有可能监听不到,或者也同样来不及。
最理想的方式我想大家都知道,就是另外一个app监听PACKAGE_REMOVED就可以了。
但是你只有一个app,并且不想你被卸载掉后在用户的手机上还残存一个app,那这个时候应该怎么办呢?
其实我们可以通过ndk,去写一个natvie的进程,这个进程通过inotify监听你的应用文件夹是否被删除,如果删除就做相应处理即可。
inotify可以监听文件是否被删除修改打开等等操作,我们知道一个app会在/data/data/目录下用包名创建一个文件夹,我们应用为com.test.test,所以监听/data/data/com.test/test这个文件夹就可以了。
最后的处理比较简单,就是上报服务器,至于上报哪些内容就要看需求了。
对于inotify的使用可参考:http://www.ibm.com/developerworks/cn/linux/l-inotify/
代码参阅如下:
#include <sys/socket.h>
#include <stdio.h>
#include <android/log.h>
#include <sys/inotify.h>
#include <sys/select.h>
#include <errno.h>
#include <stdlib.h>
#include <stdint.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include <unistd.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <stddef.h>
#include <string.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#define LOG_TAG "JNI_LEARN"
#define LOGD(fmt, args...) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, fmt, ##args)
#define DEST_PORT 12234
#define DEST_IP_ADDR "10.78.137.21"
#define CS_MOBILE_UNINSTALLID 1007
void handle_event(struct inotify_event *pevent){
if(pevent->mask & IN_DELETE_SELF){
LOGD("deleted!");
int sockfd;
struct sockaddr_in s_add;
int sin_size;
sockfd = socket(AF_INET, SOCK_DGRAM, 0);
if(-1 == sockfd)
{
LOGD("socket fail ! \r\n");
exit(1);
}
LOGD("socket ok !\r\n");
bzero(&s_add,sizeof(struct sockaddr_in));
s_add.sin_family=AF_INET;
s_add.sin_addr.s_addr = inet_addr(DEST_IP_ADDR);
s_add.sin_port=htons(DEST_PORT);
int data = CS_MOBILE_UNINSTALLID;
int dest_len = sizeof(struct sockaddr_in);
int send_num = sendto(sockfd, (void *)&data, 4, 0, (struct sockaddr *)&s_add, dest_len);
if(send_num < 0){
LOGD("sendto fail!");
}else{
LOGD("sendto ok!");
exit(0);
}
}
}
int main(int argc, char **argv)
{
LOGD("uninstall process watching...");
int fd = inotify_init();
const char* watchPath = "/data/data/com.test.test";
int wd = inotify_add_watch(fd, watchPath, IN_ALL_EVENTS);
const int BUF_LEN = 16384;
char buffer[BUF_LEN];
struct inotify_event *pevent;
while(true){
fd_set fds;
FD_ZERO(&fds);
FD_SET(fd, &fds);
//堵塞事件,当有事件发生时返回值大于0
if(select(fd +1, &fds, NULL, NULL, NULL) > 0){
int len, index = 0;
if((len = read(fd, buffer, BUF_LEN)) < 0)
break;
while(index < len){
pevent = (struct inotify_event *)(buffer + index);
LOGD("event dispatch");
//处理事件
handle_event(pevent);
index += (offsetof(struct inotify_event, name) + pevent->len);
}
}
}
return 0;
}