UITableView

UITableViewController

  • iOS 开发者通常都应该遵循的 MVC 设计模式,每个类必定是以下三种类型中的 一种 。
    模型:负责管理数据,与用户界面无关 。
    视图:负责展示界面,与模型对象无关 。
    控制器 : 负责确保用户界面和模型对象一致,并且控制应用的流程。
  • UITableView 是 一个视图对象,不负责处理应用逻辑和数据 。 当在应用中使用 UITableView 对象时,必须考虑如何搭配其他对象与 UITableView 对象一起工作 。
  • 通常情况下,UITableView 需要控制器来展示界面 。
  • UITableView 需要数据源, UITableView 对象会向数据源查询要显示的行数、显示表格 行所需的数据和其他所需的数据,没有数据源的 UITableView 对象只是空壳。凡是遵守UITableViewDataSource 协议的对象,都可以成为 UITableView 的数据源 。
  • 通常情况下,要为 UITableView 对象设置委托对象,以便能在该对象发生特定事件时做出相应处理 。 凡是遵守 UITableViewDelegate 协议的对象 , 都可以成为 UITableView的委托对象 。

1. 设置初始视图控制器为 UITableViewController

打开 Main. storyboard ,选中画布上现有的视图控制器 , 然后按下键盘上的删除键删除它,再从对象库中拖一个UITableViewController 到画布上 。接下来选中 UITableViewCont roller ,打开标识检视面板 ( identity inspector ) ,把 class 改为 ltemsViewController 。 最后打开 Items ViewController 的属性检视面板 (attributes inspector ) ,选中 Is Initial View Controller ( 是初始视图控制器)复选框。

2. 创建Item类

class Item: NSObject{  //定义 Item 类并给它设置四个属性 。Item 继承自 NSObject,如果自定义的类要与 runtime (运行时)系统交互,就需要继承自 NSObject 。
    var name: String
    var valueInDollars: Int
    var serialNumber: String?  //serialNumber 是一个可选字符串,这是因为有的 Item 对象可能没有序列号。
    let dataCreated: Date
    }
  • 结构体不支持继承,因此结构体的构造方法很直接,相反类支持继承,因此类的构造方法有一些规则。类的构造方法有两种 : 指定构造方法 ( designated initializers) 和便利构造方法 ( convenienceinitializers) 。
为 Item 类实现一个指定构造方法,并且为所有属性设置初始值 。
    init(name: String, serialNumber: String?, valueInDollars: Int) {
        self.name = name  //由于参数名和属性名是一样的,因此需要使用 self 来区分参数和属性。
        self.valueInDollars = valueInDollars
        self.serialNumber = serialNumber
        self.dataCreated = Date()
        super.init()
    }
  • 每个类都至少有一个指定构造方法,但是便利构造方法是可选的 。 可以把便利构造方法看作辅助方法 ( helpers ) ,便利构造方法总是调用当前类的其他构造方法,在构造方法名前加上 convenience 关键字,可以标识便利构造方法。
为 Item 添加 一个便利构造方法,用来创建随机的 Item 。

便利构造方法起辅助作用,可用于自定义

    convenience init(random: Bool = false) {
    if random {
        let adjectives = ["Fluffy", "Rusty", "Shiny"]
        let nouns = ["Bear", "Spork", "Mac"]
        let randomAdjective = adjectives.randomElement()!
        let randomNoun = nouns.randomElement()!

        let randomName = "\(randomAdjective) \(randomNoun)"
        let randomValue = Int.random(in: 0..<100)
        let randomSerialNumber = UUID().uuidString.components(separatedBy: "-").first!

        self.init(name: randomName, serialNumber: randomSerialNumber, valueInDollars: randomValue)
        
    }else {
        self.init(name: "", serialNumber: nil, valueInDollars: 0) }
        
    }

UITable View 数据源

  • 创建 一个叫 ItemStore 的 Swift 文件。在 ItemStore.swift 中 定义 ItemStore 类并且 添加一个用千存储 Item 的属性。
class ItemStore {  // ItemStore 并不需要使用 NSObject 的任何属性或方法。
    var allItems = [Item]()  //创建一个 allItems 变量,其类型为空数组。
}
  • 在 ItemStore.swift 中,实现 createitem() 方法来创建 Item 对象 。
@discardableResult func createItem() -> Item {  //@discardableResult 调用方法但是忽略返回值
        let newItem = Item(random: true)
        allItems.append(newItem)
        return newItem
    }

让控制器访问 ItemStore

  • 在 ItemsViewCont roller.swift 中添加一个 ItemStore 属性
class ItemsViewController: UITableViewController{  //定义一个叫 ItemsViewController 的 UITableViewController 子类
    var itemStore: ItemStore!  //添加一个 ItemStore 属性,以让控制器访问 ItemStore,此时还不能访问ItemStore,需要初始化ItemStore 属性才能访问。
}
  • 初始化 ItemsViewController 的 ItemStore 这个属性
func scene(_ scene: UIScene, willConnectTo session: UISceneSession, options connectionOptions: UIScene.ConnectionOptions) {
        // Use this method to optionally configure and attach the UIWindow `window` to the provided UIWindowScene `scene`.
        // If using a storyboard, the `window` property will automatically be initialized and attached to the scene.
        // This delegate does not imply the connecting scene or session are new (see `application:configurationForConnectingSceneSession` instead).
        guard let _ = (scene as? UIWindowScene) else { return }
        
        let itemStore = ItemStore()  //方法实例化
        let itemsController = window!.rootViewController as! ItemsViewController
        itemsController.itemStore = itemStore  //将其 ItemStore 属性设置为 ItemStore 的新实例。
    }
  • 最后 ,在 ItemStore.swift 中实现指定构造方法,并且添加 5 个随机 Item 。
init() {  //实现指定构造方法,并且添加 35 个随机 Item 。
        for _ in 0..<35 {
            createItem()
            
            //如果 crateltem( ) 没有声明@也scardableResult ,那么调用方法需要这样写:
            //let _ = createltem()  //调用方法 ,但是忽略返回值
        }

DIP原则

目的:通过倒置依赖关系来 解耦对象,可以让代码更容易维护 。

  1. 上层对象不应该依赖底层对象,它们都应该依赖抽象 。
  2. 抽象不应该依赖细节,细节应该依赖抽象。
  • 在 Homepwner 中,DIP 原则中的抽象是“存储“,“存储”是一个底层对象,它只需要知道 Item 的存储和获取即可 。 相反 ItemsViewController 是一个上层对象,它只需要知道有 一个辅助对象 ( ItemStore ) 会提供数据即可,而辅助对象会维护一组 Item 实例,并且可以创建、修改和存储 Item 。这个解耦让 ItemsViewController 不依赖于 ItemStore , 只要 遵守 “存储"的抽象原则, ItemStore 就可以被替换为以其他方式(比如从网络服务)获取 Item 对象,并且不用修改 ItemsViewController 。

  • 通常来说 ,实现 DIP 的方式是注入 。 最简单的形式是,高层对象并不指定它需要哪一个低层对象,而是通过构造方法或者属性传入,在 ItemsViewController 中,使用了属性来传入 ItemStore 。

实现数据源方法

Only two methods of this protocol are required, and they are shown in the following example code.

// Return the number of rows for the table.     
override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
   return 0
}

// Provide a cell object for each row.
override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
   // Fetch a cell of the appropriate type.
   let cell = tableView.dequeueReusableCell(withIdentifier: "cellTypeIdentifier", for: indexPath)
   
   // Configure the cell’s contents.
   cell.textLabel!.text = "Cell text"
       
   return cell
}
截屏2021-11-24 20.35.59.png
截屏2021-11-24 20.36.20.png

创建并获取UITableViewCells

    override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return itemStore.allItems.count
    }
    
    
    override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        //let cell = UITableViewCell(style: .value1, reuseIdentifier: "UItableViewCell")
        let cell = tableView.dequeueReusableCell(withIdentifier: "UITableViewCell", for: indexPath)   
        //创建一个新的 UITableViewCell 对象或重用一个 UITableViewCell 对象
        
        let item = itemStore.allItems[indexPath.row]
        cell.textLabel?.text = item.name
        cell.detailTextLabel?.text = "$\(item.valueInDollars)"
        return cell
    }

重用UITableViewCells

图片来自 iOS.Programming.The.Big.Nerd.Ranch.Guide.7th.Edition.2020.3,第 217 页.png
图片来自 iOS.Programming.The.Big.Nerd.Ranch.Guide.7th.Edition.2020.3,第 218 页.png

运行结果

截屏2021-11-24 20.47.48.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,504评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,434评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,089评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,378评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,472评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,506评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,519评论 3 413
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,292评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,738评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,022评论 2 329
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,194评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,873评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,536评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,162评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,413评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,075评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,080评论 2 352

推荐阅读更多精彩内容