android指纹java_Android

Android M指纹的资料太少,经过一段时间阅读原生Android代码,写了以下例子,贡献出来给需要帮助的人。

以下内容基于64位的高通CPU,搭载fpc1020芯片,此部分代码在原生android上做了更改,以应付工厂指纹的测试。原生android指纹录入需要采集10次(因不同的芯片而定)。

代码简单说明:

1. FingerPrintEnrollBase类:

重要的是

Java代码

public interface Listener {

void onEnrollmentHelp(CharSequence helpString);

void onEnrollmentError(int errMsgId, CharSequence errString);

void onEnrollmentProgressChange(int steps, int remaining);

}

这个接口。

1)onEnrollmentHelp函数:是下层返回的指纹操作帮助信息。

2)onEnrollmentError函数是下层返回的错误信息,errMsgId是错误信息类型,其中1表示指纹硬件不可用,3表示超时,5表示操作已取消。errString是errMsgId对应的文字信息,有需要的自己实验就知道了。

3)onEnrollmentProgressChange()函数是显示当前指纹的采集情况,steps:总共需要录入多少次;remaining是剩余多少次,我在此函数中有打印Log。

2. FingerPrintActivity是主Activity类,FingerPrintEnrollBase是指纹Base类。

需要特别说明的是,在FingerPrintEnrollBase类中有一句:private byte[] mToken = new byte[69];这个数组正常情况是通过输入密码后经由framework层向hal层申请的,我这里为了省去验证那一块的代码,通过sizeof(hw_auth_token_t)计算出来的为69,需要的请依此计算。我在代码里实验过,此数组如果不是69(在64位CPU上),将直接导致程序crach!

fpc指纹的采集流程图:

15749c37bba941aafb30e8590e8da9cb.png

源码如下:

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;

}

}

}

}

以上就是对Android 指纹资料的整理,后续继续补充相关资料,谢谢大家对本站的支持!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/574652.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

about-ie下模拟input file上传功能失效

Q:IE9下file提交到iframe中,load一直不触发,其他高级浏览器均无此问题 解决方案:不使用js模拟 input click事件,取而代之的是把真实的input设置为要触发元素的大小,进行触发onchange原因分析:低…

java把收集的数据节点_java面试题收集(04)

1、rebbitmq的使用场景有哪些?(1)单发送单接受使用场景:简单的发送与接受,没有特别的处理。(2)单发送多接受使用场景:一个发送端,多个接收端,如分布式的任务发布,要保证消息发送的可靠性&#x…

go java gc_图解Golang的GC垃圾回收算法

虽然Golang的GC自打一开始,就被人所诟病,但是经过这么多年的发展,Golang的GC已经改善了非常多,变得非常优秀了。以下是Golang GC算法的里程碑:v1.1 STWv1.3 Mark STW, Sweep 并行v1.5 三色标记法v1.8 hybrid write bar…

统计学的经典书籍指南 (转载自:http://blog.csdn.net/ysuncn/article/details/1943679)...

一、统计学基础部分 1、《统计学》 David Freedman等著,魏宗舒,施锡铨等译 中国统计出版社 据说是统计思想讲得最好的一本书,读了部分章节,受益很多。整本书几乎没有公式,但是讲到了统计思想的精髓。 2、《Mind on …

Linux Bash Shell j简单入门

BASH 的基本语法 最简单的例子 —— Hello World! 关于输入、输出和错误输出 BASH 中对变量的规定(与 C 语言的异同) BASH 中的基本流程控制语法 函数的使用 2.1 最简单的例子 —— Hello World! 几乎所有的讲解编程的书给读者的第一个例子都是 …

jpa mysql脚本迁移_JPA通过LOAD DATA LOCAL INFILE大批量导入数据到MySQL

连接层代码:import org.hibernate.Session;import org.hibernate.internal.SessionFactoryImpl;import org.springframework.jdbc.support.nativejdbc.C3P0NativeJdbcExtractor;import java.sql.Connection;import java.sql.PreparedStatement;import java.sql.SQLE…

javascript的事件冒泡,阻止事件冒泡和事件委托, 事件委托是事件冒泡的一个应用。...

2018年12月13日更新 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>Title</title><style>ul li{border: 1px solid yellow;}</style> </head> <body> <p hidden>…

php 序列化 java_PHP--序列化与反序列化详解

PHP--序列化与反序列化详解博客说明文章所涉及的资料来自互联网整理和个人总结&#xff0c;意在于个人学习和经验汇总&#xff0c;如有什么地方侵权&#xff0c;请联系本人删除&#xff0c;谢谢&#xff01;说明所有php里面的值都可以使用函数serialize()来返回一个包含字节流的…

hadoop2.x的变化

HDFS Federation&#xff08;HDFS联邦&#xff09; HDFS有两个主要层&#xff1a; Namespace 由目录、文件和块组成&#xff1b;支持所有命名空间对文件和目录的操作。 Block Storage Service 由Block Management和Storage组成。 Block Management 提供dataNode集群成员关系&am…

python enumeration_python枚举防止无效的属性分配

要使枚举类完全“只读”,只需要使用__setattr__hook防止全部的属性分配。因为元类附加到类之后它是创建的,分配正确的枚举值没有问题。就像伊桑的回答一样,我用的是EnumMeta类作为自定义元类的基础:from enum import EnumMeta, Enumclass FrozenEnumMeta(EnumMeta):"Enum …

java wps linux 安装_ubuntu安装Java开发环境

1. 从sun主页JDK for Linux版本。这里的是jdk-6u6--i586.bin.2. 用root用户登录ubuntu&#xff0c;或是在普通用户下用su命令切换用户。切换到所需的安装目录。类型&#xff1a;cd 例如&#xff0c;要在 /usr/java/ 目录中 安装软件&#xff0c;请键入&#xff1a;cd /usr…

[BZOJ 2594] [Wc2006]水管局长数据加强版 【LCT】

题目链接&#xff1a;BZOJ - 2594 题目分析 这道题如果没有删边的操作&#xff0c;那么就是 NOIP2013 货车运输&#xff0c;求两点之间的一条路径&#xff0c;使得边权最大的边的边权尽量小。 那么&#xff0c;这条路径就是最小生成树上这两点之间的路径。 然而现在有了删边操作…

vs升级c++项目遇到的一些问题

1、error C1189: #error : This file requires _WIN32_WINNT to be #defined at least to 0x0403. Value 0x0501 or higher is recommended. 在工程的stdafx.h中&#xff0c;改为如下配置&#xff1a; #ifndef WINVER #define WINVER 0x0501 #endif #ifndef _WIN32_WINNT #def…

python模糊匹配忽略大小写_在python中忽略大小写的简单方法?

如果搜索多个单词&#xff0c;则创建一个集合是有意义的&#xff1a;print(set(brown_sents).intersection(zip(repeat(most_ambiguous_word),word_class_dict[most_ambiguous_word])))输出{(word2, wordclass2), (word2, wordclass3)}要理解它的作用&#xff0c;请将脚本保存到…

IAR7.51提示秘钥无效IAR 以及 CCDebug驱动(包含win7 64bit)

今天IAR不识别我的仿真器&#xff0c;然后我感觉驱动有问题&#xff0c;就把之前的驱动卸载了&#xff0c;但是按照以前的方法按章驱动(选择路径到IAR的某个目录)&#xff0c;提示找不到驱动。。。 也不想重新装个IAR了&#xff0c;于是到CSDN上下载了这个驱动程序&#xff0c;…

cannot be resolved or is not a field

我通常的解决办法&#xff1a;1.看看是不是manifest.xml文件有错误&#xff0c;如果有的话&#xff0c;R.java是不会生成的2.一般来说R.layout.aaa.xml&#xff0c;这里的R是自己的包的R.java&#xff0c;不是android系统的R.java&#xff0c;所以如果引入包的时候有import and…

java解析时已到达文件结尾_IO流读取到文件末尾继续读取

import java.io.BufferedInputStream;import java.io.BufferedOutputStream;import java.io.File;import java.io.FileInputStream;import java.io.FileNotFoundException;import java.io.IOException;//正常情况//main()方法中第一个read()和和第二个read();是可以顺利执行完的…

百度地图动态显示查询结构

<script type"text/javascript">// 百度地图API功能function G(id) {return document.getElementById(id);}var map new BMap.Map("l-map");map.centerAndZoom("北京",12); // 初始化地图,设置城市和地图级别。var ac …

solr5.0mysql_ik扩展支持Solr配置详解

扩展ik原生代码&#xff1a;public class IKAnalyzerTokenizerFactory extends TokenizerFactory{private boolean useSmart;public boolean useSmart(){return this.useSmart;}public void setUseSmart(boolean useSmart) {this.useSmart useSmart;}public IKAnalyzerTokeniz…

apache htpasswd.exe创建密码

一、使用apache htpasswd.exe创建密码文件&#xff0c;命令请看PHP推荐教程&#xff1a;apache htpasswd命令用法详解 apache htpasswd命令用法实例 1、如何利用htpasswd命令添加用户&#xff1f; 在apache安装目录bin下找到htpasswd #usr/local/apache/bin/htpasswd -bc .pass…