Swift - Zugangskontrolle

Um den Zugriff auf Codeblöcke, Module und Abstraktion einzuschränken, erfolgt die Zugriffssteuerung. Auf Klassen, Strukturen und Aufzählungen kann über ihre Eigenschaften, Methoden, Initialisierer und Indizes über Zugriffssteuerungsmechanismen zugegriffen werden. Konstanten, Variablen und Funktionen in einem Protokoll sind eingeschränkt und ermöglichen den Zugriff als global und lokal durch Zugriffssteuerung. Die auf Eigenschaften, Typen und Funktionen angewendete Zugriffssteuerung kann als "Entitäten" bezeichnet werden.

Das Zugriffssteuerungsmodell basiert auf Modulen und Quelldateien.

Das Modul ist als eine Einheit der Codeverteilung definiert und kann mit dem Schlüsselwort 'import' importiert werden. Eine Quelldatei wird als einzelne Quellcodedatei in einem Modul definiert, um auf mehrere Typen und Funktionen zuzugreifen.

Die Sprache Swift 4 bietet drei verschiedene Zugriffsebenen. Sie sind öffentlich, intern und privat zugänglich.

S.No. Zugriffsebenen & Definition
1

Public

Ermöglicht die Verarbeitung von Entitäten in einer beliebigen Quelldatei aus ihrem definierenden Modul, einer Quelldatei aus einem anderen Modul, das das definierende Modul importiert.

2

Internal

Ermöglicht die Verwendung von Entitäten in einer Quelldatei aus ihrem definierenden Modul, jedoch nicht in einer Quelldatei außerhalb dieses Moduls.

3

Private

Beschränkt die Verwendung einer Entität auf ihre eigene definierende Quelldatei. Der private Zugriff spielt eine Rolle, um die Implementierungsdetails einer bestimmten Codefunktionalität auszublenden.

Syntax

public class SomePublicClass {}
internal class SomeInternalClass {}
private class SomePrivateClass {}

public var somePublicVariable = 0
internal let someInternalConstant = 0
private func somePrivateFunction() {}

Zugriffskontrolle für Funktionstypen

Bei einigen Funktionen sind möglicherweise Argumente innerhalb der Funktion ohne Rückgabewerte deklariert. Das folgende Programm deklariert a und b als Argumente für die Funktion sum (). Innerhalb der Funktion selbst werden die Werte für die Argumente a und b durch Aufrufen des Funktionsaufrufs sum () übergeben und ihre Werte gedruckt, wodurch Rückgabewerte eliminiert werden. Um den Rückgabetyp der Funktion als privat festzulegen, deklarieren Sie die allgemeine Zugriffsebene der Funktion mit dem Modifikator private.

private func sum(a: Int, b: Int) {
   let a = a + b
   let b = a - b
   print(a, b)
}

sum(a: 20, b: 10)
sum(a: 40, b: 10)
sum(a: 24, b: 6)

Wenn wir das obige Programm auf einem Spielplatz ausführen, erhalten wir das folgende Ergebnis:

30 20
50 40
30 24

Zugriffssteuerung für Aufzählungstypen

public enum Student {
   case Name(String)
   case Mark(Int,Int,Int)
}
var studDetails = Student.Name("Swift 4")
var studMarks = Student.Mark(98,97,95)

switch studMarks {
   case .Name(let studName):
      print("Student name is: \(studName).")
   case .Mark(let Mark1, let Mark2, let Mark3):
      print("Student Marks are: \(Mark1),\(Mark2),\(Mark3).")
   
}

Wenn wir das obige Programm auf einem Spielplatz ausführen, erhalten wir das folgende Ergebnis:

Student Marks are: 98,97,95

Die Aufzählung in der Sprache Swift 4 erhält für einzelne Fälle einer Aufzählung automatisch dieselbe Zugriffsebene. Betrachten Sie zum Beispiel den Zugriff auf den Schülernamen und die in drei Fächern gesicherten Noten. Der Aufzählungsname wird als Schüler deklariert, und die in der Aufzählungsklasse vorhandenen Mitglieder sind Namen, die zum Zeichenfolgendatentyp gehören. Noten werden als Marke1, Marke2 und Marke3 des Datentyps Integer dargestellt. Zugriff auf den Namen des Schülers oder auf die von ihm erzielten Noten. Jetzt druckt der Schalterfall den Schülernamen, wenn dieser Fallblock ausgeführt wird, andernfalls werden die vom Schüler gesicherten Markierungen gedruckt. Wenn beide Bedingungen fehlschlagen, wird der Standardblock ausgeführt.

Zugriffskontrolle für Unterklassen

Mit Swift 4 kann der Benutzer jede Klasse unterklassifizieren, auf die im aktuellen Zugriffskontext zugegriffen werden kann. Eine Unterklasse kann keine höhere Zugriffsebene als ihre Oberklasse haben. Der Benutzer darf keine öffentliche Unterklasse einer internen Oberklasse schreiben.

public class cricket {
   internal func printIt() {
      print("Welcome to Swift 4 Super Class")
   }
}

internal class tennis: cricket {
   override internal func printIt() {
      print("Welcome to Swift 4 Sub Class")
   }
}

let cricinstance = cricket()
cricinstance.printIt()

let tennisinstance = tennis()
tennisinstance.printIt()

Wenn wir das obige Programm auf einem Spielplatz ausführen, erhalten wir das folgende Ergebnis:

Welcome to Swift Super Class
Welcome to Swift Sub Class

Zugriffssteuerung für Konstanten, Variablen, Eigenschaften und Indizes

Swift 4-Konstante, -Variable oder -Eigenschaft kann nicht als öffentlicher Typ definiert werden. Es ist nicht gültig, eine öffentliche Eigenschaft mit einem privaten Typ zu schreiben. Ebenso kann ein Index nicht öffentlicher sein als sein Index- oder Rückgabetyp.

Wenn eine Konstante, Variable, Eigenschaft oder ein Index einen privaten Typ verwendet, muss die Konstante, Variable, Eigenschaft oder der Index ebenfalls als privat markiert werden.

private var privateInstance = SomePrivateClass()

Getter und Setter

Getter und Setter für Konstanten, Variablen, Eigenschaften und Indizes erhalten automatisch dieselbe Zugriffsebene wie die Konstanten, Variablen, Eigenschaften oder Indizes, zu denen sie gehören.

class Samplepgm {
   var counter: Int = 0{
      willSet(newTotal) {
         print("Total Counter is: \(newTotal)")
      }
      didSet {
         if counter > oldValue {
            print("Newly Added Counter \(counter - oldValue)")
         }
      }
   }
}

let NewCounter = Samplepgm()
NewCounter.counter = 100
NewCounter.counter = 800

Wenn wir das obige Programm auf einem Spielplatz ausführen, erhalten wir das folgende Ergebnis:

Total Counter is: 100
Newly Added Counter 100
Total Counter is: 800
Newly Added Counter 700

Zugriffssteuerung für Initialisierer und Standardinitialisierer

Benutzerdefinierte Initialisierer können einer Zugriffsebene zugewiesen werden, die kleiner oder gleich dem Typ ist, den sie initialisieren. Ein erforderlicher Initialisierer muss dieselbe Zugriffsebene haben wie die Klasse, zu der er gehört. Die Typen der Parameter eines Initialisierers können nicht privater sein als die Zugriffsebene des Initialisierers.

Um jede Unterklasse des Schlüsselworts "Initialisieren" erforderlich zu deklarieren, muss sie vor der Funktion init () definiert werden.

class classA {
   required init() {
      let a = 10
      print(a)
   }
}
class classB: classA {
   required init() {
      let b = 30
      print(b)
   }
}
let res = classA()
let print = classB()

Wenn wir das obige Programm auf einem Spielplatz ausführen, erhalten wir das folgende Ergebnis:

10
30
10

Ein Standardinitialisierer hat dieselbe Zugriffsebene wie der Typ, den er initialisiert, es sei denn, dieser Typ ist als öffentlich definiert. Wenn die Standardinitialisierung als öffentlich definiert ist, wird sie als intern betrachtet. Wenn der Benutzer einen öffentlichen Typ benötigt, der mit einem Initialisierer ohne Argumente in einem anderen Modul initialisiert werden kann, geben Sie als Teil der Definition des Typs explizit einen öffentlichen Initialisierer ohne Argumente an.

Zugriffskontrolle für Protokolle

Wenn wir ein neues Protokoll definieren, um Funktionen von einem vorhandenen Protokoll zu erben, müssen beide als dieselben Zugriffsebenen deklariert werden, um die Eigenschaften voneinander zu erben. Mit der Swift 4-Zugriffskontrolle können die Benutzer kein "öffentliches" Protokoll definieren, das von einem "internen" Protokoll erbt.

public protocol tcpprotocol {
   init(no1: Int)
}
public class mainClass {
   var no1: Int      // local storage
   init(no1: Int) {
      self.no1 = no1 // initialization
   }
}
class subClass: mainClass, tcpprotocol {
   var no2: Int
   init(no1: Int, no2 : Int) {
      self.no2 = no2
      super.init(no1:no1)
   }
   
   // Requires only one parameter for convenient method
   required override convenience init(no1: Int) {
      self.init(no1:no1, no2:0)
   }
}

let res = mainClass(no1: 20)
let print = subClass(no1: 30, no2: 50)

print("res is: \(res.no1)")
print("res is: \(print.no1)")
print("res is: \(print.no2)")

Wenn wir das obige Programm auf einem Spielplatz ausführen, erhalten wir das folgende Ergebnis:

res is: 20
res is: 30
res is: 50

Zugriffskontrolle für Erweiterungen

In Swift 4 können Benutzer keinen expliziten Zugriffsstufenmodifikator für eine Erweiterung angeben, wenn der Benutzer diese Erweiterung verwendet, um die Protokollkonformität hinzuzufügen. Die Standardzugriffsebene für jede Implementierung der Protokollanforderungen innerhalb der Erweiterung wird mit einer eigenen Protokollzugriffsebene bereitgestellt.

Zugriffskontrolle für Generika

Mit Generika kann der Benutzer Mindestzugriffsebenen angeben, um auf die Typbeschränkungen für seine Typparameter zuzugreifen.

public struct TOS<T> {
   var items = [T]()
   mutating func push(item: T) {
      items.append(item)
   }
   mutating func pop() -> T {
      return items.removeLast()
   }
}

var tos = TOS<String>()
tos.push(item: "Swift 4")
print(tos.items)

tos.push(item: "Generics")
print(tos.items)

tos.push(item: "Type Parameters")
print(tos.items)

tos.push(item: "Naming Type Parameters")
print(tos.items)
let deletetos = tos.pop()

Wenn wir das obige Programm auf einem Spielplatz ausführen, erhalten wir das folgende Ergebnis:

[Swift 4]
[Swift 4, Generics]
[Swift 4, Generics, Type Parameters]
[Swift 4, Generics, Type Parameters, Naming Type Parameters]

Zugriffskontrolle für Typ-Aliase

Der Benutzer kann Typ-Aliase definieren, um unterschiedliche Zugriffssteuerungstypen zu behandeln. Gleiche Zugriffsebene oder unterschiedliche Zugriffsebenen können vom Benutzer definiert werden. Wenn der Typalias "privat" ist, können die zugehörigen Mitglieder als "privat, intern vom öffentlichen Typ" deklariert werden. Wenn der Typalias öffentlich ist, können die Mitglieder kein Alias ​​als "interner" oder "privater" Name sein

Alle von Ihnen definierten Typ-Aliase werden zum Zwecke der Zugriffskontrolle als unterschiedliche Typen behandelt. Ein Typalias kann eine Zugriffsebene haben, die kleiner oder gleich der Zugriffsebene des Typs ist, den er aliasiert. Ein Alias ​​vom privaten Typ kann beispielsweise einen privaten, internen oder öffentlichen Typ aliasen, ein öffentlicher Alias ​​kann jedoch keinen internen oder privaten Typ aliasen.

public protocol Container {
   associatedtype ItemType
   mutating func append(item: ItemType)
   var count: Int { get }
   subscript(i: Int) -> ItemType { get }
}
struct Stack<T>: Container {
   // original Stack<T> implementation
   var items = [T]()
   mutating func push(item: T) {
      items.append(item)
   }
   mutating func pop() -> T {
      return items.removeLast()
   }
   
   // conformance to the Container protocol
   mutating func append(item: T) {
      self.push(item: item)
   }
   var count: Int {
      return items.count
   }
   subscript(i: Int) -> T {
      return items[i]
   }
}
func allItemsMatch<
   C1: Container, C2: Container
   where C1.ItemType == C2.ItemType, C1.ItemType: Equatable>
   (someContainer: C1, anotherContainer: C2) -> Bool {
   
   // check that both containers contain the same number of items
   if someContainer.count != anotherContainer.count {
      return false
   }
   
   // check each pair of items to see if they are equivalent
   for i in 0..<someContainer.count {
      if someContainer[i] != anotherContainer[i] {
         return false
      }
   }
   // all items match, so return true
   return true
}
var tos = Stack<String>()
tos.push(item: "Swift 4")
print(tos.items)

tos.push(item: "Generics")
print(tos.items)

tos.push(item: "Where Clause")
print(tos.items)

var eos = ["Swift 4", "Generics", "Where Clause"]
print(eos)

Wenn wir das obige Programm auf einem Spielplatz ausführen, erhalten wir das folgende Ergebnis:

[Swift 4]
[Swift 4, Generics]
[Swift 4, Generics, Where Clause]
[Swift 4, Generics, Where Clause]

Schnelle Kodierung und Dekodierung

Swift 4 stellt eine neue vor Codable Protokoll, mit dem Sie benutzerdefinierte Datentypen serialisieren und de-serialisieren können, ohne einen speziellen Code zu schreiben - und ohne sich Gedanken über den Verlust Ihrer Werttypen machen zu müssen.

struct Language: Codable {
   var name: String
   var version: Int
}
let swift = Language(name: "Swift", version: 4)
let java = Language(name: "java", version: 8)
let R = Language(name: "R", version: 3

Beachten Sie, dass Langauage dem codierbaren Protokoll entspricht. Jetzt konvertieren wir es mit einer einfachen Zeile in eine Json-Datendarstellung.

let encoder = JSONEncoder()
if let encoded = try? encoder.encode(java) {
   //Perform some operations on this value.
}

Swift codiert automatisch alle Werte in Ihrem Datentyp.

Sie können die Daten mit der Decoder-Funktion wie decodieren

let decoder = JSONDecoder()
if let decoded = try? decoder.decode(Language.self, from: encoded) {
   //Perform some operations on this value.
}

Sowohl JSONEncoder als auch das Gegenstück zur Eigenschaftsliste PropertyListEncoder bieten zahlreiche Optionen zum Anpassen ihrer Funktionsweise.