前言
NFC 這個(gè)詞已經(jīng)并不陌生了肃弟,前一段時(shí)間北京地鐵支持 NFC 支付一時(shí)成為霸占頭條的熱點(diǎn)。其實(shí)在 90 年代末到 2000 年初寓辱,二維碼和 NFC 就已經(jīng)相繼誕生腋粥,由于二維碼成本低廉蝗岖,技術(shù)門(mén)檻相對(duì)較低侥猩,因此,二維碼迅速搶占了移動(dòng)支付的市場(chǎng)抵赢,但 NFC 的發(fā)展并未因此停止欺劳。在 Android 端的 NFC 發(fā)展已經(jīng)非常迅猛了,可惜 Apple 遲遲未開(kāi)放接口铅鲤,在今年的 WWDC 上划提,蘋(píng)果宣布開(kāi)放其 NFC 接口 CoreNFC, 這為以后 NFC 的應(yīng)用提供了更多的可能。
正文
如果你對(duì) NFC 這項(xiàng)技術(shù)還比較陌生邢享,那么這里科普一下鹏往,NFC(Near Field Communication)近場(chǎng)通信,當(dāng)兩個(gè)設(shè)備相互靠近時(shí)能進(jìn)行信息交流驼仪。許多企業(yè)講 NFC 芯片放到卡片里,用帶有 NFC 芯片的卡片來(lái)授予權(quán)限將允許誰(shuí)有權(quán)限袜漩,比如說(shuō)進(jìn)出入公司绪爸。Apple CoreNFC 目前支持的格式有限,NFC 數(shù)據(jù)交換格式或 NDEF(通常用于當(dāng)今市場(chǎng)上的大多數(shù)平板電腦和手機(jī))宙攻,比如 Apple Pay 奠货。
CoreNFC Demo
這里我們通過(guò)一個(gè)簡(jiǎn)單的實(shí)例程序來(lái)演示怎么使用 CoreNFC,這個(gè)程序可以用來(lái)讀取存儲(chǔ)在卡片上 NDEF 格式的信息座掘。
為此递惋,我使用 Arduino Uno 與 Adafruit PN532 Shield 配對(duì),將其發(fā)送到樣品 NDEF 格式的卡上溢陪。 如果你不具備這些東西萍虽,或者根本不想在這樣的硬件上投入時(shí)間和金錢(qián),請(qǐng)嘗試找一張帶有信息的預(yù)格式化卡形真。 本文中杉编,我不會(huì)演示 NFC 格式化以及如何把數(shù)據(jù)存儲(chǔ)到 NDEF 卡 中。
Getting Started
打開(kāi) Xcode 9 創(chuàng)建一個(gè)簡(jiǎn)單的 Swift 工程。使用 Storyboard 創(chuàng)建簡(jiǎn)單的頁(yè)面:
ViewController 如下:
import UIKit
class ViewController: UIViewController {
@IBOutlet weak var messageLabel: UILabel!
override func viewDidLoad() {
super.viewDidLoad()
// Do any additional setup after loading the view, typically from a nib.
}
override func didReceiveMemoryWarning() {
super.didReceiveMemoryWarning()
// Dispose of any resources that can be recreated.
}
@IBAction func scanPressed(_ sender: Any) {
// this is our newly created IBAction
}
}
Entitlements & Privacy
我們的 app 要使用 NFC 必須進(jìn)行應(yīng)用授權(quán):前提是你得有一個(gè)有效 Apple id (交過(guò)保護(hù)費(fèi)的). 進(jìn)行應(yīng)用程序授權(quán)和隱私設(shè)置邓馒,打開(kāi)developer.apple.com嘶朱。 登錄你的帳戶,創(chuàng)建一個(gè)證書(shū) —— 注冊(cè)一個(gè)新的 APP ID光酣。應(yīng)用說(shuō)明應(yīng)該要支持 NFC 點(diǎn)擊下一步疏遏,確保你的確認(rèn)頁(yè)面如下圖:
然后再創(chuàng)建 provisioning profile :
這一步完成了,在我們剛創(chuàng)建的項(xiàng)目中導(dǎo)入 證書(shū)和描述文件救军,完成之后呢财异,我們還需進(jìn)行 Info.plist 配置 Privacy:
至此,我們的開(kāi)始工作就完成了缤言。??我們進(jìn)入 coding 階段宝当。
Core NFC
要實(shí)現(xiàn) NFC 功能,我們得接入 Core NFC framework:
import CoreNFC
目前為止胆萧,iOS模擬器尚未支持 CoreNFC庆揩。 這意味著如你嘗試導(dǎo)入CoreNFC,會(huì)收到一條錯(cuò)誤跌穗,表示沒(méi)有名為 CoreNFC 的模塊订晌。 遇到這種情況,請(qǐng)選擇你的 iPhone 或 Generic iOS Device蚌吸。接下來(lái)我們實(shí)現(xiàn) NFCNDEFReaderSessionDelegate 協(xié)議:
import UIKit
import CoreNFC
class ViewController: UIViewController, NFCNDEFReaderSessionDelegate {
@IBOutlet weak var messageLabel: UILabel!
var nfcSession: NFCNDEFReaderSession?
override func viewDidLoad() {
super.viewDidLoad()
// Do any additional setup after loading the view, typically from a nib.
}
override func didReceiveMemoryWarning() {
super.didReceiveMemoryWarning()
// Dispose of any resources that can be recreated.
}
@IBAction func scanPressed(_ sender: Any) {
}
func readerSession(_ session: NFCNDEFReaderSession, didInvalidateWithError error: Error) {
print("The session was invalidated: \(error.localizedDescription)")
}
func readerSession(_ session: NFCNDEFReaderSession, didDetectNDEFs messages: [NFCNDEFMessage]) {
// Parse the card's information
}
}
其中兩個(gè) readerSession 函數(shù)會(huì)分別告訴我們 NFC 會(huì)話成功或失敗锈拨,成功后則返回 NFCNDEFMessage 格式的通信數(shù)據(jù),失敗后會(huì)返回 error 信息羹唠。
??當(dāng)然奕枢,我們首先還需要初始化 NFCNDEFReaderSession 并開(kāi)啟 NFC 監(jiān)聽(tīng)。
@IBAction func scanPressed(_ sender: Any) {
nfcSession = NFCNDEFReaderSession.init(delegate: self, queue: nil, invalidateAfterFirstRead: true)
nfcSession?.begin()
}
然后運(yùn)行程序佩微,看看:
如果提示 Session is invalidated unexpectedly 缝彬,那么請(qǐng)仔細(xì)核對(duì) 證書(shū)、描述文件以及 Privacy 設(shè)置是否正確哺眯。
這個(gè)過(guò)程并不難谷浅,簡(jiǎn)單幾步就能搞定,下面我們來(lái)看看怎么解析獲取到的 message 奶卓。
解析 Message
首先一疯,讓我們來(lái)看一下這個(gè)函數(shù):
func readerSession(_ session: NFCNDEFReaderSession, didDetectNDEFs messages: [NFCNDEFMessage])
讓我們來(lái)看看每一個(gè) message 對(duì)象包含了哪些信息:
print(messages[0])
我們可以看到:
( // Payload one (There's only one payload in this card)
"TNF=1, /* Type Name Format */
Payload Type=<55>,
Payload ID=<>,
Payload=<0048656c 6c6f21>" /* What we're really interested in */
)
根據(jù)打印的結(jié)果我們可以看出:
. messages 是一個(gè) NFCNDEFMessages 對(duì)象的數(shù)組。
. NFCNDEFMessage 有一個(gè) NFCNDEFPayload 對(duì)象數(shù)組 records
然后我們?cè)賮?lái)看看每一個(gè) payload 又包含了哪些信息:
- identifier.
- type.
- typeNameFormat.
- payload.
這里其實(shí)我們只關(guān)心 payload 夺姑。好了墩邀,看看如何解析 records 吧。
func readerSession(_ session: NFCNDEFReaderSession, didDetectNDEFs messages: [NFCNDEFMessage]) {
var result = ""
for payload in messages[0].records {
result += String.init(data: payload.payload.advanced(by: 3), encoding: .utf8)! // 1
}
DispatchQueue.main.async {
self.messageLabel.text = result
}
}
來(lái)看看最終的效果: