我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Android KeyStore的使用

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Android KeyStore的使用

在我们App开发过程中,可能会涉及到一些敏感和安全数据需要加密的情况,比如登录token的存储。我们往往会使用一些加密算法将这些敏感数据加密之后再保存起来,需要取出来的时候再进行解密。

此时就会有一个问题:用于加解密的Key该如何存储?

  • 如果把Key和加密后的数据存到一起,那有一定的安全风险。
  • 对Key再进行一次加密,这就陷入了死循环。

为了保证安全性,Android提供了KeyStore系统来保存Key,本文就浅探一下KeyStore及其使用方法。

一、什么是KeyStore?如何保证安全性?

1、什么是KeyStore?

先来看看官方对他的定义:

This class represents a storage facility for cryptographic keys and certificates.

这个类代表加密密钥和证书的存储设施

定义其实很简单,他就是用来保存加解密的Key和证书的。

2、如何保证安全性?

安全性保护措施在官方文档里有写(Android 密钥库系统 | Android 开发者 | Android Developers (google.cn)), 我就按自己的理解总结一下:

  • Key是保存在手机系统里的,应用进程在获取Key的时候是通过系统进程获取到内存里的。也就是说只能在本应用进程里才能拿到Key,想要把Key提取出来是不可能的。
  • KeyStore还可以指定密钥的授权使用方式(比如用户安全锁验证),可保证必须在授权的情况下才能获取到Key。

总的来说就是使用者只能在应用程序运行时使用KeyStore里存放的Key,除非攻击者拿到源码打断点调试,否则无法知道Key到底是什么。这样就保证了存储Key的安全性。

二、KeyStore的使用

KeyStore支持的加密算法有很多,其中对部分算法有API Level的要求,具体可以查看Android 密钥库系统 | Android 开发者 | Android Developers (google.cn)

本文就以AES加密算法为例,简单说明一下KeyStore的使用方式。注意,本文涉及到的代码需要minSdk>=23.

先简单实现一下AES算法的加解密

1、数据加密

object AESUtil {    private const val IV_BLOCK_SIZE = 16    fun encryptAES(encryptBytes: ByteArray, encryptKey: SecretKey): ByteArray?{        try {            //创建密码器            val cipher = Cipher.getInstance("AES/CBC/PKCS7PADDING")            //用密钥初始化Cipher对象            cipher.init(Cipher.ENCRYPT_MODE, encryptKey)            val final = cipher.doFinal(encryptBytes)            // iv占前16位,加密后的数据占后面            return cipher.iv + final        } catch (e: NoSuchPaddingException) {            e.printStackTrace()        } catch (e: NoSuchAlgorithmException) {            e.printStackTrace()        } catch (e: InvalidAlgorithmParameterException) {            e.printStackTrace()        } catch (e: InvalidKeyException) {            e.printStackTrace()        } catch (e: BadPaddingException) {            e.printStackTrace()        } catch (e: IllegalBlockSizeException) {            e.printStackTrace()        }        return null    }    fun decryptAES(decryptBytes: ByteArray, decryptKey: SecretKey): ByteArray? {        try {            // 先取出IV            val iv = decryptBytes.copyOfRange(0, IV_BLOCK_SIZE)            // 取出加密后的数据            val decryptData = decryptBytes.copyOfRange(IV_BLOCK_SIZE, decryptBytes.size)            val cipher = Cipher.getInstance("AES/CBC/PKCS7PADDING")            cipher.init(Cipher.DECRYPT_MODE, decryptKey, IvParameterSpec(iv))            return cipher.doFinal(decryptData)        } catch (e: NoSuchPaddingException) {            e.printStackTrace()        } catch (e: NoSuchAlgorithmException) {            e.printStackTrace()        } catch (e: InvalidAlgorithmParameterException) {            e.printStackTrace()        } catch (e: InvalidKeyException) {            e.printStackTrace()        } catch (e: BadPaddingException) {            e.printStackTrace()        } catch (e: IllegalBlockSizeException) {            e.printStackTrace()        }        return null    }}

然后我们需要为加密生成一个Key,通过KeyGenerator来实现,先生成一个KeyGenerator

private fun getKeyGenerator(alias: String): KeyGenerator {        // 第一个参数指定加密算法,第二个参数指定Provider        val keyGenerator = KeyGenerator.getInstance("AES", "AndroidKeyStore")        val parameterSpec = KeyGenParameterSpec.Builder(            alias,            KeyProperties.PURPOSE_ENCRYPT or KeyProperties.PURPOSE_DECRYPT  //用于加密和解密        )            .setBlockModes(KeyProperties.BLOCK_MODE_CBC)  // AEC_CBC            .setUserAuthenticationRequired(false)   // 是否需要用户认证            .setEncryptionPaddings(KeyProperties.ENCRYPTION_PADDING_PKCS7)  //AES算法的PADDING, 和前面的AESUtil里保持统一            .build()        keyGenerator.init(parameterSpec)        return keyGenerator    }

这个方法里接受一个alias的参数,是生成Key的别名,这个会在之后从KeyStore里取的时候用到。

生成KeyGenerator之后,就可以生成出加解密需要的Key了:

val key: SecretKey = getKeyGenerator("myKey").generateKey()

那紧接着我们就可以对需要保护的数据进行加密然后存储。

val class="lazy" data-srcData = "hello world"val encryptData = AESUtil.encryptAES(class="lazy" data-srcData.toByteArray(), key)// 存储加密后的数据...

2、从KeyStore中获取Key解密

前面我们已经把数据加密存储好了,接下来就是拿出数据解密后使用了。

我们从KeyStore中取出我们解密的Key

    fun getKeyFromKeyStore(alias: String): SecretKey? {        // 参数为Provider        val keyStore = KeyStore.getInstance("AndroidKeyStore")        // 一定要先初始化        keyStore.load(null)        // 获取KeyStore中的所有Key的别名        val aliases = keyStore.aliases()        // KeyStore里没有key        if (!aliases.hasMoreElements()) {            return null        }        // Key的保护参数,这里为不需要密码        val protParam: KeyStore.ProtectionParameter =            KeyStore.PasswordProtection(null)        // 通过别名获取Key        val entry = keyStore.getEntry(alias, protParam) as KeyStore.SecretKeyEntry        return entry.secretKey    }

每一步的注释都写在了代码里,这里方法的alias参数就是我们之前通过KeyGenerator生成Key时生成的参数。

接着就可以拿到Key去解密了。

val decryptKey = KeyStoreUtil.getKeyFromKeyStore(KEY_ALIAS)decryptKey?.let {        // 解密数据        val decryptAES = AESUtil.decryptAES(encryptData, decryptKey)    }

到这里,KeyStore的简单使用就结束了。

三、源码分析

细心的读者可能会发现问题,在前面的使用中,**并没有把Key存入到KeyStore里的操作,为什么后面就可以直接取出来?**想要搞清楚这个问题,就必须得通过源码去解决了。

先拟定一下分析问题的思路:

  1. KeyStore是从哪里取的?
  2. KeyGenerator生成Key的时候是怎么存的?

1、KeyStore是从哪里取的

KeyStore取Key的方法主要是getEntry,这个方法的源码很清晰简单

public final Entry getEntry(String alias, ProtectionParameter protParam)                throws NoSuchAlgorithmException, UnrecoverableEntryException,                KeyStoreException {        if (alias == null) {            throw new NullPointerException("invalid null input");        }        if (!initialized) {            throw new KeyStoreException("Uninitialized keystore");        }        return keyStoreSpi.engineGetEntry(alias, protParam);    }

首先取的时候alias不能为空,这是取Key的别名,如果为空自然就不知道你要哪一个Key了。

其次会判断KeyStore是否初始化。

那核心的代码就是最后一行了。

这里的KeyStoreSpi是一个abstract类,里面实现了engineGetEntry方法。

public KeyStore.Entry engineGetEntry(String alias,                    KeyStore.ProtectionParameter protParam)            throws KeyStoreException, NoSuchAlgorithmException,            UnrecoverableEntryException {...    if ((protParam == null) || protParam instanceof KeyStore.PasswordProtection) {        if (engineIsCertificateEntry(alias)) {            throw new UnsupportedOperationException                ("trusted certificate entries are not password-protected");        } else if (engineIsKeyEntry(alias)) {            char[] password = null;            if (protParam != null) {                KeyStore.PasswordProtection pp =                    (KeyStore.PasswordProtection)protParam;                password = pp.getPassword();            }            Key key = engineGetKey(alias, password);            ....        }    }    ....}

顺着源码走就会发现,真正拿Key的实现是通过engineGetKey()方法拿的,而这个方法是个abstract方法,也就是要找到具体的实现类。

我们使用的Provider是AndroidKeyStore,对应是实现类是AndroidKeyStoreSpi。那就到里面取看一下engineGetKey()的实现

public Key engineGetKey(String alias, char[] password) throws NoSuchAlgorithmException,UnrecoverableKeyException {        try {            return AndroidKeyStoreProvider.loadAndroidKeyStoreKeyFromKeystore(mKeyStore,alias,mNamespace);        }        ....    }

里面最核心的代码也就一句话,继续深挖到AndroidKeyStoreProvider里。

public static AndroidKeyStoreKey loadAndroidKeyStoreKeyFromKeystore(            @NonNull KeyStore2 keyStore, @NonNull String alias, int namespace)            throws UnrecoverableKeyException, KeyPermanentlyInvalidatedException {        ....        final AndroidKeyStoreKey key = loadAndroidKeyStoreKeyFromKeystore(keyStore, descriptor);        if (key instanceof AndroidKeyStorePublicKey) {            return ((AndroidKeyStorePublicKey) key).getPrivateKey();        } else {            return key;        }    }

核心代码是loadAndroidKeyStoreKeyFromKeystore()方法

private static AndroidKeyStoreKey loadAndroidKeyStoreKeyFromKeystore(            @NonNull KeyStore2 keyStore, @NonNull KeyDescriptor descriptor)            throws UnrecoverableKeyException, KeyPermanentlyInvalidatedException {        KeyEntryResponse response = null;        try {            // 核心代码            response = keyStore.getKeyEntry(descriptor);        } catch (android.security.KeyStoreException e) {           ....            }        }...    }

终于快能看到最终拿Key的地方了,不过这里的keyStore要注意以下,是Android下的KeyStore2这个类。

    public KeyEntryResponse getKeyEntry(@NonNull KeyDescriptor descriptor)            throws KeyStoreException {        return handleRemoteExceptionWithRetry((service) -> service.getKeyEntry(descriptor));    }

从注释里可以看到**,KeyStore获取Key的方式是通过IKeystoreService这个服务取获取的,也就是通过系统进程获取的。**这里我们主要是查看从哪里取的,更多如何取的细节读者可以看一下IKeystoreService

2、怎么存的?

前面我们弄清楚了是从哪里取的,接下来就要看一看是怎么存进去的。

KeyStore里存Key的方法是setEntry(),我们就从这里下手看看。

public final void setEntry(String alias, Entry entry,                        ProtectionParameter protParam)                throws KeyStoreException {        if (alias == null || entry == null) {            throw new NullPointerException("invalid null input");        }        if (!initialized) {            throw new KeyStoreException("Uninitialized keystore");        }        keyStoreSpi.engineSetEntry(alias, entry, protParam);    }

可以看到,存的时候KeyStore还是交给了KeyStoreSpi。而KeyStoreSpi的核心方法是engineSetKeyEntry(),我们直接到AndroidKeyStoreSpi里去看具体的实现。

@Override    public void engineSetKeyEntry(String alias, Key key, char[] password, Certificate[] chain)            throws KeyStoreException {        if ((password != null) && (password.length > 0)) {            throw new KeyStoreException("entries cannot be protected with passwords");        }        if (key instanceof PrivateKey) {            setPrivateKeyEntry(alias, (PrivateKey) key, chain, null);        } else if (key instanceof SecretKey) {            setSecretKeyEntry(alias, (SecretKey) key, null);        } else {            throw new KeyStoreException("Only PrivateKey and SecretKey are supported");        }    }

这里简单说一下:

  • PrivateKey通常是非对称加密算法的私钥,公钥用于加密,私钥用于解密,比如RSA算法。

  • SecretKey通常是对称加密算法的密钥,加密解密都用他,比如AES算法。

接着看一下setSecretKeyEntry()方法

private void setSecretKeyEntry(String alias, SecretKey key,            java.security.KeyStore.ProtectionParameter param)            throws KeyStoreException {        ...        try {            KeyStoreSecurityLevel securityLevelInterface = mKeyStore.getSecurityLevel(                    securityLevel);            KeyDescriptor descriptor = makeKeyDescriptor(alias);            securityLevelInterface.importKey(descriptor, null ,                    importArgs, flags, keyMaterial);        } catch (android.security.KeyStoreException e) {            throw new KeyStoreException("Failed to import secret key.", e);        }    }

方法很长,但是最终存入的方法是最后这里。

可以看到,最终是KeyStoreSecurityLevel这个类通过importKey()方法去保存的。

    public KeyMetadata importKey(KeyDescriptor descriptor, KeyDescriptor attestationKey,            Collection<KeyParameter> args, int flags, byte[] keyData)            throws KeyStoreException {        return handleExceptions(() -> mSecurityLevel.importKey(descriptor, attestationKey,                args.toArray(new KeyParameter[args.size()]), flags, keyData));    }

从注释里就能看懂了,往KeyStore里导入Key

3、KeyGenerator是不是帮我们存了?

搞清楚了怎么存的,就可以去KeyGenerator的源码看看他是不是确实帮我们直接保存了。

public final SecretKey generateKey() {        if (serviceIterator == null) {            return spi.engineGenerateKey();        }        ....   }

KeyGeneratorSpi也是个abstact类,我们这里的具体实现类是AndroidKeyStoreKeyGeneratorSpi

@Override    protected SecretKey engineGenerateKey() {        ....        KeyStoreSecurityLevel iSecurityLevel = null;        try {            iSecurityLevel = mKeyStore.getSecurityLevel(securityLevel);            metadata = iSecurityLevel.generateKey(                    descriptor,                    null,                     params,                    flags,                    additionalEntropy);        } catch (android.security.KeyStoreException e) {            switch (e.getErrorCode()) {                // TODO replace with ErrorCode.HARDWARE_TYPE_UNAVAILABLE when KeyMint spec                //      becomes available.                case KeymasterDefs.KM_ERROR_HARDWARE_TYPE_UNAVAILABLE:                    throw new StrongBoxUnavailableException("Failed to generate key");                default:                    throw new ProviderException("Keystore key generation failed", e);            }        }       ....        SecretKey result = new AndroidKeyStoreSecretKey(descriptor, metadata, keyAlgorithmJCA,                iSecurityLevel);        return result;    }

这个方法也特别长,但是在最后能看到一个老朋友:KeyStoreSecurityLevel。原来最终生成Key的方法是调用了他的generateKey()方法。

    public KeyMetadata generateKey(@NonNull KeyDescriptor descriptor, KeyDescriptor attestationKey,            Collection<KeyParameter> args, int flags, byte[] entropy)            throws KeyStoreException {        return handleExceptions(() -> mSecurityLevel.generateKey(                descriptor, attestationKey, args.toArray(new KeyParameter[args.size()]),                flags, entropy));    }

KeyStore里生成一个新的Key,这里就很明显了。

KeyGenerator最终在生成Key的时候,会直接生成在KeyStore里,所以我们才可以直接取到。

四、总结

本篇文章简单介绍了什么是KeyStore,如果使用KeyGeneratorKeyStore,并对KeyStore的存取方式做了源码分析。

来源地址:https://blog.csdn.net/qq_43478882/article/details/127392947

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Android KeyStore的使用

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Android使用KeyStore对数据进行加密的示例代码

谈到 Android 安全性话题,Android Developers 官方网站给出了许多很好的建议和讲解,涵盖了存储数据、权限、网络、处理凭据、输入验证、处理用户数据、加密等方方面面 密钥的保护以及网络传输安全 应该是移动应用安全最关键的
2022-06-06

修改Android签名证书keystore的密码、别名alias以及别名密码

Eclipse ADT的Custom debug keystore自定义调试证书的时候,Android应用开发接入各种SDK时会发现,有很多SDK是需要靠package name和keystore的指纹hash来识别的(百度地图SDK、又或
2022-06-06

Android-----AppBarLayout 的使用

AppBarLayout效果: 当向下滑动屏幕是时 顶部就会出现 当向上滑动屏幕时 顶部就会折叠 向下滑动时:"既然设置了ViewPager 那么我就需要几个Fragment碎片放进去 (实际有三个item 需要几个碎片就放几个item 我
2022-06-06

Android——Fragment的使用(上)

一、前情须知 如何理解Fragment? 可以把Fragment理解成一个迷你的活动,其同样拥有布局和生命周期 但Fragment不能脱离activity存在,Fragment是activity的一个组成元素,一个activity可以拥有多
2022-06-06

android初级篇之android canvas的使用

android的canvas是用来绘制图形和文字的工具,它可以在android应用程序的视图中绘制各种形状、线条、颜色和文字。要使用canvas,首先需要创建一个继承自View的自定义视图类。然后在该类的onDraw方法中使用canvas对
2023-10-12

Android ViewPager 的使用总结

在一个窗口里面添加tab便签,完成便签切换来实现页面的切换,这样的好处是可以在同一个窗口里面有多个页面,这些页面共享同一个窗口的资源,同使用多个窗口来实现这个功能来得更加流畅!! 主要产生的类文件有activity,n个view,adapt
2022-06-06

Android ViewFlipper的简单使用

大家都使用过ViewPager,但是ViewPager还有一个兄弟,那就是ViewFlipper。两者的名字非常相似,我们可以将ViewPager理解成“一页一页的视图”,ViewFlipper则是“快速翻转的视图”,但后者的使用率却远不及
2023-05-31

Android Button的基本使用

Android Button是一个常用的用户界面控件,用于在应用程序中显示可点击的按钮。以下是Android Button的基本使用方法:1. 在XML布局文件中添加Button控件:```android:id="@+id/button_i
2023-09-15

android RadioGroup的使用方法

创建一个MainActivity.java的主类 代码如下: 2022-06-06
2022-06-06

android的RecylerView基本使用

文章目录一、RecylerView基本使用1. 添加依赖2. 添加布局3. 添加adapter4. 添加item_recycler.xml5. MainActivity代码6. 效果7. 代码地址二、RecyclerView-setLayo
2022-06-06

android的fdbus怎么使用

要在Android中使用FDBus,需要遵循以下步骤:1. 添加依赖项:在项目的`build.gradle`文件中添加以下依赖项:```implementation 'com.fineos.android:fdbus:1.0.0'```2.
2023-09-29

编程热搜

  • Android:VolumeShaper
    VolumeShaper(支持版本改一下,minsdkversion:26,android8.0(api26)进一步学习对声音的编辑,可以让音频的声音有变化的播放 VolumeShaper.Configuration的三个参数 durati
    Android:VolumeShaper
  • Android崩溃异常捕获方法
    开发中最让人头疼的是应用突然爆炸,然后跳回到桌面。而且我们常常不知道这种状况会何时出现,在应用调试阶段还好,还可以通过调试工具的日志查看错误出现在哪里。但平时使用的时候给你闹崩溃,那你就欲哭无泪了。 那么今天主要讲一下如何去捕捉系统出现的U
    Android崩溃异常捕获方法
  • android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
    系统的设置–>电池–>使用情况中,统计的能耗的使用情况也是以power_profile.xml的value作为基础参数的1、我的手机中power_profile.xml的内容: HTC t328w代码如下:
    android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
  • Android SQLite数据库基本操作方法
    程序的最主要的功能在于对数据进行操作,通过对数据进行操作来实现某个功能。而数据库就是很重要的一个方面的,Android中内置了小巧轻便,功能却很强的一个数据库–SQLite数据库。那么就来看一下在Android程序中怎么去操作SQLite数
    Android SQLite数据库基本操作方法
  • ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
    工作的时候为了方便直接打开编辑文件,一些常用的软件或者文件我们会放在桌面,但是在ubuntu20.04下直接直接拖拽文件到桌面根本没有效果,在进入桌面后发现软件列表中的软件只能收藏到面板,无法复制到桌面使用,不知道为什么会这样,似乎并不是很
    ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
  • android获取当前手机号示例程序
    代码如下: public String getLocalNumber() { TelephonyManager tManager =
    android获取当前手机号示例程序
  • Android音视频开发(三)TextureView
    简介 TextureView与SurfaceView类似,可用于显示视频或OpenGL场景。 与SurfaceView的区别 SurfaceView不能使用变换和缩放等操作,不能叠加(Overlay)两个SurfaceView。 Textu
    Android音视频开发(三)TextureView
  • android获取屏幕高度和宽度的实现方法
    本文实例讲述了android获取屏幕高度和宽度的实现方法。分享给大家供大家参考。具体分析如下: 我们需要获取Android手机或Pad的屏幕的物理尺寸,以便于界面的设计或是其他功能的实现。下面就介绍讲一讲如何获取屏幕的物理尺寸 下面的代码即
    android获取屏幕高度和宽度的实现方法
  • Android自定义popupwindow实例代码
    先来看看效果图:一、布局
  • Android第一次实验
    一、实验原理 1.1实验目标 编程实现用户名与密码的存储与调用。 1.2实验要求 设计用户登录界面、登录成功界面、用户注册界面,用户注册时,将其用户名、密码保存到SharedPreference中,登录时输入用户名、密码,读取SharedP
    Android第一次实验

目录