Android如何监控本应用被卸载

我们知道很多应用被卸载后,都会打开浏览器,请你反馈一下为何卸载他们,那他们怎么知道呢?

我知道很多人会想到监听卸载广播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这个文件夹就可以了。

最后的处理比较简单,就是上报服务器。

代码参阅如下:

#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;
}



郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。