Android如何实现NFC读取卡片信息

 更新时间:2023年11月22日 16:58:43   作者:金戈鐡馬  
这篇文章主要介绍了Android如何实现NFC读取卡片信息问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
(福利推荐:【腾讯云】服务器最新限时优惠活动,云服务器1核2G仅99元/年、2核4G仅768元/3年,立即抢购>>>:9i0i.cn/qcloud

(福利推荐:你还在原价购买阿里云服务器?现在阿里云0.8折限时抢购活动来啦!4核8G企业云服务器仅2998元/3年,立即抢购>>>:9i0i.cn/aliyun

效果图

因为朋友需要个读取NFC卡片数据的功能,所以最近看了一下Android 系统下NFC 读取卡片信息的操作。

NFC(近距离无线通信 ) 是一组近距离无线技术,通常只有在距离不超过 4 厘米时才能启动连接.借助 NFC,您可以在 NFC 标签与 Android 设备之间或者两台 Android 设备之间共享小型负载。

支持NFC的Android设备同时支持以下三种主要操作模式

  • 读取器/写入器模式:支持 NFC 设备读取和/或写入被动 NFC 标签和贴纸。
  • 点对点模式:支持 NFC 设备与其他 NFC 对等设备交换数据;- Android Beam 使用的就是此操作模式。
  • 卡模拟模式:支持 NFC 设备本身充当 NFC 卡。然后,可以通过外部 NFC 读取器(例如 NFC 销售终端)访问模拟 NFC 卡。

NFC读取卡片数据流程

  • Android 设备通常会在屏幕解锁后查找 NFC 标签(停用NFC除外)
  • 卡片接近启动标签调度系统
  • 数据通过Intent携带数据启动Activity

标签调度系统定义了三种 Intent,按优先级从高到低列出如下:    

1.  ACTION_NDEF_DISCOVERED:如果扫描到包含 NDEF 负载的标签,并且可识别其类型,则使用此 Intent 启动 Activity。这是优先级最高的 Intent,标签调度系统会尽可能尝试使用此 Intent 启动 Activity,在行不通时才会尝试使用其他 Intent。    

2.  ACTION_TECH_DISCOVERED :如果没有登记要处理 ACTION_NDEF_DISCOVERED Intent 的 Activity,则标签调度系统会尝试使用此 Intent 来启动应用。此外,如果扫描到的标签包含无法映射到 MIME 类型或 URI 的 NDEF 数据,或者该标签不包含 NDEF 数据,但它使用了已知的标签技术,那么也会直接启动此 Intent(无需先启动 ACTION_NDEF_DISCOVERED)。    

3.  ACTION_TAG_DISCOVERED:如果没有处理 ACTION_NDEF_DISCOVERED 或者 ACTION_TECH_DISCOVERED Intent 的 Activity,则使用此 Intent 启动 Activity。

  • 启动Activity 处理Intent携带的数据

实现读取北京地铁卡数据功能

1. 配置NFC权限

    <!--    API 级别 9 仅通过  所以最低是10版本-->
    <uses-sdk android:minSdkVersion="10" />
    <!--    NFC 权限  -->
    <uses-permission android:name="android.permission.NFC" />
    <!--    以便您的应用仅在那些具备 NFC 硬件的设备的 Google Play 中显示:-->
    <uses-feature
        android:name="android.hardware.nfc"
        android:required="true" />

2. 配置NFC拉起页面的过滤器选项

        <!--NFC启动的页面 -->
        <activity android:name=".NFCActivity">
            <!--  配置过滤启动类型-->
            <intent-filter>
                <action android:name="android.nfc.action.TECH_DISCOVERED" />
            </intent-filter>
            <meta-data
                android:name="android.nfc.action.TECH_DISCOVERED"
                android:resource="@xml/nfc_tech_filter" />
 
            <!--            <intent-filter>-->
            <!--                <action android:name="android.nfc.action.NDEF_DISCOVERED"/>-->
            <!--                <category android:name="android.intent.category.DEFAULT"/>-->
            <!--                <data android:scheme="http"-->
            <!--                    android:host="developer.android.com"-->
            <!--                    android:pathPrefix="/index.html" />-->
            <!--            </intent-filter>-->
 
            <!--            <intent-filter>-->
            <!--                <action android:name="android.nfc.action.NDEF_DISCOVERED"/>-->
            <!--                <category android:name="android.intent.category.DEFAULT"/>-->
            <!--                <data android:mimeType="text/plain" />-->
            <!--            </intent-filter>-->
 
        </activity>

注意 nfc_tech_filter.xml 是过滤NFC 卡片类型

<?xml version="1.0" encoding="utf-8"?>
<resources xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
    <!-- 可以处理所有Android支持的NFC类型 -->
    <tech-list>
        <tech>android.nfc.tech.IsoDep</tech>
    </tech-list>
    <tech-list>
        <tech>android.nfc.tech.NfcA</tech>
    </tech-list>
    <tech-list>
        <tech>android.nfc.tech.NfcB</tech>
    </tech-list>
    <tech-list>
        <tech>android.nfc.tech.NfcF</tech>
    </tech-list>
    <tech-list>
        <tech>android.nfc.tech.NfcV</tech>
    </tech-list>
    <tech-list>
        <tech>android.nfc.tech.Ndef</tech>
    </tech-list>
    <tech-list>
        <tech>android.nfc.tech.NdefFormatable</tech>
    </tech-list>
    <tech-list>
        <tech>android.nfc.tech.MifareUltralight</tech>
    </tech-list>
    <tech-list>
        <tech>android.nfc.tech.MifareClassic</tech>
    </tech-list>
</resources>

4. 启动页面代码

import android.content.Intent
import android.nfc.NdefMessage
import android.nfc.NdefRecord.createMime
import android.nfc.NfcAdapter
import android.nfc.NfcEvent
import android.nfc.Tag
import android.os.Bundle
import android.widget.Toast
import androidx.appcompat.app.AppCompatActivity
import androidx.databinding.DataBindingUtil
import com.wkq.nfc.databinding.ActivityMainBinding
 
/**
 * NFC 拉起页面
 */
class NFCActivity : AppCompatActivity(), NfcAdapter.CreateNdefMessageCallback {
 
    //支持的标签类型
    private var nfcAdapter: NfcAdapter? = null
    private var binding: ActivityMainBinding? = null
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        binding = DataBindingUtil.setContentView<ActivityMainBinding>(this, R.layout.activity_main)
        nfcAdapter = NfcAdapter.getDefaultAdapter(this)
        if (nfcAdapter==null){
            Toast.makeText(this, "该机型不支持NFC", Toast.LENGTH_LONG).show()   
            finish()
        }
        // Register callback  *设置一个回调,使用Android Beam(TM)动态生成要发送的NDEF消息。
        nfcAdapter?.setNdefPushMessageCallback(this, this)
    }
 
 
    override fun onResume() {
        super.onResume()
        // Check to see that the Activity started due to an Android Beam
        if (NfcAdapter.ACTION_TECH_DISCOVERED == intent.action) {
            processIntent(intent)
        }
    }
 
    override fun onPause() {
        super.onPause()
        nfcAdapter!!.disableReaderMode(this)
    }
 
    override fun onNewIntent(intent: Intent?) {
        super.onNewIntent(intent)
        setIntent(intent)
    }
 
    /**
     * 处理Intent携带的数据
     */
    private fun processIntent(intent: Intent) {
        // 处理北京公交卡的数据
        var tag = intent.extras
        if (tag==null)return
  
        var content = NFCUtil.bytesToHex((tag!!.get("android.nfc.extra.TAG") as Tag).id)
        binding?.tvContent!!.text = content
        Toast.makeText(this, "获取北京地铁卡数据:" + content, Toast.LENGTH_LONG).show()
    }
 
    override fun createNdefMessage(event: NfcEvent?): NdefMessage {
        val text = "Beam me up, Android!\n\n" +
                "Beam Time: " + System.currentTimeMillis()
        return NdefMessage(
            arrayOf(
                createMime("application/vnd.com.example.android.beam", text.toByteArray())
            )
        )
 
    }
}

这里是简单的利用NFC读取卡片数据的操作,具体的数据处理只是简单的处理了北京公交卡的数据,具体项目业务上需要读取什么卡数据需要项目中具体去处理。

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持程序员之家。

相关文章

  • Android 判断是否连接成功了指定wifi

    Android 判断是否连接成功了指定wifi

    本文主要介绍了Android 判断是否连接成功了指定wifi的相关知识。具有很好的参考价值。下面跟着小编一起来看下吧
    2017-04-04
  • Android屏幕旋转 处理Activity与AsyncTask的最佳解决方案

    Android屏幕旋转 处理Activity与AsyncTask的最佳解决方案

    运行时变更就是设备在运行时发生变化(例如屏幕旋转、键盘可用性及语言)。发生这些变化,Android会重启Activity,这时就需要保存activity的状态及与activity相关的任务,以便恢复activity的状态。为此,google提供了三种解决方案,本文将对这三种方案进行逐一介绍。
    2016-12-12
  • Android View 事件防抖的两种方案

    Android View 事件防抖的两种方案

    这篇文章主要介绍了Android View 事件防抖的两种方案,帮助大家更好的理解和学习使用Android,感兴趣的朋友可以了解下
    2021-04-04
  • Android实现伴奏录音合成MP3

    Android实现伴奏录音合成MP3

    这篇文章主要为大家详细介绍了Android实现伴奏录音合成MP3,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-01-01
  • Android Gradle依赖管理、去除重复依赖、忽略的方式

    Android Gradle依赖管理、去除重复依赖、忽略的方式

    这篇文章主要介绍了Android Gradle依赖管理、去除重复依赖、忽略的方式,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-03-03
  • Android实现环形进度条

    Android实现环形进度条

    这篇文章主要为大家详细介绍了Android实现环形进度条,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-07-07
  • PowerManagerService之自动灭屏流程解析

    PowerManagerService之自动灭屏流程解析

    这篇文章主要为大家介绍了PowerManagerService之自动灭屏流程解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-10-10
  • Android开发框架之自定义ZXing二维码扫描界面并解决取景框拉伸问题

    Android开发框架之自定义ZXing二维码扫描界面并解决取景框拉伸问题

    这篇文章主要介绍了Android开发框架之自定义ZXing二维码扫描界面并解决取景框拉伸问题的相关资料,非常不错具有参考借鉴价值,需要的朋友可以参考下
    2016-06-06
  • Android?Framework如何实现Binder

    Android?Framework如何实现Binder

    这篇文章主要介绍了Android?Framework如何实现Binder,文章围绕主题展开详细的内容介绍,具有一定的参考价值,需要的小伙伴可以参考一下
    2022-09-09
  • Android开发人脸识别登录功能

    Android开发人脸识别登录功能

    这篇文章主要介绍了Android开发人脸识别登录功能,这个很多公司都在使用,非常流行,今天小编给大家从头到尾做一个案例分享到程序员之家平台,需要的朋友参考下吧
    2019-11-11

最新评论

?


http://www.vxiaotou.com