这种作假的东西其实我是很反感的,有些无良客户居然提要求把1G的改成2G的,2G的改成4G的,
导致我现在都不相信现在手机显示出来的数据了,但是没办法,为了能混口饭吃,唉...
本文提供2种方法,并已Ram Rom为例
1.在TextView.java(frameworks/base/core/java/android/widget/TextView.java),因为三方软件我们没办法直接修改,但是从显示上来看肯定是textview,因而我们可以直接在SetText()的方法中去修改
private void setText(CharSequence text, BufferType type,
boolean notifyBefore, int oldlen) {
mTextSetFromXmlOrResourceId = false;
if (text == null) {
text = "";
}
String packageName = getContext().getPackageName();
if (packageName.equals("com.antutu.ABenchMark")
|| packageName.equals("com.cpuid.cpu_z")
|| packageName.equals("com.ludashi.benchmark")) {
//这两个是我们在frameworks/base/core/res/res/valuess/configs.xml中写死的值
String fakeRamSize = getContext().getResources().getString(com.android.internal.R.string.custom_ram_size);
String fakeRomSize = getContext().getResources().getString(com.android.internal.R.string.custom_rom_size);
//然后就是用我们自己定义的字符串去替换,三方的字符串
if (fakeRamSize != null && !fakeRamSize.equals("") && s.startsWith("Total:") && s.endsWith("MB")) {
String totals = s.substring(s.indexOf(":") + 1, s.indexOf("B") + 1);
text = s.replace(totals, fakeRamSize);
}
if (fakeRomSize != null && !fakeRomSize.equals("") && s.startsWith("Total:") && s.endsWith("GB")) {
String totals = s.substring(s.indexOf(":") + 1, s.indexOf("B") + 1);
text = s.replace(totals, fakeRomSize);
}
if (fakeRamSize != null && !fakeRamSize.equals("") && s.contains("/") && s.endsWith("MB")) {
String totals = s.substring(s.lastIndexOf("/") + 1, s.lastIndexOf("B") + 1);
text = s.replace(totals, fakeRamSize);
}
if (fakeRomSize != null && !fakeRomSize.equals("") && s.contains("/") && s.endsWith("GB")) {
String totals = s.substring(s.lastIndexOf("/") + 1, s.lastIndexOf("B") + 1);
text = s.replace(totals, fakeRomSize);
}
}
}
通过上面的方法,我们就可以把ram rom,替换成我们想要的大小,并且肯定是整数。
2. 大家都看到了,用TextView改的方法很low,对不对?这上面只是3个应用就写了这么多代码,有这么多判断,那如果市面上又出来其他主流检测工具呢???
另外,从性能上面讲,系统里有多少Textview,每个显示之前都要判断是不是三方应用,如果是三方应用,每个TextView还要判断已什么开头,我的天!如果说这不影响性能,鬼才信啊!因此,我们的第二种方法会解决这些问题。
思路是这样的,先考虑三方应用是如何获取系统的ram rom的,至于如何获取,在这里不再详述,大家可以自行百度,经过调试最后发现,这些主流的App获取Rom、Ram的方式是:
//ROM
String path = Environment.getDataDirectory().getPath();
StatFs statFs = new StatFs(path);
long blockCount = statFs.getBlockCountLong();
long blockSize = statFs.getBlockSizeLong();
long availableBlocks = statFs.getAvailableBlocksLong();
long totalBytes = statFs.getTotalBytes();
long freeBytes = statFs.getFreeBytes();
//RAM
//RAM是保存在proc/meminfo文件里面的,meminfo的大概结构式这样的
/**
MemTotal: 所有可用RAM大小。
MemFree: LowFree与HighFree的总和,被系统留着未使用的内存。
Buffers: 用来给文件做缓冲大小。
Cached: 被高速缓冲存储器(cache memory)用的内存的大小(等于diskcache minus SwapCache)。
SwapCached:被高速缓冲存储器(cache memory)用的交换空间的大小。已经被交换出来的内存,仍然被存放在swapfile中,用来在需要的时候很快的被替换而不需要再次打开I/O端口。
Active: 在活跃使用中的缓冲或高速缓冲存储器页面文件的大小,除非非常必要,否则不会被移作他用。
Inactive: 在不经常使用中的缓冲或高速缓冲存储器页面文件的大小,可能被用于其他途径。
SwapTotal: 交换空间的总大小。
SwapFree: 未被使用交换空间的大小。
Dirty: 等待被写回到磁盘的内存大小。
Writeback: 正在被写回到磁盘的内存大小。
AnonPages:未映射页的内存大小。
Mapped: 设备和文件等映射的大小。
Slab: 内核数据结构缓存的大小,可以减少申请和释放内存带来的消耗。
SReclaimable:可收回Slab的大小。
SUnreclaim:不可收回Slab的大小(SUnreclaim+SReclaimable=Slab)。
PageTables:管理内存分页页面的索引表的大小。
NFS_Unstable:不稳定页表的大小。
要获取android手机总内存大小,只需读取”/proc/meminfo”文件的第1行,并进行简单的字符串处理即可。*/
public static String getMemInfoIype(Context context, String type) {
try {
FileReader fileReader = new FileReader("/proc/meminfo");
BufferedReader bufferedReader = new BufferedReader(fileReader, 4 * 1024);
String str = null;
while ((str = bufferedReader.readLine()) != null) {
if (str.contains(type)) {
break;
}
}
bufferedReader.close();
// \\s表示 空格,回车,换行等空白符
// +号表示一个或多个的意思
String[] array = str.split("\\s+");
// 获得系统总内存,单位是KB,乘以1024转换为Byte
int length = Integer.valueOf(array[1]).intValue() * 1024;
return android.text.format.Formatter.formatFileSize(context, length);
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return null;
}
因此我们只需要在源头改变数据的大小就OK,由于我对C不是太熟悉,所以开始时我采用的是修改java代码达到目的。下面是修改Rom
// 刚才已分析,ROM用的是BlockCount和blocksize的乘积,由于blocksize固定,所以我们用一下改法
// 在对应项目的system.prop文件中添加 ro.faker.rom.size = 8L
/*
扩展一下
属性名称以“ro.”开头,那么这个属性被视为只读属性。一旦设置,属性值不能改变。
属性名称以“persist.”开头,当设置这个属性时,其值也将写入/data/property。
属性名称以“net.”开头,当设置这个属性时,“net.change”属性将会自动设置,以加入到最后修改的属性名。(这是很巧妙的。 netresolve模块的使用这个属性来追踪在net.*属性上的任何变化。)
属性“ ctrl.start ”和“ ctrl.stop ”是用来启动和停止服务。每一项服务必须在/init.rc中定义.
*/
public long getBlockCountLong() {
long fakerRomSize = SystemProperties.getLong("ro.faker.rom.size", 0L);
return fakerRomSize == 0L
? (int) mStat.f_blocks
: (int) (1024 * 1024 * 1024L * fakerRomSize / getBlockSizeLong());
}
当然,getBlockCount()这个方法最好也改一下,这是过时的方法,但是仍可以使用
这个方法一般不会有问题,因为大多数都是通过这种方法获取的,但是如果想万无一失,可以通过修改libcore\luni\src\main\native\libcore_io_Linux.cpp文件中的
static jobject makeStructStatVfs(JNIEnv* env, const struct statvfs& sb) {
static jmethodID ctor = env->GetMethodID(JniConstants::structStatVfsClass, "",
"(JJJJJJJJJJJ)V");
if (ctor == NULL) {
return NULL;
}
return env->NewObject(JniConstants::structStatVfsClass, ctor,
static_cast(sb.f_bsize),
static_cast(sb.f_frsize),
static_cast(sb.f_blocks),
static_cast(sb.f_bfree),
static_cast(sb.f_bavail),
static_cast(sb.f_files),
static_cast(sb.f_ffree),
static_cast(sb.f_favail),
static_cast(sb.f_fsid),
static_cast(sb.f_flag),
static_cast(sb.f_namemax));
}
这种改法大家有兴趣自己实现吧,尴尬而不是礼貌的微笑
下面是修改RAM的,这个搞得我头皮发麻,因为之前没搞过C和Kernel,提前说明一下,kernel层和framework层是相对独立的,所以在安卓8.1 上之前的在项目的ProjectConfig.mk中定义一个宏,在kernel层通过Makefile编译的方法不能再用了,废话不说,上代码
// 上面分析过meminfo文件了,这里就不多说了,首先你要清楚项目用的内核版本是什么,可以在手机设置里面看到 ,kernel-4.9\fs\proc\meminfo.c,这个文件生成了meminfo
// 这个方法生成的KB
static int meminfo_proc_show(struct seq_file *m, void *v)
{
struct sysinfo i;
unsigned long committed;
long cached;
......
#if defined(CONFIG_FAKER_RAM_SIZE) && (CONFIG_FAKER_RAM_SIZE> 0)
show_val_kb(m, "MemTotal: ", CONFIG_FAKER_RAM_SIZE* 1024 * 256L);
#else
show_val_kb(m, "MemTotal: ", i.totalram);
#endif
show_val_kb(m, "MemFree: ", i.freeram);
show_val_kb(m, "MemAvailable: ", available);
show_val_kb(m, "Buffers: ", i.bufferram);
......
}
/*
通过这个CONFIG_FAKER_RAM_SIZE,我们就可以随意改变RAM的大小了,这个东西的配置有讲究,注意
首先在kernel-4.9\fs\proc\Kconfig中定义,格式如下 不能错,bool string的写法类似
config FAKER_RAM_SIZE
int FAKER_RAM_SIZE
default 0
CONFIG_FAKER_RAM_SIZE这个就是,config FAKER_RAM_SIZE 合起来的,定义在对应的项目中
kernel-4.9\arch\arm64\configs\xoxo_debug_defconfig
kernel-4.9\arch\arm64\configs\xoxo_defconfig
两个文件都添加如下配置,数值就是项目中需要的RAM大小1G 就是1
CONFIG_FAKER_RAM_SIZE = 1 错误写法 不能留空格 不能留空格 不能留空格
CONFIG_FAKER_RAM_SIZE=1 正确写法
*/
到此为止,就已经实现要求了,Android O,Android P 测试OK,希望你像IG一样,永不加班!