Android 使用手机NFC的读取NFC标签数据的方法

 更新时间:2018年07月18日 08:29:08   作者:秋阳君  
这篇文章主要介绍了Android 使用手机NFC的读取NFC标签数据的方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧

一 你需要准备的:

一部有nfc的手机,一张有nfc标签的卡

二 nfc简介

nfc(近距离无线通讯技术),是由非接触式射频识别(RFID)及互连互通技术整合演变而来,通过在单一芯片上集成感应式读卡器、感应式卡片和点对点通信的功能,利用移动终端实现移动支付、电子票务、门禁、移动身份识别、防伪等应用。

三 nfc过滤标签的设置

3-1 在Manifest添加权限:

在xml里添加nfc的使用权限

<uses-permission android:name="android.permission.NFC" />

这个是限制安装权限,只给有nfc功能的手机安装(可选)

<uses-feature android:name="android.hardware.nfc" android:required="true" />

3-2 nfc的过滤方式有以下:

  • ACTION_NDEF_DISCOVERED,
  • ACTION_TECH_DISCOVERED,
  • ACTION_TAG_DISCOVERED三种。过滤器的作用是过滤掉杂质,剩下的就是我们需要的了。这三种过滤方式可同时配置,可以比方成从上到下三层,只要是符合某一层过滤器要求的,过滤完就停止往下一层。

在Activity的filter里面添加对应需要的权限:

ACTION_NDEF_DISCOVERED,

<activity>
...
      <intent-filter>
        <action android:name="android.nfc.action.NDEF_DISCOVERED" />
      </intent-filter>
...
</activity>

ACTION_TECH_DISCOVERED:

在<project-root>/res/xml(自己新建xml文件夹)下新建一个nfc_tech_filter.xml文件,添加进你需要支持的标签类型。(下面的配置项可多选)。下列示例是支持与NfcA和Ndef技术的NFC标签匹配。

<resources xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
  <tech-list>
    <resources >
  <tech-list>
    <!--<tech>android.nfc.tech.IsoDep</tech>-->
    <tech>android.nfc.tech.NfcA</tech>
    <!--<tech>android.nfc.tech.NfcB</tech>-->
    <!--<tech>android.nfc.tech.NfcF</tech>-->
    <!--<tech>android.nfc.tech.NfcV</tech>-->
    <tech>android.nfc.tech.Ndef</tech>
    <!--<tech>android.nfc.tech.NdefFormatable</tech>-->
    <!--<tech>android.nfc.tech.MifareClassic</tech>-->
    <!--<tech>android.nfc.tech.MifareUltralight</tech>-->
  </tech-list>
</resources>
  </tech-list>
</resources>
<activity>
...
      <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" />

<meta-data android:name="android.nfc.action.TECH_DISCOVERED"
  android:resource="@xml/nfc_tech_filter" />
...
</activity>
ACTION_TAG_DISCOVERED,可以添加如下权限

<activity>
···
      <intent-filter>
        <action android:name="android.nfc.action.TAG_DISCOVERED" />
      </intent-filter>
···
</activity>

3-3 识别标签的顺序

四 nfc读操作(我们读取NEDF数据,其他公交卡类型的数据可以自行研究)

1 初始化nfc工具,判断是否存在nfc和nfc是否打开

2 感应到nfc标签后,读取解析对应nfc类型的标签数据

3 回传显示

代码如下:

public class NfcActivity extends Activity {

  private static final String TAG = "NfcActivity";
  private TextView tvNFCMessage;
  private PendingIntent mPendingIntent;
  private NfcAdapter mNfcAdapter;
  private Button btnClean;

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_nfc);
    Log.i(TAG, "onCreate: ");
    btnClean = findViewById(R.id.btn_clean);
    tvNFCMessage = findViewById(R.id.tv_show_nfc);


    resolveIntent(getIntent());

    //初始化nfc
    mNfcAdapter = NfcAdapter.getDefaultAdapter(this);
    mPendingIntent = PendingIntent.getActivity(this, 0,
        new Intent(this, getClass()).addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP), 0);
    if (mNfcAdapter == null) {
      Toast.makeText(NfcActivity.this, "nfc is not available", Toast.LENGTH_SHORT).show();
      finish();
      return;
    }


    btnClean.setOnClickListener(new View.OnClickListener() {
      @Override
      public void onClick(View v) {
        tvNFCMessage.setText("");
      }
    });
  }

  @Override
  protected void onResume() {
    super.onResume();
    Log.i(TAG, "onResume: ");
    if (mNfcAdapter != null) { //有nfc功能
      if (mNfcAdapter.isEnabled()) {
        //nfc功能打开了
        //隐式启动
        mNfcAdapter.enableForegroundDispatch(this, mPendingIntent, null, null);
      } else {
        Toast.makeText(NfcActivity.this, "请打开nfc功能", Toast.LENGTH_SHORT).show();
      }
    }
  }

  @Override
  protected void onNewIntent(Intent intent) {
    super.onNewIntent(intent);
    Log.i(TAG, "onNewIntent: ");
    setIntent(intent);
    if (mNfcAdapter != null) { //有nfc功能
      if (mNfcAdapter.isEnabled()) {//nfc功能打开了
        resolveIntent(getIntent());
      } else {
        Toast.makeText(NfcActivity.this, "请打开nfc功能", Toast.LENGTH_SHORT).show();
      }
    }
  }


  @Override
  protected void onPause() {
    super.onPause();
    if (mNfcAdapter != null) {
      mNfcAdapter.disableForegroundDispatch(this);
    }
  }

  //初次判断是什么类型的NFC卡
  private void resolveIntent(Intent intent) {
    NdefMessage[] msgs = NfcUtil.getNdefMsg(intent); //重点功能,解析nfc标签中的数据

    if (msgs == null) {
      Toast.makeText(NfcActivity.this, "非NFC启动", Toast.LENGTH_SHORT).show();
    } else {
      setNFCMsgView(msgs);
    }

  }

  /**
   * 显示扫描后的信息
   *
   * @param ndefMessages ndef数据
   */
  @SuppressLint("SetTextI18n")
  private void setNFCMsgView(NdefMessage[] ndefMessages) {
    if (ndefMessages == null || ndefMessages.length == 0) {
      return;
    }

//    tvNFCMessage.setText("Payload:" + new String(ndefMessages[0].getRecords()[0].getPayload()) + "\n");

    Calendar calendar = Calendar.getInstance();
    int hour = calendar.get(Calendar.HOUR_OF_DAY);
    int minute = calendar.get(Calendar.MINUTE);
    tvNFCMessage.append(hour + ":" + minute + "\n");
    List<ParsedNdefRecord> records = NdefMessageParser.parse(ndefMessages[0]);
    final int size = records.size();
    for (int i = 0; i < size; i++) {
      ParsedNdefRecord record = records.get(i);
      tvNFCMessage.append(record.getViewText() + "\n");
    }
  }

}

解析不同类型nfc类型的数据的方法(重点方法):

//初次判断是什么类型的NFC卡
  public static NdefMessage[] getNdefMsg(Intent intent) {
    if (intent == null)
      return null;

    //nfc卡支持的格式
    Tag tag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);
    String[] temp = tag.getTechList();
    for (String s : temp) {
      Log.i(TAG, "resolveIntent tag: " + s);
    }


    String action = intent.getAction();

    if (NfcAdapter.ACTION_NDEF_DISCOVERED.equals(action) ||
        NfcAdapter.ACTION_TECH_DISCOVERED.equals(action) ||
        NfcAdapter.ACTION_TAG_DISCOVERED.equals(action)) {
      Parcelable[] rawMessage = intent.getParcelableArrayExtra(NfcAdapter.EXTRA_NDEF_MESSAGES);
      NdefMessage[] ndefMessages;

      // 判断是哪种类型的数据 默认为NDEF格式
      if (rawMessage != null) {
        Log.i(TAG, "getNdefMsg: ndef格式 ");
        ndefMessages = new NdefMessage[rawMessage.length];
        for (int i = 0; i < rawMessage.length; i++) {
          ndefMessages[i] = (NdefMessage) rawMessage[i];
        }
      } else {
        //未知类型 (公交卡类型)
        Log.i(TAG, "getNdefMsg: 未知类型");
        //对应的解析操作,在Github上有
      }


      return ndefMessages;
    }

    return null;
  }

最后,附上我的demo地址,欢迎大家学习下载,有什么问题也欢迎找我讨论:

https://github.com/younger96/NFCRead

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • Android点击Button实现功能的几种方法总结

    Android点击Button实现功能的几种方法总结

    当Button有多个或者Button的使用次数很多时,我们需要采用绑定监听器的做法,其实,绑定监听器也有几种方法,不过,我在这里就不一一列举了,毕竟那些方法在实际的应用中也不常见
    2013-10-10
  • Android测试方法总结

    Android测试方法总结

    在这篇文章中我们给大家总结了Android测试方法以及需要注意的地方,有兴趣的朋友参考学习下。
    2018-07-07
  • android计算器代码示例分享

    android计算器代码示例分享

    这篇文章主要介绍了android实现的简单计算器代码示例,大家参考使用吧
    2014-01-01
  • Android获取设备传感器的方法

    Android获取设备传感器的方法

    这篇文章主要为大家详细介绍了Android获取设备传感器的方法,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-09-09
  • Recyclerview添加头布局和尾布局、item点击事件详解

    Recyclerview添加头布局和尾布局、item点击事件详解

    这篇文章主要为大家详细介绍了Recyclerview添加头布局和尾布局、item点击事件的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-08-08
  • 一文详解无痕埋点在Android中的实现

    一文详解无痕埋点在Android中的实现

    很多时候因为产品都会要获取用户的行为,需要客户端进行相关的埋点,下面这篇文章主要给大家介绍了关于无痕埋点在Android中实现的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2023-02-02
  • Android实现简易闹钟功能

    Android实现简易闹钟功能

    这篇文章主要为大家详细介绍了Android实现简易闹钟功能,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-06-06
  • Android LayerDrawable使用实例

    Android LayerDrawable使用实例

    这篇文章主要介绍了Android LayerDrawable使用实例,本文讲解了LayerDrawable的作用、LayerDrawable的原理、LayerDrawableLayerDrawable的使用实例等,需要的朋友可以参考下
    2015-06-06
  • Android实现相机拍摄、选择、图片裁剪功能

    Android实现相机拍摄、选择、图片裁剪功能

    自定义控件,重写ImageView 功能实现:点击圆形头像之后可以实现相册上传或者开启相机,然后把得到的图片经过剪裁,把剪裁过的图片设置为头像的背景图,需要的朋友可以参考下
    2016-09-09
  • Android实现消息提醒小红点效果

    Android实现消息提醒小红点效果

    这篇文章主要为大家详细介绍了Android实现消息提醒小红点效果,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-11-11

最新评论