我想把数组的数据合并到我的结构变量中
下面是代码
我想在rest->distance中添加距离数组值
distance = ["12.44","45.32","56.1","54.22"]
将此距离数组合并为结构变量距离
var rest : [Restaurants] = []
var distance : [String] = []
struct Restaurants {
var name:String
var lat:Double
var long:Double
var distance:String?
}
let rest1 = Restaurants(name: "English Tea House", lat: 31.461812, long: 74.272524, distance: nil)
let rest2 = Restaurants(name: "Cafe Barbera", lat: 31.474536, long: 74.268103, distance: nil)
let rest3 = Restaurants(name: "Butler's Chocolate", lat: 31.467505, long: 74.251908, distance: nil)
let rest4 = Restaurants(name: "Freddy's Cafe", lat: 31.461312, long: 74.272124, distance: nil)
let rest5 = Restaurants(name: "Arcadian Cafe", lat: 31.464536, long: 74.268603, distance: nil)
let rest6 = Restaurants(name: "Big Moes", lat: 31.467305, long: 74.256908, distance: nil)
rest.append(rest1)
rest.append(rest2)
rest.append(rest3)
rest.append(rest4)
rest.append(rest5)
rest.append(rest6)
for location in rest {
distance.append(findDistance(from: location.lat, long: location.long))
}
// I want to add distance array values in rest -> distance
func findDistance(from lat: Double, long: Double) -> Double {
let source = CLLocation(latitude: 31.461512, longitude: 74.272024)
let destination = CLLocation(latitude: lat, longitude: long)
let distanceInMeters = source.distance(from: destination)
return distanceInMeters
}
最佳答案
这里有很多事情要做,所以我会告诉你解决方法。
首先,rest
是一个糟糕的变量名。当我第一次读它的时候,我以为这是“剩下的”,就像其他的东西一样,我在寻找“主要”数据在哪里。击键不需要花费美元,您只需输入restaurants
。
其次,创建一个空数组,并手动将所有这些餐馆追加到该数组中。相反,您只需从数组文字创建一个数组,它直接包含所有餐馆。不需要分开
为了回答您的直接问题,您可以使用zip
将rest
和distance
一起迭代,但问题是rest1
,rest2
。。。变数,注定要失败。当您复制/粘贴一堆行,而忘记更改其中一行,意外地写入rest.append(rest6); rest.append(rest6)
,忘记rest7
时会发生什么?
第三,您的findDistance(from:long:)
函数接受两个Double
s(纬度和经度),并使用它们构造一个CLLocation
。但是,当您查看这个函数的使用位置时,您已经有了一个CLLocation
,它被分解成两个Double
s,只需要findDistance(from:long:)
立即将它们重新组合成一个CLLocation
。相反,只要让findDistance
直接接受CLLocation
。
第四,Restaurants
数据类型是miss-named。不是很多餐馆。它是一家餐厅的模型。据此命名:Restaurant
应用这些改进(并且在路上固定一堆数据),我们得到:
struct Restaurant {
var name: String
var lat: Double
var long: Double
var distance: String?
}
let restaurants = [
Restaurant(name: "English Tea House", lat: 31.461812, long: 74.272524, distance: nil),
Restaurant(name: "Cafe Barbera", lat: 31.474536, long: 74.268103, distance: nil),
Restaurant(name: "Butler's Chocolate", lat: 31.467505, long: 74.251908, distance: nil,
Restaurant(name: "Freddy's Cafe", lat: 31.461312, long: 74.272124, distance: nil),
Restaurant(name: "Arcadian Cafe", lat: 31.464536, long: 74.268603, distance: nil),
Restaurant(name: "Big Moes", lat: 31.467305, long: 74.256908, distance: nil),
]
var distances = [Double]()
for location in restaurants {
distance.append(findDistance(to: location))
}
func findDistance(to destination: CLLocation) -> Double {
let source = CLLocation(latitude: 31.461512, longitude: 74.272024)
let distanceInMeters = source.distance(from: destination)
return distanceInMeters
}
直接问题
在这里,我将演示我处理直接问题的过程。但是,不要使用这些。所有这些都是卑鄙的设计试图解决潜在的设计缺陷。我展示它们是为了演示连续的优化是什么样子。
现在,为了解决直接的问题:
首次尝试:使用
zip(_:_:)
第一种解决方法可能是使用
zip
并行迭代restaurants
和distances
,然后从每个距离对每个restaurant
进行变异。像这样的:for (restaurant, distance) in zip(restaurants, distances) {
restaurant.distance = distance
}
不过,这是行不通的。因为
Restaurant
是一个值类型,所以循环中的restaurant
变量是数组中变量的副本。设置其距离会使副本发生变化,但不会影响数组中的原始副本。第二次尝试:人工索引
我们可以通过循环索引来解决这一问题,尽管方式不那么漂亮:
for i in restaurants.indices {
restaurants[i] = distances[i]
}
第三次尝试:跳过
distances
数组。第二次尝试是有效的,但是如果
distances
的唯一目的是用一堆值填充它,只是立即使用并丢弃它们,那么为什么我们都有数组呢?for i in restaurants.indices {
restaurant.location = findDistance(to: location)
}
不过,这仍然不太好。
Restaurant
数据类型经历两个阶段的初始化,即acode smell。首先我们用nil
位置初始化它,然后将其设置为实际位置。为什么要麻烦?为什么我们不直接设定地点?let distance = findDistance(to: location)
let restaurants = [
Restaurant(name: "English Tea House", lat: 31.461812, long: 74.272524, distance: distance),
Restaurant(name: "Cafe Barbera", lat: 31.474536, long: 74.268103, distance: distance),
Restaurant(name: "Butler's Chocolate", lat: 31.467505, long: 74.251908, distance: distance),
Restaurant(name: "Freddy's Cafe", lat: 31.461312, long: 74.272124, distance: distance),
Restaurant(name: "Arcadian Cafe", lat: 31.464536, long: 74.268603, distance: distance),
Restaurant(name: "Big Moes", lat: 31.467305, long: 74.256908, distance: nil),
]
但这仍然不是好的设计。。。
用
findDistance(to:)
和Restaurant.distance
修复缺陷findDistance(to:)
到底做了什么?在内部,它与一些硬编码、未命名的位置有一定的距离。当我说“抄送”时,我会期待与我保持距离。如果它距离某个引用点a有一定距离,那么我希望API的拼写应该类似于CLLocation(latitude: 31.461512, longitude: 74.272024)
,或者类似于这样。此外,为什么存储与其他对象的距离是
someRestaurant.distance
的工作?如果我想要someResaurant.distanceFromNorthPole
,Restaurant
?API会变得非常臃肿,而restaurant.distanceToSouthPole
类型最终会做得太多。如果我restaurant.distanceToEquator
?我可以移动,当我移动时,一个预先计算的存储值如何跟上我?解决办法是根本不存储距离。相反,提供一个API,该数据类型的用户可以使用它计算到自己选择的任何点的距离。
struct Restaurant {
var name: String
var lat: Double
var long: Double
func distance(from other: CLLocation) -> Double {
let selfLocation = CLLocation(latitude: self.lat, longitude: self.long)
return selfLocation.distance(from: other)
}
}
let restaurants = [
Restaurant(name: "English Tea House", lat: 31.461812, long: 74.272524),
Restaurant(name: "Cafe Barbera", lat: 31.474536, long: 74.268103),
Restaurant(name: "Butler's Chocolate", lat: 31.467505, long: 74.251908),
Restaurant(name: "Freddy's Cafe", lat: 31.461312, long: 74.272124),
Restaurant(name: "Arcadian Cafe", lat: 31.464536, long: 74.268603),
Restaurant(name: "Big Moes", lat: 31.467305, long: 74.256908),
]
let pointA = CLLocation(latitude: 31.461512, longitude: 74.272024) // Name me!
// and now, whenever you need the distance to pointA:
for restaurant in restaurants {
let distanceFromA = restaurant.distance(from: pointA)
// This is for the purpose of a simple example only. Never hard code units like this,
// Use `Measurement` and `MeasurementFormatter` to create textual representations
// of measurements in the correct units, language and format for the user's locale.
print("\(restaurant.name) is \(distanceFromA) meters from \(pointA)")
}
令人惊讶的是,这仍然不是我们能做到的最好的!
不要把双打储存起来
这就是
restaurant
的作用。注意,几乎所有使用restaurant.distanceToMe
和CLLocation
的都需要先将其装箱到lat
。所以让我们直接存储它,而不是分离单个组件并独立地传递它们。这可以防止像long
这样的错误。struct Restaurant {
var name: String
var location: CLLocation
func distance(from other: CLLocation) -> Double {
return self.location.distance(from: other)
}
}
但是,如果您这样做,初始化器现在需要一个
CLLocation
而不是两个单独的useLocation(lat: self.lat, long: self.long /* oops! */)
/CLLocation
lat
s。这对于与位置api交互(其中long
是用于交换位置信息的“通用货币”类型)更好,但对于硬编码的位置(如您的餐厅)则更麻烦,因为你所有的初始化器调用都会被一堆对Double
的调用所膨胀:let restaurants = [
Restaurant(name: "English Tea House", CLLocation(latitude: 31.461812, longitude: 74.272524)),
Restaurant(name: "Cafe Barbera", CLLocation(latitude: 31.474536, longitude: 74.268103)),
Restaurant(name: "Butler's Chocolate", CLLocation(latitude: 31.467505), longitude: 74.251908)),
Restaurant(name: "Freddy's Cafe", CLLocation(latitude: 31.461312, longitude: 74.272124)),
Restaurant(name: "Arcadian Cafe", CLLocation(latitude: 31.464536, longitude: 74.268603)),
Restaurant(name: "Big Moes", CLLocation(latitude: 31.467305, longitude: 74.256908)),
]
为了解决这个问题,我们可以将
CLLocation
塞进一个小的初始化器中,以方便使用。我是在CLLocation.init(latitude:longitude:)
的扩展中这样做的,而不是直接在CLLocation.init(latitude:longitude:)
的初始声明中这样做,因为这样做会保留编译器生成的初始值设定项(称为“成员级初始值设定项”),否则将被替换:extension Restaurant {
init(name: String, lat: Double, long: Double) {
self.init(name: name, location: CLLocation(latitude: lat, long))
}
}
这使我们可以重新获得以前的好语法:
let restaurants = [
Restaurant(name: "English Tea House", lat: 31.461812, long: 74.272524),
Restaurant(name: "Cafe Barbera", lat: 31.474536, long: 74.268103),
Restaurant(name: "Butler's Chocolate", lat: 31.467505, long: 74.251908),
Restaurant(name: "Freddy's Cafe", lat: 31.461312, long: 74.272124),
Restaurant(name: "Arcadian Cafe", lat: 31.464536, long: 74.268603),
Restaurant(name: "Big Moes", lat: 31.467305, long: 74.256908),
]
易变性
在你的应用程序的一个实例的生命周期内,餐厅名称和位置不太可能改变,因此不需要保持它们的可变。所以让我们修正一下:
struct Restaurant {
var name: String
var location: CLLocation
func distance(from other: CLLocation) -> Double {
return self.location.distance(from: other)
}
}
最后。。。
我们已经到了最后阶段。一个名为
Restaurant
的程序,它不需要两阶段初始化,为用户可能喜欢的任何点提供最新的距离数据,并且由于Restaurant
和Restaurant
被粘在一起而不易出现复制粘贴错误。struct Restaurant {
var name: String
var location: CLLocation
func distance(from other: CLLocation) -> Double {
return self.location.distance(from: other)
}
}
extension Restaurant {
init(name: String, lat: Double, long: Double) {
self.init(name: name, location: CLLocation(latitude: lat, long))
}
}
let restaurants = [
Restaurant(name: "English Tea House", lat: 31.461812, long: 74.272524),
Restaurant(name: "Cafe Barbera", lat: 31.474536, long: 74.268103),
Restaurant(name: "Butler's Chocolate", lat: 31.467505, long: 74.251908,
Restaurant(name: "Freddy's Cafe", lat: 31.461312, long: 74.272124),
Restaurant(name: "Arcadian Cafe", lat: 31.464536, long: 74.268603),
Restaurant(name: "Big Moes", lat: 31.467305, long: 74.256908),
]
let pointA = CLLocation(latitude: 31.461512, longitude: 74.272024) // Name me!
// and now, whenever you need the distance to pointA you can do this (for example):
for restaurant in restaurants {
let distanceFromA = restaurant.distance(from: pointA)
// This is for the purpose of a simple example only. Never hard code units like this,
// Use `Measurement` and `MeasurementFormatter` to create textual representations
// of measurements in the correct units, language and format for the user's locale.
print("\(restaurant.name) is \(distanceFromA) meters from \(pointA)")
}
关于swift - 将数组数据追加到Struct变量中,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/55893353/