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. 启动页面代码
package com.wkq.nfc

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读取卡片数据的操作,具体的数据处理只是简单的处理了北京公交卡的数据,具体项目业务上需要读取什么卡数据需要项目中具体去处理.

参考:https://www.jianshu.com/p/c7961cea3b5b

### 回答1: Android调用NFC读取NFC卡片的过程如下:首先,需要确认设备的硬件支持NFC功能,并且设备的操作系统版本必须是Android 4.0或更高版本。然后,在Android应用程序中,需要在AndroidManifest.xml文件中添加必要的权限和NFC相关的配置。 接下来,需要在应用程序中注册一个NFC适配器对象,以便与设备的NFC芯片进行通信。然后,可以通过检测NFC设备是否处于范围内来启动应用程序的特定操作,例如读取卡片中的数据。 读取NFC卡片的过程包括以下步骤:首先,需要创建一个PendingIntent对象,以便在检测到NFC设备时接收通知。然后,使用NFC适配器对象调用enableForegroundDispatch()方法,将PendingIntent对象传递给它。这将使应用程序在检测到NFC设备时接收到通知,并初始化NFC操作。 在读取NFC卡片之前,需要创建一个IntentFilter对象,以过滤掉不相关的NFC标签。可以使用ACTION_NDEF_DISCOVERED和ACTION_TECH_DISCOVERED等值来设置过滤器。然后,将IntentFilter对象传递给NFC适配器的enableForegroundDispatch()方法。 当NFC适配器检测到NFC设备时,将调用应用程序的onNewIntent()方法。在onNewIntent()方法中,可以通过调用getParcelableExtra()方法获取传递给PendingIntent对象的Intent,并从中提取出NFC标签的数据。 最后,可以使用读取到的NFC标签数据进行其他的处理操作,例如解析数据、显示在应用界面上或将其发送到远程服务器等。 在读取NFC卡片的过程中需要注意的是,NFC设备必须处于活动状态,并且应用程序必须拥有前台调度权限,才能成功读取NFC卡片。此外,由于不同的NFC卡片类型和厂商可能有不同的协议和数据格式,因此需要根据具体的NFC卡片来实现相应的处理逻辑。 ### 回答2: Android调用NFC读取NFC卡片的过程如下: 首先,需要在AndroidManifest.xml文件中添加必要的权限和特性声明,以确保应用能够访问NFC功能。例如,需要添加以下权限声明: ``` <uses-permission android:name="android.permission.NFC" /> ``` 在应用的Activity中,需要注册一个NFC监听器,以接收NFC设备的读取事件。可以通过以下方式实现: ``` private NfcAdapter mNfcAdapter; ... @Override protected void onCreate(Bundle savedInstanceState) { ... mNfcAdapter = NfcAdapter.getDefaultAdapter(this); ... } ... @Override public void onResume() { super.onResume(); if (mNfcAdapter != null) { Intent intent = new Intent(this, this.getClass()).addFlags(Intent.FLAG_RECEIVER_REPLACE_PENDING); PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, intent, 0); IntentFilter[] intentFilters = new IntentFilter[]{}; mNfcAdapter.enableForegroundDispatch(this, pendingIntent, intentFilters, null); } } ... @Override public void onPause() { super.onPause(); if (mNfcAdapter != null) { mNfcAdapter.disableForegroundDispatch(this); } } ... @Override public void onNewIntent(Intent intent) { super.onNewIntent(intent); if (intent.hasExtra(NfcAdapter.EXTRA_TAG)) { Tag tag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG); // 在此处理读取到的NFC卡片数据 } } ``` 以上代码中,`mNfcAdapter`是NFC适配器实例,`onResume()`方法用于在应用进入前台时启用前台调度,以便接收NFC设备事件,`onPause()`方法用于在应用进入后台时禁用前台调度,`onNewIntent()`方法用于处理读取到的NFC卡片数据。 在`onNewIntent()`方法中,可以通过`intent.getParcelableExtra(NfcAdapter.EXTRA_TAG)`来获取到读取到的NFC标签,并进行相应的操作,例如读取和写入卡片数据等。 需要注意的是,以上代码中只展示了基本的NFC读取流程,具体的读取和处理操作需要根据需求进一步实现。 ### 回答3: Android调用NFC读取NFC卡片需要经过以下的步骤: 首先,确保设备支持NFC功能。大多数Android设备都已经集成了NFC芯片,但仍然需要在设备的设置中确认是否启用NFC功能。 然后,在Android的Manifest文件中添加必要的权限和配置。这些权限包括NFC权限,以及与NFC相关的Intent过滤器。例如,可以添加以下代码来声明NFC权限: <uses-permission android:name="android.permission.NFC"/> 同时,还需要添加一个Intent过滤器,以便应用程序能够响应NFC相关的意图。以下是示例代码: <intent-filter> <action android:name="android.nfc.action.TECH_DISCOVERED"/> </intent-filter> 接下来,需要创建一个能够响应NFC意图的活动或服务。这可以通过继承Android提供的NFC活动类或服务类来实现。在这个活动或服务中,可以使用NFC适配器来检测和连接到NFC标签。以下是示例代码: NfcAdapter nfcAdapter = NfcAdapter.getDefaultAdapter(this); if (nfcAdapter != null && nfcAdapter.isEnabled()) { // NFC已经启用,可以进行相关操作 } else { // 设备不支持或未启用NFC功能 } 在检测到NFC标签后,可以通过NFC适配器获取NFC标签的ID或数据。例如,可以使用以下代码获取NFC标签的ID: Tag tag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG); String tagId = bytesToHexString(tag.getId()); 最后,可以根据需要对NFC标签的ID或数据进行进一步的处理。例如,可以将ID与系统中已知的标签ID进行比较,或者读取和写入NFC标签的数据。 总之,Android调用NFC读取NFC卡片需要确认设备支持NFC功能、添加必要的权限和配置、创建能够响应NFC意图的活动或服务、使用NFC适配器进行NFC标签的检测和连接,并对NFC标签的ID或数据进行处理。以上是一个简单的概述,更详细的实现可以参考Android开发文档中关于NFC的相关内容。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值