欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Android?常見獲取設備標識方法總結

 更新時間:2024年09月25日 15:19:36   作者:ChenYhong  
隨著Android系統版本更新,Google對用戶隱私保護增強,限制獲取設備標識,文中測試DeviceID、ANDROID_ID、Serial、MAC地址等方法在不同API級別的表現,感興趣的朋友跟隨小編一起看看吧

在最近的開發(fā)工作中,有個功能點需要獲取設備標識作為用戶ID,理想的標識應該是不可重置的。然而,在Android系統版本的不斷迭代中,Google對用戶隱私的保護力度持續(xù)加強,針對獲取設備標識方法的限制也越來越嚴格,官方建議盡量選用用戶可重置的唯一標識,例如AAID。

為了盡量滿足需求,對以往常見的四種獲取設備標識的方法進行了測試,本文對測試結果進行記錄。

獲取設備標識

DeviceID(IMEI、MEID)

穩(wěn)定的標識符,即使恢復出廠設置也不會被重置。在不同版本的Android設備上,獲取DeviceID的方法及效果略有不同,下面分別介紹一下。

API 23-28

AndroidManifest中配置READ_PHONE_STATE權限,代碼如下:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android">
    <uses-permission android:name="android.permission.READ_PHONE_STATE" />
    <application
        ......
        >
        ......
    </application>
</manifest>

在API 23(Android6.0)到API 28(Android 9.0)的設備上,可以通過TelephonyManager.getDeviceId()方法獲取DeviceID。 另外在API 26(Android8.0)或更高版本的設備上,還可以通過TelephonyManager.getImei()TelephonyManager.getMeid()獲取設備標識。需要在運行時向用戶申請授予READ_PHONE_STATE權限。示例代碼如下:

class DeviceIdExampleActivity : AppCompatActivity() {
    private lateinit var binding: LayoutDeviceIdExampleActivityBinding
    private val deviceIdPermissionRequestLauncher = registerForActivityResult(/* contract = */ ActivityResultContracts.RequestPermission()) {
        // 無論是否授權,都進行獲取,作對比
        getDeviceID()
    }
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        binding = LayoutDeviceIdExampleActivityBinding.inflate(layoutInflater).also {
            setContentView(it.root)
        }
        binding.btnGetDeviceId.setOnClickListener {
            if (ActivityCompat.checkSelfPermission(this, Manifest.permission.READ_PHONE_STATE) != PackageManager.PERMISSION_GRANTED) {
                deviceIdPermissionRequestLauncher.launch(Manifest.permission.READ_PHONE_STATE)
            } else {
                getDeviceID()
            }
        }
    }
    private fun getDeviceID() {
        val telephonyManager = getSystemService(TelephonyManager::class.java)
        val deviceIDStrBuilder = StringBuilder()
        try {
            deviceIDStrBuilder.append("deviceId:").append(telephonyManager.deviceId)
        } catch (e: Exception) {
            e.printStackTrace()
        }
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
            try {
                // 設備類型為GSM且有權限獲取時不為空
                deviceIDStrBuilder.append("\nIMEI:").append(telephonyManager.imei)
            } catch (e: Exception) {
                e.printStackTrace()
            }
            try {
                // 設備類型為CDMA且有權限獲取時不為空
                deviceIDStrBuilder.append("\nMEID:").append(telephonyManager.meid)
            } catch (e: Exception) {
                e.printStackTrace()
            }
        }
        binding.tvResultValue.text = deviceIDStrBuilder
    }
}

測試設備為Android Studio模擬器,API為28,測試效果如下:

API 29以上

從API 29(Android10.0)開始,即使用戶授予READ_PHONE_STATE權限,上述三種方法返回值均為空字符串。

測試設備為Android Studio模擬器,API為29,測試效果如下:

在API 29以上也有仍能獲取Device ID的方法,需要與運營商或者手機廠商有深度合作,具體可以參考官方文檔

ANDROID_ID

設備首次啟動時生成,恢復出廠設置時會被重置,無需任何權限即可獲取。獲取代碼如下:

class DeviceIdExampleActivity : AppCompatActivity() {
    private lateinit var binding: LayoutDeviceIdExampleActivityBinding
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        binding = LayoutDeviceIdExampleActivityBinding.inflate(layoutInflater).also {
            setContentView(it.root)
        }
        binding.btnGetAndroidId.setOnClickListener {
            binding.tvResultValue.text = "ANDROID_ID:${Settings.Secure.getString(contentResolver, Settings.Secure.ANDROID_ID)}"
        }
    }
}

測試設備為Pixel 7,API為34,測試效果如下:

需要注意的是,根據查詢到的資料來看,不同廠商的設備可能會出現返回空值或者返回值相同的情況。另外,ANDROID_ID可以手動進行修改

Serial(序列號)

設備硬件序列號,通常在沒有root的情況下不會重置,在不同版本的Android設備上,獲取Serial的方法及效果略有不同,下面分別介紹一下。

API 23-25

在API 23-25版本的設備上無需配置權限,獲取代碼如下:

class DeviceIdExampleActivity : AppCompatActivity() {
    private lateinit var binding: LayoutDeviceIdExampleActivityBinding
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        binding = LayoutDeviceIdExampleActivityBinding.inflate(layoutInflater).also {
            setContentView(it.root)
        }
        binding.btnGetDeviceSerial.setOnClickListener {
            try {
                binding.tvResultValue.text = "deviceSerial:${Build.SERIAL}"
            } catch (e: Exception) {
                e.printStackTrace()
            }
        }
    }
}

測試設備為Android Studio模擬器,API為25,測試效果如下:

API 26-28

從API 26(Android8.0)開始,android.os.Build.SERIAL總是返回UNKNOWN,改為使用android.os.Build.getSerial()方法獲取,另外還需配置READ_PHONE_STATE權限并且在運行時動態(tài)申請用戶授權。示例代碼如下:

class DeviceIdExampleActivity : AppCompatActivity() {
    private lateinit var binding: LayoutDeviceIdExampleActivityBinding
    private val serialPermissionRequestLauncher = registerForActivityResult(/* contract = */ ActivityResultContracts.RequestPermission()) {
        // 無論是否授權,都進行獲取,作對比
        getSerial()
    }
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        binding = LayoutDeviceIdExampleActivityBinding.inflate(layoutInflater).also {
            setContentView(it.root)
        }
        binding.btnGetDeviceSerial.setOnClickListener {
            if (ActivityCompat.checkSelfPermission(this, Manifest.permission.READ_PHONE_STATE) != PackageManager.PERMISSION_GRANTED) {
                serialPermissionRequestLauncher.launch(Manifest.permission.READ_PHONE_STATE)
            } else {
                getSerial()
            }
        }
    }
    private fun getSerial() {
        val serialStrBuilder = StringBuilder()
        try {
            serialStrBuilder.append("Build.SERIAL:").append(Build.SERIAL)
        } catch (e: Exception) {
            e.printStackTrace()
        }
        try {
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
                serialStrBuilder.append("\nBuild.getSerial():").append(Build.getSerial())
            }
        } catch (e: Exception) {
            e.printStackTrace()
        }
        binding.tvResultValue.text = serialStrBuilder
    }
}

測試設備為Android Studio模擬器,API為28,測試效果如下:

API 29以上

從API 29(Android10.0)開始,獲取Serial和獲取DeviceID一樣,文中列舉的方法無法正常獲取,但官方也提供了相應的解決方法。

測試設備為Android Studio模擬器,API為29,測試效果如下:

MAC地址

用于標識網絡接口的硬件地址,通常在設備出廠時分配。有root權限時可以進行修改,因此存在重復的可能性。在不同版本的Android設備上,獲取MAC地址的效果略有不同,下面分別介紹一下。

API 23-28

使用NetworkInterface.getNetworkInterfaces()方法來獲取MAC地址,無需任何權限,示例代碼如下:

class DeviceIdExampleActivity : AppCompatActivity() {
    private lateinit var binding: LayoutDeviceIdExampleActivityBinding
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        binding = LayoutDeviceIdExampleActivityBinding.inflate(layoutInflater).also {
            setContentView(it.root)
        }
        binding.btnGetMacAddress.setOnClickListener {
            val networkInterfaces = NetworkInterface.getNetworkInterfaces()
            while (networkInterfaces.hasMoreElements()) {
                val network = networkInterfaces.nextElement()
                if (network.name == "wlan0") {
                    binding.tvResultValue.text = "macAddress:${network.hardwareAddress?.joinToString(":") { macAddress -> String.format("%02X", macAddress) }}"
                    break
                }
            }
        }
    }
}

測試設備為samsung Galaxy S8,API為28,測試效果如下:

關閉WI-FI

WI-FI

熱點

API 29

獲取代碼和上面一致,但是從API 29(Android10.0)開始,連接不同的WI-FI時會獲取到不同的MAC地址。

測試設備為samsung Galaxy S9,API為29,測試效果如下:

WI-FI

熱點

API 30以上

從API 30(Android11.0)開始,已無法獲取到MAC地址。

測試設備為Pixel 7,API為34,測試效果如下:

小結

文中測試的四種設備標識,確定不會重復的應該只有DeviceId和MAC地址,但是二者從API 29開始也都不可靠了。當然,在市面上仍然有不少API 29以下的設備,在選擇唯一標識方案時,低版本的設備仍然可以采用DeviceID或MAC地址,高版本則采用官方推薦的AAID等方案。

到此這篇關于Android 常見獲取設備標識方法現狀的文章就介紹到這了,更多相關Android設備標識內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

最新評論