gpt4 book ai didi

ios - 使用 Swift 对 TableViewController 中的字典数据进行分组和排序

转载 作者:行者123 更新时间:2023-11-30 14:07:39 24 4
gpt4 key购买 nike

我有一个如下所示的字典数据结构,我试图将它们分组在我的 TableViewController 中,以便 A 组显示以 title = A 开头的 MyData ,并在同时显示 sectionIndexTitlesForTableView 以及从标题获取的可用字母。

[这就是我想要实现的目标]

Prototype

我尝试删除字典中标题元素的所有第一个字母,并使用下面的代码将它们保存在一个集合中,但是当我运行我的应用程序时,我得到的结果在我的表中重复。

我对 Swift 还很陌生,很高兴能得到关于如何实现这一目标的指导。

这是我的词典数据:

var data: [[String:AnyObject]] =

[
[
"id": "1",
"title": "A Title",
"alphabet": "A",
"Detail": "This is a String"
],
[
"id": "2",
"title": "A Title Again",
"alphabet": "A",
"Detail": "This is a String"
],
[
"id": "3",
"title": "B Title",
"alphabet": "B",
"Detail": "This is a String"
],
[
"id": "4",
"title": "B Title Again",
"alphabet": "B",
"Detail": "This is a String"
]
]

这是我的尝试:

class Index: UITableViewController {

var MyData = data

var letters = Set<String>()


override func viewDidLoad() {
super.viewDidLoad()
for element in MyData {
var title = element["title"] as? String
let letter = title?.substringToIndex(advance(title!.startIndex, 1))
letters.insert(letter!)
}


MyData = MyData.sort { element1, element2 in
let title1 = element1["title"] as? String
let title2 = element2["title"] as? String

return title1 < title2
}

}

// MARK: - Table view data source

override func numberOfSectionsInTableView(tableView: UITableView) -> Int {
// #warning Incomplete implementation, return the number of sections
return letters.count
}

override func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {

return self.MyData.count
}


override func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
let cell = self.tableView.dequeueReusableCellWithIdentifier("Cell", forIndexPath: indexPath) as UITableViewCell?

cell!.textLabel?.text = (MyData[indexPath.row]["title"] as! String)

return cell!

}

最佳答案

问题是 numberOfRowsInSection 必须返回每个部分的行数,在您的示例中,第 0 部分为 2,第 1 部分为 2

您可以使用键值编码方法valueForKey来收集您的字母集,该方法经常被误认为objectForKey
与为给定键返回一个值的 objectForKey 不同,valueForKey 返回数组中所有成员的键 alphabet 的值。

此代码创建字母的Set来清除重复项,将其返回到Array并对其进行排序。

let letters = (data as NSArray).valueForKey("alphabet") as! [String]
let filteredLetters = Set<String>(letters)
let sortedLetters = Array(filteredLetters).sorted {$0 < $1}

如果 alphabet 以及其他键的所有值都保证为 String,则无需将它们转换为可选值。

然后在numberOfRowsInSection中,您必须过滤每个部分的项目数

override func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
return data.filter { ($0["alphabet"] as! String) == sortedLetters[section] }.count
}

请注意,表达式 sortedLetters[section] 不需要进行转换,因为编译器知道这是一个 String 数组。

当然,您还必须为 cellForRowAtIndexPath 中的部分检索适当的项目,这是相当昂贵的,因为主数组将被过滤多次。
我建议将 viewDidLoad() 中的 data 转换为一个新字典,其中字母作为键,数组包含以此特定字母开头的项目作为值。这是就速度和性能而言的最佳解决方案。

这是一个完整的解决方案(不显示用于快速搜索的字母)

class TableViewController: UITableViewController {

let data: [[String:String]] =

[
[
"id": "1",
"title": "A Title",
"alphabet": "A",
"Detail": "This is a String"
],
[
"id": "2",
"title": "A Title Again",
"alphabet": "A",
"Detail": "This is a String"
],
[
"id": "3",
"title": "B Title",
"alphabet": "B",
"Detail": "This is a String"
],
[
"id": "4",
"title": "B Title Again",
"alphabet": "B",
"Detail": "This is a String"
]
]


var letters = [String]()
var dataSource = [String:AnyObject]()

override func viewDidLoad() {
super.viewDidLoad()

for value in data {
let letter = value["alphabet"]!
if dataSource[letter] == nil {
letters.append(letter)
dataSource[letter] = [[String:AnyObject]]()
}
var array = dataSource[letter] as! [[String:AnyObject]]
array.append(value)
dataSource.updateValue(array, forKey: letter)
}
letters.sorted {$0 < $1}
tableView.reloadData()

}

// MARK: - Table view data source

override func numberOfSectionsInTableView(tableView: UITableView) -> Int {
return letters.count
}

override func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
let letter = letters[section]
return dataSource[letter]!.count
}


override func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
let cell = tableView.dequeueReusableCellWithIdentifier("Cell", forIndexPath: indexPath) as! UITableViewCell

let letter = letters[indexPath.section]
let letterArray = dataSource[letter]! as! [[String:AnyObject]]
let item = letterArray [indexPath.row]
if let title = item["title"] as? String {
cell.textLabel?.text = title
}
return cell
}

override func tableView(tableView: UITableView, titleForHeaderInSection section: Int) -> String? {
return letters[section]
}
}

关于ios - 使用 Swift 对 TableViewController 中的字典数据进行分组和排序,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32166237/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com