Android 指纹识别

指纹识别相关api

FingerprintManagaerCompat

指纹识别的核心包装类

初始化方法

 FingerprintManager fingerprintManager = FingerprintManagerCompat.from(context);
 

用于判断指纹录入情况的方法

//判断指纹识别的硬件是否存在
fingerprintManager.isHardwareDetected();

//判断用户是否录入了指纹
fingerprintManager.hasEnrolledFingerprints();

发起指纹验证

fingerprintManager.authenticate(FingerprintManagerCompat.CryptoObject crypto, 
int flags,
CancellationSignal cancel,
FingerprintManagerCompat.AuthenticationCallback callback, Handler handler)

# 参数详解

FingerprintManagerCompat.CryptoObject
//这是一个加密类的包装类,在java中加密所用的类通是Cipher,Signature,Mac

flags  
//官方文档指明这是一个optinal,并且这里should be 0    也就是必须是0

CancellationSignal  
//这是一个信号类,直接new CancellationSignal()创建一个新对象,然后通过这个对象可以取消指纹扫描等操作

FingerprintManagerCompat.AuthenticationCallback callback
//这是指纹识别结果的监听类,所有的结果处理都将在这个类中


Handler handler   
//处理结果用的handler,通常只是用来告知在哪个线程中返回结果,一般设置为null,因为默认会在主线程中回调扫描结果。

FingerprintManagerCompat.CryptoObject

构造方法

FingerprintManagerCompat.CryptoObject(Signature signature)
FingerprintManagerCompat.CryptoObject(Cipher cipher)
FingerprintManagerCompat.CryptoObject(Mac mac)
//三个构造方法分别传入mac,signature,Cipher

Cipher/Signature/Mac详解

FingerprintManagerCompat.AuthenticationCallback

void    onAuthenticationError(int errMsgId, CharSequence errString)
//指纹扫描出错,这个错误是指打开指纹扫描失败或者扫描中断等系统错误,无法补救

void    onAuthenticationFailed()
//这个错误是指扫描过程完成,但是指纹不匹配

void    onAuthenticationHelp(int helpMsgId, CharSequence helpString)
//这是当开启指纹识别时遇到的可补救的错误时提示用户做相应操作的方法,例如你的指纹识别传感器有灰尘等赃物,helpString将会提示你进行清理

void    onAuthenticationSucceeded(FingerprintManagerCompat.AuthenticationResult result)
//指纹识别成功的回调,result是一个包装了CryptoObject对象的结果对象

指纹识别编码示例

  1. 首先我们需要创建一个Cipher(也可以是Signature或者是Mac)

创建一个工具类用于获取cipher,由于这里指定的加密方式是AES+CBC模式,CBC加密方式的Cipher会创建一个初始向量 byte[]数组,因此在获取相应的解密Cipher时,要把这个byte[]数据传递进去。

@RequiresApi(api = Build.VERSION_CODES.M)
public class KeyGenTool {
    private static final String ANDROID_KEY_STORE = "AndroidKeyStore";
    private static final String KEY_ALGORITHM = KeyProperties.KEY_ALGORITHM_AES;
    private static final String KEY_BLOCK_MODE = KeyProperties.BLOCK_MODE_CBC;
    private static final String KEY_PADDING = KeyProperties.ENCRYPTION_PADDING_PKCS7;
    private static final String TRANSFORMATION = KEY_ALGORITHM + "/" + KEY_BLOCK_MODE + "/" + KEY_PADDING;
    private final String KEY_ALIAS;

    public KeyGenTool(Context context) {
        KEY_ALIAS = context.getPackageName()+"fingerprint";
    }

    public Cipher getEncryptCipher() {
        Cipher cipher = null;
        try {
            cipher = Cipher.getInstance(TRANSFORMATION);
            cipher.init(Cipher.ENCRYPT_MODE, getKey());
        } catch (Exception e) {
            e.printStackTrace();
        }
        return cipher;
    }

    /**
     * 获取解密的cipher
     *
     * @param initializeVector 加密cipher的一些参数
     *                      包括initialize vector(AES加密中 以CBC模式加密需要一个初始的数据块,解密时同样需要这个初始块)
     * @return
     */
    public Cipher getDecryptCipher(byte[] initializeVector) {
        Cipher cipher = null;
        try {
            cipher = Cipher.getInstance(TRANSFORMATION);
            IvParameterSpec ivParameterSpec = new IvParameterSpec(initializeVector);
            cipher.init(Cipher.DECRYPT_MODE, getKey(), ivParameterSpec);
        } catch (Exception e) {
            e.printStackTrace();
        }
        return cipher;
    }

    /**
     * 获取key,首先从秘钥库中根据别名获取key,如果秘钥库中不存在,则创建一个key,并存入秘钥库中
     * @return
     * @throws Exception
     */
    private SecretKey getKey() throws Exception {
        SecretKey secretKey = null;
        KeyStore keyStore = KeyStore.getInstance(ANDROID_KEY_STORE);
        keyStore.load(null);
        if (keyStore.isKeyEntry(KEY_ALIAS)) {
            KeyStore.SecretKeyEntry secretKeyEntry = (KeyStore.SecretKeyEntry) keyStore.getEntry(KEY_ALIAS, null);
            secretKey = secretKeyEntry.getSecretKey();
        } else {
            secretKey = createKey();
        }

        return secretKey;
    }


    /**
     * 在Android中,key的创建之后必须存储在秘钥库才能使用
     * @return
     * @throws Exception
     */
    private SecretKey createKey() throws Exception {
        //在创建KeyGenerator的时候,第二个参数指定provider为AndroidKeyStore,这样创建的key就会被存放在这个秘钥库中
        KeyGenerator keyGenerator = KeyGenerator.getInstance(KEY_ALGORITHM, ANDROID_KEY_STORE);
        KeyGenParameterSpec spec = new KeyGenParameterSpec
                .Builder(KEY_ALIAS, KeyProperties.PURPOSE_ENCRYPT | KeyProperties.PURPOSE_DECRYPT)
                .setBlockModes(KEY_BLOCK_MODE)
                .setEncryptionPaddings(KEY_PADDING)
               //这个设置为true,表示这个key必须是通过了用户认证才可以使用
                .setUserAuthenticationRequired(true)
                .build();
        keyGenerator.init(spec);
        return keyGenerator.generateKey();
    }

}
  1. 权限
<uses-permission android:name="android.permission.USE_FINGERPRINT" />
  1. 创建FingerprintManager并开启加密验证

这里加密的字符串为简单的helloworld,并且将加密以后的字节数组通过base64转换成字符串保存在SharePreference
这里不单保存了加密过的字符串,还保存了

@RequiresApi(api = Build.VERSION_CODES.M)
public void encrypt(View v) {
    mCancelSignal = new CancellationSignal();
    FingerprintManagerCompat.CryptoObject object = new FingerprintManagerCompat.CryptoObject(mKeyGenTool.getEncryptCipher());
    mFingerprintManagerCompat.authenticate(object, 0, mCancelSignal
            , new FingerprintManagerCompat.AuthenticationCallback() {
                @Override
                public void onAuthenticationError(int errMsgId, CharSequence errString) {
                    super.onAuthenticationError(errMsgId, errString);
                }

                @Override
                public void onAuthenticationHelp(int helpMsgId, CharSequence helpString) {
                    super.onAuthenticationHelp(helpMsgId, helpString);
                }

                @Override
                public void onAuthenticationSucceeded(FingerprintManagerCompat.AuthenticationResult result) {
                    try {
                       Cipher cipher = result.getCryptoObject().getCipher();
                            byte[] bytes = cipher.doFinal("helloworld".getBytes());
                            //保存加密过后的字符串
                            SharePreferenceTool.getInstance().saveStr("hello", Base64.encodeToString(bytes,Base64.URL_SAFE));
                            //保存用于做AES-CBC加密变换的初始向量数组
                            byte[] iv = cipher.getIV();
                            SharePreferenceTool.getInstance().saveStr("iv", Base64.encodeToString(iv,Base64.URL_SAFE));
                         
                    }catch (Exception e){
                        e.printStackTrace();
                    }
                }

                @Override
                public void onAuthenticationFailed() {
                    super.onAuthenticationFailed();
                }
            },null);
}
  1. 解密

@RequiresApi(api = Build.VERSION_CODES.M)
    public void decrypt(View v) {

        //保存用于做AES-CBC
        String ivStr = SharePreferenceTool.getInstance().getStr("iv");
        byte[] iv = Base64.decode(ivStr, Base64.URL_SAFE);

        mCancelSignal = new CancellationSignal();
        FingerprintManagerCompat.CryptoObject object = new FingerprintManagerCompat.CryptoObject(mKeyGenTool.getDecryptCipher(iv));
        mFingerprintManagerCompat.authenticate(object, 0, mCancelSignal
                , new FingerprintManagerCompat.AuthenticationCallback() {
                    @Override
                    public void onAuthenticationError(int errMsgId, CharSequence errString) {
                        super.onAuthenticationError(errMsgId, errString);
                    }

                    @Override
                    public void onAuthenticationHelp(int helpMsgId, CharSequence helpString) {
                        super.onAuthenticationHelp(helpMsgId, helpString);
                    }

                    @Override
                    public void onAuthenticationSucceeded(FingerprintManagerCompat.AuthenticationResult result) {
                        try {

                            Cipher cipher = result.getCryptoObject().getCipher();
                            String text = SharePreferenceTool.getInstance().getStr("hello");
                            byte[] input = Base64.decode(text, Base64.URL_SAFE);
                            byte[] bytes = cipher.doFinal(input);
                            //保存加密过后的字符串
                            Log.d("TestFingerprintActivity", "解密的字符串为: " + new String(bytes));
                            SharePreferenceTool.getInstance().saveStr("hello", text);

                            byte[] iv = cipher.getIV();
                            Log.d("TestFingerprintActivity", "iv:" + Base64.encodeToString(iv, Base64.URL_SAFE));

                            //保存用于做AES-CBC
                            Toast.makeText(TestFingerprintActivity.this, "解密成功", Toast.LENGTH_SHORT).show();
                        } catch (Exception e) {
                            e.printStackTrace();
                        }
                    }

                    @Override
                    public void onAuthenticationFailed() {
                        super.onAuthenticationFailed();
                    }
                }, null);

       /* try {
            String text = SharePreferenceTool.getInstance().getStr("hello");
            byte[] input = Base64.decode(text, Base64.URL_SAFE);

            //保存用于做AES-CBC
            String ivStr = SharePreferenceTool.getInstance().getStr("iv");
            byte[] iv = Base64.decode(ivStr, Base64.URL_SAFE);

            Cipher cipher = mKeyGenTool.getDecryptCipher(iv);
            byte[] output = cipher.doFinal(input);

            Log.d("TestFingerprintActivity", "加密后的字符串为: " + new String(output));
            Toast.makeText(this, "解密成功", Toast.LENGTH_SHORT).show();
        } catch (Exception e) {
            e.printStackTrace();
        }*/
    }

解密操作将保存的初始化向量提供给keyGenTool 创建出一个解密用的Cipher,然后发起指纹验证,当指纹验证成功之后取出Cipher进行解密,注意到底下注释掉的一行代码。

指纹验证如何保证安全性

  1. 为什么Cipher需要包装传递给authenticate()方法

Cipher传递给指纹验证方法,再取出来做加密解密,和直接用Cipher加密解密有什么区别呢?问题的关键还是在创建的Key上,创建keyGenerator时,有一个方法setUserAuthenticationRequired(true),也就是说这个key秘钥必须用户验证了才可以使用的,所以使用这种key初始化的的Cipher如果直接用于加密或者解密,会报出错误W/System.err: javax.crypto.IllegalBlockSizeException,仔细查看错误栈信息会发现它是由于android.security.KeyStoreException: Key user not authenticated这个错误引起的。而当我们将Cipher包装传递给指纹验证方法时,其内部验证了用户的身份,也就解除了Cipher中的key的使用限制,因此在回调方法中就可以使用该Cipher来加解密了。

fingerprintauthenticate.png
  1. 如何保存相关信息以配合指纹验证身份

指纹验证只是一个认证用户身份的方式,由于用于加密解密的实际操作其实是委托给CryptoObject内部的Cipher,因此主要的工作还是在Cipher的处理上,以指纹支付为例,加密方式是AES-CBC的对称加密:

  1. 首先是是加密,当用户选择开通指纹验证登录时,首先会要求用户输入支付密码,接下来代码中初始化一个Cipher用于加密操作,接着要求用户验证指纹,指纹验证成功后将支付密码通过Cipher加密并将做Base64转换成字符串,同时将Cipher的初始向量byte[] iv一起转换成字符串,将这两个字符串存储到服务器。同时,用于初始化Cipherkey保存在Android内部秘钥库AndroidKeyStore中,外部应用程序无法获取。
  2. 当用户发起支付时,首先从服务器获取之前加密过的密码字符串和初始向量字符串,同时从本地秘钥库AndroidKeyStore中通过alias别名取出之前存储的key,用初始向量key初始化一个Cipher,接下来发起指纹支付,指纹验证通过后便可用这个Cipher解密字符串获取真正的支付密码,然后传递给服务器验证。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,684评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,143评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,214评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,788评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,796评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,665评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,027评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,679评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,346评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,664评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,766评论 1 331
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,412评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,015评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,974评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,073评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,501评论 2 343

推荐阅读更多精彩内容

  • 概述 之前一直对加密相关的算法知之甚少,只知道类似DES、RSA等加密算法能对数据传输进行加密,且各种加密算法各有...
    Henryzhu阅读 3,005评论 0 14
  • 指纹识别大致应用在几种场景1,系统解锁2,应用锁3,支付认证4,普通的登录认证 指纹识别需要手机硬件支持才能使用。...
    ReleaseYH阅读 4,838评论 0 6
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,596评论 18 139
  • 唯美食与爱不可辜负--佚名 美食使人心情愉悦!享受美食的时间是快乐的,但是等待美食出炉的时间才是最...
  • 会有一段比较难熬的时期,坚持就是胜利。
    MiumiuWong阅读 131评论 0 0