美文网首页Android开发Android开发经验谈Android开发
Android 使用手机NFC的读取NFC标签数据

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

作者: 秋阳君 | 来源:发表于2018-07-10 15:21 被阅读35次

    一 你需要准备的:一部有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 识别标签的顺序


    image.png

    四 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 使用手机NFC的读取NFC标签数据

        本文链接:https://www.haomeiwen.com/subject/rhjvuftx.html