[Android]APK程序卸载二次确认的实现

严正声明

       本人本着技术开放,思想分享的目的,撰写本文。文章仅供参考之用,请勿使之于非法或有害于社会和谐之用。

Sodino

2011-01-24

Android上能不能实现卸载时提示呢,比如卸载某某软件时,做个用户调查卸载的原因。 我以前想着是的不行的,以前的想法是: Windows上卸载时能实现此功能是因为有些程序的卸载是自己实现的,非系统操作。 但android上目前来说还不支持,系统卸载时,还没发现有啥接口可以和目标卸载程序交互。
呵呵,今天鼓捣LogCat,发现还是可以的。 实现基础是: 1.通过注册权限,能够获取LogCat的输出流的输出信息。 2.进入系统的卸载界面时,"打包安装程序(com.android.packageinstaller)"会输出如下信息
01-22 16:29:15.250: INFO/ActivityManager(147): Starting activity: Intent { act=android.intent.action.DELETE dat=package:lab.sodino.uninstallhint cmp=com.android.packageinstaller/.UninstallerActivity }
好了,有这句话就足够了。截取输出流信息,当获取字符串中包含"android.intent.action.DELETE"和"<you_package>"时,就启动卸载提示页面。
话就这么多了。接下来看效果图,上代码。

[java] view plaincopyprint?

  1. package lab.sodino.uninstallhint;
  2. import android.app.Activity;
  3. import android.content.Intent;
  4. import android.os.Bundle;
  5. import android.os.Handler;
  6. import android.os.Message;
  7. import android.view.View;
  8. import android.widget.Button;
  9. import android.widget.TextView;
  10. /**
  11. * @author Sodino E-mail:[email protected]
  12. * @version Time:2011-1-12 上午10:09:59
  13. */
  14. public class MainActivity extends Activity implements LogcatObserver {
  15. private TextView txtInfo;
  16. private Handler handler;
  17. public void onCreate(Bundle savedInstanceState) {
  18. super.onCreate(savedInstanceState);
  19. setContentView(R.layout.main);
  20. Button btnScannerLogcat = (Button) findViewById(R.id.btnScanLogcat);
  21. btnScannerLogcat.setOnClickListener(new Button.OnClickListener() {
  22. public void onClick(View view) {
  23. // 开启Logcat流监听
  24. LogcatScanner.startScanLogcatInfo(MainActivity.this);
  25. }
  26. });
  27. Button btnUninstallMe = (Button) findViewById(R.id.btnUninstallMe);
  28. btnUninstallMe.setOnClickListener(new Button.OnClickListener() {
  29. public void onClick(View view) {
  30. // 调用应用程序信息
  31. Intent intent = new Intent(Intent.ACTION_VIEW);
  32. // com.android.settings/.InstalledAppDetails
  33. intent.setClassName("com.android.settings", "com.android.settings.InstalledAppDetails");
  34. intent.putExtra("pkg", "lab.sodino.uninstallhint");
  35. startActivity(intent);
  36. }
  37. });
  38. txtInfo = (TextView) findViewById(R.id.txtInfo);
  39. handler = new Handler() {
  40. public void handleMessage(Message msg) {
  41. txtInfo.append(String.valueOf(msg.obj) + "/n");
  42. }
  43. };
  44. }
  45. public void handleNewLine(String info) {
  46. Message msg = new Message();
  47. msg.obj = info;
  48. handler.sendMessage(msg);
  49. if (info.contains("android.intent.action.DELETE") && info.contains(getPackageName())) {
  50. // 启动删除提示
  51. Intent intent = new Intent();
  52. intent.setClass(this, UninstallWarningActivity.class);
  53. intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
  54. startActivity(intent);
  55. }
  56. }
  57. }

package lab.sodino.uninstallhint;
import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;
/**
* @author Sodino E-mail:sodinoopen@hotmail.com
* @version Time:2011-1-12 上午10:09:59
*/
public class MainActivity extends Activity implements LogcatObserver {
private TextView txtInfo;
private Handler handler;
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
Button btnScannerLogcat = (Button) findViewById(R.id.btnScanLogcat);
btnScannerLogcat.setOnClickListener(new Button.OnClickListener() {
public void onClick(View view) {
// 开启Logcat流监听
LogcatScanner.startScanLogcatInfo(MainActivity.this);
}
});
Button btnUninstallMe = (Button) findViewById(R.id.btnUninstallMe);
btnUninstallMe.setOnClickListener(new Button.OnClickListener() {
public void onClick(View view) {
// 调用应用程序信息
Intent intent = new Intent(Intent.ACTION_VIEW);
// com.android.settings/.InstalledAppDetails
intent.setClassName("com.android.settings", "com.android.settings.InstalledAppDetails");
intent.putExtra("pkg", "lab.sodino.uninstallhint");
startActivity(intent);
}
});
txtInfo = (TextView) findViewById(R.id.txtInfo);
handler = new Handler() {
public void handleMessage(Message msg) {
txtInfo.append(String.valueOf(msg.obj) + "/n");
}
};
}
public void handleNewLine(String info) {
Message msg = new Message();
msg.obj = info;
handler.sendMessage(msg);
if (info.contains("android.intent.action.DELETE") && info.contains(getPackageName())) {
// 启动删除提示
Intent intent = new Intent();
intent.setClass(this, UninstallWarningActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(intent);
}
}
}

[java] view plaincopyprint?

  1. package lab.sodino.uninstallhint;
  2. import java.io.DataInputStream;
  3. import java.io.IOException;
  4. import java.io.InputStream;
  5. /**
  6. *@author Sodino Email:[email protected]<br/>
  7. *@version 2011-1-22 上午11:10:56
  8. */
  9. public class LogcatScanner {
  10. private static AndroidLogcatScanner scannerThead;
  11. public final static void startScanLogcatInfo(LogcatObserver observer) {
  12. if (scannerThead == null) {
  13. scannerThead = new AndroidLogcatScanner(observer);
  14. scannerThead.start();
  15. LogOut.out(LogcatScanner.class, "scannerThread.start()");
  16. }
  17. }
  18. static class AndroidLogcatScanner extends Thread {
  19. private LogcatObserver observer;
  20. public AndroidLogcatScanner(LogcatObserver observer) {
  21. this.observer = observer;
  22. }
  23. public void run() {
  24. String[] cmds = { "logcat", "-c" };
  25. String shellCmd = "logcat";
  26. Process process = null;
  27. InputStream is = null;
  28. DataInputStream dis = null;
  29. String line = "";
  30. Runtime runtime = Runtime.getRuntime();
  31. try {
  32. observer.handleNewLine(line);
  33. int waitValue;
  34. waitValue = runtime.exec(cmds).waitFor();
  35. observer.handleNewLine("waitValue=" + waitValue + "/n Has do Clear logcat cache.");
  36. process = runtime.exec(shellCmd);
  37. is = process.getInputStream();
  38. dis = new DataInputStream(is);
  39. while ((line = dis.readLine()) != null) {
  40. observer.handleNewLine(line);
  41. }
  42. } catch (InterruptedException e) {
  43. e.printStackTrace();
  44. } catch (IOException ie) {
  45. ie.printStackTrace();
  46. } finally {
  47. try {
  48. if (dis != null) {
  49. dis.close();
  50. }
  51. if (is != null) {
  52. is.close();
  53. }
  54. if (process != null) {
  55. process.destroy();
  56. }
  57. } catch (Exception e) {
  58. e.printStackTrace();
  59. }
  60. }
  61. }
  62. }
  63. }

package lab.sodino.uninstallhint;
import java.io.DataInputStream;
import java.io.IOException;
import java.io.InputStream;
/**
*@author Sodino Email:sodinoopen@hotmail<br/>
*@version 2011-1-22 上午11:10:56
*/
public class LogcatScanner {
private static AndroidLogcatScanner scannerThead;
public final static void startScanLogcatInfo(LogcatObserver observer) {
if (scannerThead == null) {
scannerThead = new AndroidLogcatScanner(observer);
scannerThead.start();
LogOut.out(LogcatScanner.class, "scannerThread.start()");
}
}
static class AndroidLogcatScanner extends Thread {
private LogcatObserver observer;
public AndroidLogcatScanner(LogcatObserver observer) {
this.observer = observer;
}
public void run() {
String[] cmds = { "logcat", "-c" };
String shellCmd = "logcat";
Process process = null;
InputStream is = null;
DataInputStream dis = null;
String line = "";
Runtime runtime = Runtime.getRuntime();
try {
observer.handleNewLine(line);
int waitValue;
waitValue = runtime.exec(cmds).waitFor();
observer.handleNewLine("waitValue=" + waitValue + "/n Has do Clear logcat cache.");
process = runtime.exec(shellCmd);
is = process.getInputStream();
dis = new DataInputStream(is);
while ((line = dis.readLine()) != null) {
observer.handleNewLine(line);
}
} catch (InterruptedException e) {
e.printStackTrace();
} catch (IOException ie) {
ie.printStackTrace();
} finally {
try {
if (dis != null) {
dis.close();
}
if (is != null) {
is.close();
}
if (process != null) {
process.destroy();
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
}
}

[java] view plaincopyprint?

  1. package lab.sodino.uninstallhint;
  2. /**
  3. * @author Sodino E-mail:[email protected]
  4. * @version Time:2011-1-22 下午03:00:54
  5. */
  6. public interface LogcatObserver {
  7. public void handleNewLine(String line);
  8. }

package lab.sodino.uninstallhint;
/**
* @author Sodino E-mail:sodinoopen@hotmail.com
* @version Time:2011-1-22 下午03:00:54
*/
public interface LogcatObserver {
public void handleNewLine(String line);
}

[java] view plaincopyprint?

  1. package lab.sodino.uninstallhint;
  2. import android.app.Activity;
  3. import android.app.ActivityManager;
  4. import android.content.Context;
  5. import android.content.Intent;
  6. import android.os.Bundle;
  7. import android.view.View;
  8. import android.widget.Button;
  9. /**
  10. * @author Sodino E-mail:[email protected]
  11. * @version Time:2011-1-12 上午10:26:09
  12. */
  13. public class UninstallWarningActivity extends Activity {
  14. public void onCreate(Bundle savedInstanceState) {
  15. super.onCreate(savedInstanceState);
  16. setContentView(R.layout.uninstall_warning);
  17. Button btnContinue = (Button) findViewById(R.id.btnContinue);
  18. btnContinue.setOnClickListener(new Button.OnClickListener() {
  19. public void onClick(View view) {
  20. UninstallWarningActivity.this.finish();
  21. }
  22. });
  23. Button btnCancel = (Button) findViewById(R.id.btnCancel);
  24. btnCancel.setOnClickListener(new Button.OnClickListener() {
  25. public void onClick(View view) {
  26. UninstallWarningActivity.this.finish();
  27. ActivityManager actMag = (ActivityManager) UninstallWarningActivity.this
  28. .getSystemService(Context.ACTIVITY_SERVICE);
  29. //杀掉系统的打包安装程序。
  30. if (android.os.Build.VERSION.SDK_INT < 8) {
  31. actMag.restartPackage("com.android.packageinstaller");
  32. } else {
  33. actMag.killBackgroundProcesses("com.android.packageinstaller");
  34. }
  35. Intent i = new Intent(Intent.ACTION_MAIN);
  36. i.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
  37. i.addCategory(Intent.CATEGORY_HOME);
  38. startActivity(i);
  39. }
  40. });
  41. }
  42. }

package lab.sodino.uninstallhint;
import android.app.Activity;
import android.app.ActivityManager;
import android.content.Context;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
/**
* @author Sodino E-mail:sodinoopen@hotmail.com
* @version Time:2011-1-12 上午10:26:09
*/
public class UninstallWarningActivity extends Activity {
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.uninstall_warning);
Button btnContinue = (Button) findViewById(R.id.btnContinue);
btnContinue.setOnClickListener(new Button.OnClickListener() {
public void onClick(View view) {
UninstallWarningActivity.this.finish();
}
});
Button btnCancel = (Button) findViewById(R.id.btnCancel);
btnCancel.setOnClickListener(new Button.OnClickListener() {
public void onClick(View view) {
UninstallWarningActivity.this.finish();
ActivityManager actMag = (ActivityManager) UninstallWarningActivity.this
.getSystemService(Context.ACTIVITY_SERVICE);
//杀掉系统的打包安装程序。
if (android.os.Build.VERSION.SDK_INT < 8) {
actMag.restartPackage("com.android.packageinstaller");
} else {
actMag.killBackgroundProcesses("com.android.packageinstaller");
}
Intent i = new Intent(Intent.ACTION_MAIN);
i.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
i.addCategory(Intent.CATEGORY_HOME);
startActivity(i);
}
});
}
}

最后在AndroidManifest.xml中添加上权限。

[xhtml] view plaincopyprint?

  1. <uses-permission android:name="android.permission.READ_LOGS" />
  2. <uses-permission android:name="android.permission.KILL_BACKGROUND_PROCESSES"/>
  3. <uses-permission android:name="android.permission.RESTART_PACKAGES"/>
时间: 2024-11-04 01:52:20

[Android]APK程序卸载二次确认的实现的相关文章

Android apk程序调用其它的APK程序

Intent mIntent = new Intent(); ComponentName comp = new ComponentName("启动的APK包名","启动的APK入口类(包名加类名)"); mIntent.setComponent(comp); mIntent.setAction("android.intent.action.MAIN"); startActivity(mIntent);

Android apk程序调用其他的APK程序

Intent mIntent = new Intent(); ComponentName comp = new ComponentName("启动的APK包名","启动的APK入口类(包名加类名)"); mIntent.setComponent(comp); mIntent.setAction("android.intent.action.MAIN"); startActivity(mIntent);

Android apk的安装、卸载、更新升级(通过Eclipse实现静默安装)

一.通过Intent消息机制发送消息,调用系统应用进行,实现apk的安装/卸载 . (1) 调用系统的安装应用,让系统自动进行apk的安装 String fileName = "/data/data/com.zlc.ipanel.operate/FileOperate.apk";  Uri uri = Uri.fromFile(new File(fileName));  Intent intent = new Intent(Intent.ACTION_VIEW);  intent.se

反编译Android APK及防止APK程序被反编译

原文出处 反编译Android APK及防止APK程序被反编译 怎么逆向工程对Android Apk 进行反编译 google Android开发是开源的,开发过程中有些时候会遇到一些功能,自己不知道该怎么做,然而别的软件里面已经有了,这个时候可以采用反编译的方式,解开其他的程序,来了解一些它 的做法,同时啊,还可以借鉴别人的软件结构,资源文件:作为一个开发者,你可能会很想知道这些效果界面是怎么去实现的,这时,你便可以对改应用的APK进行反编译查看.此方式主要目的是为了促进开发者学习,借鉴好的代

Android apk应用程序签名

Android apk应用程序签名 分类: Android 2012-11-25 19:33 570人阅读 评论(0) 收藏 举报 一.Android Apk签名 Apk签名首先要有一个keystore的签名用的文件. keystore是由jdk自带的工具keytool生成的. 具体生成方式参考: 开始->运行->cmd->cd到你安装的jdk的目录里,我的是C:/Program Files/Java/jdk1.6.0_10/bin 然后输入: keytool -genkey -alia

android 5.0之二 第一个程序

一:四大核心组件 Activity(活动) 是Android应用中负责与用户交互的组件 Service(服务) Service通常位于后台运行,它一般不需要与用户交互,因此Service组件没有图形用户界面 BroadcastReceiver(广播接收器) 用于接收系统中其它组件发送的广播 ContentProvider(内容提供者) 多个应用程序之间进行实时的数据交换的前提 二:工程目录结构 Android按照用途将不同的内容分别存放在不同的目录中 src文件夹 存放开发者编写的源代码 存放导

密西西比河谷州立大学:Android应用程序开发(二)

第二讲 Hello World 密西西比河谷州立大学:Android应用程序开发(二),码迷,mamicode.com

Android插件化(二):使用DexClassLoader动态加载assets中的apk

Android插件化(二):使用DexClassLoader动态加载assets中的apk 简介 上一篇博客讲到,我们可以使用MultiDex.java加载离线的apk文件.需要注意的是,apk中的类是加载到当前的PathClassLoader当中的,如果apk文件过多,可能会出现ANR的情况.那么,我们能不能使用DexClassLoader加载apk呢?当然是可以的!首先看一下Doc文档. A class loader that loads classes from .jar and .apk

Android apk动态加载机制的研究(二):资源加载和activity生命周期管理

出处:http://blog.csdn.net/singwhatiwanna/article/details/23387079 (来自singwhatiwanna的csdn博客) 前言 为了更好地阅读本文,你需要先阅读Android apk动态加载机制的研究这篇文章,在此文中,博主分析了Android中apk的动态加载机制,并在文章的最后指出需要解决的两个复杂问题:资源的访问和activity生命周期的管理,而本文将会分析这两个复杂问题的解决方法.需要说明的一点是,我们不可能调起任何一个未安装的