iOS NFC刷卡功能如何实现?iOS NFC开发全攻略

长按可调倍速

一群草台班子,让 99% 的门禁卡都在裸奔:如何破解门禁卡?【柴知道】

近场通信(NFC)技术为iOS应用带来了与物理世界互动的全新维度,它允许设备在几厘米范围内安全地交换数据、读取标签或模拟卡片,对于iOS开发者而言,掌握Core NFC框架是解锁门禁控制、信息交互、支付集成、资产追踪等丰富场景的关键。要在iOS应用中实现NFC功能,核心在于熟练运用Apple提供的Core NFC框架,通过配置正确的权限、处理NFC会话生命周期、解析或构建符合NDEF(NFC Data Exchange Format)标准的数据负载,并遵循苹果严格的隐私与后台限制策略。

iOS NFC刷卡功能如何实现

核心功能与技术限制

  • 读取NFC标签 (NFC Tag Reading):这是Core NFC最基本也是最广泛使用的功能,应用可以读取嵌入在物品(如海报、产品包装、名片、工牌)中的被动NFC标签(NDEF格式),读取通常在应用处于前台且用户明确触发(如点击按钮)时进行。
  • 读取NFC数据交换格式 (NDEF):Core NFC主要专注于读取和解析符合NDEF标准的数据,NDEF消息由一个或多个NDEF记录组成,每个记录包含:
    • 类型名称格式 (TNF):标识记录类型的分类(如NFC论坛定义的类型、外部类型、绝对URI、MIME类型等)。
    • 类型 (Type):描述记录内容的更具体标识符(T 是文本记录,U 是URI记录,sp 是智能海报记录)。
    • 有效载荷 (Payload):记录携带的实际数据(如URL字符串、文本内容、自定义二进制数据)。
  • 写入NFC标签 (NFC Tag Writing – 部分支持):从iOS 13开始,Core NFC支持写入可写的NFC标签(NDEF格式),开发者可以将自定义的NDEF消息写入到标签中,这需要标签本身是可写的(例如NTAG213, 215, 216等),且应用在前台由用户触发。
  • 技术限制
    • 设备要求:需要配备NFC读/写器的iPhone(iPhone 7及更新机型),iPhone 7/7 Plus仅支持读取(不支持写入)。
    • 后台限制:iOS严格限制NFC在后台的使用,应用在前台运行时才能发起NFC会话。唯一的例外是“后台标签读取”:如果应用注册了特定的NDEF记录类型(如URL、文本、通用控制记录),当用户将iPhone靠近包含该类型记录的标签时,系统会短暂唤醒应用(即使应用在后台或设备锁屏状态)并传递记录内容,这常用于快速启动应用特定功能。
    • 会话时长限制:单个NFC会话有超时限制(通常60秒左右),超时后需要用户重新触发。
    • 卡模拟限制:Core NFC 支持将iPhone模拟为NFC标签或智能卡(如门禁卡、公交卡),此功能由Apple的Core NFC + Wallet框架(用于Apple Pay、交通卡)或CarKey等特定API实现,对第三方开发者开放极其有限(通常需要MFi认证或特定合作伙伴关系)。

开发环境准备

  1. Xcode:使用最新稳定版本。
  2. 设备:配备NFC读/写器的iPhone (iPhone 7或更新机型,写操作需iPhone XS/XR或更新且iOS 13+),模拟器不支持NFC测试。
  3. 添加Capability
    • 在Xcode中打开你的项目。
    • 选择你的应用Target。
    • 转到 Signing & Capabilities 标签页。
    • 点击 + Capability
    • 搜索并添加 Near Field Communication Tag Reading,这会自动在项目配置和 Entitlements 文件中添加必要的权限。
  4. 配置Info.plist
    • 必须提供使用NFC的原因描述,否则应用会被拒绝或在运行时崩溃。
    • 添加键 NFCReaderUsageDescription (或 Privacy - NFC Scan Usage Description),值为字符串(“扫描NFC标签以获取产品信息”或“读取工牌进行身份验证”)。
    • 对于后台标签读取,还需额外配置:
      • 添加键 com.apple.developer.nfc.readersession.iso7816.select-identifiers (Array类型)。
      • 在这个数组中添加你的应用希望系统在后台唤醒它时能处理的NDEF记录类型对应的应用ID(Application IDs),你需要注册你希望处理的特定URI协议方案(如 myapp://)、MIME类型或NFC论坛外部记录类型(需反向域名格式,如 com.example.mytype),这告诉系统:“当检测到包含这种记录的标签时,请唤醒我的应用来处理”。

实战场景:NFC标签读取

  1. 导入框架

    import CoreNFC
  2. 检查设备支持

    iOS NFC刷卡功能如何实现

    guard NFCNDEFReaderSession.readingAvailable else {
        // 设备不支持NFC或系统版本过低
        showAlert("此设备不支持NFC标签读取功能。")
        return
    }
  3. 创建并启动NDEF读取会话

    class NFCReaderViewController: UIViewController, NFCNDEFReaderSessionDelegate {
        var nfcSession: NFCNDEFReaderSession?
        @IBAction func startScanning(_ sender: UIButton) {
            // 创建一个新的NDEF读取会话,指定委托和是否在检测到标签后暂停(true会暂停,false会连续读取)
            nfcSession = NFCNDEFReaderSession(delegate: self, queue: nil, invalidateAfterFirstRead: true)
            nfcSession?.alertMessage = "将iPhone靠近NFC标签..." // 会话期间显示的提示信息
            nfcSession?.begin() // 开始会话,系统会显示扫描界面
        }
        // MARK: - NFCNDEFReaderSessionDelegate
        // 会话因错误或用户取消而失效时调用
        func readerSession(_ session: NFCNDEFReaderSession, didInvalidateWithError error: Error) {
            // 处理错误(如session超时、用户取消、设备不支持、权限问题)
            DispatchQueue.main.async {
                if let nfcError = error as? NFCReaderError {
                    switch nfcError.code {
                    case .readerSessionInvalidationErrorUserCanceled:
                        // 用户主动取消
                        break
                    default:
                        self.showAlert("NFC会话出错: (nfcError.localizedDescription)")
                    }
                }
            }
            self.nfcSession = nil // 释放会话
        }
        // 成功检测到NDEF标签并读取到消息时调用
        func readerSession(_ session: NFCNDEFReaderSession, didDetectNDEFs messages: [NFCNDEFMessage]) {
            // 一个标签可能包含多条NDEF记录(messages数组可能有多个元素,但通常一个标签对应一个message)
            for message in messages {
                for record in message.records {
                    // 解析每条NDEF记录
                    self.processNDEFRecord(record)
                }
            }
            // 如果invalidateAfterFirstRead=true,会话会自动结束,否则需要手动session.invalidate()
        }
        // 处理单条NDEF记录的核心方法
        private func processNDEFRecord(_ record: NFCNDEFRecord) {
            DispatchQueue.main.async {
                switch record.typeNameFormat {
                case .nfcWellKnown:
                    // 处理NFC论坛预定义类型
                    if let typeString = String(data: record.type, encoding: .ascii) {
                        switch typeString {
                        case "T": // 文本记录
                            self.handleTextRecord(record.payload)
                        case "U": // URI记录
                            self.handleURIRecord(record.payload)
                        case "Sp": // 智能海报记录 (通常包含URL和文本/标题)
                            // 智能海报记录本身是一个NDEF消息,需要递归解析
                            if let spMessage = try? NFCNDEFMessage(data: record.payload) {
                                for spRecord in spMessage.records {
                                    self.processNDEFRecord(spRecord) // 递归解析内部记录
                                }
                            }
                        default:
                            print("未知的Well Known类型: (typeString)")
                        }
                    }
                case .absoluteURI:
                    // 绝对URI记录(整个payload就是URI字符串)
                    if let uriString = String(data: record.payload, encoding: .utf8) {
                        self.handleAbsoluteURI(uriString)
                    }
                case .media:
                    // MIME类型记录 (record.type 包含MIME类型字符串)
                    if let mimeType = String(data: record.type, encoding: .utf8),
                       let payloadData = record.payload {
                        self.handleMediaRecord(mimeType: mimeType, data: payloadData)
                    }
                case .nfcExternal:
                    // NFC论坛外部记录类型 (自定义类型,通常以反向域名格式定义在record.type中)
                    if let externalType = String(data: record.type, encoding: .utf8),
                       let payloadData = record.payload {
                        self.handleExternalRecord(type: externalType, data: payloadData)
                    }
                case .empty, .unchanged, .unknown, .reserved:
                    // 通常忽略或简单记录
                    print("未处理或未知的记录类型格式: (record.typeNameFormat)")
                @unknown default:
                    break
                }
            }
        }
        // 示例:解析文本记录 (Payload结构: [状态字节(包含语言编码长度)][ISO/IANA语言码][实际文本])
        private func handleTextRecord(_ payload: Data) {
            guard payload.count > 1 else { return }
            let statusByte = payload[0]
            let languageCodeLength = Int(statusByte & 0x3F) // 低6位表示语言码长度
            guard payload.count > languageCodeLength else { return }
            if let languageCode = String(data: payload.subdata(in: 1..<(1 + languageCodeLength)), encoding: .utf8),
               let text = String(data: payload.subdata(in: (1 + languageCodeLength)..<payload.count), encoding: .utf8) {
                print("文本 ((languageCode)): (text)")
                // 更新UI显示文本
            }
        }
        // 示例:解析URI记录 (Payload结构: [URI标识码][URI字符串])
        private func handleURIRecord(_ payload: Data) {
            guard !payload.isEmpty else { return }
            let uriIdentifier = payload[0]
            let uriSuffix = String(data: payload.subdata(in: 1..<payload.count), encoding: .utf8) ?? ""
            var fullURI: String
            // 根据NFC论坛URI标识码表解析前缀
            switch uriIdentifier {
            case 0x00: fullURI = uriSuffix // 无前缀
            case 0x01: fullURI = "http://www.(uriSuffix)"
            case 0x02: fullURI = "https://www.(uriSuffix)"
            case 0x03: fullURI = "http://(uriSuffix)"
            case 0x04: fullURI = "https://(uriSuffix)"
            case 0x05: fullURI = "tel:(uriSuffix)"
            case 0x06: fullURI = "mailto:(uriSuffix)"
            case 0x07: fullURI = "ftp://anonymous:anonymous@(uriSuffix)"
            case 0x08: fullURI = "ftp://ftp.(uriSuffix)"
            case 0x09: fullURI = "ftps://(uriSuffix)"
                // ... 其他标识码参考NFC论坛规范
            default:
                // 0x0A - 0x1F 保留,0x20 - 0xFF 为自定义URI协议,通常直接拼接
                if uriIdentifier >= 0x20 {
                    if let prefix = String(data: Data([uriIdentifier]), encoding: .ascii) {
                        fullURI = prefix + uriSuffix
                    } else {
                        fullURI = uriSuffix
                    }
                } else {
                    fullURI = uriSuffix
                }
            }
            print("URI: (fullURI)")
            // 处理URI(如打开网页、拨打电话、发邮件等)
        }
        // ... 其他记录类型的处理方法(handleMediaRecord, handleExternalRecord等)类似
    }

进阶应用:NFC数据写入

  1. 前提:确保设备支持(iPhone XS/XR或更新,iOS 13+),标签是可写的NDEF标签。

  2. 创建NDEF消息

    // 创建一个包含一条文本记录的NDEF消息
    func createTextNDEFMessage(text: String, languageCode: String = "en") -> NFCNDEFMessage? {
        // 1. 创建文本记录负载: [状态字节][语言码][文本]
        let statusByte: UInt8 = UInt8(languageCode.count) // 假设只用了低6位,UTF-8编码
        var payload = Data([statusByte])
        payload.append(contentsOf: languageCode.utf8)
        payload.append(contentsOf: text.utf8)
        // 2. 创建文本记录 (TNF=.nfcWellKnown, Type="T")
        guard let textRecord = NFCNDEFPayload(
            format: .nfcWellKnown,
            type: "T".data(using: .ascii)!, // "T" for Text
            identifier: Data(), // 标识符通常为空
            payload: payload
        ) else {
            print("创建文本记录失败")
            return nil
        }
        // 3. 将记录放入消息
        return NFCNDEFMessage(records: [textRecord])
    }
    // 创建一个包含一条URI记录的NDEF消息 (使用https://)
    func createURINDEFMessage(urlString: String) -> NFCNDEFMessage? {
        // 1. 处理URI前缀标识码 (这里简化,假设是完整的https://开头)
        var uriPayload = Data()
        let prefixCode: UInt8 = 0x04 // 代表 "https://"
        uriPayload.append(prefixCode)
        uriPayload.append(contentsOf: urlString.dropFirst(8).utf8) // 移除"https://" (8字符)
        // 2. 创建URI记录 (TNF=.nfcWellKnown, Type="U")
        guard let uriRecord = NFCNDEFPayload(
            format: .nfcWellKnown,
            type: "U".data(using: .ascii)!, // "U" for URI
            identifier: Data(),
            payload: uriPayload
        ) else {
            print("创建URI记录失败")
            return nil
        }
        return NFCNDEFMessage(records: [uriRecord])
    }
  3. 创建并启动NDEF写入会话

    iOS NFC刷卡功能如何实现

    class NFCWriterViewController: UIViewController, NFCNDEFReaderSessionDelegate {
        var nfcSession: NFCNDEFReaderSession?
        var messageToWrite: NFCNDEFMessage? // 要写入的NDEF消息
        @IBAction func startWriting(_ sender: UIButton) {
            guard let message = createTextNDEFMessage(text: "你好,NFC世界!") else { // 或 createURINDEFMessage
                showAlert("创建要写入的消息失败")
                return
            }
            messageToWrite = message
            // 创建支持写入的会话
            nfcSession = NFCNDEFReaderSession(delegate: self, queue: nil, invalidateAfterFirstRead: false) // 设置为false以便写入
            nfcSession?.alertMessage = "将iPhone靠近可写的NFC标签进行写入..."
            nfcSession?.begin()
        }
        // MARK: - NFCNDEFReaderSessionDelegate (didInvalidateWithError同上)
        // 检测到标签(可能是读也可能是写)
        func readerSession(_ session: NFCNDEFReaderSession, didDetectNDEFs messages: [NFCNDEFMessage]) {
            // 写入操作通常不需要处理读取到的消息,除非你想先读取再覆盖
            session.alertMessage = "检测到标签,准备写入..."
        }
        // 检测到标签(更通用的方法,iOS 11后推荐)
        func readerSession(_ session: NFCNDEFReaderSession, didDetect tags: [NFCNDEFTag]) {
            guard let tag = tags.first else { // 通常处理检测到的第一个标签
                session.invalidate(errorMessage: "未检测到有效标签或检测到多个标签。")
                return
            }
            session.connect(to: tag) { [weak self] (error: Error?) in
                guard error == nil else {
                    session.invalidate(errorMessage: "连接标签失败: (error!.localizedDescription)")
                    return
                }
                // 1. 查询标签容量和可写性
                tag.queryNDEFStatus { (status: NFCNDEFStatus, capacity: Int, error: Error?) in
                    guard error == nil else {
                        session.invalidate(errorMessage: "查询标签状态失败")
                        return
                    }
                    guard status == .readWrite || status == .readWriteMBR else {
                        session.invalidate(errorMessage: "标签不可写或状态不支持写入 ((status))")
                        return
                    }
                    guard let message = self?.messageToWrite,
                          message.length <= capacity else {
                        session.invalidate(errorMessage: "要写入的消息 ((messageToWrite?.length ?? 0) 字节) 超出标签容量 ((capacity) 字节)")
                        return
                    }
                    // 2. 执行写入操作
                    tag.writeNDEF(message) { (error: Error?) in
                        if let error = error {
                            session.invalidate(errorMessage: "写入失败: (error.localizedDescription)")
                        } else {
                            session.alertMessage = "写入成功!"
                            session.invalidate() // 手动结束会话
                        }
                    }
                }
            }
        }
    }

行业应用与创新思考

  • 零售与营销:产品包装上的NFC标签,扫描获取详细信息、防伪验证、促销链接、视频展示。
  • 智能家居/物联网:轻触NFC标签配置Wi-Fi设备(如智能灯泡、插座)、触发场景(如“晚安模式”)。
  • 身份认证与门禁:读取员工卡/学生证信息进行身份验证(需后端配合),简化签到流程。
  • 物流与资产管理:追踪货物位置、读取资产标签信息进行盘点和维护记录。
  • 文化展览与旅游:博物馆展品旁的标签提供多语言解说、视频资料;景点导览地图。
  • 创意交互:艺术装置、互动海报、游戏道具。

专业建议与避坑指南

  1. 用户体验至上:明确告知用户何时需要靠近标签,提供清晰的扫描提示和反馈(成功/失败),避免让用户“盲扫”。
  2. 优雅处理错误:全面处理各种可能的错误情况(设备不支持、无权限、标签不支持、读写失败、超时、用户取消),给予友好的错误提示。
  3. 后台读取精确定义:仔细规划你的“后台标签读取”策略,只注册真正需要唤醒应用的特定记录类型,避免滥用导致用户困扰或电池消耗,后台读取传递的数据量有限,设计轻量级的启动逻辑。
  4. 安全与隐私
    • 绝不在NDEF标签中存储敏感信息(密码、个人身份信息),使用标签作为“钥匙”去安全地获取后端数据。
    • 对读取到的外部数据(尤其是URL)进行严格验证和清理,防止注入攻击。
    • 清晰告知用户应用如何使用NFC数据。
  5. 优化写入流程:写入操作比读取更易出错(标签位置移动、标签不可写、容量不足),确保用户界面有明确的写入状态指示和错误恢复机制。
  6. 兼容性与测试:在不同型号的iPhone和不同类型的NFC标签上进行充分测试,注意旧机型(iPhone 7)的功能限制。
  7. 探索自定义格式:在NDEF标准框架内(尤其是使用nfcExternalmedia类型),可以定义应用专属的数据结构进行更复杂的交互。

掌握iOS NFC开发,意味着你能够为你的应用构建无缝连接数字与物理世界的桥梁,从简单的信息获取到复杂的交互流程,Core NFC框架提供了稳定而强大的基础,关键在于理解其运作机制、遵循平台规范、注重用户体验,并充分发挥创造力去解决实际问题。

你的NFC应用构想是什么?你打算利用它来解决哪个具体的场景痛点?或者你在开发过程中遇到了哪些独特的挑战?欢迎在评论区分享你的想法和经验!

首发原创文章,作者:世雄 - 原生数据库架构专家,如若转载,请注明出处:https://idctop.com/article/30983.html

(0)
上一篇 2026年2月14日 09:29
下一篇 2026年2月14日 09:32

相关推荐

  • ip摄像头开发难吗?ip摄像头开发教程

    IP摄像头开发的本质,是在有限算力与网络带宽限制下,对视频数据采集、编码、传输及交互的全链路优化过程,成功的开发方案并非单纯依赖硬件堆砌,而是取决于视频流媒体架构设计的合理性、低延迟传输协议的选型以及端侧边缘计算能力的深度挖掘, 这一领域的技术门槛,正从单纯的嵌入式开发向“音视频算法+网络传输+AI推理”的复合……

    2026年3月21日
    3500
  • Unity3d游戏开发入门难吗?零基础怎么学Unity3D

    Unity3d游戏开发入门的核心在于建立清晰的开发工作流与扎实的脚本逻辑基础,而非单纯掌握软件操作,对于初学者而言,最有效的学习路径是以C#编程语言为内核,以Unity引擎为工具框架,通过“最小可行性产品(MVP)”的开发实战,快速构建游戏开发的闭环认知,这一过程要求开发者从底层逻辑理解游戏对象的交互机制,避免……

    2026年3月28日
    2300
  • miui7.5开发版发布,哪些新功能令人期待?体验升级背后有何秘密?

    MIUI 7.5 开发版深度刷机与体验指南MIUI 7.5开发版,作为小米在Android 5.x时代为发烧友定制的先锋系统,曾以其活跃的功能迭代和深度可玩性风靡一时,虽然官方已停止维护,但对于怀旧玩家、特定设备持有者或系统研究者而言,它仍具有独特价值,本指南将提供一套完整、安全且符合当前环境的刷机方案,助你重……

    2026年2月6日
    6930
  • javaweb开发实战经典怎么样?javaweb开发实战经典pdf下载

    JavaWeb开发的核心价值在于构建高可用、高并发、易维护的企业级应用体系,掌握JavaWeb开发实战经典,不仅是掌握Servlet或JSP等基础技术,更是深入理解MVC设计模式、数据库优化、框架整合以及分布式架构演进的过程,真正的实战能力,体现在从需求分析到系统上线的全生命周期管理,以及解决复杂业务场景下的技……

    2026年4月2日
    1700
  • 小米开发版如何升级|稳定版刷机教程一步到位

    小米开发版升级的核心步骤是:解锁Bootloader -> 下载对应机型的开发版ROM -> 通过线刷(Fastboot模式)或卡刷(Recovery模式)方式刷入系统, 开发版系统更新频率高,包含最新功能,但也可能不稳定,仅推荐发烧友和开发者使用,操作前务必备份重要数据并承担风险, 理解开发版与稳……

    2026年2月8日
    8200
  • unity可以做安卓开发吗?Unity安卓开发教程详解

    Unity作为全球领先的实时3D开发引擎,在移动端的游戏与应用开发中占据主导地位,而安卓系统则是全球市场份额最大的移动操作系统,Unity 安卓开发的核心价值在于“一次开发,多端部署”的高效性,但真正决定产品成败的关键,在于开发者是否具备跨越引擎与原生平台鸿沟的深度整合能力, 仅仅掌握Unity引擎内的C#脚本……

    2026年3月13日
    5700
  • 南通开发区驰加具体位置在哪,驰加做一次保养多少钱

    开发一套定制化的汽车服务门店管理系统是实现数字化运营的核心,针对南通开发区驰加这类专业汽车服务场景,程序开发不仅要解决基础的数据记录问题,更要构建一套集预约、库存、客户管理于一体的闭环生态,本教程将基于实际业务场景,详细阐述从需求分析到系统落地的全流程开发方案,旨在为技术人员提供一套具备高可用性和扩展性的架构指……

    2026年2月21日
    6000
  • BOA开发怎么进行,嵌入式BOA服务器移植教程详解

    BOA Web Server 是嵌入式 Linux 系统中实现远程设备管理和监控的首选解决方案,它通过提供轻量级、高性能且支持 CGI 的 HTTP 服务,完美解决了资源受限环境下的网络交互难题,在嵌入式开发领域,掌握 BOA 的移植、配置及 CGI 交互编程,是构建智能化物联网设备的关键技术路径,本文将深入剖……

    2026年2月17日
    12300
  • luci开发难吗?luci开发教程入门指南

    LuCI 开发的核心在于理解MVC架构与OpenWrt系统的深度集成,掌握这一关键点,便能高效构建出功能强大且用户友好的路由器管理界面,开发过程并非简单的页面制作,而是涉及后端数据交互、前端渲染优化以及系统配置文件读写的系统工程,LuCI 开发的架构逻辑与技术底座LuCI作为OpenWrt上的Web管理界面,其……

    2026年3月28日
    1900
  • 微信端网页开发教程,如何高效掌握最新技术要点?

    微信端网页开发(H5)因其触达用户便捷、开发周期相对较短、迭代灵活等优势,已成为连接微信生态内用户的重要桥梁,微信内置浏览器(X5内核)的特殊性以及微信生态的规则,给开发者带来了独特的挑战,掌握微信端网页开发的核心要点与最佳实践,是打造流畅用户体验、实现业务目标的关键, 微信环境特殊性:理解你的“容器”微信内置……

    2026年2月6日
    5200

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

评论列表(3条)

  • 老光5712的头像
    老光5712 2026年2月18日 16:09

    这篇文章的内容非常有价值,我从中学习到了很多新的知识和观点。作者的写作风格简洁明了,却又不失深度,

    • 快乐user378的头像
      快乐user378 2026年2月18日 17:41

      @老光5712读了这篇文章,我深有感触。作者对标签的理解非常深刻,论述也很有逻辑性。内容既有理论深度,又有实践指导意义,

    • 萌老2547的头像
      萌老2547 2026年2月18日 19:33

      @老光5712这篇文章的内容非常有价值,我从中学习到了很多新的知识和观点。作者的写作风格简洁明了,却又不失深度,