आईओएस कंबाइन फ्यूचर पब्लिशर के साथ एपीआई कॉल करना

May 06 2023
कंबाइन फ्रेमवर्क क्या है? iOS कंबाइन फ्रेमवर्क को Apple द्वारा 2019 में iOS 13 के हिस्से के रूप में पेश किया गया था। यह समय के साथ अतुल्यकालिक घटनाओं को संसाधित करने के लिए एक घोषणात्मक स्विफ्ट फ्रेमवर्क है।

कंबाइन फ्रेमवर्क क्या है?

iOS कंबाइन फ्रेमवर्क को Apple द्वारा 2019 में iOS 13 के हिस्से के रूप में पेश किया गया था। यह समय के साथ अतुल्यकालिक घटनाओं को संसाधित करने के लिए एक घोषणात्मक स्विफ्ट फ्रेमवर्क है। इसका मतलब है कि यह आपको प्रतिक्रियाशील प्रोग्रामिंग कोड लिखने की अनुमति देता है, जो अतुल्यकालिक डेटा स्ट्रीम को संभालने के लिए एक लोकप्रिय प्रतिमान है।

कंबाइन फ्रेमवर्क से पहले, iOS डेवलपर्स को रिएक्टिव कोड लिखने के लिए RxSwift और ReactiveSwift जैसे थर्ड-पार्टी रिएक्टिव फ्रेमवर्क पर निर्भर रहना पड़ता था। हालाँकि, कम्बाइन फ्रेमवर्क की शुरुआत के साथ, iOS डेवलपर्स अब बाहरी पुस्तकालयों पर निर्भर हुए बिना प्रतिक्रियाशील प्रोग्रामिंग लिख सकते हैं।

इसका एक फायदा यह है कि कंबाइन फ्रेमवर्क Apple द्वारा बनाया और समर्थित है, जिसका अर्थ है कि यह एक उच्च-प्रदर्शन और शक्तिशाली ढांचा है। इसके अतिरिक्त, कंबाइन फ्रेमवर्क को अन्य Apple फ्रेमवर्क और API जैसे UIKit और SwiftUI के साथ मूल रूप से काम करने के लिए डिज़ाइन किया गया है।

यदि आप पहले से ही प्रतिक्रियाशील प्रोग्रामिंग से परिचित हैं, तो आप पाएंगे कि कंबाइन फ्रेमवर्क डेटा स्ट्रीम का प्रतिनिधित्व करने और घटनाओं को संभालने के लिए प्रकाशकों, ग्राहकों और ऑपरेटरों जैसी परिचित अवधारणाओं का उपयोग करता है। हालाँकि, भले ही आप प्रतिक्रियाशील प्रोग्रामिंग के लिए नए हों, कंबाइन फ्रेमवर्क एसिंक्रोनस डेटा स्ट्रीम को संभालने के लिए एक एकीकृत और आसानी से सीखने वाला मॉडल प्रदान करता है।

हम कंबाइन का उपयोग करके क्या कर सकते हैं?

कंबाइन फ्रेमवर्क अतुल्यकालिक डेटा स्ट्रीम को संभालने के लिए एक घोषणात्मक मॉडल प्रदान करता है। इसका मतलब यह है कि हम क्लोजर, प्रतिनिधियों, केवीओ, और अधिसूचनाओं को अधिक कुशलता से संभालने के लिए ढांचे के एपीआई का उपयोग कर सकते हैं , और शक्तिशाली तरीकों से डेटा स्ट्रीम को रूपांतरित, फ़िल्टर और संयोजित कर सकते हैं।

उदाहरण के लिए, यदि हमें घटनाओं की एक श्रृंखला को संभालने की आवश्यकता है, जैसे कि UI को अपडेट करने से पहले नेटवर्क अनुरोध के पूरा होने की प्रतीक्षा करना, तो हम अधिक संक्षिप्त और पठनीय कोड लिखने के लिए कंबाइन फ्रेमवर्क का उपयोग कर सकते हैं। हम एक प्रकाशक बना सकते हैं जो नेटवर्क अनुरोध पूरा होने पर एक मूल्य का उत्सर्जन करता है, और यूआई को अपडेट करने के लिए उस प्रकाशक की सदस्यता लेता है। यह हमारे कोड को अधिक सुव्यवस्थित और तर्क करने में आसान बना सकता है।

इस ट्यूटोरियल के बाकी हिस्सों में, हम कम्बाइन फ्रेमवर्क की मूल बातों का पता लगाएंगे और आपको दिखाएंगे कि एपीआई कॉल लिखने के लिए इसका उपयोग कैसे करें।

संयोजन कैसे काम करता है?

कंबाइन फ्रेमवर्क निम्नलिखित तीन प्रमुख अवधारणाओं के आधार पर कार्य करता है

1. प्रकाशक

2. सब्सक्राइबर

3. संचालिका

कंबाइन मुख्य रूप से प्रकाशक और ग्राहक मॉडल के रूप में काम कर रहा है।

प्रकाशक एक प्रोटोकॉल है जो मूल्य प्रकार को दो संबद्ध प्रकारों के साथ परिभाषित करता है : आउटपुट और विफलता । प्रकाशक ग्राहकों के पंजीकरण की अनुमति देते हैं और उन ग्राहकों के लिए मूल्यों का उत्सर्जन करते हैं। प्रकाशकों को डेटा स्ट्रीम के स्रोत के रूप में माना जा सकता है।

सब्सक्राइबर भी एक प्रोटोकॉल है, लेकिन यह एक रेफरेंस टाइप है। सदस्य प्रकाशकों से मूल्य प्राप्त कर सकते हैं और जब प्रकाशक मूल्यों का उत्सर्जन करना समाप्त कर लेते हैं तो उन्हें सूचित किया जा सकता है। सब्सक्राइबर के दो संबद्ध प्रकार हैं: इनपुट और विफलता । सब्सक्राइबर्स को डेटा स्ट्रीम के गंतव्य के रूप में माना जा सकता है।

यहां पब्लिशर और सब्सक्राइबर मॉडल को मिलाने का पैटर्न दिया गया है।

नमूना

अंत में, ऑपरेटर प्रकाशकों के विस्तार हैं। ऑपरेटर वे विधियाँ हैं जिन्हें प्रकाशक द्वारा बुलाया जाता है और प्रकाशक को एक मूल्य लौटाता है। ऑपरेटरों को प्रकाशकों के संबद्ध प्रकारों को ग्राहकों तक पहुँचाने से पहले उन्हें बदलने या हेरफेर करने के लिए इस्तेमाल किया जा सकता है। एक प्रकाशक के लिए कई ऑपरेटर हो सकते हैं, जो जटिल डेटा परिवर्तन और जोड़-तोड़ की अनुमति देते हैं।

प्रकाशकों और सदस्यों के साथ ऑपरेटरों को कैसे जोड़ा जाए, इसका चित्रण

कंबाइन फ्रेमवर्क का उपयोग करके एपीआई कॉल करना

यह कंबाइन फ्रेमवर्क का त्वरित परिचय था। अब आइए एक वास्तविक दुनिया के उदाहरण में गोता लगाएँ कि एपीआई कॉल करने के लिए इसका उपयोग कैसे किया जाए।

इस उदाहरण में, हम Apple RSS फ़ीड से संयुक्त राज्य अमेरिका में उपयोग किए जाने वाले शीर्ष 50 iOS निःशुल्क ऐप्स प्राप्त करेंगे। आप इस वेबसाइट पर भविष्य में उपयोग के लिए अपना स्वयं का RSS फ़ीड बना सकते हैं:
https://rss.applemarketingtools.com/

इसे पूरा करने के लिए, हम निम्नलिखित यूआरएल को कॉल करेंगे और परिणाम को टेबलव्यू में लोड करेंगे:
https://rss.applemarketingtools.com/api/v2/us/apps/top-free/50/apps.json

यहाँ JSON संरचना है:

{
  "feed": {
    "title": "Top Free Apps",
    "id": "https://rss.applemarketingtools.com/api/v2/us/apps/top-free/2/apps.json",
    "author": {
      "name": "Apple",
      "url": "https://www.apple.com/"
    },
    "links": [
      {
        "self": "https://rss.applemarketingtools.com/api/v2/us/apps/top-free/2/apps.json"
      }
    ],
    "copyright": "Copyright © 2023 Apple Inc. All rights reserved.",
    "country": "us",
    "icon": "https://www.apple.com/favicon.ico",
    "updated": "Mon, 1 May 2023 17:37:11 +0000",
    "results": [
      {
        "artistName": "Temu",
        "id": "1641486558",
        "name": "Temu: Shop Like a Billionaire",
        "releaseDate": "2022-08-31",
        "kind": "apps",
        "artworkUrl100": "https://is5-ssl.mzstatic.com/image/thumb/Purple116/v4/da/78/29/da7829de-6cdf-d3da-853f-24fb53186635/AppIcon-1x_U007emarketing-0-7-0-0-P3-85-220.png/100x100bb.png",
        "genres": [],
        "url": "https://apps.apple.com/us/app/temu-shop-like-a-billionaire/id1641486558"
      },
      {
        "artistName": "Bytedance Pte. Ltd",
        "id": "1500855883",
        "name": "CapCut - Video Editor",
        "releaseDate": "2020-04-14",
        "kind": "apps",
        "artworkUrl100": "https://is4-ssl.mzstatic.com/image/thumb/Purple126/v4/9d/17/31/9d1731e5-bab9-31c6-7cc9-ba2cb0d7690f/AppIcon-0-0-1x_U007emarketing-0-0-0-7-0-0-sRGB-0-0-0-GLES2_U002c0-512MB-85-220-0-0.png/100x100bb.png",
        "genres": [],
        "url": "https://apps.apple.com/us/app/capcut-video-editor/id1500855883"
      }
    ]
  }
}

प्रोजेक्ट बनाने के चरण:

  1. उपरोक्त JSON डेटा को डिकोड करने के लिए, एक API मॉडल स्ट्रक्चर बनाया जाएगा। फ़ाइल का नाम AppResponse.swift होगा , और यह इस तरह दिखेगी:
  2. import Foundation
    
    struct AppsResponse: Codable {
        let feed: Feed
    }
    
    // MARK: - Feed
    struct Feed: Codable {
        let results: [App]
    }
    
    // MARK: - Result
    public struct App: Codable {
        let artistName, id, name, releaseDate: String
        let artworkUrl100: String
        let url: String
    }
    

    import UIKit
    import Combine  // (1)
    
    enum NetworkError: Error {  // (2)
        case invalidURL
        case responseError
        case unknown
    }
    
    extension NetworkError: LocalizedError {
        var errorDescription: String? {
            switch self {
            case .invalidURL:
                return NSLocalizedString("Invalid URL", comment: "")
            case .responseError:
                return NSLocalizedString("Unexpected status code", comment: "")
            case .unknown:
                return NSLocalizedString("Unknown error", comment: "")
            }
        }
    }
    
    class WebServiceManager: NSObject {  
        static let shared = WebServiceManager()
    
        private var cancellables = Set<AnyCancellable>() // (3)
        
         func getData<T: Decodable>(endpoint: String, id: Int? = nil, type: T.Type) -> Future<T, Error> {
            return Future<T, Error> { [weak self] promise in  // (4) -> Future Publisher
                guard let self = self, let url = URL(string: endpoint) else {  
                    return promise(.failure(NetworkError.invalidURL))
                }
                print("URL is \(url.absoluteString)")
                URLSession.shared.dataTaskPublisher(for: url) // (5) -> Publisher
                    .tryMap { (data, response) -> Data in  // (6) -> Operator
                        guard let httpResponse = response as? HTTPURLResponse, 200...299 ~= httpResponse.statusCode else {
                            throw NetworkError.responseError
                        }
                        return data
                    }
                    .decode(type: T.self, decoder: JSONDecoder())  // (7) -> Operator
                    .receive(on: RunLoop.main) // (8) -> Sheduler Operator
                    .sink(receiveCompletion: { (completion) in  // (9) -> Subscriber
                        if case let .failure(error) = completion {
                            switch error {
                            case let decodingError as DecodingError:
                                promise(.failure(decodingError))
                            case let apiError as NetworkError:
                                promise(.failure(apiError))
                            default:
                                promise(.failure(NetworkError.unknown))
                            }
                        }
                    }, receiveValue: {  data in  // (10)
                        print(data)
                        promise(.success(data)
        ) })
                    .store(in: &self.cancellables)  // (11)
            }
        }
    }
    

(2)। NetworkError को परिभाषित करना: एक गणना को परिभाषित किया गया है जो त्रुटि प्रोटोकॉल के अनुरूप है। एपीआई कॉल के दौरान होने वाली विभिन्न त्रुटियों का प्रतिनिधित्व करने के लिए इस गणना का उपयोग किया जाता है।

(3)। डिक्लेयर सेट <AnyCancellable> ऑब्जेक्ट: एपीआई कॉल के दौरान किए गए किसी भी कंबाइन सब्सक्रिप्शन का ट्रैक रखने के लिए AnyCancellable ऑब्जेक्ट्स का एक सेट बनाया गया है। एपीआई कॉल पूर्ण या रद्द होने पर ये सदस्यताएं रद्द कर दी जाएंगी।

(4)। वापसी भविष्य: भविष्य एक प्रकार का प्रकाशक है। यह एसिंक्रोनस ऑपरेशन का उपयोग कर रहा है और आउटपुट और त्रुटि देता है । यह एपीआई कॉल से डेटा प्राप्त होने तक प्रतीक्षा कर रहा है।

(5)। URLSession.shared.dataTaskPublisher (): भविष्य के प्रकाशक के अंदर हम एपीआई से डेटा प्राप्त करने के लिए एक और प्रकाशक ग्राहक मॉडल लिख रहे हैं। dataTaskPublisher() URL सत्र डेटा कार्य के लिए प्रकाशक है। यह एक यूआरएल लेता है और डेटा और यूआरएल रेस्पॉन्स का एक टपल देता है।

(6)। tryMap (): यह एक कंबाइन ऑपरेटर है जो अपस्ट्रीम प्रकाशक द्वारा उत्सर्जित डेटा को रूपांतरित करता है और एक त्रुटि फेंक सकता है। इस मामले में, यह प्रतिक्रिया के HTTP स्थिति कोड की जांच करता है और 200-299 रेंज में नहीं होने पर त्रुटि फेंकता है: यदि कोई त्रुटि नहीं है, तो यह डेटा उत्सर्जित करता है।

(7)। डिकोड (): यह एक कंबाइन ऑपरेटर है जो JSONDecoder का उपयोग करके अपस्ट्रीम डेटा को डिकोड करता है और निर्दिष्ट सामान्य प्रकार T का मान लौटाता है । यदि डिकोडिंग विफल हो जाती है, तो यह एक त्रुटि फेंक देगा।

(8)। प्राप्त करें (चालू: RunLoop.main): यह एक संयोजन ऑपरेटर है जो शेड्यूलर को निर्दिष्ट करता है जिस पर मान प्राप्त करना है। इस मामले में, यह मुख्य थ्रेड के रनलूप को निर्दिष्ट करता है, जो यूजर इंटरफेस को अपडेट करने के लिए जरूरी है।

(9)। सिंक (प्राप्त करें: प्राप्त करें:): यह एक संयोजन ऑपरेटर है जो प्रकाशक के मूल्यों और त्रुटियों को प्राप्त करने के लिए एक ग्राहक बनाता है।

(10)। रिसीव वैल्यू को हैंडल करना: यह सिंक के अंदर क्लोजर है और इसे तब निष्पादित किया जाता है जब फ्यूचर सफलतापूर्वक टाइप टी का मान पैदा करता है । इस मामले में, यह केवल डेटा को कंसोल पर प्रिंट करता है और प्राप्त मूल्य के साथ वादे को पूरा करता है।

(11)। store(in: &self.cancellables): यह AnyCancellable का एक तरीका है जो सब्सक्रिप्शन को कैंसिलेबल सेट में जोड़ता है। यह सदस्यता को बाद में जरूरत पड़ने पर रद्द करने की अनुमति देता है।

3. आइए देखें कि UITableViewController उपवर्ग के लिए डेटा पुनर्प्राप्त करने के लिए उपरोक्त विधि को कैसे कॉल करें।

private var apps:[App]? = nil
    private var cancellables = Set<AnyCancellable>() // (1)
    
    private func getApps() {
        let spinner = UIActivityIndicatorView(style: .medium)
        spinner.hidesWhenStopped = true
            spinner.startAnimating()
            tableView.backgroundView = spinner
        
        let appURL = "https://rss.applemarketingtools.com/api/v2/us/apps/top-free/50/apps.json"
         
        WebServiceManager.shared.getData(endpoint: appURL, type: AppResponse.self) // (2) -> Publisher
            .sink { completion in // (3) -> Subscriber
                spinner.stopAnimating()
            switch completion {
            case .failure(let err):
                print("Error is \(err.localizedDescription)")
            case .finished:
                print("Finished")
            }
        } receiveValue: { [weak self] appResponse in
            self?.apps = appResponse.feed.results
            self?.tableView.reloadData()
        } .store(in: &cancellables) // (4)
    }

(1)। आपके द्वारा बनाई जाने वाली सदस्यताओं को रखने के लिए रद्द करने योग्य का एक नया खाली सेट बनाएं।

(2)। AppURL और AppsResponse.self प्रकार में गुजरते हुए WebServiceManager की getData विधि को कॉल करें । यह एक भविष्य प्रकाशक लौटाएगा जो दिए गए समापन बिंदु से डेटा पुनर्प्राप्त करने के लिए एपीआई अनुरोध करेगा।

(3)। सिंक पद्धति का उपयोग करके भविष्य के प्रकाशक के लिए दो ग्राहकों को संलग्न करें। एक ग्राहक कंप्लीशन इवेंट को हैंडल करता है, जबकि दूसरा डेटा इवेंट को हैंडल करता है।

(4)। सब्सक्रिप्शन को स्मृति में रखने के लिए रद्द करने योग्य सेट में संग्रहीत करें और यदि आवश्यक हो तो उन्हें रद्द करने की अनुमति दें।

यह काफी हद तक ट्यूटोरियल है। आप निम्न लिंक का उपयोग करके स्रोत कोड डाउनलोड कर सकते हैं।

निष्कर्ष

अंत में, iOS कंबाइन फ्रेमवर्क iOS अनुप्रयोगों में प्रतिक्रियाशील प्रोग्रामिंग लिखने के लिए एक शक्तिशाली उपकरण है। अपने प्रकाशक-सब्सक्राइबर मॉडल और उपयोग में आसान ऑपरेटरों के साथ, आईओएस कंबाइन फ्रेमवर्क डेवलपर्स को एक घोषणात्मक और पठनीय तरीके से डेटा की धाराओं को आसानी से प्रबंधित और हेरफेर करने में सक्षम बनाता है। हालांकि नए उपयोगकर्ताओं के लिए सीखने की अवस्था हो सकती है, अपने कोडबेस में कंबाइन को शामिल करने से अधिक कुशल और रखरखाव योग्य कोड हो सकता है। चाहे आप छोटी या बड़ी परियोजनाओं पर काम कर रहे हों, कंबाइन का उपयोग करके आप अपनी प्रतिक्रियाशील प्रोग्रामिंग को सरल बना सकते हैं। मुझे उम्मीद है कि इस ट्यूटोरियल ने आपको कंबाइन का एक उपयोगी परिचय दिया है और आप इसे अपनी परियोजनाओं में कैसे उपयोग कर सकते हैं, और आपको इसकी अधिक क्षमताओं का पता लगाने के लिए प्रेरित करता है। यदि आपके कोई प्रश्न या प्रतिक्रिया हैं, तो कृपया उन्हें नीचे टिप्पणी में छोड़ दें।

पढ़ने के लिए धन्यवाद, और हैप्पी कोडिंग!