क्या स्विफ्ट भाषा में एक अमूर्त वर्ग बनाने का एक तरीका है, या क्या यह उद्देश्य-सी की तरह एक सीमा है? मैं एक सार वर्ग के रूप में जावा को परिभाषित करता है, जो एक सार वर्ग बनाना चाहता हूं।
क्या स्विफ्ट भाषा में एक अमूर्त वर्ग बनाने का एक तरीका है, या क्या यह उद्देश्य-सी की तरह एक सीमा है? मैं एक सार वर्ग के रूप में जावा को परिभाषित करता है, जो एक सार वर्ग बनाना चाहता हूं।
जवाबों:
स्विफ्ट में कोई सार कक्षाएं नहीं हैं (बस उद्देश्य-सी की तरह)। आपका सबसे अच्छा शर्त एक प्रोटोकॉल का उपयोग करने वाला है , जो एक जावा इंटरफेस की तरह है।
स्विफ्ट 2.0 के साथ, आप फिर प्रोटोकॉल एक्सटेंशन का उपयोग करके विधि कार्यान्वयन और गणना की गई संपत्ति कार्यान्वयन जोड़ सकते हैं। आपके एकमात्र प्रतिबंध हैं कि आप सदस्य चर या स्थिरांक प्रदान नहीं कर सकते हैं और कोई गतिशील प्रेषण नहीं है ।
इस तकनीक का एक उदाहरण होगा:
protocol Employee {
var annualSalary: Int {get}
}
extension Employee {
var biweeklySalary: Int {
return self.annualSalary / 26
}
func logSalary() {
print("$\(self.annualSalary) per year or $\(self.biweeklySalary) biweekly")
}
}
struct SoftwareEngineer: Employee {
var annualSalary: Int
func logSalary() {
print("overridden")
}
}
let sarah = SoftwareEngineer(annualSalary: 100000)
sarah.logSalary() // prints: overridden
(sarah as Employee).logSalary() // prints: $100000 per year or $3846 biweekly
ध्यान दें कि यह "एब्सट्रैक्ट क्लास" प्रदान कर रहा है, यहां तक कि स्ट्रक्चर्स के लिए भी, लेकिन कक्षाएं भी उसी प्रोटोकॉल को लागू कर सकती हैं।
यह भी ध्यान दें कि कर्मचारी प्रोटोकॉल को लागू करने वाले प्रत्येक वर्ग या संरचना को फिर से वार्षिक संपत्ति घोषित करनी होगी।
सबसे महत्वपूर्ण बात, ध्यान दें कि कोई गतिशील प्रेषण नहीं है । जब logSalary
इसे इंस्टेंस के रूप में संग्रहीत किया जाता है तो SoftwareEngineer
इसे विधि का ओवरराइड संस्करण कहा जाता है । जब इसे कॉल logSalary
करने के बाद इसे इंस्टेंस पर Employee
कॉल किया जाता है , तो यह ओरिजिनल इम्प्लीमेंटेशन कहता है (यह डायनामिकली ओवरराइड किए गए वर्जन पर नहीं जाता है, हालांकि इंस्टेंस वास्तव में ए है Software Engineer
।
अधिक जानकारी के लिए, उस सुविधा के बारे में महान WWDC वीडियो देखें: स्विफ्ट में मूल्य प्रकारों के साथ बिल्डिंग बेहतर ऐप्स
protocol Animal { var property : Int { get set } }
। यदि आप नहीं चाहते कि आप सेट को छोड़ भी सकें, तो संपत्ति को एक सेटर करने की जरूरत नहीं है
func logSalary()
कर्मचारी प्रोटोकॉल घोषणा में जोड़ते हैं , तो उदाहरण overridden
दोनों कॉल के लिए प्रिंट करता है logSalary()
। यह स्विफ्ट 3.1 में है। इस प्रकार आपको बहुरूपता का लाभ मिलता है। दोनों ही मामलों में सही विधि कहा जाता है।
ध्यान दें कि यह उत्तर स्विफ्ट 2.0 और इसके बाद के संस्करण पर लक्षित है
आप प्रोटोकॉल और प्रोटोकॉल एक्सटेंशन के साथ समान व्यवहार प्राप्त कर सकते हैं।
सबसे पहले, आप एक प्रोटोकॉल लिखते हैं जो उन सभी तरीकों के लिए एक इंटरफ़ेस के रूप में कार्य करता है जिन्हें सभी प्रकारों में लागू किया जाना है जो इसके अनुरूप हैं।
protocol Drivable {
var speed: Float { get set }
}
फिर आप उन सभी प्रकारों में डिफ़ॉल्ट व्यवहार जोड़ सकते हैं जो इसके अनुरूप हैं
extension Drivable {
func accelerate(by: Float) {
speed += by
}
}
अब आप लागू करके नए प्रकार बना सकते हैं Drivable
।
struct Car: Drivable {
var speed: Float = 0.0
init() {}
}
let c = Car()
c.accelerate(10)
तो मूल रूप से आपको मिलता है:
Drivable
लागूspeed
Drivable
( accelerate
) के अनुरूप सभी प्रकारों के लिए डिफ़ॉल्ट-व्यवहार लागू कर सकते हैंDrivable
यह सिर्फ एक प्रोटोकॉल के बाद से तुरंत नहीं होने की गारंटी हैयह मॉडल वास्तव में लक्षणों की तरह बहुत अधिक व्यवहार करता है, जिसका अर्थ है कि आप कई प्रोटोकॉल के अनुरूप हो सकते हैं और उनमें से किसी के डिफ़ॉल्ट कार्यान्वयन को ले सकते हैं, जबकि एक अमूर्त सुपरक्लास के साथ आप एक साधारण वर्ग पदानुक्रम तक सीमित होते हैं।
UICollectionViewDatasource
,। मैं सभी बॉयलरप्लेट को हटाकर अलग-अलग प्रोटोकॉल / एक्सटेंशन में इनकैप्सुलेट करना चाहता हूं और फिर कई वर्गों द्वारा फिर से उपयोग करना चाहता हूं। वास्तव में, टेम्प्लेट पैटर्न यहां बिल्कुल सही होगा, लेकिन ...
मुझे लगता है कि यह जावा abstract
या C # के सबसे करीब है abstract
:
class AbstractClass {
private init() {
}
}
ध्यान दें, private
काम करने के लिए संशोधक के लिए, आपको एक अलग स्विफ्ट फ़ाइल में इस वर्ग को परिभाषित करना होगा।
संपादित करें: फिर भी, यह कोड एक सार विधि घोषित करने की अनुमति नहीं देता है और इस प्रकार इसके कार्यान्वयन को मजबूर करता है।
सबसे आसान तरीका fatalError("Not Implemented")
प्रोटोकॉल एक्सटेंशन पर अमूर्त विधि (चर नहीं) में कॉल का उपयोग करना है ।
protocol MyInterface {
func myMethod() -> String
}
extension MyInterface {
func myMethod() -> String {
fatalError("Not Implemented")
}
}
class MyConcreteClass: MyInterface {
func myMethod() -> String {
return "The output"
}
}
MyConcreteClass().myMethod()
(MyConcreteClass() as MyInterface).myMethod()
लेकिन यह करता है! कुंजी myMethod
प्रोटोकॉल घोषणा में शामिल है; अन्यथा कॉल क्रैश हो जाता है।
कई हफ्तों तक संघर्ष करने के बाद, मुझे आखिरकार एहसास हुआ कि स्विफ्ट में जावा / पीएचपी अमूर्त वर्ग का अनुवाद कैसे किया जाए:
public class AbstractClass: NSObject {
internal override init(){}
public func getFoodToEat()->String
{
if(self._iAmHungry())
{
return self._myFavoriteFood();
}else{
return "";
}
}
private func _myFavoriteFood()->String
{
return "Sandwich";
}
internal func _iAmHungry()->Bool
{
fatalError(__FUNCTION__ + "Must be overridden");
return false;
}
}
public class ConcreteClass: AbstractClass, IConcreteClass {
private var _hungry: Bool = false;
public override init() {
super.init();
}
public func starve()->Void
{
self._hungry = true;
}
public override func _iAmHungry()->Bool
{
return self._hungry;
}
}
public protocol IConcreteClass
{
func _iAmHungry()->Bool;
}
class ConcreteClassTest: XCTestCase {
func testExample() {
var concreteClass: ConcreteClass = ConcreteClass();
XCTAssertEqual("", concreteClass.getFoodToEat());
concreteClass.starve();
XCTAssertEqual("Sandwich", concreteClass.getFoodToEat());
}
}
हालाँकि मुझे लगता है कि Apple ने अमूर्त वर्ग को लागू नहीं किया क्योंकि यह आम तौर पर डेलिगेट + प्रोटोकॉल पैटर्न का उपयोग करता है। उदाहरण के लिए उपरोक्त पैटर्न इसी तरह बेहतर होगा:
import UIKit
public class GoldenSpoonChild
{
private var delegate: IStomach!;
internal init(){}
internal func setup(delegate: IStomach)
{
self.delegate = delegate;
}
public func getFoodToEat()->String
{
if(self.delegate.iAmHungry())
{
return self._myFavoriteFood();
}else{
return "";
}
}
private func _myFavoriteFood()->String
{
return "Sandwich";
}
}
public class Mother: GoldenSpoonChild, IStomach
{
private var _hungry: Bool = false;
public override init()
{
super.init();
super.setup(self);
}
public func makeFamilyHungry()->Void
{
self._hungry = true;
}
public func iAmHungry()->Bool
{
return self._hungry;
}
}
protocol IStomach
{
func iAmHungry()->Bool;
}
class DelegateTest: XCTestCase {
func testGetFood() {
var concreteClass: Mother = Mother();
XCTAssertEqual("", concreteClass.getFoodToEat());
concreteClass.makeFamilyHungry();
XCTAssertEqual("Sandwich", concreteClass.getFoodToEat());
}
}
मुझे इस तरह के पैटर्न की आवश्यकता थी क्योंकि मैं UITableViewController जैसे viewWillAppear आदि में कुछ तरीकों को सामान्य करना चाहता था।
प्रोटोकॉल का उपयोग करके अमूर्त वर्गों का अनुकरण करने का एक तरीका है। यह एक उदाहरण है:
protocol MyProtocol {
func doIt()
}
class BaseClass {
weak var myDelegate: MyProtocol?
init() {
...
}
func myFunc() {
...
self.myDelegate?.doIt()
...
}
}
class ChildClass: BaseClass, MyProtocol {
override init(){
super.init()
self.myDelegate = self
}
func doIt() {
// Custom implementation
}
}
एक और तरीका है कि आप अमूर्त वर्ग को कैसे लागू कर सकते हैं, इनिशियलज़र को ब्लॉक करना है। मैंने इसे इस तरह किया है:
class Element:CALayer { // IT'S ABSTRACT CLASS
override init(){
super.init()
if self.dynamicType === Element.self {
fatalError("Element is abstract class, do not try to create instance of this class")
}
}
}
मैं एक Weather
सार वर्ग बनाने की कोशिश कर रहा था , लेकिन प्रोटोकॉल का उपयोग करना आदर्श नहीं था क्योंकि मुझे init
बार-बार एक ही तरीके को लिखना पड़ता था। प्रोटोकॉल का विस्तार और एक लेखनinit
विधि यह समस्या थी, खासकर जब से मैं इसके NSObject
अनुरूप उपयोग कर रहा थाNSCoding
।
इसलिए मैं इस NSCoding
अनुरूपता के लिए आया :
required init?(coder aDecoder: NSCoder) {
guard type(of: self) != Weather.self else {
fatalError("<Weather> This is an abstract class. Use a subclass of `Weather`.")
}
// Initialize...
}
के रूप में init
:
fileprivate init(param: Any...) {
// Initialize
}
बेस क्लास की सभी संपत्तियों और विधियों को प्रोटोकॉल एक्सटेंशन के कार्यान्वयन के लिए स्थानांतरित करें, जहां सेल्फ बेस क्लास के लिए बाधा है। आप बेस क्लास के सभी तरीकों और संपत्तियों तक पहुंच प्राप्त करेंगे। इसके अतिरिक्त संकलक व्युत्पन्न वर्गों के लिए प्रोटोकॉल में अमूर्त विधियों और गुणों के कार्यान्वयन की जांच करते हैं
protocol Commom:class{
var tableView:UITableView {get};
func update();
}
class Base{
var total:Int = 0;
}
extension Common where Self:Base{
func update(){
total += 1;
tableView.reloadData();
}
}
class Derived:Base,Common{
var tableView:UITableView{
return owner.tableView;
}
}
कोई गतिशील प्रेषण की सीमा के साथ, आप ऐसा कुछ कर सकते हैं:
import Foundation
protocol foo {
static var instance: foo? { get }
func prt()
}
extension foo {
func prt() {
if Thread.callStackSymbols.count > 30 {
print("super")
} else {
Self.instance?.prt()
}
}
}
class foo1 : foo {
static var instance : foo? = nil
init() {
foo1.instance = self
}
func prt() {
print("foo1")
}
}
class foo2 : foo {
static var instance : foo? = nil
init() {
foo2.instance = self
}
func prt() {
print("foo2")
}
}
class foo3 : foo {
static var instance : foo? = nil
init() {
foo3.instance = self
}
}
var f1 : foo = foo1()
f1.prt()
var f2 : foo = foo2()
f2.prt()
var f3 : foo = foo3()
f3.prt()