NFC开发入门篇

﹥>﹥吖頭↗ 提交于 2019-12-07 11:29:50

之前做过一些关于NFC的项目,比如手机羊城通卡芯片的充值,粤通卡的充值(与微信的羊城通充值功能类似),NFC功能是需要硬件的支持的,不过现在越来越多手机支持NFC功能了,所以这里简单的介绍它的开发流程吧。

一、NFC简介

NFC(Near Field Communication)近距离无线通信技术。工作模式分为两种:卡模式(Card emulation)、点对点模式(P2P mode)。这里主要是介绍卡模式的的开发,卡模式的开发又分为机卡通道和非机卡通道两种,机卡通道就是手机卡里面集成了相关的芯片,一卡多用,卡离开手机之后无法工作,比如手机卡里面又附带羊城通的芯片的功能,这样既可以打电话又可以刷公交和地铁。非机卡通道就是卡和手机是分开使用的,就是平常使用单独一张公交卡一样。

二、开发流程

1、非机卡通道模式

(1)在AndroidManifest.xml中声明权限

<uses-permission android:name=”android.permission.NFC” />
<uses-feature android:name=”android.hardware.nfc” android:required=”true” />

(2)在AndroidManifest.xml 中的activity中声明可以处理的NFC Tag。
当手机开启了NFC,并且检测到一个TAG后,TAG分发系统会自动创建一个封装了NFC TAG信息的intent。如果多于一个应用程序能够处理这个intent的话,那么手机就会弹出一个框,让用户选择处理该TAG的Activity。 TAG分发系统定义了3中intent。按优先级从高到低排列为:
NDEF_DISCOVERED, TECH_DISCOVERED,TAG_DISCOVERED
当Android设备检测到有NFC Tag靠近时,会根据Action申明的顺序给对应的Activity 发送含NFC消息的 Intent。在某个Activity中添加以下的代码:
这里写图片描述

(3)编写xml文件。
在res目录下新建xml,编写上面红色框的xml文件:nfc_tech_filter,声明要处理的NFC Tag。
这里写图片描述

(4)测试
打开Android手机的NFC,运行程序,并把羊城通靠近手机背部,这样就可以看到可以处理NFC卡的应用。
这里写图片描述

(5)设置Activity
在onCreate()方法里添加以下代码。

nfcAdapter = NfcAdapter.getDefaultAdapter(this);
pendingIntent = PendingIntent.getActivity(this, 0, new Intent(this,getClass()).addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP), 0);
//区分系统版本
sysVersion = Integer.parseInt(VERSION.SDK);
if(sysVersion<19){
    onNewIntent(getIntent());
}

在onPause里面解除

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

重写OnnewIntent方法,在低于4.4系统时会用到这方法。

@Override
    protected void onNewIntent(Intent intent) {
        super.onNewIntent(intent);
        //获取数据
        final Parcelable p = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);
        Log.d("NFCTAG", intent.getAction());
        board.setText((p != null) ? CardReader.load(p) : null);
        if(p==null){
            board.setText("the intent=="+intent);
        }else{
            //board.setText("the intent !=null");
        }
    }

重写onResume方法

public static String[][] TECHLISTS;
public static IntentFilter[] FILTERS;
static {
   try {
TECHLISTS = new String[][] { { IsoDep.class.getName() },{ NfcV.class.getName() }, { NfcF.class.getName() }, };
FILTERS = new IntentFilter[] { new IntentFilter(
                    NfcAdapter.ACTION_TECH_DISCOVERED, "*/*") };
        } catch (Exception e) {
        }
    }
//4.4以上系统,在这个页面,多次发现标签,onresume只执行一次,4.4以下的会执行多次,但是onNewIntent()和enableReaderMode()都能够执行多次
@Override
    protected void onResume() {
        super.onResume();
        if (nfcAdapter != null){
            nfcAdapter.enableForegroundDispatch(this, pendingIntent,
                    CardReader.FILTERS, CardReader.TECHLISTS);
            enableReaderMode();
        }
        Log.e("NFC----", IsoDep.class.getName());
    }

(6)与卡片的交互方法的实现

 public String tagDiscovered(Tag tag) {
        Log.i(TAG, "New tag discovered");
        IsoDep isoDep = IsoDep.get(tag);
        System.out.println("isodep=="+isoDep);
        if (isoDep != null) {
            try {
                isoDep.connect();
                //读余额
                byte[] bal = new byte[]{(byte) 0x80,(byte)                         0x5c,(byte) 0x00,(byte) 0x02,(byte) 0x04};
                byte[] result = isoDep.transceive(bal);
                // 如果连接成功将返回 0x9000 
                int resultLength = result.length;
                byte[] statusWord = { result[resultLength - 2],result[resultLength - 1] };
                byte[] payload = Arrays.copyOf(result,resultLength - 2);
                if (Arrays.equals(SELECT_OK_SW, statusWord)) {
                //余额结果保存在前4位
                    int n = Util.toInt(payload, 0, 4);
                    if (n > 100000 || n < -100000)
                        n -= 0x80000000;
                    String s = toAmountString(n / 100.0f);//余额
                    return "余额=="+s;
                }
            } catch (IOException e) {
                Log.e(TAG, "Error communicating with card: " + e.toString());       
            }
        }
        return null;
    }

    public static String load(Parcelable parcelable) {
        // 从Parcelable筛选出各类NFC标准数据
        final Tag tag = (Tag) parcelable;
        return tagDiscovered(tag);
    }

至此非机卡通道方式的代码设置结束。

2、机卡通道模式

(1)选择API
使用的API是Open mobile api,这是基于android平台的用于APP与SIM卡建立通信连接的规范。添加的方法和添加其它版本API一样。
(2) 添加权限

<uses-permission android:name="org.simalliance.openmobileapi.SMARTCARD"/>
<uses-library android:name="org.simalliance.openmobileapi" 
             android:required="true" />

(3)在Activity类实现以下代码

SEService mSeService = new SEService(this, this);

实现回调的方法:

@Override
    public void serviceConnected(SEService service) {
//      DebugLog.i(LOG_TAG, "service connected");
//      可以读取卡的数据了
        NFCResponeController.getCardInfo(mSeService);
    }

    @Override
    protected void onDestroy() {
        if (mSeService != null && mSeService.isConnected()) {
            mSeService.shutdown();
        }
        super.onDestroy();
    }

/**
     * 传输API访问卡,执行command命令
     * @param aid AID标识字节
     * @param command 执行命令字节 至少4个字节
     * @param mseService SE服务
     * @return resulteStr 卡返回字节
     */
    public static byte[] queryCardCommand(byte[] aid,byte[] command,SEService mseService)
    {
        try {
              Reader[] readers = mseService.getReaders();
              if (readers.length < 1)
                 return null;
              Session session = readers[0].openSession();
              Channel channel = session.openLogicalChannel(aid);
              byte[] respApdu = channel.transmit(command);
              channel.close();
//            DebugLog.d(DebugLog_TAG,"读卡接口结果:"+NFCUtil.bytesToHexString(respApdu));
              return respApdu;
           }catch(java.lang.IllegalArgumentException e)
            {
//               DebugLog.e(DebugLog_TAG, "ERROR<<<<<<<<发送命令至少4个字节");
                 return null;
            }
            catch (Exception e) {
//                DebugLog.e(DebugLog_TAG, "Error occured:", e);
                  return null;
            } 
    }

3、参考内容

参考的博客

三、总结

刚开始接触的时候会不知道怎么入手,但这些基本的操作是需要掌握的,用多几次之后,就可以算入门了。

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!