Android项目实战--手机卫士35--清除程序缓存

2024-06-09 15:48

本文主要是介绍Android项目实战--手机卫士35--清除程序缓存,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!



最新实战教程,让你了解Android自动化刷量、作弊与防作弊的那些事,案例:刷友盟统计、批量注册苹果帐号




因为一直忙着工作的事,也很久没有更新我们这个项目了,今天就给大家讲一下那个清除缓存的功能,其实清除缓存是有两种的,一种是清除手机rom里面的缓存,一种是清除手机sd卡里面的缓存,我们今天主要讲的就是第一种

ps:这里来一个知识扫盲,就是手机里面的rom和ram啦,如果已经知道了的,就可以跳过啦,我们去买手机,有时候经常会被那些销售人员忽悠的,说什么8G的内存啦,什么的,其实他这里面说的极大可能就是你手机里面rom的大小啦,rom就是read only menory(只读存储器)你可以把它当成你电脑上的硬盘,不过它只能读取而已,ram就是random access menory(随机存取器)这个就相当于你电脑的内存啦,所以那个销售人员说的手机内存有多大的时候,我们一定要问清楚啦,不要被人蒙了


好啦,回归正题,我们今天讲的那个缓存是清除手机rom里面的缓存,其实也是挺简单的,只要知道了要怎么做之后,至于那个清除sd卡里面的缓存的话,这个我到时会给大家说一下是怎样清理的,具体我就不写了,好,先来看一下我们要做的效果

     


上的第一个图就是我们把我们的应用的一些信息给读取出来了,但那个界面不怎么好看,大家可以自己优化一下,当我们点击了要清理缓存的条目时,我们就会进入到系统设置里面的一个界面,因为清理rom里面的缓存是要root才行的,而我们没有root,那么就只要借助系统里面的功能啦,到时我也会教大家怎样root手机的

好啦,废话不多说,我们直接进入代码阶段

首先,我们先新建一个model类用来存放缓存的信息

com.xiaobin.security.domain.CacheInfo

package com.xiaobin.security.domain;import android.graphics.drawable.Drawable;public class CacheInfo
{private String name;private String packageName;private Drawable icon;//应用大小private String codeSize;//数据大小private String dataSize;//缓存大小private String cacheSize;public String getName(){return name;}public void setName(String name){this.name = name;}public String getPackageName(){return packageName;}public void setPackageName(String packageName){this.packageName = packageName;}public Drawable getIcon(){return icon;}public void setIcon(Drawable icon){this.icon = icon;}public String getCodeSize(){return codeSize;}public void setCodeSize(String codeSize){this.codeSize = codeSize;}public String getDataSize(){return dataSize;}public void setDataSize(String dataSize){this.dataSize = dataSize;}public String getCacheSize(){return cacheSize;}public void setCacheSize(String cacheSize){this.cacheSize = cacheSize;}}

写完model类之后呢,我们就要把我们的应用的信息读取出来啦,比如读取应用大小啊,缓存大小啊,数据大小啊,这几个信息,

那么,怎样去读取这些信息呢,其实PackageManagerj里面有一个私有的方法的,叫getPackageSizeInfo,

所以现在思路清楚啦,我们就是要通过反射来调用这个方法,然后,getPackageSizeInfo这个方法要传递一个IPackageStatsObserver.Stub对象的,这个对象我们一看,就知道是一个AIDL的对象啦,所以,我们就要把相应的AIDL文件给放到src目录下面啦,我们就是在它里面拿到缓存大小,这 些数据的啦,

好啦,说得有点玄,直接上代码更清晰

	/*** 通过AIDL的方法来获取到应用的缓存信息,getPackageSizeInfo是PackageManager里面的一个私有方法来的* 我们通过反射就可以调用到它的了,但是这个方法里面会传递一个IPackageStatsObserver.Stub的对象* 里面就可能通过AIDL来获取我们想要的信息了* * 因为这样的调用是异步的,所以当我们完成获取完这些信息之后,我们就通过handler来发送一个消息* 来通知我们的应用,通过getCacheInfos来获取到我们的Vector* * 为什么要用Vector呢,因为下面的方法是异步的,也就是有可能是多线程操作,所以我们就用了线程安全的Vector* * @param cacheInfo* @param position*/private void initDataSize(final CacheInfo cacheInfo, final int position){try{Method method = PackageManager.class.getMethod("getPackageSizeInfo", new Class[] { String.class,IPackageStatsObserver.class });method.invoke(packageManager,new Object[] { cacheInfo.getPackageName(),new IPackageStatsObserver.Stub(){@Overridepublic void onGetStatsCompleted(PackageStats pStats, boolean succeeded)throws RemoteException{System.out.println("onGetStatsCompleted" + position);long cacheSize = pStats.cacheSize;long codeSize = pStats.codeSize;long dataSize = pStats.dataSize;cacheInfo.setCacheSize(TextFormater.dataSizeFormat(cacheSize));cacheInfo.setCodeSize(TextFormater.dataSizeFormat(codeSize));cacheInfo.setDataSize(TextFormater.dataSizeFormat(dataSize));cacheInfos.add(cacheInfo);if (position == (size - 1)){// 当完全获取完信息之后,发送一个成功的消息// 1对应的就是CacheClearActivity里面的FINISHhandler.sendEmptyMessage(1);}}} });}catch (Exception e){e.printStackTrace();}}

有一点是要注意的,获取缓存大小这些信息的内部实现是异步的,所以我们用一个vector来存放信息,免得会出来一些特殊的情况


好啦,下面我们把完整的类粘出来,这个类写得有点复杂,主要是因为获取缓存大小这些信息的内部实现是异步的,我们要保证数据的正确性,所以可能就写得有点难理解

大家如果看不懂这个类的话,就欢迎留言

com.xiaobin.security.engine.CacheInfoProvider

package com.xiaobin.security.engine;import java.lang.reflect.Method;
import java.util.List;
import java.util.Vector;import android.content.Context;
import android.content.pm.ApplicationInfo;
import android.content.pm.IPackageStatsObserver;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.content.pm.PackageStats;
import android.graphics.drawable.Drawable;
import android.os.Handler;
import android.os.RemoteException;import com.xiaobin.security.domain.CacheInfo;
import com.xiaobin.security.utils.TextFormater;public class CacheInfoProvider
{private Handler handler;private PackageManager packageManager;private Vector<CacheInfo> cacheInfos;private int size = 0;public CacheInfoProvider(Handler handler, Context context){// 拿到一个包管理器packageManager = context.getPackageManager();this.handler = handler;cacheInfos = new Vector<CacheInfo>();}public void initCacheInfos(){// 获取到所有安装了的应用程序的信息,包括那些卸载了的,但没有清除数据的应用程序List<PackageInfo> packageInfos = packageManager.getInstalledPackages(PackageManager.GET_UNINSTALLED_PACKAGES);size = packageInfos.size();for (int i = 0; i < size; i++){PackageInfo packageInfo = packageInfos.get(i);CacheInfo cacheInfo = new CacheInfo();// 拿到包名String packageName = packageInfo.packageName;cacheInfo.setPackageName(packageName);// 拿到应用程序的信息ApplicationInfo applicationInfo = packageInfo.applicationInfo;// 拿到应用程序的程序名String name = applicationInfo.loadLabel(packageManager).toString();cacheInfo.setName(name);// 拿到应用程序的图标Drawable icon = applicationInfo.loadIcon(packageManager);cacheInfo.setIcon(icon);initDataSize(cacheInfo, i);}}/*** 通过AIDL的方法来获取到应用的缓存信息,getPackageSizeInfo是PackageManager里面的一个私有方法来的* 我们通过反射就可以调用到它的了,但是这个方法里面会传递一个IPackageStatsObserver.Stub的对象* 里面就可能通过AIDL来获取我们想要的信息了* * 因为这样的调用是异步的,所以当我们完成获取完这些信息之后,我们就通过handler来发送一个消息* 来通知我们的应用,通过getCacheInfos来获取到我们的Vector* * 为什么要用Vector呢,因为下面的方法是异步的,也就是有可能是多线程操作,所以我们就用了线程安全的Vector* * @param cacheInfo* @param position*/private void initDataSize(final CacheInfo cacheInfo, final int position){try{Method method = PackageManager.class.getMethod("getPackageSizeInfo", new Class[] { String.class,IPackageStatsObserver.class });method.invoke(packageManager,new Object[] { cacheInfo.getPackageName(),new IPackageStatsObserver.Stub(){@Overridepublic void onGetStatsCompleted(PackageStats pStats, boolean succeeded)throws RemoteException{System.out.println("onGetStatsCompleted" + position);long cacheSize = pStats.cacheSize;long codeSize = pStats.codeSize;long dataSize = pStats.dataSize;cacheInfo.setCacheSize(TextFormater.dataSizeFormat(cacheSize));cacheInfo.setCodeSize(TextFormater.dataSizeFormat(codeSize));cacheInfo.setDataSize(TextFormater.dataSizeFormat(dataSize));cacheInfos.add(cacheInfo);if (position == (size - 1)){// 当完全获取完信息之后,发送一个成功的消息// 1对应的就是CacheClearActivity里面的FINISHhandler.sendEmptyMessage(1);}}} });}catch (Exception e){e.printStackTrace();}}public Vector<CacheInfo> getCacheInfos(){return cacheInfos;}public void setCacheInfos(Vector<CacheInfo> cacheInfos){this.cacheInfos = cacheInfos;}}

其实主要的思想是这们的,当完全获取完所有应用的缓存大小这些信息的时候,我们就给activity发送一个消息,然后activity里面的handler处理这个消息,然后就通过getCacheInfos这个方法,拿到所有已经填充好信息的CacheInfos对象的集合啦,这样,我们就可以回到activity里面填充数据啦


上面我已经说过了,如果我们要自己删除rom的缓存的话,那就是要root权限的,但我们现在没有,所以,我们就要通过系统设置里面的一个功能来进行清除啦

				/*** Android2.3打开settings里面的那个应用的详细界面* 后来我又查了一个Android4.1的,也是这样写的,所有应该是2.3之后,都是这样写的了,* 但是这只是猜测,各位有空的可以去下载Android Settings的代码看一下* 这样就可以做成多个版本的适配了* <intent-filter> * <action android:name="android.settings.APPLICATION_DETAILS_SETTINGS" /> * <category android:name="android.intent.category.DEFAULT" /> * <data android:scheme="package" /> * </intent-filter>*//*** Android2.2打开settings里面的那个应用的详细界面* 用这个版本来打开的话,就要加多一句把包名设置进去的* intent.putExtra("pkg", packageName);* <intent-filter> * <action android:name="android.intent.action.VIEW" /> * <category android:name="android.intent.category.DEFAULT" /> * <category android:name="android.intent.category.VOICE_LAUNCH" />* </intent-filter>*/Intent intent = new Intent();intent.setAction("android.settings.APPLICATION_DETAILS_SETTINGS");intent.addCategory("android.intent.category.DEFAULT");intent.setData(Uri.parse("package:" + cacheInfos.get(position).getPackageName()));startActivity(intent);

上面的注释已经很详细了,我就不多说啦


下面是完整的activity代码,布局文件就不放啦,因为写得太难看了

com.xiaobin.security.ui.CacheClearActivity

package com.xiaobin.security.ui;import java.util.Vector;import android.annotation.SuppressLint;
import android.app.ListActivity;
import android.content.Intent;
import android.net.Uri;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.view.View;
import android.view.ViewGroup;
import android.view.Window;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.BaseAdapter;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.ListView;
import android.widget.TextView;import com.xiaobin.security.R;
import com.xiaobin.security.domain.CacheInfo;
import com.xiaobin.security.engine.CacheInfoProvider;public class CacheClearActivity extends ListActivity
{private static final int LOADING = 0;private static final int FINISH = 1;private CacheInfoProvider provider;private ListView lv_list;private LinearLayout ll_load;private Vector<CacheInfo> cacheInfos;@SuppressLint("HandlerLeak")private Handler handler = new Handler(){public void handleMessage(Message msg){switch (msg.what){case LOADING:ll_load.setVisibility(View.VISIBLE);break;case FINISH:ll_load.setVisibility(View.INVISIBLE);// 当加载完成之后,就调用provider里面的get方法,// 这样就可以得到一个加载完成后的数据了cacheInfos = provider.getCacheInfos();lv_list.setAdapter(new CacheAdapter());break;default:break;}}};@Overrideprotected void onCreate(Bundle savedInstanceState){super.onCreate(savedInstanceState);requestWindowFeature(Window.FEATURE_NO_TITLE);setContentView(R.layout.cache_clear);provider = new CacheInfoProvider(handler, this);lv_list = getListView();ll_load = (LinearLayout) findViewById(R.id.ll_cache_clear_load);lv_list.setOnItemClickListener(new OnItemClickListener(){@Overridepublic void onItemClick(AdapterView<?> parent, View view,int position, long id){/*** Android2.3打开settings里面的那个应用的详细界面* 后来我又查了一个Android4.1的,也是这样写的,所有应该是2.3之后,都是这样写的了,* 但是这只是猜测,各位有空的可以去下载Android Settings的代码看一下* 这样就可以做成多个版本的适配了* <intent-filter> * <action android:name="android.settings.APPLICATION_DETAILS_SETTINGS" /> * <category android:name="android.intent.category.DEFAULT" /> * <data android:scheme="package" /> * </intent-filter>*//*** Android2.2打开settings里面的那个应用的详细界面* 用这个版本来打开的话,就要加多一句把包名设置进去的* intent.putExtra("pkg", packageName);* <intent-filter> * <action android:name="android.intent.action.VIEW" /> * <category android:name="android.intent.category.DEFAULT" /> * <category android:name="android.intent.category.VOICE_LAUNCH" />* </intent-filter>*/Intent intent = new Intent();intent.setAction("android.settings.APPLICATION_DETAILS_SETTINGS");intent.addCategory("android.intent.category.DEFAULT");intent.setData(Uri.parse("package:" + cacheInfos.get(position).getPackageName()));startActivity(intent);}});loadData();}private void loadData(){ll_load.setVisibility(View.VISIBLE);new Thread(new Runnable(){@Overridepublic void run(){provider.initCacheInfos();}}).start();}// =======================================================================private class CacheAdapter extends BaseAdapter{@Overridepublic int getCount(){return cacheInfos.size();}@Overridepublic Object getItem(int position){return cacheInfos.get(position);}@Overridepublic long getItemId(int position){return position;}@Overridepublic View getView(int position, View convertView, ViewGroup parent){View view;ViewHolder holder;CacheInfo info = cacheInfos.get(position);if (convertView == null){view = View.inflate(CacheClearActivity.this,R.layout.cache_clear_item, null);holder = new ViewHolder();holder.iv_icon = (ImageView) view.findViewById(R.id.iv_cache_icon);holder.tv_name = (TextView) view.findViewById(R.id.tv_cache_name);holder.tv_code = (TextView) view.findViewById(R.id.tv_cache_code);holder.tv_data = (TextView) view.findViewById(R.id.tv_cache_data);holder.tv_cache = (TextView) view.findViewById(R.id.tv_cache_cache);view.setTag(holder);}else{view = convertView;holder = (ViewHolder) view.getTag();}holder.iv_icon.setImageDrawable(info.getIcon());holder.tv_name.setText(info.getName());holder.tv_code.setText("应用大小:" + info.getCodeSize());holder.tv_data.setText("数据大小:" + info.getDataSize());holder.tv_cache.setText("缓存大小:" + info.getCacheSize());return view;}}private class ViewHolder{ImageView iv_icon;TextView tv_name;TextView tv_cache;TextView tv_code;TextView tv_data;}}

好啦,到这里为止,我们的清除rom里面的缓存,就已经是完成的啦,最后,我给大家说一下清除sd卡里面的缓存是怎样的

大家都知道,我们装的大部分应用,都会有sd卡里面建一个目录,然后装一些应用的信息的,而这些,就是这些应用对应存放缓存的目录啦,

要清除sd卡里面的缓存,其实就是要有一个数据库啦,它专门收录市面上一些常用的应用,在sd卡上建立的目录名称,然后再通过这个数据库,然后对sd卡里面的目录清除的而已,这个方法是比较的麻烦的,还要有数据库的支持才行,所以大家可以自己试试


最后,和大家说一下

为了方便大家的交流,我创建了一个群,这样子大家有什么疑问也可以在群上交流

群号是298440981


今天源码下载

这篇关于Android项目实战--手机卫士35--清除程序缓存的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1045615

相关文章

vite搭建vue3项目的搭建步骤

《vite搭建vue3项目的搭建步骤》本文主要介绍了vite搭建vue3项目的搭建步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学... 目录1.确保Nodejs环境2.使用vite-cli工具3.进入项目安装依赖1.确保Nodejs环境

Python版本信息获取方法详解与实战

《Python版本信息获取方法详解与实战》在Python开发中,获取Python版本号是调试、兼容性检查和版本控制的重要基础操作,本文详细介绍了如何使用sys和platform模块获取Python的主... 目录1. python版本号获取基础2. 使用sys模块获取版本信息2.1 sys模块概述2.1.1

MyBatis延迟加载与多级缓存全解析

《MyBatis延迟加载与多级缓存全解析》文章介绍MyBatis的延迟加载与多级缓存机制,延迟加载按需加载关联数据提升性能,一级缓存会话级默认开启,二级缓存工厂级支持跨会话共享,增删改操作会清空对应缓... 目录MyBATis延迟加载策略一对多示例一对多示例MyBatis框架的缓存一级缓存二级缓存MyBat

前端缓存策略的自解方案全解析

《前端缓存策略的自解方案全解析》缓存从来都是前端的一个痛点,很多前端搞不清楚缓存到底是何物,:本文主要介绍前端缓存的自解方案,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录一、为什么“清缓存”成了技术圈的梗二、先给缓存“把个脉”:浏览器到底缓存了谁?三、设计思路:把“发版”做成“自愈”四、代码

idea+spring boot创建项目的搭建全过程

《idea+springboot创建项目的搭建全过程》SpringBoot是Spring社区发布的一个开源项目,旨在帮助开发者快速并且更简单的构建项目,:本文主要介绍idea+springb... 目录一.idea四种搭建方式1.Javaidea命名规范2JavaWebTomcat的安装一.明确tomcat

pycharm跑python项目易出错的问题总结

《pycharm跑python项目易出错的问题总结》:本文主要介绍pycharm跑python项目易出错问题的相关资料,当你在PyCharm中运行Python程序时遇到报错,可以按照以下步骤进行排... 1. 一定不要在pycharm终端里面创建环境安装别人的项目子模块等,有可能出现的问题就是你不报错都安装

Python爬虫HTTPS使用requests,httpx,aiohttp实战中的证书异步等问题

《Python爬虫HTTPS使用requests,httpx,aiohttp实战中的证书异步等问题》在爬虫工程里,“HTTPS”是绕不开的话题,HTTPS为传输加密提供保护,同时也给爬虫带来证书校验、... 目录一、核心问题与优先级检查(先问三件事)二、基础示例:requests 与证书处理三、高并发选型:

java程序远程debug原理与配置全过程

《java程序远程debug原理与配置全过程》文章介绍了Java远程调试的JPDA体系,包含JVMTI监控JVM、JDWP传输调试命令、JDI提供调试接口,通过-Xdebug、-Xrunjdwp参数配... 目录背景组成模块间联系IBM对三个模块的详细介绍编程使用总结背景日常工作中,每个程序员都会遇到bu

uni-app小程序项目中实现前端图片压缩实现方式(附详细代码)

《uni-app小程序项目中实现前端图片压缩实现方式(附详细代码)》在uni-app开发中,文件上传和图片处理是很常见的需求,但也经常会遇到各种问题,下面:本文主要介绍uni-app小程序项目中实... 目录方式一:使用<canvas>实现图片压缩(推荐,兼容性好)示例代码(小程序平台):方式二:使用uni

MyCat分库分表的项目实践

《MyCat分库分表的项目实践》分库分表解决大数据量和高并发性能瓶颈,MyCat作为中间件支持分片、读写分离与事务处理,本文就来介绍一下MyCat分库分表的实践,感兴趣的可以了解一下... 目录一、为什么要分库分表?二、分库分表的常见方案三、MyCat简介四、MyCat分库分表深度解析1. 架构原理2. 分