├── .gitignore ├── .idea ├── compiler.xml ├── copyright │ └── profiles_settings.xml ├── gradle.xml ├── misc.xml ├── modules.xml └── runConfigurations.xml ├── README.md ├── app ├── .gitignore ├── CMakeLists.txt ├── build.gradle ├── proguard-rules.pro └── src │ ├── androidTest │ └── java │ │ └── unstall │ │ └── yyh │ │ └── com │ │ └── a360installtolistener │ │ └── ExampleInstrumentedTest.java │ ├── main │ ├── AndroidManifest.xml │ ├── cpp │ │ └── native-lib.cpp │ ├── java │ │ └── unstall │ │ │ └── yyh │ │ │ └── com │ │ │ └── a360installtolistener │ │ │ └── MainActivity.java │ └── res │ │ ├── layout │ │ └── activity_main.xml │ │ ├── mipmap-hdpi │ │ └── ic_launcher.png │ │ ├── mipmap-mdpi │ │ └── ic_launcher.png │ │ ├── mipmap-xhdpi │ │ └── ic_launcher.png │ │ ├── mipmap-xxhdpi │ │ └── ic_launcher.png │ │ ├── mipmap-xxxhdpi │ │ └── ic_launcher.png │ │ ├── values-w820dp │ │ └── dimens.xml │ │ └── values │ │ ├── colors.xml │ │ ├── dimens.xml │ │ ├── strings.xml │ │ └── styles.xml │ └── test │ └── java │ └── unstall │ └── yyh │ └── com │ └── a360installtolistener │ └── ExampleUnitTest.java ├── build.gradle ├── gradle.properties ├── gradle └── wrapper │ ├── gradle-wrapper.jar │ └── gradle-wrapper.properties ├── gradlew ├── gradlew.bat └── settings.gradle /.gitignore: -------------------------------------------------------------------------------- 1 | *.iml 2 | .gradle 3 | /local.properties 4 | /.idea/workspace.xml 5 | /.idea/libraries 6 | .DS_Store 7 | /build 8 | /captures 9 | .externalNativeBuild 10 | -------------------------------------------------------------------------------- /.idea/compiler.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | -------------------------------------------------------------------------------- /.idea/copyright/profiles_settings.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | -------------------------------------------------------------------------------- /.idea/gradle.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 17 | 18 | -------------------------------------------------------------------------------- /.idea/misc.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 19 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | 44 | 46 | 47 | 48 | 49 | 50 | 1.8 51 | 52 | 57 | 58 | 59 | 60 | 61 | 62 | -------------------------------------------------------------------------------- /.idea/modules.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | -------------------------------------------------------------------------------- /.idea/runConfigurations.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 11 | 12 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | 2 | #### NDK项目实战仿360手机助手卸载监听提交报告。 3 | 4 | 5 | 效果图如下: 6 | 7 | 8 | ![效果图](http://upload-images.jianshu.io/upload_images/4614633-12cb0be0690df358.gif?imageMogr2/auto-orient/strip) 9 | 10 | 11 | ### 目的 12 | 13 | 需要监听什么时候卸载,卸载后执行部分操作 14 | 15 | ### 针对于Android的系统,我们可以试想有一下策略 16 | 17 | 18 | - ****1、监听系统卸载广播**** 19 | 20 | 只能监听到其他应用的卸载广播,无法监听到自己是否被卸载。 21 | 22 | 内存 --》 监听 别人 23 | . ACTION_PACKAGE_REMOVED 24 | 2.ACTION_PACKAGE_REMOVED 25 | 26 | 27 | 28 | 29 | - 2、****卸载**** 30 | 31 | 通过系统log (正在被安装的包程序不能接收到这个广播) 32 | 33 | 34 | 监听自身 35 | 36 | - 3、****Java线程**** 37 | 38 | 轮训 监听 监听/data/data/{package-name}目录是否存在 39 | 40 | 41 | - 4、****C进程**** 42 | 43 | 监听/data/data/{package-name}目录是否存在 跳转到网页 44 | 45 | 46 | ![image](http://s14.sinaimg.cn/orignal/49b22c67045842f500f0d) 47 | 48 | - 5、****静默安装另外的apk**** 49 | 50 | 监听自己是否被卸载 可以,但是前提需要(root) 51 | 52 | 从上面分析来看 53 | 54 | - 如果开启线程 ,则耗资源比较大 55 | - File 的监听 可以试试 56 | 57 | 58 | ### 360卸载监听的方式: 59 | 60 | 监听 /data/data/{package-name}目录是否存在 61 | 62 | #### 总结 63 | 64 | 从前四种方案可以看到,单纯的Java层代码是无法监听自身卸载的。既然Java层无法实现,我们试着使用C语言在底层实现。 65 | ****借助Linux进程fork出来的C进程在应用被卸载后不会被销毁****,监听/data/data/{package-name}目录是否存在,如果不存在,就证明应用被卸载了。 66 | 67 | 68 | ## 说干就干,开始撸码 69 | 70 | 1. ****fork()子进程**** 71 | 2. ****创建监听文件**** 72 | 3. ****初始化inotify实例**** 73 | 4. ****注册监听事件**** 74 | 5. ****调用read函数开始监听**** 75 | 6. ****卸载反馈统计**** 76 | 77 | ### 实现原理 78 | 79 | 是仿照FileObserve监听文件的方式,在本地方式中进行拦截和操作 80 | 81 | 接下老我么来看下FileObserve的源码 82 | 83 | FileObserve一进来的时候就开启了一个线程,如下: 84 | 85 | ![FileObserve进来初始化.png](http://upload-images.jianshu.io/upload_images/4614633-a3d118624f3fd0cb.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240) 86 | 87 | 利用本利方法进行监听: 88 | 89 | 90 | ![FileObserve.png](http://upload-images.jianshu.io/upload_images/4614633-08b0f4cbd6d5185f.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240) 91 | 92 | 93 | 在线程中进行操作: 94 | 95 | ![执行操作.png](http://upload-images.jianshu.io/upload_images/4614633-78b15015fa45a698.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240) 96 | 97 | 98 | C中fork代码如下: 99 | ``` 100 | JNIEXPORT 101 | void JNICALL 102 | Java_unstall_yyh_com_a360installtolistener_MainActivity_callUnInstallListener(JNIEnv *env,jobject obj,jint versionSdk,jstring path){ 103 | LOGD("------------------------"); 104 | LOGF("------------------------"); 105 | const char * path_str = env->GetStringUTFChars(path,0); 106 | pid_t pid = fork(); 107 | if(pid < 0){ 108 | LOGD("克隆失败"); 109 | }else if(pid > 0){ 110 | LOGD("父进程"); 111 | }else{ 112 | LOGD("子进程!"); 113 | //*******************在这里进程操作***************** 114 | LOGD("你好,终端研发部"); 115 | int fuileDescript = inotify_init(); 116 | int watch = inotify_add_watch(fuileDescript,path_str,IN_DELETE_SELF); 117 | void * p = malloc(sizeof(struct inotify_event)); 118 | read(fuileDescript,p, sizeof(struct inotify_event)); 119 | inotify_rm_watch(fuileDescript,watch); 120 | LOGD(LOG_TAG,"接下来进行操作,来条状网页!!!"); 121 | if(versionSdk< 17){ 122 | //am start -a android.intent.action.VIEW -d http://gityuan.com 123 | execlp("am","am","start","-a","android.intent.action.VIEW","-d","https://mp.weixin.qq.com/s?__biz=MzI3OTU0MzI4MQ==&mid=2247484366&idx=2&sn=a015497277d2a6380a80fdc9031ca51c&chksm=eb476f50dc30e64620fbb8a7ce0aebc445638c5f1097763e0da36fc40beb85fb256d980af440&scene=18#wechat_redirect",NULL); 124 | }else{ 125 | execlp("am","am","start","--user","0","-a","android.intent.action.VIEW","-d", "https://mp.weixin.qq.com/s?__biz=MzI3OTU0MzI4MQ==&mid=2247484366&idx=2&sn=a015497277d2a6380a80fdc9031ca51c&chksm=eb476f50dc30e64620fbb8a7ce0aebc445638c5f1097763e0da36fc40beb85fb256d980af440&scene=18#wechat_redirect",NULL); 126 | } 127 | } 128 | env->ReleaseStringUTFChars(path,path_str); 129 | } 130 | ``` 131 | 132 | 最后卸载监听的是我技术号里的一篇文章。 133 | 134 | ### 总结 135 | - 6.0之后的就不能再进行监听卸载了。 136 | - 凡是360手机助手能支持的该方式也基本支持 137 | - 由于部分厂家修改底层源码导致部分手机无法监听下载 138 | - 通过实验360手机助手也不能适配所有的6.0之前的手机,如小米红木手机 139 | 140 | 141 | 142 | 如果对ndk不了的同学们可以一下参考博客: 143 | 144 | NDK探究之旅: 145 | 146 | [NDK和JNI初探及其联系和区别](http://mp.weixin.qq.com/s?__biz=MzI3OTU0MzI4MQ==&mid=2247484266&idx=1&sn=b4f9390e0c45a04c331365b7faf8ef99&chksm=eb476ff4dc30e6e2b80e431c89bec941907bf5b308431097bb725ece700771b73faff581ac5d&scene=21#wechat_redirect) 147 | 148 | [NDK探究之旅《一》——对jni和NDK的认识](http://blog.csdn.net/androidstarjack/article/details/71375042) 149 | 150 | [NDK探究之旅《二》——C语言的基本认识](http://blog.csdn.net/androidstarjack/article/details/71156474) 151 | 152 | [NDK探究之旅《三》—C语言的输入输出函数](http://blog.csdn.net/androidstarjack/article/details/71514125) 153 | 154 | [NDK探究之旅《四》——指针的强化理解](http://blog.csdn.net/androidstarjack/article/details/71514125) 155 | 156 | [NDK探究之旅《五》——指针和数组之间的关系](http://blog.csdn.net/androidstarjack/article/details/71699350) 157 | 158 | [NDK探究之旅《六》—函数的指针、结构体、枚举、宏定义](http://blog.csdn.net/androidstarjack/article/details/72455218) 159 | 160 | [NDK探究之旅《七》—函数指针,宏定义的优缺点及应用场景](http://blog.csdn.net/androidstarjack/article/details/72455218) 161 | 162 | [NDK探究之旅《八》——jni的开发流程规范及环境配置](http://blog.csdn.net/androidstarjack/article/details/72615926) 163 | 164 | [NDK探究之旅《九》——jni开发中常见的错误及其注意事项](http://blog.csdn.net/androidstarjack/article/details/72758094) 165 | 166 | [NDK探究之旅《十》——ndk项目实战之Androidstudio开发经验总结](http://blog.csdn.net/androidstarjack/article/details/72813995) 167 | 168 | [NDK探究之旅《十一》—C代码调用Java代码之项目实战](http://blog.csdn.net/androidstarjack/article/details/72827991) 169 | 170 | 171 | 172 | >GitHub下载链接: 173 | > 174 | >https://github.com/androidstarjack/360InstallToListener 175 | > 176 | >下载慢? 177 | > 178 | >http://download.csdn.net/download/androidstarjack/9966557 179 | 180 | ### 相信自己,没有做不到的,只有想不到的 181 | 如果你觉得此文对您有所帮助,欢迎入群 QQ交流群 :232203809 182 | 微信公众号:终端研发部 183 | 184 | 185 | ![技术+职场](http://upload-images.jianshu.io/upload_images/4614633-977d06f49c7ba7be.jpg?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240) 186 | 187 | (欢迎关注学习和交流) 188 | -------------------------------------------------------------------------------- /app/.gitignore: -------------------------------------------------------------------------------- 1 | /build 2 | -------------------------------------------------------------------------------- /app/CMakeLists.txt: -------------------------------------------------------------------------------- 1 | # Sets the minimum version of CMake required to build the native 2 | # library. You should either keep the default value or only pass a 3 | # value of 3.4.0 or lower. 4 | 5 | cmake_minimum_required(VERSION 3.4.1) 6 | 7 | # Creates and names a library, sets it as either STATIC 8 | # or SHARED, and provides the relative paths to its source code. 9 | # You can define multiple libraries, and CMake builds it for you. 10 | # Gradle automatically packages shared libraries with your APK. 11 | 12 | add_library( # Sets the name of the library. 13 | native-lib 14 | 15 | # Sets the library as a shared library. 16 | SHARED 17 | 18 | # Provides a relative path to your source file(s). 19 | # Associated headers in the same location as their source 20 | # file are automatically included. 21 | src/main/cpp/native-lib.cpp ) 22 | 23 | # Searches for a specified prebuilt library and stores the path as a 24 | # variable. Because system libraries are included in the search path by 25 | # default, you only need to specify the name of the public NDK library 26 | # you want to add. CMake verifies that the library exists before 27 | # completing its build. 28 | 29 | find_library( # Sets the name of the path variable. 30 | log-lib 31 | 32 | # Specifies the name of the NDK library that 33 | # you want CMake to locate. 34 | log ) 35 | 36 | # Specifies libraries CMake should link to your target library. You 37 | # can link multiple libraries, such as libraries you define in the 38 | # build script, prebuilt third-party libraries, or system libraries. 39 | 40 | target_link_libraries( # Specifies the target library. 41 | native-lib 42 | 43 | # Links the target library to the log library 44 | # included in the NDK. 45 | ${log-lib} ) 46 | -------------------------------------------------------------------------------- /app/build.gradle: -------------------------------------------------------------------------------- 1 | apply plugin: 'com.android.application' 2 | 3 | android { 4 | compileSdkVersion 25 5 | buildToolsVersion "25.0.3" 6 | defaultConfig { 7 | applicationId "unstall.yyh.com.a360installtolistener" 8 | minSdkVersion 19 9 | targetSdkVersion 25 10 | versionCode 1 11 | versionName "1.0" 12 | testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner" 13 | externalNativeBuild { 14 | cmake { 15 | cppFlags "-frtti -fexceptions" 16 | } 17 | } 18 | } 19 | buildTypes { 20 | release { 21 | minifyEnabled false 22 | proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' 23 | } 24 | } 25 | externalNativeBuild { 26 | cmake { 27 | path "CMakeLists.txt" 28 | } 29 | } 30 | } 31 | 32 | dependencies { 33 | compile fileTree(dir: 'libs', include: ['*.jar']) 34 | androidTestCompile('com.android.support.test.espresso:espresso-core:2.2.2', { 35 | exclude group: 'com.android.support', module: 'support-annotations' 36 | }) 37 | compile 'com.android.support:appcompat-v7:25.3.1' 38 | testCompile 'junit:junit:4.12' 39 | } 40 | -------------------------------------------------------------------------------- /app/proguard-rules.pro: -------------------------------------------------------------------------------- 1 | # Add project specific ProGuard rules here. 2 | # By default, the flags in this file are appended to flags specified 3 | # in D:\sdk/tools/proguard/proguard-android.txt 4 | # You can edit the include path and order by changing the proguardFiles 5 | # directive in build.gradle. 6 | # 7 | # For more details, see 8 | # http://developer.android.com/guide/developing/tools/proguard.html 9 | 10 | # Add any project specific keep options here: 11 | 12 | # If your project uses WebView with JS, uncomment the following 13 | # and specify the fully qualified class name to the JavaScript interface 14 | # class: 15 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview { 16 | # public *; 17 | #} 18 | -------------------------------------------------------------------------------- /app/src/androidTest/java/unstall/yyh/com/a360installtolistener/ExampleInstrumentedTest.java: -------------------------------------------------------------------------------- 1 | package unstall.yyh.com.a360installtolistener; 2 | 3 | import android.content.Context; 4 | import android.support.test.InstrumentationRegistry; 5 | import android.support.test.runner.AndroidJUnit4; 6 | 7 | import org.junit.Test; 8 | import org.junit.runner.RunWith; 9 | 10 | import static org.junit.Assert.*; 11 | 12 | /** 13 | * Instrumentation test, which will execute on an Android device. 14 | * 15 | * @see Testing documentation 16 | */ 17 | @RunWith(AndroidJUnit4.class) 18 | public class ExampleInstrumentedTest { 19 | @Test 20 | public void useAppContext() throws Exception { 21 | // Context of the app under test. 22 | Context appContext = InstrumentationRegistry.getTargetContext(); 23 | 24 | assertEquals("unstall.yyh.com.a360installtolistener", appContext.getPackageName()); 25 | } 26 | } 27 | -------------------------------------------------------------------------------- /app/src/main/AndroidManifest.xml: -------------------------------------------------------------------------------- 1 | 2 | 4 | 5 | 6 | 7 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | -------------------------------------------------------------------------------- /app/src/main/cpp/native-lib.cpp: -------------------------------------------------------------------------------- 1 | #include 2 | #include 3 | #include 4 | #include 5 | #define LOG_TAG "yuyahao1" 6 | #include 7 | #include 8 | 9 | #define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, __VA_ARGS__) 10 | #define LOGF(...) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, __VA_ARGS__) 11 | 12 | 13 | 14 | extern "C"{ 15 | JNIEXPORT jstring JNICALL 16 | Java_unstall_yyh_com_a360installtolistener_MainActivity_stringFromJNI( 17 | JNIEnv *env, 18 | jobject /* this */) { 19 | std::string hello = "欢迎关注终端研发部:


" 20 | "ndk项目实战—高仿360手机助手之卸载监听"; 21 | LOGD(LOG_TAG,"我进入到这个减小的局面了"); 22 | return env->NewStringUTF(hello.c_str()); 23 | } 24 | JNIEXPORT 25 | void JNICALL 26 | Java_unstall_yyh_com_a360installtolistener_MainActivity_callUnInstallListener(JNIEnv *env,jobject obj,jint versionSdk,jstring path){ 27 | LOGD("------------------------"); 28 | LOGF("------------------------"); 29 | const char * path_str = env->GetStringUTFChars(path,0); 30 | pid_t pid = fork(); 31 | if(pid < 0){ 32 | LOGD("克隆失败"); 33 | }else if(pid > 0){ 34 | LOGD("父进程"); 35 | }else{ 36 | LOGD("子进程!"); 37 | //*******************在这里进程操作***************** 38 | LOGD("你好,终端研发部"); 39 | int fuileDescript = inotify_init(); 40 | int watch = inotify_add_watch(fuileDescript,path_str,IN_DELETE_SELF); 41 | void * p = malloc(sizeof(struct inotify_event)); 42 | read(fuileDescript,p, sizeof(struct inotify_event)); 43 | inotify_rm_watch(fuileDescript,watch); 44 | LOGD(LOG_TAG,"接下来进行操作,来条状网页!!!"); 45 | if(versionSdk< 17){ 46 | //am start -a android.intent.action.VIEW -d http://gityuan.com 47 | execlp("am","am","start","-a","android.intent.action.VIEW","-d","https://mp.weixin.qq.com/s?__biz=MzI3OTU0MzI4MQ==&mid=2247484366&idx=2&sn=a015497277d2a6380a80fdc9031ca51c&chksm=eb476f50dc30e64620fbb8a7ce0aebc445638c5f1097763e0da36fc40beb85fb256d980af440&scene=18#wechat_redirect",NULL); 48 | }else{ 49 | execlp("am","am","start","--user","0","-a","android.intent.action.VIEW","-d", "https://mp.weixin.qq.com/s?__biz=MzI3OTU0MzI4MQ==&mid=2247484366&idx=2&sn=a015497277d2a6380a80fdc9031ca51c&chksm=eb476f50dc30e64620fbb8a7ce0aebc445638c5f1097763e0da36fc40beb85fb256d980af440&scene=18#wechat_redirect",NULL); 50 | } 51 | } 52 | env->ReleaseStringUTFChars(path,path_str); 53 | } 54 | 55 | 56 | 57 | } 58 | 59 | 60 | -------------------------------------------------------------------------------- /app/src/main/java/unstall/yyh/com/a360installtolistener/MainActivity.java: -------------------------------------------------------------------------------- 1 | package unstall.yyh.com.a360installtolistener; 2 | 3 | import android.os.Build; 4 | import android.os.FileObserver; 5 | import android.support.v7.app.AppCompatActivity; 6 | import android.os.Bundle; 7 | import android.text.Html; 8 | import android.view.View; 9 | import android.widget.DatePicker; 10 | import android.widget.NumberPicker; 11 | import android.widget.TextView; 12 | /** 13 | * 类功能描述:
14 | * NDK项目实战—高仿360手机助手之卸载监听 15 | * 博客地址:http://blog.csdn.net/androidstarjack 16 | * 公众号:终端研发部 17 | * @author yuyahao 18 | * @version 1.0

修改时间:
修改备注:
19 | */ 20 | public class MainActivity extends AppCompatActivity { 21 | 22 | // Used to load the 'native-lib' library on application startup. 23 | static { 24 | System.loadLibrary("native-lib"); 25 | } 26 | 27 | @Override 28 | protected void onCreate(Bundle savedInstanceState) { 29 | super.onCreate(savedInstanceState); 30 | setContentView(R.layout.activity_main); 31 | // Example of a call to a native method 32 | final TextView tv = (TextView) findViewById(R.id.sample_text); 33 | tv.setText(Html.fromHtml("点我试试看:\n

" + 34 | " NDK项目实战—高仿360手机助手之卸载监听")); 35 | callUnInstallListener(Build.VERSION.SDK_INT,"data/data/unstall.yyh.com.a360installtolistener"); 36 | tv.setOnClickListener(new View.OnClickListener() { 37 | @Override 38 | public void onClick(View v) { 39 | tv.setText(Html.fromHtml(stringFromJNI())); 40 | DatePicker s; 41 | NumberPicker ss; 42 | } 43 | }); 44 | } 45 | 46 | /** 47 | * A native method that is implemented by the 'native-lib' native library, 48 | * which is packaged with this application. 49 | */ 50 | public native String stringFromJNI(); 51 | 52 | /** 53 | * 自己写的一个监听的本地方法 54 | * @param versionSdk 55 | * @param path 56 | */ 57 | private native void callUnInstallListener(int versionSdk,String path); 58 | } 59 | -------------------------------------------------------------------------------- /app/src/main/res/layout/activity_main.xml: -------------------------------------------------------------------------------- 1 | 2 | 12 | 13 | 21 | 22 | 23 | -------------------------------------------------------------------------------- /app/src/main/res/mipmap-hdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/codeGoogler/360InstallToListener/eb063976b82d06179d4dac696f33a6039cc9bf83/app/src/main/res/mipmap-hdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-mdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/codeGoogler/360InstallToListener/eb063976b82d06179d4dac696f33a6039cc9bf83/app/src/main/res/mipmap-mdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/codeGoogler/360InstallToListener/eb063976b82d06179d4dac696f33a6039cc9bf83/app/src/main/res/mipmap-xhdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xxhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/codeGoogler/360InstallToListener/eb063976b82d06179d4dac696f33a6039cc9bf83/app/src/main/res/mipmap-xxhdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xxxhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/codeGoogler/360InstallToListener/eb063976b82d06179d4dac696f33a6039cc9bf83/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/values-w820dp/dimens.xml: -------------------------------------------------------------------------------- 1 | 2 | 5 | 64dp 6 | 7 | -------------------------------------------------------------------------------- /app/src/main/res/values/colors.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | #3F51B5 4 | #303F9F 5 | #FF4081 6 | 7 | -------------------------------------------------------------------------------- /app/src/main/res/values/dimens.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 16dp 4 | 16dp 5 | 6 | -------------------------------------------------------------------------------- /app/src/main/res/values/strings.xml: -------------------------------------------------------------------------------- 1 | 2 | 360InstallToListener 3 | 点我试试看: 4 | NDK项目实战—高仿360手机助手之卸载监听 5 | 6 | -------------------------------------------------------------------------------- /app/src/main/res/values/styles.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 10 | 11 | 12 | -------------------------------------------------------------------------------- /app/src/test/java/unstall/yyh/com/a360installtolistener/ExampleUnitTest.java: -------------------------------------------------------------------------------- 1 | package unstall.yyh.com.a360installtolistener; 2 | 3 | import org.junit.Test; 4 | 5 | import static org.junit.Assert.*; 6 | 7 | /** 8 | * Example local unit test, which will execute on the development machine (host). 9 | * 10 | * @see Testing documentation 11 | */ 12 | public class ExampleUnitTest { 13 | @Test 14 | public void addition_isCorrect() throws Exception { 15 | assertEquals(4, 2 + 2); 16 | } 17 | } -------------------------------------------------------------------------------- /build.gradle: -------------------------------------------------------------------------------- 1 | // Top-level build file where you can add configuration options common to all sub-projects/modules. 2 | 3 | buildscript { 4 | repositories { 5 | jcenter() 6 | } 7 | dependencies { 8 | classpath 'com.android.tools.build:gradle:2.2.2' 9 | 10 | // NOTE: Do not place your application dependencies here; they belong 11 | // in the individual module build.gradle files 12 | } 13 | } 14 | 15 | allprojects { 16 | repositories { 17 | jcenter() 18 | } 19 | } 20 | 21 | task clean(type: Delete) { 22 | delete rootProject.buildDir 23 | } 24 | -------------------------------------------------------------------------------- /gradle.properties: -------------------------------------------------------------------------------- 1 | # Project-wide Gradle settings. 2 | 3 | # IDE (e.g. Android Studio) users: 4 | # Gradle settings configured through the IDE *will override* 5 | # any settings specified in this file. 6 | 7 | # For more details on how to configure your build environment visit 8 | # http://www.gradle.org/docs/current/userguide/build_environment.html 9 | 10 | # Specifies the JVM arguments used for the daemon process. 11 | # The setting is particularly useful for tweaking memory settings. 12 | org.gradle.jvmargs=-Xmx1536m 13 | 14 | # When configured, Gradle will run in incubating parallel mode. 15 | # This option should only be used with decoupled projects. More details, visit 16 | # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects 17 | # org.gradle.parallel=true 18 | -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.jar: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/codeGoogler/360InstallToListener/eb063976b82d06179d4dac696f33a6039cc9bf83/gradle/wrapper/gradle-wrapper.jar -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.properties: -------------------------------------------------------------------------------- 1 | #Mon Dec 28 10:00:20 PST 2015 2 | distributionBase=GRADLE_USER_HOME 3 | distributionPath=wrapper/dists 4 | zipStoreBase=GRADLE_USER_HOME 5 | zipStorePath=wrapper/dists 6 | distributionUrl=https\://services.gradle.org/distributions/gradle-2.14.1-all.zip 7 | -------------------------------------------------------------------------------- /gradlew: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env bash 2 | 3 | ############################################################################## 4 | ## 5 | ## Gradle start up script for UN*X 6 | ## 7 | ############################################################################## 8 | 9 | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 10 | DEFAULT_JVM_OPTS="" 11 | 12 | APP_NAME="Gradle" 13 | APP_BASE_NAME=`basename "$0"` 14 | 15 | # Use the maximum available, or set MAX_FD != -1 to use that value. 16 | MAX_FD="maximum" 17 | 18 | warn ( ) { 19 | echo "$*" 20 | } 21 | 22 | die ( ) { 23 | echo 24 | echo "$*" 25 | echo 26 | exit 1 27 | } 28 | 29 | # OS specific support (must be 'true' or 'false'). 30 | cygwin=false 31 | msys=false 32 | darwin=false 33 | case "`uname`" in 34 | CYGWIN* ) 35 | cygwin=true 36 | ;; 37 | Darwin* ) 38 | darwin=true 39 | ;; 40 | MINGW* ) 41 | msys=true 42 | ;; 43 | esac 44 | 45 | # Attempt to set APP_HOME 46 | # Resolve links: $0 may be a link 47 | PRG="$0" 48 | # Need this for relative symlinks. 49 | while [ -h "$PRG" ] ; do 50 | ls=`ls -ld "$PRG"` 51 | link=`expr "$ls" : '.*-> \(.*\)$'` 52 | if expr "$link" : '/.*' > /dev/null; then 53 | PRG="$link" 54 | else 55 | PRG=`dirname "$PRG"`"/$link" 56 | fi 57 | done 58 | SAVED="`pwd`" 59 | cd "`dirname \"$PRG\"`/" >/dev/null 60 | APP_HOME="`pwd -P`" 61 | cd "$SAVED" >/dev/null 62 | 63 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar 64 | 65 | # Determine the Java command to use to start the JVM. 66 | if [ -n "$JAVA_HOME" ] ; then 67 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then 68 | # IBM's JDK on AIX uses strange locations for the executables 69 | JAVACMD="$JAVA_HOME/jre/sh/java" 70 | else 71 | JAVACMD="$JAVA_HOME/bin/java" 72 | fi 73 | if [ ! -x "$JAVACMD" ] ; then 74 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME 75 | 76 | Please set the JAVA_HOME variable in your environment to match the 77 | location of your Java installation." 78 | fi 79 | else 80 | JAVACMD="java" 81 | which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 82 | 83 | Please set the JAVA_HOME variable in your environment to match the 84 | location of your Java installation." 85 | fi 86 | 87 | # Increase the maximum file descriptors if we can. 88 | if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then 89 | MAX_FD_LIMIT=`ulimit -H -n` 90 | if [ $? -eq 0 ] ; then 91 | if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then 92 | MAX_FD="$MAX_FD_LIMIT" 93 | fi 94 | ulimit -n $MAX_FD 95 | if [ $? -ne 0 ] ; then 96 | warn "Could not set maximum file descriptor limit: $MAX_FD" 97 | fi 98 | else 99 | warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" 100 | fi 101 | fi 102 | 103 | # For Darwin, add options to specify how the application appears in the dock 104 | if $darwin; then 105 | GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" 106 | fi 107 | 108 | # For Cygwin, switch paths to Windows format before running java 109 | if $cygwin ; then 110 | APP_HOME=`cygpath --path --mixed "$APP_HOME"` 111 | CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` 112 | JAVACMD=`cygpath --unix "$JAVACMD"` 113 | 114 | # We build the pattern for arguments to be converted via cygpath 115 | ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` 116 | SEP="" 117 | for dir in $ROOTDIRSRAW ; do 118 | ROOTDIRS="$ROOTDIRS$SEP$dir" 119 | SEP="|" 120 | done 121 | OURCYGPATTERN="(^($ROOTDIRS))" 122 | # Add a user-defined pattern to the cygpath arguments 123 | if [ "$GRADLE_CYGPATTERN" != "" ] ; then 124 | OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" 125 | fi 126 | # Now convert the arguments - kludge to limit ourselves to /bin/sh 127 | i=0 128 | for arg in "$@" ; do 129 | CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` 130 | CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option 131 | 132 | if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition 133 | eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` 134 | else 135 | eval `echo args$i`="\"$arg\"" 136 | fi 137 | i=$((i+1)) 138 | done 139 | case $i in 140 | (0) set -- ;; 141 | (1) set -- "$args0" ;; 142 | (2) set -- "$args0" "$args1" ;; 143 | (3) set -- "$args0" "$args1" "$args2" ;; 144 | (4) set -- "$args0" "$args1" "$args2" "$args3" ;; 145 | (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; 146 | (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; 147 | (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; 148 | (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; 149 | (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; 150 | esac 151 | fi 152 | 153 | # Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules 154 | function splitJvmOpts() { 155 | JVM_OPTS=("$@") 156 | } 157 | eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS 158 | JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME" 159 | 160 | exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@" 161 | -------------------------------------------------------------------------------- /gradlew.bat: -------------------------------------------------------------------------------- 1 | @if "%DEBUG%" == "" @echo off 2 | @rem ########################################################################## 3 | @rem 4 | @rem Gradle startup script for Windows 5 | @rem 6 | @rem ########################################################################## 7 | 8 | @rem Set local scope for the variables with windows NT shell 9 | if "%OS%"=="Windows_NT" setlocal 10 | 11 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 12 | set DEFAULT_JVM_OPTS= 13 | 14 | set DIRNAME=%~dp0 15 | if "%DIRNAME%" == "" set DIRNAME=. 16 | set APP_BASE_NAME=%~n0 17 | set APP_HOME=%DIRNAME% 18 | 19 | @rem Find java.exe 20 | if defined JAVA_HOME goto findJavaFromJavaHome 21 | 22 | set JAVA_EXE=java.exe 23 | %JAVA_EXE% -version >NUL 2>&1 24 | if "%ERRORLEVEL%" == "0" goto init 25 | 26 | echo. 27 | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 28 | echo. 29 | echo Please set the JAVA_HOME variable in your environment to match the 30 | echo location of your Java installation. 31 | 32 | goto fail 33 | 34 | :findJavaFromJavaHome 35 | set JAVA_HOME=%JAVA_HOME:"=% 36 | set JAVA_EXE=%JAVA_HOME%/bin/java.exe 37 | 38 | if exist "%JAVA_EXE%" goto init 39 | 40 | echo. 41 | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 42 | echo. 43 | echo Please set the JAVA_HOME variable in your environment to match the 44 | echo location of your Java installation. 45 | 46 | goto fail 47 | 48 | :init 49 | @rem Get command-line arguments, handling Windowz variants 50 | 51 | if not "%OS%" == "Windows_NT" goto win9xME_args 52 | if "%@eval[2+2]" == "4" goto 4NT_args 53 | 54 | :win9xME_args 55 | @rem Slurp the command line arguments. 56 | set CMD_LINE_ARGS= 57 | set _SKIP=2 58 | 59 | :win9xME_args_slurp 60 | if "x%~1" == "x" goto execute 61 | 62 | set CMD_LINE_ARGS=%* 63 | goto execute 64 | 65 | :4NT_args 66 | @rem Get arguments from the 4NT Shell from JP Software 67 | set CMD_LINE_ARGS=%$ 68 | 69 | :execute 70 | @rem Setup the command line 71 | 72 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar 73 | 74 | @rem Execute Gradle 75 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS% 76 | 77 | :end 78 | @rem End local scope for the variables with windows NT shell 79 | if "%ERRORLEVEL%"=="0" goto mainEnd 80 | 81 | :fail 82 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of 83 | rem the _cmd.exe /c_ return code! 84 | if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 85 | exit /b 1 86 | 87 | :mainEnd 88 | if "%OS%"=="Windows_NT" endlocal 89 | 90 | :omega 91 | -------------------------------------------------------------------------------- /settings.gradle: -------------------------------------------------------------------------------- 1 | include ':app' 2 | --------------------------------------------------------------------------------