Android设备基础信息获取 源码修改方式 APK开发
APK 获取设备信息
头文件
import java.io.BufferedReader;
import java.io.File;
import java.io.FileFilter;
import java.io.FileInputStream;
import java.io.FileReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.util.ArrayList;
获取设备型号
public static String getDeviceModel() {
return Build.MODEL;
}
获取设备制造商
public static String getManufacturer() {
return Build.MANUFACTURER;
}
获取Android版本号
public static String getAndroidVersion() {
return Build.VERSION.RELEASE;
}
获取SDK版本信息
public static int getDeviceSDK(){
return android.os.Build.VERSION.SDK_INT;
}
获取当前设备安全补丁级别日期
public static String getSecurityPatchLevel() {
return Build.VERSION.SECURITY_PATCH;
}
获取设备制造商
public static String getDeviceMANUFACTURER() {
return Build.SOC_MANUFACTURER;
}
获取构建的内部版本
内部版本
Build ID
在rb-a3568/build/make/core/build_id.mk
目录下
public static String getBuildID() {
return Build.ID;
}
获取显示信息
public static String getDisplay() {
return Build.DISPLAY;
}
获取设备硬件名
public static String getHardware() {
return Build.HARDWARE;
}
获取设备CPU架构
public static String getCpuArchitecture() {
return Build.CPU_ABI;
}
获取CPU名称
通过读取
Linux文件
的形式获取
public static String getCpuName() {
String str1 = "/proc/cpuinfo";
String str2 = "";
String cpuName = "";
try {
FileReader fileReader = new FileReader(str1);
BufferedReader bufferedReader = new BufferedReader(fileReader);
while ((str2 = bufferedReader.readLine()) != null) {
// 为空跳过
if (TextUtils.isEmpty(str2)) {
continue;
}
// 进行 分割 对比
// 使用split(":\\s+", 2)方法将字符串str2按冒号和后续空格分割成最多两部分。例如,"Hardware: Intel Core i7"会被分成["Hardware", "Intel Core i7"]。
String[] arrayOfString = str2.split(":\\s+", 2);
if (TextUtils.equals(arrayOfString[0].trim(), "Hardware")) {
cpuName = arrayOfString[1];
break;
}
}
bufferedReader.close();
fileReader.close();
} catch (IOException e) {
e.printStackTrace();
}
return cpuName;
}
获取CPU核数
public static String getCpuCores() {
return Build.CPU_ABI;
}
获取CPU频率
public static final int DEVICEINFO_UNKNOWN = -1;
/**
* Method for reading the clock speed of a CPU core on the device. Will read from either
* {@code /sys/devices/system/cpu/cpu0/cpufreq/cpuinfo_max_freq} or {@code /proc/cpuinfo}.
*
* @return Clock speed of a core on the device, or -1 in the event of an error.
* 获取 CPU 频率
*/
public static ArrayList<Integer> getCPUFreqMHzs() {
int maxFreq = DEVICEINFO_UNKNOWN;
int curFreq = 0;
ArrayList<Integer> arrayList = new ArrayList<Integer>();
try {
int coreNum = getNumberOfCPUCores();
for (int i = 0; i < coreNum; i++) {
String filename =
"/sys/devices/system/cpu/cpu" + i + "/cpufreq/cpuinfo_max_freq";
File cpuInfoMaxFreqFile = new File(filename);
if (cpuInfoMaxFreqFile.exists() && cpuInfoMaxFreqFile.canRead()) {
byte[] buffer = new byte[128];
FileInputStream stream = new FileInputStream(cpuInfoMaxFreqFile);
try {
stream.read(buffer);
int endIndex = 0;
// Trim the first number out of the byte buffer.
while (Character.isDigit(buffer[endIndex]) && endIndex < buffer.length) {
endIndex++;
}
String str = new String(buffer, 0, endIndex);
// 频率是按照1000计算
curFreq = Integer.parseInt(str) / 1000;
arrayList.add(curFreq);
} catch (NumberFormatException e) {
} catch (IOException e) {
throw new RuntimeException(e);
} finally {
stream.close();
}
}
}
if (maxFreq == DEVICEINFO_UNKNOWN && arrayList.size() == 0) {
FileInputStream stream = new FileInputStream("/proc/cpuinfo");
try {
int freqBound = parseFileForValue("cpu MHz", stream);
curFreq = freqBound;
arrayList.add(curFreq);
} finally {
stream.close();
}
}
} catch (IOException e) {
}
return arrayList;
}
/**
* Reads the number of CPU cores from the first available information from
* {@code /sys/devices/system/cpu/possible}, {@code /sys/devices/system/cpu/present},
* then {@code /sys/devices/system/cpu/}.
*
* @return Number of CPU cores in the phone, or DEVICEINFO_UKNOWN = -1 in the event of an error.
*/
public static int getNumberOfCPUCores() {
int coreNumber = -1;
if (coreNumber != DEVICEINFO_UNKNOWN) {
return coreNumber;
}
if (Build.VERSION.SDK_INT <= Build.VERSION_CODES.GINGERBREAD_MR1) {
// Gingerbread doesn't support giving a single application access to both cores, but a
// handful of devices (Atrix 4G and Droid X2 for example) were released with a dual-core
// chipset and Gingerbread; that can let an app in the background run without impacting
// the foreground application. But for our purposes, it makes them single core.
coreNumber = 1;
return coreNumber;
}
int cores;
try {
cores = getCoresFromFileInfo("/sys/devices/system/cpu/present");
if (cores == DEVICEINFO_UNKNOWN) {
cores = new File("/sys/devices/system/cpu/").listFiles(CPU_FILTER).length;;
}
} catch (SecurityException e) {
cores = DEVICEINFO_UNKNOWN;
} catch (NullPointerException e) {
cores = DEVICEINFO_UNKNOWN;
}
coreNumber = cores;
return coreNumber;
}
/**
* Tries to read file contents from the file location to determine the number of cores on device.
* @param fileLocation The location of the file with CPU information
* @return Number of CPU cores in the phone, or DEVICEINFO_UKNOWN = -1 in the event of an error.
*/
private static int getCoresFromFileInfo(String fileLocation) {
InputStream is = null;
try {
is = new FileInputStream(fileLocation);
BufferedReader buf = new BufferedReader(new InputStreamReader(is));
String fileContents = buf.readLine();
buf.close();
return getCoresFromFileString(fileContents);
} catch (IOException e) {
return DEVICEINFO_UNKNOWN;
} finally {
if (is != null) {
try {
is.close();
} catch (IOException e) {
// Do nothing.
}
}
}
}
/**
* Converts from a CPU core information format to number of cores.
* @param str The CPU core information string, in the format of "0-N"
* @return The number of cores represented by this string
*/
private static int getCoresFromFileString(String str) {
if (str == null || !str.matches("0-[\\d]+$")) {
return DEVICEINFO_UNKNOWN;
}
return Integer.valueOf(str.substring(2)) + 1;
}
private static final FileFilter CPU_FILTER = new FileFilter() {
@Override
public boolean accept(File pathname) {
String path = pathname.getName();
//regex is slow, so checking char by char.
if (path.startsWith("cpu")) {
for (int i = 3; i < path.length(); i++) {
if (!Character.isDigit(path.charAt(i))) {
return false;
}
}
return true;
}
return false;
}
};
/**
* Helper method for reading values from system files, using a minimised buffer.
*
* @param textToMatch - Text in the system files to read for.
* @param stream - FileInputStream of the system file being read from.
* @return A numerical value following textToMatch in specified the system file.
* -1 in the event of a failure.
*/
private static int parseFileForValue(String textToMatch, FileInputStream stream) {
byte[] buffer = new byte[1024];
try {
int length = stream.read(buffer);
for (int i = 0; i < length; i++) {
if (buffer[i] == '\n' || i == 0) {
if (buffer[i] == '\n') i++;
for (int j = i; j < length; j++) {
int textIndex = j - i;
//Text doesn't match query at some point.
if (buffer[j] != textToMatch.charAt(textIndex)) {
break;
}
//Text matches query here.
if (textIndex == textToMatch.length() - 1) {
return extractValue(buffer, j);
}
}
}
}
} catch (IOException e) {
//Ignore any exceptions and fall through to return unknown value.
} catch (NumberFormatException e) {
}
return DEVICEINFO_UNKNOWN;
}
/**
* Helper method used by {@link #parseFileForValue(String, FileInputStream) parseFileForValue}. Parses
* the next available number after the match in the file being read and returns it as an integer.
* @param index - The index in the buffer array to begin looking.
* @return The next number on that line in the buffer, returned as an int. Returns
* DEVICEINFO_UNKNOWN = -1 in the event that no more numbers exist on the same line.
*/
private static int extractValue(byte[] buffer, int index) {
while (index < buffer.length && buffer[index] != '\n') {
if (Character.isDigit(buffer[index])) {
int start = index;
index++;
while (index < buffer.length && Character.isDigit(buffer[index])) {
index++;
}
String str = new String(buffer, 0, start, index - start);
return Integer.parseInt(str);
}
index++;
}
return DEVICEINFO_UNKNOWN;
}
获取设备内存信息
// 获取设备总内存
public static String getTotalMemory(Context context) {
ActivityManager am = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);
ActivityManager.MemoryInfo mi = new ActivityManager.MemoryInfo();
am.getMemoryInfo(mi);
return Formatter.formatFileSize(context, mi.totalMem);// 将获取的内存大小规格化
}
// 获取设备剩余内存
public static String getAvailableMemory(Context context) {
ActivityManager am = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);
ActivityManager.MemoryInfo mi = new ActivityManager.MemoryInfo();
am.getMemoryInfo(mi);
return Formatter.formatFileSize(context, mi.availMem);// 将获取的内存大小规格化
}
// 格式化文件大小,单位为MB
private static String formatFileSize(long sizeInBytes) {
return Formatter.formatFileSize(null, sizeInBytes);
}
获取主板品牌
public static String getBRAND() {
return Build.BRAND;
}
统一输出日志
// 统一日志输出方法
public static void logDeviceInfo(Context context) {
if (true) { // 仅在 Debug 模式打印日志,避免正式版泄露设备信息
Log.i(TAG, "===== 设备信息 =====");
Log.i(TAG, String.format("设备型号: %s", getDeviceModel()));
Log.i(TAG, String.format("Android 版本: %s (SDK %d)", getAndroidVersion(), getDeviceSDK()));
Log.i(TAG, String.format("安全补丁级别: %s", getSecurityPatchLevel()));
Log.i(TAG, String.format("Build 号: %s", getBuildID()));
Log.i(TAG, String.format("屏幕显示信息: %s", getDisplay()));
Log.i(TAG, "===== 产品信息 =====");
Log.i(TAG, String.format("硬件信息: %s", getHardware()));
Log.i(TAG, String.format("制造商: %s", getDeviceMANUFACTURER()));
Log.i(TAG, String.format("品牌: %s", getBRAND()));
Log.i(TAG, "===== 内存信息 =====");
Log.i(TAG, String.format("总内存: %s", getTotalMemory(context)));
Log.i(TAG, String.format("可用内存: %s", getAvailableMemory(context)));
Log.i(TAG, "======CPU信息=======");
Log.i(TAG, String.format("CPU 名称: %s", getCpuName()));
Log.i(TAG, String.format("CPU 架构: %s", getCpuArchitecture()));
Log.i(TAG, String.format("CPU 核数: %s", getNumberOfCPUCores()));
Log.i(TAG, String.format("CPU 频率: %s", getCPUFreqMHzs()));
}
}
获取存储信息
// 获取设备存储 信息
// 获取StorageManager实例
private String GetStorageInfo(Map<String,String> m_Device_Map) throws IOException {
String Storege_info = "";
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
StorageManager storageManager = (StorageManager) getSystemService(Context.STORAGE_SERVICE);
StorageStatsManager storageStatsManager = (StorageStatsManager)getSystemService(STORAGE_STATS_SERVICE);
List<StorageVolume> volumeList = storageManager.getStorageVolumes();
Log.i(Tag,"获取的存储数量" + volumeList.size());
for (StorageVolume volume : volumeList) {
if (null != volume ) {
String label = volume.getDescription(this); //这个其实就是U盘的名称
String status = volume.getState(); //设备挂载的状态,如:mounted、unmounted
boolean isEmulated = volume.isEmulated(); //是否是内部存储设备
boolean isRemovable = volume.isRemovable(); //是否是可移除的外部存储设备
String mPath=""; //设备的路径
Log.i(Tag,"name:"+label);
Log.i(Tag,"status:"+status);
Log.i(Tag,"isEmulated:"+isEmulated);
Log.i(Tag,"isRemovable:"+isRemovable);
File file;
try {
long totalSpace = 0;
long availSpace = 0;
file = volume.getDirectory();
mPath = file.getPath();
Log.i(Tag,"mPath:"+mPath);
// 判断存储路径是否指向系统盘
String path = volume.getDirectory().getAbsolutePath();
if (mPath.contains("/storage/emulated/0") ) {
// 读取系统所在占用空间
try {
totalSpace = storageStatsManager.getTotalBytes(StorageManager.UUID_DEFAULT);//总空间大小
availSpace = storageStatsManager.getFreeBytes(StorageManager.UUID_DEFAULT);//可用空间大小
long systemBytes = totalSpace - availSpace;//系统所占不可用空间大小
Log.i(Tag,"totalBytes:"+ formatFileSize(totalSpace));
Log.i(Tag,"isEmulated:"+ formatFileSize(availSpace));
Log.i(Tag,"systemBytes:"+ formatFileSize(systemBytes));
} catch (IOException e) {
e.printStackTrace();
}
}
else {
// 通过读取挂载文件夹的形式读取文件空间
totalSpace = file.getTotalSpace();
availSpace = file.getFreeSpace();
Log.i(Tag, "可用的block数目::" + SDCardUtils.formatFileSize(totalSpace) + ",剩余空间:" + SDCardUtils.formatFileSize(availSpace));
}
// 设备名称 对应的空间容量
m_Device_Map.put(label,"Storage:" + formatFileSize(totalSpace) );
}catch (Exception e) {
e.toString();
}
}
else
{
Log.i(Tag,"Not volume");
}
}
}
//想看外部存储时,替换uuid即可
return Storege_info;
}
/**
* 格式化文件大小并转换为可视化单位(B, KB, MB, GB, TB)
*/
private String formatFileSize(long sizeInBytes) {
if (sizeInBytes <= 0) {
return "0 B";
}
// 定义单位 列表
final String[] units = new String[] {"B", "KB", "MB", "GB", "TB"};
// 计算单位的指数
int unitIndex = 0;
double size = sizeInBytes;
// 根据大小调整单位
while (size >= 1000 && unitIndex < units.length - 1) {
size /= 1000;
unitIndex++;
}
// 保留两位小数
return String.format("%.2f %s", size, units[unitIndex]);
}
设备信息源码位置修改
build\make\core\sysprop.mk
- 搜索到数据很多,根据当前使用的
Rockchip 3568
的芯片,找到3568
对应的文件夹下的device/rockchip/rk356x/rk3568_t/rk3568_t.mk
- 根据需求修改相应的参数