Android 指紋識(shí)別開(kāi)發(fā)實(shí)例
Android M指紋的資料太少,經(jīng)過(guò)一段時(shí)間閱讀原生Android代碼,寫了以下例子,貢獻(xiàn)出來(lái)給需要幫助的人。
以下內(nèi)容基于64位的高通CPU,搭載fpc1020芯片,此部分代碼在原生android上做了更改,以應(yīng)付工廠指紋的測(cè)試。原生android指紋錄入需要采集10次(因不同的芯片而定)。
代碼簡(jiǎn)單說(shuō)明:
1. FingerPrintEnrollBase類:
重要的是
Java代碼
public interface Listener { void onEnrollmentHelp(CharSequence helpString); void onEnrollmentError(int errMsgId, CharSequence errString); void onEnrollmentProgressChange(int steps, int remaining); }
這個(gè)接口。
1)onEnrollmentHelp函數(shù):是下層返回的指紋操作幫助信息。
2)onEnrollmentError函數(shù)是下層返回的錯(cuò)誤信息,errMsgId是錯(cuò)誤信息類型,其中1表示指紋硬件不可用,3表示超時(shí),5表示操作已取消。errString是errMsgId對(duì)應(yīng)的文字信息,有需要的自己實(shí)驗(yàn)就知道了。
3)onEnrollmentProgressChange()函數(shù)是顯示當(dāng)前指紋的采集情況,steps:總共需要錄入多少次;remaining是剩余多少次,我在此函數(shù)中有打印Log。
2. FingerPrintActivity是主Activity類,F(xiàn)ingerPrintEnrollBase是指紋Base類。
需要特別說(shuō)明的是,在FingerPrintEnrollBase類中有一句:private byte[] mToken = new byte[69];這個(gè)數(shù)組正常情況是通過(guò)輸入密碼后經(jīng)由framework層向hal層申請(qǐng)的,我這里為了省去驗(yàn)證那一塊的代碼,通過(guò)sizeof(hw_auth_token_t)計(jì)算出來(lái)的為69,需要的請(qǐng)依此計(jì)算。我在代碼里實(shí)驗(yàn)過(guò),此數(shù)組如果不是69(在64位CPU上),將直接導(dǎo)致程序crach!
fpc指紋的采集流程圖:
源碼如下:
FingerPrintEnrollBase:
Java代碼
/* * Copyright (C) 2015 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License */ package com.mediatek.test; import android.app.Activity; import android.app.Fragment; import android.content.Context; import android.hardware.fingerprint.FingerprintManager; import android.os.Bundle; import android.os.CancellationSignal; import android.os.Handler; import android.util.Log; /** * local fragment to handle the state around fingerprint enrollment. */ public class FingerPrintEnrollBase extends Fragment { private int mEnrollmentSteps = -1; private int mEnrollmentRemaining = 0; private Listener mListener; private boolean mEnrolling; private CancellationSignal mEnrollmentCancel; private Handler mHandler = new Handler(); private byte[] mToken = new byte[69];//if byte length != sizeof(hw_auth_token_t), throws Exception private boolean mDone; private static final String TAG = "FingerPrintEnrollBase"; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setRetainInstance(true); } @Override public void onAttach(Activity activity) { super.onAttach(activity); } @Override public void onStart() { super.onStart(); Log.d(TAG, "mToken length="+mToken.length); if (!mEnrolling) { FingerprintManager mFpM = (FingerprintManager) getActivity().getSystemService(Context.FINGERPRINT_SERVICE); startEnrollment(mFpM); } } @Override public void onStop() { super.onStop(); if (!getActivity().isChangingConfigurations()) { cancelEnrollment(); } } protected void startEnrollment(FingerprintManager mFpM) { mHandler.removeCallbacks(mTimeoutRunnable); mEnrollmentSteps = -1; mEnrollmentCancel = new CancellationSignal(); if(mFpM != null){ mFpM.enroll(mToken, mEnrollmentCancel, 0, mEnrollmentCallback); } mEnrolling = true; } protected void cancelEnrollment() { mHandler.removeCallbacks(mTimeoutRunnable); if (mEnrolling) { mEnrollmentCancel.cancel(); mEnrolling = false; mEnrollmentSteps = -1; } } public void setListener(Listener listener) { mListener = listener; } public int getEnrollmentSteps() { return mEnrollmentSteps; } public int getEnrollmentRemaining() { return mEnrollmentRemaining; } public boolean isDone() { return mDone; } private FingerprintManager.EnrollmentCallback mEnrollmentCallback = new FingerprintManager.EnrollmentCallback() { @Override public void onEnrollmentProgress(int remaining) { if (mEnrollmentSteps == -1) { mEnrollmentSteps = remaining; } mEnrollmentRemaining = remaining; mDone = remaining == 0; if (mListener != null) { mListener.onEnrollmentProgressChange(mEnrollmentSteps, remaining); } } @Override public void onEnrollmentHelp(int helpMsgId, CharSequence helpString) { if (mListener != null) { mListener.onEnrollmentHelp(helpString); } } @Override public void onEnrollmentError(int errMsgId, CharSequence errString) { if (mListener != null) { mListener.onEnrollmentError(errMsgId, errString); } } }; private final Runnable mTimeoutRunnable = new Runnable() { @Override public void run() { cancelEnrollment(); } }; public interface Listener { void onEnrollmentHelp(CharSequence helpString); void onEnrollmentError(int errMsgId, CharSequence errString); void onEnrollmentProgressChange(int steps, int remaining); } }
FingerPrintActivity:
Java代碼
package com.mediatek.test; import java.io.FileInputStream; import java.io.IOException; import java.util.List; import java.util.Timer; import java.util.TimerTask; import android.R.color; import android.app.Activity; import android.app.AlertDialog; import android.content.Context; import android.content.DialogInterface; import android.content.Intent; import android.graphics.Color; import android.os.Bundle; import android.os.CancellationSignal; import android.os.Handler; import android.os.Message; import android.text.Spannable; import android.text.style.ForegroundColorSpan; import android.util.Log; import android.view.KeyEvent; import android.view.View; import android.view.View.OnClickListener; import android.widget.Button; import android.widget.ImageView; import android.widget.ProgressBar; import android.widget.TableRow; import android.widget.TextView; import android.widget.Toast; import com.mediatek.test.R; import android.hardware.fingerprint.Fingerprint; import android.hardware.fingerprint.FingerprintManager; public class FingerPrintActivity extends Activity implements FingerPrintEnrollBase.Listener{ private static final String TAG = "FingerPrintActivity"; private MyHandler mHandler = new MyHandler(); private TextView fingerPrintAcquisitionTextView, fingerPrintEnrollmentErrorView, fingerPrintEnrollmentHelpView; private FingerPrintEnrollBase fpeb; private FingerprintManager mFingerprintManager; @Override protected int setTitleId() { return R.string.tittle_fingerprint; } @Override protected Activity getActivity() { return this; } @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.fingerprint); fingerPrintAcquisitionTextView = (TextView)findViewById(R.id.fingerprint_message); fingerPrintEnrollmentErrorView= (TextView)findViewById(R.id.fingerprint_enrollment_error); fingerPrintEnrollmentHelpView= (TextView)findViewById(R.id.fingerprint_enrollment_help); mFingerprintManager = (FingerprintManager) getActivity().getSystemService( Context.FINGERPRINT_SERVICE); } private void sendMsgToHandler(int msgType, String remark){ Message msg = Message.obtain(); msg.what = msgType; Bundle bundle = new Bundle(); bundle.putString("result", remark); msg.setData(bundle); mHandler.sendMessage(msg); } @Override public void onEnrollmentHelp(CharSequence helpString) { Log.d(TAG, "==onEnrollmentHelp="+helpString.toString()); sendMsgToHandler(3, helpString.toString()); } @Override public void onEnrollmentError(int errMsgId, CharSequence errString) { Log.d(TAG, "==onEnrollmentError errMsgId="+errMsgId+" errString="+errString.toString()); if(errMsgId == 3){//time out postEnrollment(); }else if(errMsgId == 1){//hardware lose efficacy startEnrollment(); }else if(errMsgId == 5){//operation cancel }else{ sendMsgToHandler(4, errString.toString()); } } @Override public void onEnrollmentProgressChange(int steps, int remaining) { Log.d(TAG, "===============onEnrollmentProgressChange======="); Log.d(TAG, "steps" + steps + "||remaining=" + remaining); fingerPrintEnrollmentHelpView.setText(""); fingerPrintEnrollmentErrorView.setText(""); sendMsgToHandler(2, getString(R.string.fingerprint_validate_success)); buttonS.setEnabled(true); stopFingerprint(); } private void startEnrollment(){ if (fpeb == null) { fpeb = new FingerPrintEnrollBase(); } fpeb.setListener(this); fpeb.startEnrollment(mFingerprintManager); } private void postEnrollment(){ if(mFingerprintManager != null){ mFingerprintManager.postEnroll(); } } private void stopFingerprint() { if (fpeb != null){ fpeb.cancelEnrollment(); onStop(); } } @Override protected void onStart() { super.onStart(); } @Override protected void onStop() { super.onStop(); if(fpeb != null){ fpeb.setListener(null); } } @Override protected void onResume() { super.onResume(); if(mFingerprintManager == null){ mFingerprintManager = (FingerprintManager) getActivity().getSystemService( Context.FINGERPRINT_SERVICE); } startEnrollment(); } @Override protected void onPause() { super.onPause(); postEnrollment(); mFingerprintManager = null; stopFingerprint(); fpeb = null; } class MyHandler extends Handler{ @Override public void handleMessage(Message msg) { Bundle bundle = null; super.handleMessage(msg); switch (msg.what) { case 0: removeMessages(0); bundle = msg.getData(); String result = bundle.getString("result"); selfTestView.setText(result); selfTestView.setTextColor(result.contains("PASS")?Color.GREEN:Color.RED); break; case 1: removeMessages(1); bundle = msg.getData(); String rs = bundle.getString("result"); checkBroadView.setText(bundle.getString("result")); checkBroadView.setTextColor(rs.contains("PASS")?Color.GREEN:Color.RED); break; case 2: removeMessages(2); bundle = msg.getData(); String fingerprint = bundle.getString("result"); fingerPrintAcquisitionTextView.setText(bundle.getString("result")); fingerPrintAcquisitionTextView.setTextColor(fingerprint.contains("PASS")?Color.GREEN:Color.WHITE); mButtonSelftest.setEnabled(true); mButtonCheckerboardTest.setEnabled(true); break; case 3: removeMessages(3); bundle = msg.getData(); String enrollmentHelp = bundle.getString("result"); fingerPrintEnrollmentHelpView.setText(bundle.getString("result")); break; case 4: removeMessages(4); bundle = msg.getData(); String enrollmentError = bundle.getString("result"); fingerPrintEnrollmentErrorView.setText(bundle.getString("result")); fingerPrintAcquisitionTextView.setText(""); break; default: break; } } } }
以上就是對(duì)Android 指紋資料的整理,后續(xù)繼續(xù)補(bǔ)充相關(guān)資料,謝謝大家對(duì)本站的支持!
- Android 指紋識(shí)別詳解及實(shí)現(xiàn)方法
- Android獲取apk簽名指紋的md5值(防止重新被打包)的實(shí)現(xiàn)方法
- android 指紋識(shí)別調(diào)用實(shí)現(xiàn)方法及示例代碼
- Android中的指紋識(shí)別demo開(kāi)發(fā)實(shí)例
- Android 6.0指紋識(shí)別App開(kāi)發(fā)案例
- Android7.0指紋服務(wù)FingerprintService實(shí)例介紹
- Android指紋識(shí)別API初試
- Android6.0指紋識(shí)別開(kāi)發(fā)實(shí)例詳解
- Android6.0指紋識(shí)別開(kāi)發(fā)案例
- Android指紋登錄工具類封裝
相關(guān)文章
Android單選按鈕對(duì)話框用法實(shí)例分析
這篇文章主要介紹了Android單選按鈕對(duì)話框用法,以完整實(shí)例形式分析布局及對(duì)話框類的相關(guān)使用技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-09-09Android開(kāi)發(fā)實(shí)現(xiàn)仿京東商品搜索選項(xiàng)卡彈窗功能
這篇文章主要介紹了Android開(kāi)發(fā)實(shí)現(xiàn)仿京東商品搜索選項(xiàng)卡彈窗功能,涉及Android布局及事件響應(yīng)相關(guān)操作技巧,需要的朋友可以參考下2017-11-11Android 設(shè)置Edittext獲取焦點(diǎn)并彈出軟鍵盤
本文主要介紹了Android設(shè)置Edittext獲取焦點(diǎn)并彈出軟鍵盤的實(shí)現(xiàn)代碼。具有很好的參考價(jià)值。下面跟著小編一起來(lái)看下吧2017-04-04Android內(nèi)存優(yōu)化操作方法梳理總結(jié)
這篇文章主要介紹了Android 內(nèi)存優(yōu)化知識(shí)點(diǎn)梳理總結(jié),Android 操作系統(tǒng)給每個(gè)進(jìn)程都會(huì)分配指定額度的內(nèi)存空間,App 使用內(nèi)存來(lái)進(jìn)行快速的文件訪問(wèn)交互,長(zhǎng)時(shí)間如此便需要優(yōu)化策略,文章分享優(yōu)化知識(shí)點(diǎn)總結(jié),需要的朋友可以參考一下2022-11-11Flutter 滾動(dòng)監(jiān)聽(tīng)及實(shí)戰(zhàn)appBar滾動(dòng)漸變的實(shí)現(xiàn)
這篇文章主要介紹了Flutter 滾動(dòng)監(jiān)聽(tīng)及實(shí)戰(zhàn)appBar滾動(dòng)漸變,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-09-09android現(xiàn)有項(xiàng)目輕微改動(dòng)適配平板心得
這篇文章主要介紹了android現(xiàn)有項(xiàng)目輕微改動(dòng)適配平板心得,對(duì)android適配感興趣的同學(xué)可以參考下2021-04-04Android 表情面板和軟鍵盤切換時(shí)跳閃問(wèn)題的解決方法
這篇文章主要介紹了Android 表情面板和軟鍵盤切換時(shí)跳閃問(wèn)題的解決方法,需要的朋友可以參考下2017-08-08