本文旨在提供一个健壮且兼容性强的方案,用于在android应用中获取设备序列号。由于不同设备制造商和Android版本在序列号的存储方式上存在差异,直接依赖单一方法可能导致部分设备无法获取。本文将介绍一种通过尝试多种系统属性来获取序列号的方法,并提供相应的代码示例和注意事项,确保应用能够尽可能地获取到设备序列号。
序列号获取的挑战
在Android开发中,获取设备序列号看似简单,但实际情况却复杂得多。不同厂商的设备可能将序列号存储在不同的系统属性中,而 Build.SERIAL 在某些情况下可能返回 “unknown” 或需要特定的权限才能访问。因此,一个可靠的解决方案需要考虑这些差异,并尝试多种方法来获取序列号。
解决方案:尝试多种系统属性
以下代码展示了一种尝试多种系统属性来获取设备序列号的方法。该方法通过反射访问 android.os.SystemProperties 类,并依次尝试不同的属性名称,直到成功获取序列号或所有属性都尝试完毕。
import android.Manifest import android.content.Context import android.content.pm.PackageManager import android.os.Build import androidx.core.app.ActivityCompat import androidx.core.content.ContextCompat import android.widget.Toast fun getDeviceSerial(applicationContext: Context): String { var serialNumber: String = "" try { val c = Class.forName("android.os.SystemProperties") val get = c.getMethod("get", String::class.Java) serialNumber = get.invoke(c, "gsm.sn1") as String when (serialNumber) { "" -> serialNumber = get.invoke(c, "ril.serialnumber") as String } when (serialNumber) { "" -> serialNumber = get.invoke(c, "ro.serialno") as String } when (serialNumber) { "" -> serialNumber = get.invoke(c, "sys.serialnumber") as String } when (serialNumber) { "" -> serialNumber = get.invoke(c, "ro.boot.serialno") as String } when (serialNumber) { "" -> serialNumber = get.invoke(c, "ro.ril.oem.sno") as String } when (serialNumber) { "" -> serialNumber = get.invoke(c, "ril.cdma.esn") as String } when (serialNumber) { "" -> serialNumber = get.invoke(c, "vendor.gsm.serial") as String } when (serialNumber) { "" -> serialNumber = get.invoke(c, "ro.boot.un") as String } when (serialNumber) { "" -> serialNumber = get.invoke(c, "ro.boot.uniqueno") as String } when (serialNumber) { "" -> serialNumber = get.invoke(c, "ro.ril.oem.wifimac") as String } when (serialNumber) { "" -> serialNumber = get.invoke(c, "ro.ril.oem.btmac") as String } @Suppress("DEPRECATION") when (serialNumber) { "" -> serialNumber = Build.SERIAL } when (serialNumber) { "" -> serialNumber = "" } } catch (e: Exception) { e.printStackTrace() Toast.makeText(applicationContext, e.message, Toast.LENGTH_LONG).show() serialNumber = "" } if (serialNumber == "unknown") { try { val c = Class.forName("android.os.SystemProperties") val get = c.getMethod( "get", String::class.java, String::class.java ) serialNumber = get.invoke(c, "ril.serialnumber", "unknown") as String } catch (ignored: Exception) { Toast.makeText( applicationContext, "ignored ${ignored.message}", Toast.LENGTH_LONG ) .show() } } try { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O && serialNumber == "unknown") { if (ActivityCompat.checkSelfPermission( applicationContext, Manifest.permission.READ_PHONE_STATE ) == PackageManager.PERMISSION_GRANTED ) { serialNumber = Build.getSerial() } } } catch (e: Exception) { serialNumber = "" } return serialNumber } fun openTelePhony(context: Context?) { context?.let { var a = "" try { //check permission if (ContextCompat.checkSelfPermission( context, Manifest.permission.CALL_PHONE ) == PackageManager.PERMISSION_GRANTED ) { a = getDeviceSerial(context) } else { a = Build.SERIAL } System.out.print(a) } catch (e: Exception) { System.out.print() } } }
代码解释:
- 权限检查: 在Android 8.0 (API level 26) 及以上,Build.getSerial() 需要 READ_PHONE_STATE 权限。 代码中包含权限检查和请求的逻辑(虽然在提供的openTelePhony函数中,权限不足时直接使用Build.SERIAL可能返回”unknown”,应考虑处理这种情况)。
- 反射机制: 使用反射访问 android.os.SystemProperties 类,避免直接依赖该类的API,提高代码的兼容性。
- 多种属性尝试: 依次尝试多个常见的系统属性,例如 gsm.sn1, ril.serialnumber, ro.serialno 等。
- 错误处理: 使用 try-catch 块捕获可能出现的异常,例如 ClassNotFoundException 或 NoSuchMethodException,保证程序的健壮性。
注意事项
-
权限声明: 在 AndroidManifest.xml 文件中声明 READ_PHONE_STATE 权限。
<uses-permission android:name="android.permission.READ_PHONE_STATE" />
-
权限请求: 在运行时请求 READ_PHONE_STATE 权限。请参考Android官方文档了解如何在运行时请求权限。
-
“unknown” 值处理: 即使使用了多种方法,仍然可能无法获取到序列号,或者获取到 “unknown” 值。 在应用中需要对这种情况进行妥善处理,例如使用其他唯一标识符作为备选方案。
-
设备兼容性: 不同设备的系统属性可能存在差异。 建议在多种设备上进行测试,并根据实际情况调整代码中的属性列表。
-
Android 10 (API level 29) 及更高版本: 从 Android 10 开始,访问设备序列号受到更多限制,并且需要满足特定条件才能获取。请务必仔细阅读 Android 官方文档,了解相关限制和最佳实践。
总结
获取Android设备序列号是一个具有挑战性的任务,但通过尝试多种系统属性、进行适当的错误处理和权限管理,可以大大提高获取序列号的成功率。请务必根据实际情况进行测试和调整,并遵守Android平台的安全和隐私政策。
评论(已关闭)
评论已关闭