Android O Nfc Enable详解(上)

摘要】随着Google于去年八月二十二日正式发布Android 8.0版本 Oreo,各种新的功能和变化也在等待着开发者的学习和研究。整体来说,这次的改动还是很大的,比如Camera的重新实现,HIDL机制的引入等。下面主要基于Android O的实现来分析NFC是如何Enable的。

这里写图片描述

上图是从Settings中启动NFC到JNI层的关系

这里首先说一下NFC的核心处理服务NfcService是如何初始化的,不同于蓝牙Service只在enable的时候才会启动,NfcService在手机初始化的过程中就会被启动,原因在于NfcApplication的Manifest里声明了android:persistent=”true” 这个属性。

熟悉Android启动流程的朋友应该知道,init过程中,zygote进程会start server

//zygote进程start system-server
service zygote /system/bin/app_process64 -Xzygote /system/bin --zygote --start-system-server

在SystemServer启动之后,首先会启动BootstrapServices和CoreService,之后会调用
startOtherServices

// Start services.
        try {
            traceBeginAndSlog("StartServices");
            startBootstrapServices();
            startCoreServices();
            startOtherServices();

在这个方法里面,一系列系统服务的systemReady方法会被调用,例如PowerManagerService,PackageManagerService以及DisplayManagerService等,当然,上面提到的ActivityManagerService的systemReady方法也会被调用,在这里第三方的code将会被运行

  // We now tell the activity manager it is okay to run third party
        // code.  It will call back into us once it has gotten to the state
        // where third party code can really run (but before it has actually
        // started launching the initial applications), for us to complete our
        // initialization.
        mActivityManagerService.systemReady(() -> {
            Slog.i(TAG, "Making services ready");
            traceBeginAndSlog("StartActivityManagerReadyPhase");
            mSystemServiceManager.startBootPhase(
                    SystemService.PHASE_ACTIVITY_MANAGER_READY);

在AMS的systemReady中,会调用startPersistentApps,最终在这个方法里,persistent app会被start,在NfcApplication start的onCreate中,NfcService会被初始化。

private void startPersistentApps(int matchFlags) {
        if (mFactoryTest == FactoryTest.FACTORY_TEST_LOW_LEVEL) return;

        synchronized (this) {
            try {
                final List<ApplicationInfo> apps = AppGlobals.getPackageManager()
                        .getPersistentApplications(STOCK_PM_FLAGS | matchFlags).getList();
        //NfcApplication的onCreate中
        if (UserHandle.myUserId() == 0 && isMainProcess) {
            mNfcService = new NfcService(this);

需要注意一点,因为NfcApplication有android:persistent=”true”这个属性,就会导致com.android.nfc这个进程是常驻的,没办法被kill掉(AMS监听到意外挂掉的应用是persistent的,会尝试重启这个应用)

OK,言归正传,当用户在Settings中switch nfc的开关,首先会调用NfcAdapter的enable
这是一个bind调用,在之前NfcService的初始化过程中,ServiceManager会把NfcAdapterService加入到其队列里面,它继承了INfcAdapter.Stub

mNfcAdapter = new NfcAdapterService();

final class NfcAdapterService extends INfcAdapter.Stub {
        @Override
        public boolean enable() throws RemoteException {
            NfcPermissions.enforceAdminPermissions(mContext);
            int val =  mDeviceHost.GetDefaultSE();
            Log.i(TAG, "getDefaultSE " + val);

// Make sure this is only called when object construction is complete.
        ServiceManager.addService(SERVICE_NAME, mNfcAdapter);

因此会走到上述代码中,而在NfcService#NfcAdapterService中,nfc的enable的执行是通过一个异步Task完成的

new EnableDisableTask().execute(TASK_ENABLE);

这里说明一点,在Android的上古时代(Android 1.6),AsyncTask是多任务并发执行的,但是
出现了很多线程同步的问题,为了避免这些错误,在3,2之后,又改成了默认单任务执行。
所以,同一时刻,只能有一个enable或者disable的操作在队列中中执行,其他的task都在队列中等待。

AsyncTask顺利执行,会走到enableInternal中,这个是上层实际开始enable NFC的地方,我们看看这里面发生了哪些操作。

首先,在进行device的initialize之前,会生成一个watchDog线程作为监听,默认的timeout值是90秒,如果初始化异常,会调用自身的abort方法将自身的进程kill掉重新启动

WatchDogThread watchDog = new WatchDogThread("enableInternal", timeout);
            watchDog.start();

//一旦触发watchDog,会调用doAbort方法kill掉com.android.nfc进程,重启            
mDeviceHost.doAbort(getName());            

接下来会执行mDeviceHost.initialize(),DeviceHost是一个对外的Interface,包含了很多的NFC方法,NativeNfcManager实现了这个接口。
在NativeNfcManager的方法里,我们可以看到,预先加载了一个name为nqnfc_nci_jni的动态链接库。通过查看Nfc根目录下的nci/jni子目录的Android.mk我们可以看到,所有jni目录下面的文件都会被编译进一个so库,这个库就是nqnfc_nci_jni

//如果动态链接库的name不是以lib开头的,系统会给name自动加上lib作为名称的头部
LOCAL_MODULE := libnqnfc_nci_jni

static {
        System.loadLibrary("nqnfc_nci_jni");
    }

private native boolean doInitialize();

通过JNI,最终会走到nfcManager_doInitialize,这个方法比较耗时,一般会在1s~3s之间,是底层的逻辑初始化,包括NFA(NFC For Android)以及GKI(General Kernel Interface)等,也是Nfc Enable最重要的部分,关于这块,在下一篇文章中我们再具体讨论吧~

注:本人水平有限,欢迎各位大牛批评指正!

  • 3
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
### 回答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的相关内容。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值