Skip to main content

Tất cả Keywords trong Swift

Nay rảnh rỗi nên mình xin được mạn phép clone 1 bản từ tác giả Jordan Morgan, (bài viết gốc ở đây) nói về tất cả keywords trong Swift, hy vọng nó sẽ giúp ích cho các bạn trong quá trình phỏng vấn cũng như trong công việc. Tuy nhiên 1 số từ mình sẽ để bản tiếng Anh, vì dịch ra nó hơi “chuối", và 1 số định nghĩa mình cũng ko dịch sát nghĩa, mà dịch theo cách hiểu của riêng mình. Có thiếu sót gì thì mình mong được các bạn bổ sung thêm :D.

Declaration Keywords

associatedtype: Cho phép tạo 1 tên bất kỳ cho 1 loại biến trong khai báo của 1 protocol. Biến này được qui định là loại nào khi protocol đó được adopt.
protocol Entertainment
{
    associatedtype MediaType
}class Foo : Entertainment
{
    typealias MediaType = String // Bất cứ loại nào đều được
}
class : là thành phần không thể thiếu trong mọi ứng dụng, chúng giúp chúng ta tổ chức và quản lý code thành những khối, nó có 1 số điểm khác struct như sau:
  • Tính kế thừa.
  • Cho phép ép kiểu hoặc check kiểu lúc chương trình chạy runtime.
  • Tính huỷ: cho phép instance của một class phải phóng bất cứ tài nguyên nào mà nó đã gán, hàm huỷ được gọi ngay trước khi instance của một class được giải phóng (trả lại bộ nhớ đã được cấp phát tới ram).
  • Class là kiểu tham chiếu, và chính vì nó là kiểu tham chiếu nên nó có thêm toán tử đồng nhất thức (===), có nghĩa rằng hai biến hoặc hằng của kiểu class tham chiếu tới chính xác cùng một instance của class.
class Person
{
    var name:String
    var age:Int
    var gender:String
}
deinit: được gọi ngay khi instance của 1 class được giải phóng vùng nhớ trong vùng nhớ heap.
class Person
{
    var name:String
    var age:Int
    var gender:Stringdeinit
   {
        // Giải phóng vùng nhớ trong heap.
   }
}
enum : Giúp định nghĩa 1 tập hợp có số phần tử cố định và được liệt kê sẵn. Enum là kiểu tham trị.
enum Gender
{
    case male
    case female
}
extension : Cho phép mở rộng thêm hàm từ 1 class hoặc struct hoặc enum hoặc protocol.
class Person
{
    var name:String = ""
    var age:Int = 0
    var gender:String = ""
}extension Person
{
    func printInfo()
    {
        print("My name is \(name), I'm \(age) years old and I'm a \(gender).")
    }
}
fileprivate : là 1 access control giới hạn trong 1 file, thường dùng cho extension.
class Person
{
    fileprivate var jobTitle:String = ""
}extension Person
{// nếu mà khai báo bằng private là compiler báo lỗi
    func printJobTitle()
    {
        print("My job is \(jobTitle)")
    }
}
func : cho phép khai báo 1 hàm.
func addNumbers(num1:Int, num2:Int) -> Int
{
    return num1+num2
}
import : Cho phép nhúng 1 framework hoặc 1 module khác vào trong module hiện có.
import UIKit//All of UIKit's code is now available
class Foo {}
init : là quá trình chuẩn bị một instance của class, struct, hoặc enum để sử dụng.
class Person 
{
    init()
    {
        //Set default values, prep for use, etc.
    }
}
inout : Tham số chỉ tồn tại trong phạm vi của hàm, nên sử dụng inout sẽ giúp thay đổi giá trị của một tham số của hàm và sự thay đổi đó vẫn còn khi hàm kết thúc. Khi gọi hàm phải đặt dấu (&) ngay trước tên biến để cho trình biên dịch biết biến đó có thể thay dổi trong hàm.
func swapInts(inout a: Int, inout _ b:Int) {  
    let temp = a       
    a = b        
    b = temp    
}
internal : Một access control cho phép sử dụng trong 1 module, không ghi gì thì mặc định là internal đó.
class Person
{
    internal var jobTitle:String = ""
}let aPerson = Person()
aPerson.jobTitle = "This can set anywhere in the application"
let : định nghĩa 1 biến bằng let là sau này không có thay đổi giá trị nó được đó.
let constantString = "This cannot be mutated going forward"
open : Một access control cho phép object được sử dụng ở ngoài module khác . Thường viết bằng open khi tạo ra các framework.
open var foo: String? 
operator : các toán tử cho phép chúng ta kiểm tra, thay đổi hoặc kết hợp các giá trị lại với nhau.
let foo = 5
let anotherFoo = -foo 
 
let box = 5 + 3let didPassCheckAll = didPassCheckOne && didPassCheckTwo// Toán tử 3 ngôi giúp code ngắn hơn
let isLegalDrinkingAgeInUS = age >= 21 ? true : false
private : Một access control cho phép chỉ xài trong 1 scope.
class Person
{
    private var jobTitle = ""
}extension Person
{// Chỗ này sẽ compile lỗi
    func printJobTitle()
    {
        print("My job is \(jobTitle)")
    }
}
protocol : định nghĩa các giao thức, class, struct hay enum mà adopt procotol này thì phải theo chuẩn của nó.
protocol Blog
{
    var wordCount:Int { get set }
    func printReaderStats()
}class TTIDGPost : Blog
{
    var wordCount: Int
    
    init(wordCount: Int)
    {
        self.wordCount = wordCount
    }func printReaderStats()
    {
        //Print out some stats on the post
    }
}
public : Một access control cho phép object sử dụng ở tất cả các file trong cùng 1 module.
public var foo: String? 
static : biến nào mà được định nghĩa bằng từ khoá static thì nó tồn tại suốt trong 1 chương trình và không bị huỷ, ngoài ra sử dụng static thì ta có thể thao tác biến hoặc hàm bằng class hoặc struct hoặc enum đó luôn.
class Person
{
    var jobTitle: String?static func assignRandomName(_ aPerson: Person)
    {
        aPerson.jobTitle = "Some random job"
    }
}let somePerson = Person()
Person.assignRandomName(somePerson)
//somePerson.jobTitle is now "Some random job"
struct : cũng là thành phần không thể thiếu trong mọi ứng dụng như class, chúng giúp chúng ta tổ chức và quản lý code thành những khối, nó có 1 số điểm khác class như sau:
  • Không có tính kế thừa.
  • Là kiểu tham trị.
  • Không có tính huỷ.
struct Person
{
    var name: String
    var age: Int
    var gender: String
}
subscript : cho phép truy xuất đến 1 thành viên nằm trong một collection, list hoặc sequence.
var postMetrics = ["Likes":422, "ReadPercentage":0.58, "Views":3409]
let postLikes = postMetrics["Likes"]
typealias : Cho phép đặt 1 tên khác của 1 kiểu sẵn có.
typealias JSONDictionary = [String: AnyObject]func parseJSON(_ deserializedData: JSONDictionary){}
var : định nghĩa 1 biến có thể thay đổi được giá trị.
var mutableString = ""
mutableString = "Mutated"

Keywords in Statements

break : kết thúc chương trình trong loop, if hoặc switch
for idx in 0...3
{
    if idx % 2 == 0
    {
        break
    }
}
case : một trường hợp trong câu lệnh switch.
let box = 1switch box
{
case 0:
    print("Box equals 0")
case 1:
    print("Box equals 1")
default:
    print("Box doesn't equal 0 or 1")
}
continue : trong vòng lặp nếu gặp continue thì nó sẽ bỏ qua trường hợp đó (ở đây trong câu lệnh if) rồi tiếp tục duyệt tiếp.
for idx in 0...3
{
    if idx % 2 == 0
    {
        continue // Thoát khúc này thôi
    }
    
    print("This code never fires on even numbers") // Vẫn in ra nhé
}
default : xét case từ trên xuống mà ko thấy cái nào dính thì bay vào default.
let box = 1switch box
{
case 0:
    print("Box equals 0")
case 1:
    print("Box equals 1") // In mỗi thằg này ra thôi
default:
    print("Covers any scenario that doesn't get addressed above.")
}
defer : sử dụng khi muốn đảm bảo thực thi 1 đoạn code nào đó ngay khi hàm kết thúc.
func test()
{
    defer
    {
        print("2") // thằng này print sau
    }
    print("1") // thằng này print trước}
do : thực thi đoạn chương trình nào đó có khả năng xảy ra lỗi .
do
{
    try expression
    //statements
}
catch someError ex
{
    //Handle error
}
else : Nếu không phải thằng A thì là thằng B thôi.
if val > 1
{
    print("val is greater than 1")
}
else
{
    print("val is not greater than 1")
}
fallthrough : xét trên xuống, nếu khớp 1 case nào đó vẫn tiếp tục xét tiếp 1 case liền kề
let box = 0switch box
{
case 0:
    print(0) // In ra 0
    fallthrough
case 1:
    print(1) // In ra 1
case 2:
    print(2) // Không có được in ra 
default:
    print("default")
}
for : giúp lặp các phần tử trong 1 sequence hoặc array hoặc các kí tự trong 1 String.
for _ in 0..<3 { print ("This prints 3 times") }
guard : đảm bảo điều kiện nào đó đúng, vừa có thể unwrap biến optional. Thật ra xài if let cũng được, nhưng xài guard đỡ phải lồng vào trong 1 scope nhìn rối mắt.
private func printRecordFromLastName(userLastName: String?) 
{
    guard let name = userLastName, userLastName != "Null" else
    {
        //Sorry Bill Null, find a new job
        return
    }//Party on
    print(dataStore.findByLastName(name))
}
if : đảm bảo 1 hoặc nhiều điều kiện khi thực hiện 1 hoặc nhiều câu lệnh nào đó.
if 1 > 2
{
    print("This will never execute")
}
in : đi chung với for ở trên
for _ in 0..<3 { print ("This prints 3 times") }
repeat : Y chang do while trong C/C++. Thực thi đoạn chương trình ít nhất 1 lần trước khi lặp qua điều kiện
repeat
{
    print("Always executes at least once before the condition is considered")
}
while 1 > 2
return : thoát khỏi hàm luôn, và có thể trả về giá trị nào đó tuỳ chúng ta khai báo.
func doNothing()
{
    return //Immediately leaves the contextlet anInt = 0
    print("This never prints \(anInt)")
}
func returnName() -> String?
{
    return self.userName //Returns the value of userName
}
switch : xét và so sánh giá trị xem nó match với case nào, hoặc thích thì dùng if + else if + else.
let box = 1switch box
{
case 0:
    print("Box equals 0")
    fallthrough
case 1:
    print("Box equals 0 or 1")
default:
    print("Box doesn't equal 0 or 1")
}
where : dùng để ghép với for kiểm tra xem có đúng điều kiện không, hoặc dùng cho 1 loại generic type phải conform đúng protocol nào đó.
protocol Nameable
{
    var name: String {get set}
}func createdFormattedName<T: Nameable>(_ namedEntity: T) -> String where T: Equatable
{
    // Chỉ có thực thể nào mà vừa conform Nameable và Equatable thì mới dùng được hàm này
    return "This things name is " + namedEntity.name
}
for i in 0…3 where i % 2 == 0
{
    print(i) //Prints 0 and 2
}
while : loop điều kiện nào đó cho tới khi nó sai thì thôi.
while foo != bar
{
    print("Keeps going until the foo == bar")
}

Expressions and Types Keywords

Any : để hiện diện cho bất kỳ loại thuộc tính nào của đối tượng, bao gồm cả hàm.
var anything = [Any]()anything.append("Any Swift type can be added")
anything.append(0)
anything.append({(foo: String) -> String in "Passed in \(foo)"})
as : dùng để ép kiểu.
var anything = [Any]()anything.append("Any Swift type can be added")
anything.append(0)
anything.append({(foo: String) -> String in "Passed in \(foo)" })let intInstance = anything[1] as? Int
hoặc
var anything = [Any]()anything.append("Any Swift type can be added")
anything.append(0)
anything.append({(foo: String) -> String in "Passed in \(foo)" })for thing in anything
{
    switch thing
    {
    case 0 as Int:
        print("It's zero and an Int type")
    case let someInt as Int:
        print("It's an Int that's not zero but \(someInt)")
    default:
        print("Who knows what it is")
    }
}
catch : nếu trong mệnh đề clause xảy ra lỗi, thì catch sẽ xử lý lỗi đó, ta có thể catch nhiều trường hợp khác nhau như trong ví dụ dưới.
do
{
    try haveAWeekend(4)
}
catch WeekendError.Overtime(let hoursWorked)
{
    print(“You worked \(hoursWorked) more than you should have”)
}
catch WeekendError.WorkAllWeekend
{
    print(“You worked 48 hours :-0“)
}
catch
{
    print(“Gulping the weekend exception”)
}
false : biến kiểu Bool, ko phải là true.
let alwaysFalse = false
let alwaysTrue = trueif alwaysFalse { print("Won't print, alwaysFalse is false 😉")}
is : kiểm tra xem có phải loại subclass nào đó hay không.
class Person {}
class Programmer : Person {}
class Nurse : Person {}let people = [Programmer(), Nurse()]for aPerson in people
{
    if aPerson is Programmer
    {
        print("This person is a dev")
    }
    else if aPerson is Nurse
    {
        print("This person is a nurse")
    }
}
nil : Represents a stateless value for any type in Swift. *Different from Objective-C’s nil, which is a pointer to a nonexistent object. (chỗ này mình không dịch)
class Person{}
struct Place{}//Literally any Swift type or instance can be nil
var statelessPerson: Person? = nil
var statelessPlace: Place? = nil
var statelessInt: Int? = nil
var statelessString: String? = nil
rethrows : 1 hàm ném ra 1 error nếu 1 tham số trong hàm ném ra 1 error.
func networkCall(onComplete:() throws -> Void) rethrows
{
    do
    {
        try onComplete()
    }
    catch
    {
        throw SomeError.error
    }
}
super : là một biến tham chiếu mà được sử dụng để tham chiếu đến đối tượng lớp cha gần nhất.
class Person
{
    func printName()
    {
        print("Printing a name. ")
    }
}class Programmer : Person
{
    override func printName()
    {
        super.printName() // Super là Person đó 
        print("Hello World!")
    }
}let aDev = Programmer()
aDev.printName() //"Printing a name. Hello World!"
self : là thực thể của loại mà mình đang xài đó ( có thể là class, struct hoặc enum)
class Person
{
    func printSelf()
    {
        print("This is me: \(self)")
    }
}let aPerson = Person()
aPerson.printSelf() //"This is me: Person"
Self : Xài trong protocol, Self chính là class, struct hoặc enum nào đó conform protocol kia.
protocol Printable
{
    func printTypeTwice(otherMe:Self)
}struct Foo : Printable
{
    func printTypeTwice(otherMe: Foo)
    {
        print("I am me plus \(otherMe)")
    }
}let aFoo = Foo()
let anotherFoo = Foo()aFoo.printTypeTwice(otherMe: anotherFoo) //I am me plus Foo()
throw : ném ra 1 error từ 1 hàm.
enum WeekendError: Error
{
    case Overtime
    case WorkAllWeekend
}func workOvertime () throws
{
    throw WeekendError.Overtime
}
throws : Chỉ ra rằng 1 hàm có thể ném ra 1 lỗi nào đó.
enum WeekendError: Error
{
    case Overtime
    case WorkAllWeekend
}func workOvertime () throws
{
    throw WeekendError.Overtime
}//"throws" indicates in the function's signature that I need use try, try? or try!
try workOvertime()
true giá trị true trong kiểu Bool.
let alwaysFalse = false
let alwaysTrue = trueif alwaysTrue { print("Always prints")}
try : Khi gọi hàm thì ta phải thêm try hoặc try? hoặc try! trước hàm đó.
let aResult = try dangerousFunction() //có lỗi là phải catch lại 
let aResult = try! dangerousFunction() //có lỗi là crash luôn 
if let aResult = try? dangerousFunction() //Unwrap 1 giá trị optional từ hàm này trả về

Keywords Using Patterns

_ : Nói chung là bỏ qua giá trị khi nó match dc đó.
for _ in 0..<3
{
    print("Just loop 3 times, index has no meaning")
}
hoặc
let _ = Singleton() //Ignore value or unused variable

Keywords Using #

#available : Kiểm tra điều kiện lúc runtime, ví dụ như kiểm tra version ios.
if #available(iOS 10, *)
{
    print("iOS 10 APIs are available")
}
#colorLiteral : ra bộ chọn màu.
let aColor = #colorLiteral //Brings up color picker
#column : Biết mình đang ở cột nào.
class Person
{
    func printInfo()
    {
        print("Some person info - on column \(#column)") 
    }
}let aPerson = Person()
aPerson.printInfo() //Some person info - on column 53
#if #elseif #else #endif : Kiểm tra điều kiện như là đang xài hệ điều hành nào, version bao nhiêu…
#if os(iOS)
    print("Compiled for an iOS device")
#elseif os(macOS)
    print("Compiled on a mac computer")
#endif
#file : Trả về tên file chứa nó.
class Person
{
    func printInfo()
    {
        print("Some person info - inside file \(#file)") 
    }
}let aPerson = Person()
aPerson.printInfo() //Some person info - inside file /*file path to the Playground file I wrote it in*/
#function :trả về tên của function chứa nó, trong getter, setter thì sẽ trả về tên thuộc tính, trong 1 số hàm như init hoặc subscript thì sẽ trả về tên của từ khoá đó, nếu đặt ở ngay đầu file thì sẽ trả về tên của module hiện tại.
class Person
{
    func printInfo()
    {
        print("Some person info - inside function \(#function)") 
    }
}let aPerson = Person()
aPerson.printInfo() //Some person info - inside function printInfo()
#imageLiteral : show nguyên kho hình lên cho mình chọn.
let anImage = #imageLiteral //Brings up a picker to select an image inside the playground file
#line : Trả về số dòng hiện tại.
class Person
{
    func printInfo()
    {
        print("Some person info - on line number \(#line)") 
    }
}let aPerson = Person()
aPerson.printInfo() //Some person info - on line number 5
#selector : nói chung là đảm bảo hàm nào đó tồn tại và gắn vào 1 action nào đó.
//Static checking occurs to make sure doAnObjCMethod exists
control.sendAction(#selector(doAnObjCMethod), to: target, forEvent: event)
#sourceLocation : gọi dòng này xong là reset số dòng với tên file, không còn đúng như trước lúc gọi.
#sourceLocation(file:"foo.swift", line:6)//Reports new values
print(#file)
print(#line)//This resets the source code location back to the default values numbering and filename
#sourceLocation()print(#file)
print(#line)

Keywords For Specific Context(s)

convenience : đúng theo nghĩa của nó là tiện lợi, chạy dựa trên designated initializer, mình có viết 1 bài về nó ở đây.
class Person
{
    var name:Stringinit(_ name:String)
    {
        self.name = name
    }convenience init()
    {
        self.init("No Name")
    }
}let me = Person()
print(me.name)//Prints "No Name"
dynamic : Nói chung là chạy bằng Objective-C runtime và Swift runtime sẽ cho ra kết quả khác nhau, thêm dynamic vào thì coi như nó chạy bằng Objective-C runtime.
class Person
{
    //Implicitly has the "objc" attribute now too
    //This is helpful for interop with libs or
    //Frameworks that rely on or are built
    //Around Obj-C "magic" (i.e. some KVO/KVC/Swizzling)
    dynamic var name:String?
}
didSet : sau khi gán xong thì nó sẽ thực hiện câu lệnh bên trong didSet
var data = [1,2,3]
{
    didSet
    {
        tableView.reloadData()
    }
}
final : Ngăn không cho kế thừa nữa.
final class Person {}
class Programmer : Person {} //Compile time error
get : dùng để trả về giá trị 1 giá trị nào đó.
class Person
{
    var name:String
    {
        get { return self.name }
        set { self.name = newValue}
    }    var indirectSetName:String
    {
        get
        {
            if let aFullTitle = self.fullTitle
            {
                return aFullTitle
            }
            return ""
        }        set (newTitle)
        {
            //If newTitle was absent, newValue could be used
            self.fullTitle = "\(self.name) :\(newTitle)"
        }     }
}
indirect : Chỉ ra enum có 1 case khác liên quan tới giá trị của 1 hoặc nhiều case trong enum đó
indirect enum Entertainment
{
    case eventType(String)
    case oneEvent(Entertainment)
    case twoEvents(Entertainment, Entertainment)
}let dinner = Entertainment.eventType("Dinner")
let movie = Entertainment.eventType("Movie")let dateNight = Entertainment.twoEvents(dinner, movie)
lazy : biến nào có lazy thì nó chỉ được tính toán khi nó được gọi ra, giúp tiết kiệm bộ nhớ hơn.
class Person
{
    lazy var personalityTraits = {
        //Some crazy expensive database  hit
        return ["Nice", "Funny"]
    }()
}let aPerson = Person()
aPerson.personalityTraits //Database hit only happens now once it's accessed for the first time
mutating : Cho phép thay đổi giá trị của thuộc tính của struct hoặc enum.
struct Person
{
    var job = ""mutating func assignJob(newJob:String)
    {
        self = Person(job: newJob)
    }
}var aPerson = Person()
aPerson.job //""aPerson.assignJob(newJob: "iOS Engineer at Buffer")
aPerson.job //iOS Engineer at Buffer
nonmutating : Chỉ ra rằng hàm setter không thay đổi được instance chứa nó.
enum Paygrade
{
    case Junior, Middle, Senior, Master    var experiencePay:String?
    {
        get
        {
            database.payForGrade(String(describing:self))
        }        nonmutating set
        {
            if let newPay = newValue
            {
                database.editPayForGrade(String(describing:self), newSalary:newPay)
            }
        }
    }
}let currentPay = Paygrade.Middle//Updates Middle range pay to 45k, but doesn't mutate experiencePay
currentPay.experiencePay = "$45,000"
optional : Sử dụng để khai báo hàm optional trong protocol.
@objc protocol Foo
{
    func requiredFunction()
    @objc optional func optionalFunction()
}class Person : Foo
{
    func requiredFunction()
    {
        print("Conformance is now valid")
    }
}
override : Chỉ ra rằng lớp con đang ghi đè lên biến hoặc hàm của lớp cha
class Person
{
    func printInfo()
    {
        print("I'm just a person!")
    }
}class Programmer : Person
{
    override func printInfo()
    {
        print("I'm a person who is a dev!")
    }
}let aPerson = Person()
let aDev = Programmer()aPerson.printInfo() //I'm just a person!
aDev.printInfo() //I'm a person who is a dev!
required : đảm bảo rằng mọi lớp con phải thực thi hàm khởi tạo cho trước.
class Person
{
    var name:String?required init(_ name:String)
    {
        self.name = name
    }
}class Programmer : Person
{
    //Excluding this init(name:String) would be a compiler error
    required init(_ name: String)
    {
        super.init(name)
    }
}
set : trong hàm set có thể làm gì thì làm, set giá trị cho chính property chứa nó hoặc property khác cũng được.
class Person
{
    var name:String
    {
        get { return self.name }
        set { self.name = newValue}
    }    var indirectSetName:String
    {
        get
        {
            if let aFullTitle = self.fullTitle
            {
                return aFullTitle
            }
            return ""
        }        set (newTitle)
        {
            //If newTitle was absent, newValue could be used
            self.fullTitle = "\(self.name) :\(newTitle)"
        }
    }
}
Type : có thể là class type, struct type, enum type hoặc protocol type.
class Person {}
class Programmer : Person {}let aDev:Programmer.Type = Programmer.self
unowned : Cho phép 1 instance tham chiếu tới 1 instance khác mà không làm tăng reference count, đảm bảo rằng instance đã tham chiếu tới có lifetime bằng hoặc lâu hơn chính nó.
class Person
{
    var occupation:Job?
}//Khi nào người mất thì job mới mất
class Job
{
    unowned let employee:Person    init(with employee:Person)
    {
        self.employee = employee
    }
}
weak : Cho phép 1 instance tham chiếu tới 1 instance khác mà không làm tăng reference count, nhưng instance đã tham chiếu kia có lifetime ngắn hơn (có thể bị huỷ trước)
class Person
{
    var residence:House?
}class House
{
    weak var occupant:Person?
}var me:Person? = Person()
var myHome:House? = House()me!.residence = myHome
myHome!.occupant = meme = nil
myHome!.occupant //Is now nil
willSet : bên trên có cái didSet là khi nào gán xong mới chạy mấy câu lệnh bên trong, còn willSet thì được gọi ngay trước khi biến được gán đâu đó. Nó có newValue là giá trị sẽ được gán đó.
class Person
{
    var name:String?
    {
        willSet(newValue) {print("I've got a new name, it's \(newValue)!")}
    }
}let aPerson = Person()
aPerson.name = "Jordan" //Prints out "I've got a new name, it's Jordan!" right before name is assigned to

Vậy là hết rồi

Nếu các bạn thấy hữu ích thì follow và clap cho mình nhé ^^. Thanks.

Comments

Popular posts from this blog

MVVM và VIPER: Con đường trở thành Senior

Trong bài viết trước chúng ta đã tìm hiểu về MVC và MVP để ứng dụng cho một iOS App đơn giản. Bài này chúng ta sẽ tiếp tục ứng dụng 2 mô hình MVVM và VIPER . Nhắc lại là ứng dụng của chúng ta cụ thể khi chạy sẽ như sau: Source code đầy đủ cho tất cả mô hình MVC, MVP, MVVM và VIPER các bạn có thể download tại đây . MVVM MVVM có thể nói là mô hình kiến trúc được rất nhiều các cư dân trong cộng đồng ưa chuộng. Điểm tinh hoa của kiến trúc này là ở ViewModel , mặc dù rất giống với Presenter trong MVP tuy nhiên có 2 điều làm nên tên tuổi của kiến trúc này đó là: ViewModel không hề biết gì về View , một ViewModel có thể được sử dụng cho nhiều View (one-to-many). ViewModel sử dụng Observer design pattern để liên lạc với View (thường được gọi là binding data , có thể là 1 chiều hoặc 2 chiều tùy nhu cầu ứng dụng). Chính đặc điểm này MVVM thường được phối hợp với các thư viện hỗ trợ Reactive Programming hay Event/Data Stream , đây là triết lý lập trình hiện đại và hiệu...

Alamofire vs URLSession

Alamofire vs URLSession: a comparison for networking in Swift Alamofire and URLSession both help you to make network requests in Swift. The URLSession API is part of the foundation framework, whereas Alamofire needs to be added as an external dependency. Many  developers  doubt  whether it’s needed to include an extra dependency on something basic like networking in Swift. In the end, it’s perfectly doable to implement a networking layer with the great URLSession API’s which are available nowadays. This blog post is here to compare both frameworks and to find out when to add Alamofire as an external dependency. Build better iOS apps faster Looking for a great mobile CI/CD solution that has tons of iOS-specific tools, smooth code signing, and even real device testing? Learn more about Bitrise’s iOS specific solutions! This shows the real power of Alamofire as the framework makes a lot of things easier. What is Alamofire? Where URLSession...

Frame vs Bounds in iOS

This article is a repost of an answer I wrote on Stack Overflow . Short description frame = a view’s location and size using the parent view’s coordinate system ( important for placing the view in the parent) bounds = a view’s location and size using its own coordinate system (important for placing the view’s content or subviews within itself) Details To help me remember frame , I think of a picture frame on a wall . The picture frame is like the border of a view. I can hang the picture anywhere I want on the wall. In the same way, I can put a view anywhere I want inside a parent view (also called a superview). The parent view is like the wall. The origin of the coordinate system in iOS is the top left. We can put our view at the origin of the superview by setting the view frame’s x-y coordinates to (0, 0), which is like hanging our picture in the very top left corner of the wall. To move it right, increase x, to move it down increase y. To help me remember bound...