Android下获取设备唯一标识(UDID, DeviceID)

本文详细阐述了在Android环境下获取设备唯一标识的方法,包括通过`getDeviceId()`函数、`Secure.getString()`方法以及在获取失败时采用随机生成UUID的兼容策略,并解决了特定设备的兼容性问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

android下获取设备唯一标识原本非常简单(至少不会像iOS一样禁用这个,禁用那个),但是由于设备的多样性需要考虑的东西也对应复杂起来。

先附上完整代码

protected static final String PREFS_FILE = "gank_device_id.xml";  
    protected static final String PREFS_DEVICE_ID = "gank_device_id";
    protected static String uuid;
    static public String getUDID()
    {
    	if( uuid ==null ) {
            synchronized (GankMainActivity.class) {  
                if( uuid == null) {  
                    final SharedPreferences prefs = s_instance.getSharedPreferences( PREFS_FILE, 0);  
                    final String id = prefs.getString(PREFS_DEVICE_ID, null );  
  
                    if (id != null) {  
                        // Use the ids previously computed and stored in the prefs file  
                        uuid = id;  
                    } else {  
  
                        final String androidId = Secure.getString(s_instance.getContentResolver(), Secure.ANDROID_ID);  
  
                        // Use the Android ID unless it's broken, in which case fallback on deviceId,  
                        // unless it's not available, then fallback on a random number which we store  
                        // to a prefs file  
                        try {  
                            if (!"9774d56d682e549c".equals(androidId)) {  
                                uuid = UUID.nameUUIDFromBytes(androidId.getBytes("utf8")).toString();  
                            } else {  
                                final String deviceId = ((TelephonyManager) s_instance.getSystemService( Context.TELEPHONY_SERVICE )).getDeviceId();  
                                uuid = deviceId!=null ? UUID.nameUUIDFromBytes(deviceId.getBytes("utf8")).toString() : UUID.randomUUID().toString();  
                            }  
                        } catch (UnsupportedEncodingException e) {  
                            throw new RuntimeException(e);  
                        }  
  
                        // Write the value out to the prefs file  
                        prefs.edit().putString(PREFS_DEVICE_ID, uuid).commit();  
                    }
                }  
            }  
        }
    	return uuid;
    }

1、正常情况下可以通过((TelephonyManager) s_instance.getSystemService( Context.TELEPHONY_SERVICE )).getDeviceId();  来获取,但是某些平板电脑此函数会返回空

2、通过 Secure.getString(s_instance.getContentResolver(), Secure.ANDROID_ID);   也可以获取到一个id,但是android2.2或者是某些山寨手机使用这个也是有问题的,它会返回一个固定的值 9774d56d682e549c

3、如果前两个都没有获取到udid,那么就在程序启动的时候创建一个随机的uuid,然后保存起来。这个算是兼容方案,当然这样的设备并不会很多。


### 获取唯一设备ID的挑战 在开发过程中,获取一个能够唯一标识设备的ID是一个常见的需求。然而,在Android和iOS平台上,由于隐私政策和技术限制的影响,这一目标变得愈发困难。 #### Android 平台上的解决方案 在Android平台上,虽然存在多种可能用于识别设备的方式,但由于系统更新以及对用户隐私保护的要求日益严格,许多传统方法已经不再适用[^1]。例如,IMEI、MAC地址和Android ID曾被广泛使用,但这些方式要么受到权限限制,要么无法满足跨应用的一致性和稳定性要求[^2]。 一种推荐的做法是创建并存储自定义的应用级UUID作为设备标识符。这种方法可以确保即使操作系统版本发生变化或者硬件特性不可访问的情况下仍然有效: ```java public class DeviceIdManager { private static final String PREFS_FILE = "device_id.xml"; private static final String PREFS_DEVICE_ID = "device_id"; public static String getDeviceId(Context context) { SharedPreferences prefs = context.getSharedPreferences(PREFS_FILE, Context.MODE_PRIVATE); String deviceId = prefs.getString(PREFS_DEVICE_ID, null); if (deviceId == null) { deviceId = UUID.randomUUID().toString(); SharedPreferences.Editor editor = prefs.edit(); editor.putString(PREFS_DEVICE_ID, deviceId); editor.apply(); } return deviceId; } } ``` 此代码片段展示了如何利用SharedPreferences保存一个随机生成的UUID,并将其作为设备ID返回。 #### iOS 平台上的解决方案 对于iOS来说,苹果公司出于隐私考虑早已停止支持UDID等可以直接追踪用户的标识方案。目前官方建议使用的Identifier For Vendor(IDFV),它由App Store分配给同一供应商下的所有应用程序共享同一个值直到卸载最后一个关联程序为止;还有Advertising Identifier(IDFA),主要用于广告目的且允许用户重置或禁用其跟踪功能。 如果需要更持久可靠的解决办法,则可以通过Keychain服务来储存自行产生的GUID形式的数据串当作专属标志码: ```swift import UIKit import Security func save(key: String, data: Data) -> OSStatus { let query: [String : Any] = [ kSecClass as String : kSecClassGenericPassword, kSecAttrAccount as String : key, kSecValueData as String : data ] SecItemDelete(query as CFDictionary) return SecItemAdd(query as CFDictionary, nil) } func load(key: String) -> Data? { let query: [String : Any] = [ kSecClass as String : kSecClassGenericPassword, kSecAttrAccount as String : key, kSecReturnData as String : true, kSecMatchLimit as String : kSecMatchLimitOne ] var result: AnyObject? SecItemCopyMatching(query as CFDictionary, &result) return result as? Data } let deviceIDKey = "com.example.deviceid" if let storedData = load(key: deviceIDKey), let id = NSString(data:storedData, encoding:String.Encoding.utf8.rawValue){ print(id) } else{ let newID = NSUUID().uuidString.data(using:.utf8)! _ = save(key: deviceIDKey, data:newID) print(String(data:newID,encoding:.utf8)!) } ``` 上述Swift代码实现了基于Keychain的安全数据存取机制,从而达到长期保留特定字符串的目的。 ### 结论 综上所述,在当前环境下无论是Android还是iOS都无法简单直接地获得完全意义上的永久型物理装置编号。因此采用软件层面构建逻辑实体号成为主流趋势之一。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值