|
|
@@ -1,4 +1,4 @@
|
|
|
-# 结构体,集合和高阶函数
|
|
|
+# 结构体、集合和高阶函数
|
|
|
|
|
|
通常你在应用中定义了一个结构体,那么你也可能需要这个结构体的(指针)对象集合,比如:
|
|
|
|
|
|
@@ -12,7 +12,6 @@ type Car struct {
|
|
|
}
|
|
|
|
|
|
type Cars []*Car
|
|
|
-
|
|
|
```
|
|
|
|
|
|
在定义所需功能时我们可以利用函数可以作为(其它函数的)参数的事实来使用高阶函数,例如:
|
|
|
@@ -26,7 +25,6 @@ func (cs Cars) Process(f func(car *Car)) {
|
|
|
f(c)
|
|
|
}
|
|
|
}
|
|
|
-
|
|
|
```
|
|
|
|
|
|
2)在上面的基础上,实现一个查找函数来获取子集合,并在 `Process()` 中传入一个闭包执行(这样就可以访问局部切片 `cars`):
|
|
|
@@ -43,7 +41,6 @@ func (cs Cars) FindAll(f func(car *Car) bool) Cars {
|
|
|
)
|
|
|
return cars
|
|
|
}
|
|
|
-
|
|
|
```
|
|
|
|
|
|
3)实现 Map 功能,产出除 car 对象以外的东西:
|
|
|
@@ -55,11 +52,10 @@ func (cs Cars) Map(f func(car *Car) Any) []Any {
|
|
|
ix := 0
|
|
|
cs.Process(func(c *Car) {
|
|
|
result[ix] = f(c)
|
|
|
- ix++
|
|
|
+ ix++
|
|
|
})
|
|
|
return result
|
|
|
}
|
|
|
-
|
|
|
```
|
|
|
|
|
|
现在我们可以定义下面这样的具体查询:
|
|
|
@@ -68,7 +64,6 @@ func (cs Cars) Map(f func(car *Car) Any) []Any {
|
|
|
allNewBMWs := allCars.FindAll(func(car *Car) bool {
|
|
|
return (car.Manufacturer == “BMW”) && (car.BuildYear > 2010)
|
|
|
})
|
|
|
-
|
|
|
```
|
|
|
|
|
|
4)我们也可以根据入参返回不同的函数。也许我们想根据不同的厂商添加汽车到不同的集合,但是这可能会是多变的。所以我们可以定义一个函数来产生特定的添加函数和 map 集:
|
|
|
@@ -92,7 +87,6 @@ func MakeSortedAppender(manufacturers[]string)(func(car*Car),map[string]Cars) {
|
|
|
}
|
|
|
return appender, sortedCars
|
|
|
}
|
|
|
-
|
|
|
```
|
|
|
|
|
|
现在我们可以用它把汽车分类为独立的集合,像这样:
|
|
|
@@ -102,7 +96,6 @@ manufacturers := []string{“Ford”, “Aston Martin”, “Land Rover”, “B
|
|
|
sortedAppender, sortedCars := MakeSortedAppender(manufacturers)
|
|
|
allUnsortedCars.Process(sortedAppender)
|
|
|
BMWCount := len(sortedCars[“BMW”])
|
|
|
-
|
|
|
```
|
|
|
|
|
|
我们让这些代码在下面的程序 cars.go(此处只展示了 main() 中的代码,别的代码已经在上面展示)中执行:
|
|
|
@@ -117,7 +110,6 @@ import (
|
|
|
"fmt"
|
|
|
)
|
|
|
|
|
|
-
|
|
|
type Any interface{}
|
|
|
type Car struct {
|
|
|
Model string
|
|
|
@@ -182,12 +174,12 @@ func (cs Cars) Map(f func(car *Car) Any) []Any {
|
|
|
func MakeSortedAppender(manufacturers []string) (func(car *Car), map[string]Cars) {
|
|
|
// Prepare maps of sorted cars.
|
|
|
sortedCars := make(map[string]Cars)
|
|
|
-
|
|
|
+
|
|
|
for _, m := range manufacturers {
|
|
|
sortedCars[m] = make([]*Car, 0)
|
|
|
}
|
|
|
sortedCars["Default"] = make([]*Car, 0)
|
|
|
-
|
|
|
+
|
|
|
// Prepare appender function:
|
|
|
appender := func(c *Car) {
|
|
|
if _, ok := sortedCars[c.Manufacturer]; ok {
|
|
|
@@ -198,14 +190,15 @@ func MakeSortedAppender(manufacturers []string) (func(car *Car), map[string]Cars
|
|
|
}
|
|
|
return appender, sortedCars
|
|
|
}
|
|
|
+```
|
|
|
|
|
|
-/* Output:
|
|
|
+输出:
|
|
|
+
|
|
|
+```
|
|
|
AllCars: [0xf8400038a0 0xf840003bd0 0xf840003ba0 0xf840003b70]
|
|
|
New BMWs: [0xf840003bd0]
|
|
|
Map sortedCars: map[Default:[0xf840003ba0] Jaguar:[] Land Rover:[] BMW:[0xf840003bd0 0xf840003b70] Aston Martin:[] Ford:[0xf8400038a0]]
|
|
|
We have 2 BMWs
|
|
|
-*/
|
|
|
-
|
|
|
```
|
|
|
|
|
|
|