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

相关推荐

  • flash 开发android怎么操作?Android开发还能用Flash吗

    Flash开发Android应用的核心结论在于:尽管Adobe已停止官方支持,但通过现代跨平台框架与AIR运行时的深度结合,开发者依然能够高效构建高性能的移动应用,且在游戏开发与多媒体交互领域具备独特优势,这一方案不仅解决了兼容性问题,还能显著降低开发成本,尤其适合需要快速迭代的中小团队,技术可行性:Flash……

    2026年3月27日
    6900
  • 系统开发的任务有哪些,系统开发主要做什么工作

    系统开发的任务核心在于将模糊的业务需求转化为可运行的软件实体,这一过程不仅是代码的编写,更是对业务逻辑的深度解构与技术重构,成功的系统开发必须实现业务流程数字化、数据资产价值化以及系统架构可扩展化,这三者构成了系统开发的终极目标,开发团队需在有限的资源与时间内,通过科学的工程化管理,交付高质量、高可用、易维护的……

    2026年3月12日
    11600
  • 香港ZJI独立服务器560元方案怎么样?香港独服实测对比

    在当前的建站与业务部署环境中,独立服务器的性能直接决定了核心业务的稳定性与并发处理能力,本次针对香港ZJI独立服务器月付560元方案进行了深度实测,该机房地处香港核心网络枢纽,主打直连大陆优化线路,以下为详尽的性能拆解与数据比对, 核心硬件与配置解析本款560元/月方案定位为高性价比入门级独立服务器,硬件配置兼……

    2026年4月29日
    3600
  • 美国LOCVPSVPS测评,22.2元/月方案实测对比,LOCVPS月付22元VPS值得买吗

    美国LOCVPS作为国内站长群体中颇具知名度的老牌主机商,其主打的洛杉矶MC机房方案一直以性价比著称,本次针对其月付22.2元的促销方案进行了深度实测,从硬件性能、网络质量到路由节点进行全面剖析,并结合2026年最新优惠活动进行解析,为建站及外贸从业者提供客观的选购参考, 测评方案核心参数与活动说明本次实测基于……

    2026年4月28日
    2400
  • stm32f0开发难吗?新手入门教程详解

    STM32F0系列作为ARM Cortex-M0内核的典型代表,以其高性价比和低功耗特性,成为众多工业控制与消费电子项目的首选方案,高效完成STM32F0开发的核心在于充分利用其硬件特性,配合模块化的软件架构,从而在缩短开发周期的同时确保系统运行的稳定性, 相较于F1或F4系列,F0系列虽然在处理性能上有所取舍……

    2026年3月10日
    10000
  • 开发发票资质怎么办理?办理开发票资质需要什么条件

    企业及个体工商户合规开具发票的前提,是必须具备合法的税务登记资质与相应的经营许可,这构成了开发票资质的核心要件,不具备这一基础资质的主体,无法独立开票,只能申请税务机关代开,核心结论在于:合法的开票资质并非单一证照,而是税务登记、税种核定、票种核定及硬件设施配置的综合体现,企业必须完成这一闭环,才能在法律框架内……

    2026年4月1日
    7500
  • lt开发是什么意思?lt开发流程详解

    LT开发的核心价值在于通过系统化的技术架构与精细化的流程管理,实现产品从概念到落地的全生命周期高效交付,其本质是以用户需求为导向,以技术可行性为基石,以商业价值为终局的工程化实践,成功的LT开发项目必然遵循“需求精准定义—架构科学设计—代码规范实现—测试全面覆盖—运维持续迭代”的闭环逻辑,任何环节的缺失或弱化都……

    2026年3月28日
    6700
  • 郑州定制开发多少钱?郑州网站建设哪家好?

    在郑州寻求软件开发服务,选择标准化产品往往难以契合企业独特的业务流程、管理挑战或市场定位,这正是郑州定制开发的核心价值所在——构建专属的数字化解决方案,驱动业务实现突破性成长,本文将深入解析郑州定制开发的完整路径与关键考量, 为何郑州企业更需要定制化解决方案?郑州作为中原经济区的核心引擎,汇聚了制造、物流、商贸……

    2026年2月14日
    9630
  • 开发桌面应用程序用什么语言好?桌面软件开发教程

    在当今软件开发领域,C语言依然是开发高性能桌面应用程序的首选工具之一,其底层控制能力、执行效率以及跨平台特性,使其在系统级应用、嵌入式软件和高性能工具开发中占据不可替代的地位,本文将深入探讨C语言在桌面应用程序开发中的核心优势、关键技术以及实践方法,帮助开发者掌握高效开发的精髓,C语言开发桌面应用程序的核心优势……

    2026年4月7日
    4500
  • 北京开发商电话是多少?北京知名开发商联系方式大全

    获取北京开发商电话最直接且有效的途径,是通过北京市住房和城乡建设委员会的官方备案系统与正规房产交易平台的公示信息,这不仅能确保联系方式的真实性与时效性,更是规避中介骚扰、直接对接项目开发主体的关键手段,对于购房者、合作伙伴以及行业研究者而言,掌握一手开发商联系方式,意味着掌握了信息对称的主动权,为何官方渠道是获……

    2026年3月21日
    7200

发表回复

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

评论列表(3条)

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

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

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

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

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

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