Android卸载反馈

Android卸载反馈

最初记得是在360安全卫士中出现的,在手机上卸载他的应用之后浏览器就会弹出一个反馈页面,让用户进行反馈,感觉这种功能对于产品改进特别有帮助。
但是仔细一想该怎么去实现却犯愁了,最开始想这也简单啊,不就是监听下自身被卸载就可以了,应该系统会有卸载的广播,可惜没有。甚至其他的一些
方法也是不行的,因为你程序都被卸载了,你的代码怎么会执行呢?皮之不存,毛将焉附。那360是怎样实现的呢?说句真心话360做的产品还是非常有创新
的,虽然有些时候他会损坏用户的利益,不过但从技术方面,着实让人信服。

既然Java实现不了,那就得考虑下其他的了,自然最先想到的就是JNI了,可惜C的部分不懂,上网搜了很多资料和介绍,找得到可以通过一下方式实现:
- 通过c中的fork方法来复制一个子进程。复制出来的子进程在父进程被销毁后,仍然可以存在。
pid_t fpid = fork()被调用前,就一个进程执行该段代码,这条语句执行之后,就会有两个进程执行该代码,两个进程执行没有固定先后顺序,只要看
系统调度策略,fork()函数的特别之处在于调用一次会返回两次结果:
- 返回值大于0,当前是父进程。
- 返回0,当前是子进程。
- 返回小于0的负值。(出错了,可能是内存不足或者是进程数已经达到系统最大值)

  • fork出子进程后让子进程一直去监听/data/data/packageName是否存在,如果不存在了,那就说明程序被卸载了,但是这样一直去轮训判断肯定会浪费
    系统资源的,当然也会更加费电,对用户来讲肯定是有损害的。所以这种技术最好也不好用,不然大家的手机以后还能了得。万恶的产品。

  • 得到程序被卸载之后弹出浏览器打开指定反馈页面。这就要用到am命令了,最早知道这个命令是在开发TV版的时候,遥控器找不到了,程序安装后无法
    打开了,用该命令就不怕了,哈哈。但是在c中怎么执行am命令呢?这就要用到execlp()函数,该函数就是c中执行系统命令的函数。

好了,主要的内容上面都分析完了,下面上代码。

  • Java层定义natvie方法。
public class MainActivity extends Activity {
    private static final String TAG = "@@@";

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        String packageDir = "/data/data/" + getPackageName();
        initUninstallFeedback(packageDir, Build.VERSION.SDK_INT);
    }

    private native void initUninstallFeedback(String packagePath, int sdkVersion);

    static {
        System.loadLibrary("uninstall_feedback");
    }
}
  • 创建jni目录,增加Android.mk以及uninstall_feedback.c文件。
    Android.mk的内容:
LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)

LOCAL_MODULE    := uninstall_feedback
LOCAL_SRC_FILES := uninstall_feedback.c

LOCAL_C_INCLUDES := $(LOCAL_PATH)/include
LOCAL_LDLIBS += -L$(SYSROOT)/usr/lib -llog

include $(BUILD_SHARED_LIBRARY)

uninstall_feedback.c的实现:

/**
 * 将Java中的String转换成c中的char字符串
 */
char* Jstring2CStr(JNIEnv* env, jstring jstr) {
    char* rtn = NULL;
    jclass clsstring = (*env)->FindClass(env, "java/lang/String"); //String
    jstring strencode = (*env)->NewStringUTF(env, "GB2312"); // 得到一个java字符串 "GB2312"
    jmethodID mid = (*env)->GetMethodID(env, clsstring, "getBytes",
            "(Ljava/lang/String;)[B"); //[ String.getBytes("gb2312");
    jbyteArray barr = (jbyteArray)(*env)->CallObjectMethod(env, jstr, mid,
            strencode); // String .getByte("GB2312");
    jsize alen = (*env)->GetArrayLength(env, barr); // byte数组的长度
    jbyte* ba = (*env)->GetByteArrayElements(env, barr, JNI_FALSE);
    if (alen > 0) {
        rtn = (char*) malloc(alen + 1); //""
        memcpy(rtn, ba, alen);
        rtn[alen] = 0;
    }
    (*env)->ReleaseByteArrayElements(env, barr, ba, 0); //
    return rtn;
}

void Java_com_charon_uninstallfeedback_MainActivity_initUninstallFeedback(
        JNIEnv* env, jobject thiz, jstring packageDir, jint sdkVersion) {

    char * pd = Jstring2CStr(env, packageDir);

    //fork子进程,以执行轮询任务
    pid_t pid = fork();

    if (pid < 0) {
        // fork失败了
    } else if (pid == 0) {
        // 可以一直采用一直判断文件是否存在的方式去判断,但是这样效率稍低,下面使用监听的方式,死循环,每个一秒判断一次,这样太浪费资源了。
        int check = 1;
        while (check) {
            FILE* file = fopen(pd, "rt");
            if (file == NULL) {
                if (sdkVersion >= 17) {
                    // Android4.2系统之后支持多用户操作,所以得指定用户
                    execlp("am", "am", "start", "--user", "0", "-a",
                            "android.intent.action.VIEW", "-d",
                            "http://shouji.360.cn/web/uninstall/uninstall.html",
                            (char*) NULL);
                } else {
                    // Android4.2以前的版本无需指定用户
                    execlp("am", "am", "start", "-a",
                        "android.intent.action.VIEW", "-d",
                            "http://shouji.360.cn/web/uninstall/uninstall.html",
                            (char*) NULL);
                }
                check = 0;
            } else {
            }
            sleep(1);
        }
    } else {
    }
}
  • 编译so文件。Windows下要用cygwin来操作。
    上面的介绍是在Eclipse中进行的,用ndk-build命令来编译so。具体请看之前写的JNI基础这篇文章。

源码请见Github

有关如何在Android Stuido中进行ndk开发请看另一篇文章。


评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值