( รวดเร็ว 2.x )
นอกจากนี้คุณยังสามารถขยายอาร์เรย์เพื่อให้สอดคล้องกับโปรโตคอลที่มีสีฟ้า rpints สำหรับวิธีการประเภททั่วไปเช่นโปรโตคอลที่มีกำหนดเองของคุณ utils การทำงานสำหรับทุกองค์ประกอบของอาร์เรย์ทั่วไปสอดคล้องกับข้อ จำกัด MyTypes
บางชนิดกล่าวว่าโปรโตคอล โบนัสที่ใช้วิธีนี้คือคุณสามารถเขียนฟังก์ชั่นที่ใช้อาร์กิวเมนต์อาเรย์ทั่วไปโดยมีข้อ จำกัด ว่าอาร์กิวเมนต์อาเรย์เหล่านี้จะต้องเป็นไปตามโปรโตคอลยูทิลิตี้ฟังก์ชั่นที่คุณกำหนดเองMyFunctionalUtils
ว่าสิ่งเหล่านี้อาร์กิวเมนต์อาร์เรย์ต้องเป็นไปตามที่กำหนดเองสาธารณูปโภคฟังก์ชั่นโพรโทคอของคุณบอกว่าโปรโตคอล
คุณสามารถรับพฤติกรรมนี้ได้โดยปริยายโดยพิมพ์ข้อ จำกัด ขององค์ประกอบอาร์เรย์ให้MyTypes
หรือ --- ตามที่ฉันจะแสดงในวิธีที่ฉันอธิบายด้านล่าง --- ค่อนข้างเรียบร้อยอย่างชัดเจนแจ้งให้ฟังก์ชั่นอาร์เรย์ทั่วไปของคุณโดยตรงแสดงอาร์เรย์อินพุตนั้น เป็นไปMyFunctionalUtils
ตาม
เราเริ่มต้นด้วยโปรโตคอลMyTypes
เพื่อใช้เป็นข้อ จำกัด ประเภท ขยายประเภทที่คุณต้องการให้พอดีกับ generics ของคุณด้วยโปรโตคอลนี้ (ตัวอย่างด้านล่างขยายประเภทพื้นฐานInt
และDouble
รวมถึงประเภทที่กำหนดเองMyCustomType
)
/* Used as type constraint for Generator.Element */
protocol MyTypes {
var intValue: Int { get }
init(_ value: Int)
func *(lhs: Self, rhs: Self) -> Self
func +=(inout lhs: Self, rhs: Self)
}
extension Int : MyTypes { var intValue: Int { return self } }
extension Double : MyTypes { var intValue: Int { return Int(self) } }
// ...
/* Custom type conforming to MyTypes type constraint */
struct MyCustomType : MyTypes {
var myInt : Int? = 0
var intValue: Int {
return myInt ?? 0
}
init(_ value: Int) {
myInt = value
}
}
func *(lhs: MyCustomType, rhs: MyCustomType) -> MyCustomType {
return MyCustomType(lhs.intValue * rhs.intValue)
}
func +=(inout lhs: MyCustomType, rhs: MyCustomType) {
lhs.myInt = (lhs.myInt ?? 0) + (rhs.myInt ?? 0)
}
พิธีสารMyFunctionalUtils
(ถือพิมพ์เขียวเพิ่มฟังก์ชันอรรถประโยชน์อาเรย์ทั่วไปเพิ่มเติมของเรา) และหลังจากนั้นการขยาย Array โดยMyFunctionalUtils
; การใช้งานของวิธีการพิมพ์สีฟ้า:
/* Protocol holding our function utilities, to be used as extension
o Array: blueprints for utility methods where Generator.Element
is constrained to MyTypes */
protocol MyFunctionalUtils {
func foo<T: MyTypes>(a: [T]) -> Int?
// ...
}
/* Extend array by protocol MyFunctionalUtils and implement blue-prints
therein for conformance */
extension Array : MyFunctionalUtils {
func foo<T: MyTypes>(a: [T]) -> Int? {
/* [T] is Self? proceed, otherwise return nil */
if let b = self.first {
if b is T && self.count == a.count {
var myMultSum: T = T(0)
for (i, sElem) in self.enumerate() {
myMultSum += (sElem as! T) * a[i]
}
return myMultSum.intValue
}
}
return nil
}
}
ในที่สุดการทดสอบและสองตัวอย่างแสดงฟังก์ชั่นการใช้อาร์เรย์ทั่วไปด้วยกรณีดังต่อไปนี้ตามลำดับ
แสดงการยืนยันโดยนัยว่าพารามิเตอร์อาร์เรย์สอดคล้องกับโปรโตคอล 'MyFunctionalUtils' ผ่านประเภทที่ จำกัด องค์ประกอบของอาร์เรย์ให้เป็น 'MyTypes' (ฟังก์ชันbar1
)
แสดงอย่างชัดเจนว่าพารามิเตอร์อาร์เรย์สอดคล้องกับโปรโตคอล 'MyFunctionalUtils' (ฟังก์ชันbar2
)
การทดสอบและตัวอย่างดังต่อไปนี้:
/* Tests & examples */
let arr1d : [Double] = [1.0, 2.0, 3.0]
let arr2d : [Double] = [-3.0, -2.0, 1.0]
let arr1my : [MyCustomType] = [MyCustomType(1), MyCustomType(2), MyCustomType(3)]
let arr2my : [MyCustomType] = [MyCustomType(-3), MyCustomType(-2), MyCustomType(1)]
/* constrain array elements to MyTypes, hence _implicitly_ constraining
array parameters to protocol MyFunctionalUtils. However, this
conformance is not apparent just by looking at the function signature... */
func bar1<U: MyTypes> (arr1: [U], _ arr2: [U]) -> Int? {
return arr1.foo(arr2)
}
let myInt1d = bar1(arr1d, arr2d) // -4, OK
let myInt1my = bar1(arr1my, arr2my) // -4, OK
/* constrain the array itself to protocol MyFunctionalUtils; here, we
see directly in the function signature that conformance to
MyFunctionalUtils is given for valid array parameters */
func bar2<T: MyTypes, U: protocol<MyFunctionalUtils, _ArrayType> where U.Generator.Element == T> (arr1: U, _ arr2: U) -> Int? {
// OK, type U behaves as array type with elements T (=MyTypes)
var a = arr1
var b = arr2
a.append(T(2)) // add 2*7 to multsum
b.append(T(7))
return a.foo(Array(b))
/* Ok! */
}
let myInt2d = bar2(arr1d, arr2d) // 10, OK
let myInt2my = bar2(arr1my, arr2my) // 10, OK
extension T[]
บิตเดียวกันเมื่อคลิก Command บนประเภท Array ใน XCode แต่ไม่เห็นวิธีที่จะนำไปใช้โดยไม่ได้รับข้อผิดพลาด