我们知道很多应用被卸载后,都会打开浏览器,请你反馈一下为何卸载他们,那他们怎么知道呢?
我知道很多人会想到监听卸载广播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 1007void 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;}