Реализация считывания NFC/RFID-меток товаров в мобильном приложении
NFC в телефоне — это HF RFID на 13.56 МГц. Читает те же чипы что и стационарные HF-ридеры: MIFARE Classic/Ultralight, NTAG213/215/216, ICODE SLI, ISO 15693. Для считывания товарных меток в ритейле, верификации подлинности или складских операций — встроенный NFC телефона работает без внешнего оборудования. Дальность — 1–5 сантиметров. Не UHF, не массовое считывание. Зато работает на любом современном смартфоне.
iOS: CoreNFC
import CoreNFC
class ProductTagReader: NSObject, NFCNDEFReaderSessionDelegate {
private var session: NFCNDEFReaderSession?
var onProductFound: ((ProductInfo) -> Void)?
func startReading() {
guard NFCNDEFReaderSession.readingAvailable else {
showError("NFC недоступен на этом устройстве")
return
}
session = NFCNDEFReaderSession(delegate: self, queue: nil, invalidateAfterFirstRead: false)
session?.alertMessage = "Приложите телефон к метке товара"
session?.begin()
}
func readerSession(_ session: NFCNDEFReaderSession, didDetectNDEFs messages: [NFCNDEFMessage]) {
for message in messages {
for record in message.records {
guard record.typeNameFormat == .nfcWellKnown,
let type = String(data: record.type, encoding: .utf8),
type == "U" else { continue }
// URL-запись в NDEF — стандарт для товарных меток
if let urlString = parseNDEFUrl(record.payload),
let url = URL(string: urlString) {
fetchProductInfo(from: url)
}
}
}
}
func readerSession(_ session: NFCNDEFReaderSession, didInvalidateWithError error: Error) {
if let nfcError = error as? NFCReaderError,
nfcError.code != .readerSessionInvalidationErrorFirstNDEFTagRead,
nfcError.code != .readerSessionInvalidationErrorUserCanceled {
showError("Ошибка NFC: \(nfcError.localizedDescription)")
}
}
}
invalidateAfterFirstRead: false — сессия не закрывается после первого считывания. Полезно для последовательной проверки нескольких товаров без повторного запуска сессии.
Для NTAG/MIFARE без NDEF — NFCTagReaderSession с pollingOption: [.iso14443]:
func tagReaderSession(_ session: NFCTagReaderSession, didDetect tags: [NFCTag]) {
guard let tag = tags.first else { return }
session.connect(to: tag) { [weak self] error in
if let error = error {
session.invalidate(errorMessage: "Ошибка: \(error.localizedDescription)")
return
}
switch tag {
case .miFare(let mifareTag):
let uid = mifareTag.identifier.hexString
self?.lookupProduct(uid: uid)
case .iso15693(let isoTag):
// ICODE SLI для коробочных меток в логистике
let uid = isoTag.identifier.hexString
self?.lookupProduct(uid: uid)
default:
session.invalidate(errorMessage: "Неподдерживаемый тип метки")
}
}
}
Android: NFC Foreground Dispatch
class ProductScanActivity : AppCompatActivity() {
private lateinit var nfcAdapter: NfcAdapter
private lateinit var pendingIntent: PendingIntent
private lateinit var filters: Array<IntentFilter>
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
nfcAdapter = NfcAdapter.getDefaultAdapter(this)
?: run { showNoNfcMessage(); return }
pendingIntent = PendingIntent.getActivity(
this, 0,
Intent(this, javaClass).addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP),
PendingIntent.FLAG_MUTABLE
)
filters = arrayOf(IntentFilter(NfcAdapter.ACTION_NDEF_DISCOVERED).apply {
addDataType("*/*")
})
}
override fun onResume() {
super.onResume()
nfcAdapter.enableForegroundDispatch(this, pendingIntent, filters, null)
}
override fun onPause() {
super.onPause()
nfcAdapter.disableForegroundDispatch(this)
}
override fun onNewIntent(intent: Intent) {
super.onNewIntent(intent)
when (intent.action) {
NfcAdapter.ACTION_NDEF_DISCOVERED -> handleNdefTag(intent)
NfcAdapter.ACTION_TAG_DISCOVERED -> handleRawTag(intent)
}
}
private fun handleNdefTag(intent: Intent) {
val messages = intent.getParcelableArrayExtra(NfcAdapter.EXTRA_NDEF_MESSAGES)
?.filterIsInstance<NdefMessage>() ?: return
messages.flatMap { it.records.toList() }
.filter { it.tnf == NdefRecord.TNF_WELL_KNOWN && it.type.contentEquals(NdefRecord.RTD_URI) }
.forEach { record ->
val url = parseNdefUri(record.payload)
viewModel.loadProduct(url)
}
}
}
Foreground dispatch перехватывает NFC-теги пока приложение активно. Без него Android показывает системный диалог выбора приложения.
Форматы меток и что в них хранить
| Чип | Память | Типичное применение |
|---|---|---|
| NTAG213 | 144 байт | URL на страницу товара |
| NTAG215 | 504 байт | URL + JSON с базовыми атрибутами |
| NTAG216 | 888 байт | Расширенные данные, история |
| MIFARE Ultralight | 48 байт | Только UID (нет места для данных) |
Для верификации подлинности: на метке хранится подписанный токен, приложение верифицирует подпись публичным ключом без обращения к серверу:
// ECDSA верификация токена с метки
func verifyAuthTag(_ signedPayload: Data) -> Bool {
let publicKey = getEmbeddedPublicKey() // зашит в bundle приложения
return SecKeyVerifySignature(
publicKey,
.ecdsaSignatureMessageX962SHA256,
productId as CFData,
signature as CFData,
nil
)
}
Сроки
Считывание NDEF-URL и загрузка информации о товаре: 2–3 дня. Кастомные форматы тегов с верификацией подписи и офлайн-базой: 1–2 недели.







