回到简体

This commit is contained in:
chai2010
2016-02-15 11:06:34 +08:00
parent 9e878f9944
commit 2b37b23285
177 changed files with 2354 additions and 2354 deletions

View File

@@ -1,6 +1,6 @@
## 6.2. 基於指針對象的方法
## 6.2. 基于指针对象的方法
當調用一個函數時,會對其每一個參數值進行拷,如果一個函數需要更新一個變量,或者函的其中一個參數實在太大我希望能避免進行這種默認的拷貝,這種情況下我就需要用到指了。對應到我們這里用更新接收器的象的方法,當這個接受者量本身比較大時,我就可以用其指而不是對象來聲明方法,如下:
当调用一个函数时,会对其每一个参数值进行拷,如果一个函数需要更新一个变量,或者函的其中一个参数实在太大我希望能避免进行这种默认的拷贝,这种情况下我就需要用到指了。对应到我们这里用更新接收器的象的方法,当这个接受者量本身比较大时,我就可以用其指而不是对象来声明方法,如下:
```go
func (p *Point) ScaleBy(factor float64) {
@@ -9,18 +9,18 @@ func (p *Point) ScaleBy(factor float64) {
}
```
這個方法的名字是`(*Point).ScaleBy`里的括是必的;有括號的話這個表達式可能被理解`*(Point.ScaleBy)`
这个方法的名字是`(*Point).ScaleBy`里的括是必的;有括号的话这个表达式可能被理解`*(Point.ScaleBy)`
現實的程序里,一般會約定如果Point這個類有一個指針作爲接收器的方法,那所有Point的方法都必有一個指針接收器,使是那些不需要這個指針接收器的函。我們在這里打破了這個約定隻是爲了展示一下兩種方法的同而已。
现实的程序里,一般会约定如果Point这个类有一个指针作为接收器的方法,那所有Point的方法都必有一个指针接收器,使是那些不需要这个指针接收器的函。我们在这里打破了这个约定只是为了展示一下两种方法的同而已。
隻有類型(Point)和指向他的指(*Point),才是可能會出現在接收器明里的兩種接收器。此外,了避免歧,在明方法,如果一個類型名本身是一個指針的話,是不允其出在接收器中的,比如下面這個例子:
只有类型(Point)和指向他的指(*Point),才是可能会出现在接收器明里的两种接收器。此外,了避免歧,在明方法,如果一个类型名本身是一个指针的话,是不允其出在接收器中的,比如下面这个例子:
```go
type P *int
func (P) f() { /* ... */ } // compile error: invalid receiver type
```
想要調用指針類型方法`(*Point).ScaleBy`要提供一Point型的指針卽可,像下面這樣
想要用指针类型方法`(*Point).ScaleBy`要提供一Point型的指针即可,像下面这样
```go
r := &Point{1, 2}
@@ -28,7 +28,7 @@ r.ScaleBy(2)
fmt.Println(*r) // "{2, 4}"
```
或者這樣
或者这样
```go
p := Point{1, 2}
@@ -36,7 +36,7 @@ pptr := &p
pptr.ScaleBy(2)
fmt.Println(p) // "{2, 4}"
```
或者這樣:
或者这样:
```go
p := Point{1, 2}
@@ -44,51 +44,51 @@ p := Point{1, 2}
fmt.Println(p) // "{2, 4}"
```
過後面兩種方法有些笨拙。幸的是go言本身在這種地方會幫到我。如果接收器p是一Point型的量,且其方法需要一Point指針作爲接收器,我可以用下面這種簡短的法:
过后面两种方法有些笨拙。幸的是go言本身在这种地方会帮到我。如果接收器p是一Point型的量,且其方法需要一Point指针作为接收器,我可以用下面这种简短的法:
```go
p.ScaleBy(2)
```
編譯器會隱式地幫我們&p去調用ScaleBy這個方法。這種簡寫方法隻適用於“變量”包括struct里的字段比如p.X以及array和slice的元素比如perim[0]。我不能通過一個無法取到地址的接收器來調用指方法,比如臨時變量的存地址就無法獲取得到:
编译器会隐式地帮我们&p去用ScaleBy这个方法。这种简写方法只适用于“变量”包括struct里的字段比如p.X以及array和slice的元素比如perim[0]。我不能通过一个无法取到地址的接收器来调用指方法,比如临时变量的存地址就无法获取得到:
```go
Point{1, 2}.ScaleBy(2) // compile error: can't take address of Point literal
```
但是我可以用一`*Point`這樣的接收器來調用Point的方法爲我們可以通地址找到這個變量,要用解引用符`*`取到該變量卽可。編譯器在里也會給我們隱式地插入`*`這個操作符,所以下面這兩種寫法等的:
但是我可以用一`*Point`这样的接收器来调用Point的方法为我们可以通地址找到这个变量,要用解引用符`*`取到该变量即可。编译器在里也会给我们隐式地插入`*`这个操作符,所以下面这两种写法等的:
```Go
pptr.Distance(q)
(*pptr).Distance(q)
```
里的幾個例子可能你有些惑,所以我們總結一下:在每一合法的方法調用表式中,也就是下面三種情況里的任意一種情況都是可以的:
里的几个例子可能你有些惑,所以我们总结一下:在每一合法的方法用表式中,也就是下面三种情况里的任意一种情况都是可以的:
是接收器的實際參數和其接收器的形式參數相同,比如者都是型T或者都是`*T`
是接收器的实际参数和其接收器的形式参数相同,比如者都是型T或者都是`*T`
```go
Point{1, 2}.Distance(q) // Point
pptr.ScaleBy(2) // *Point
```
或者接收器形參是類型T但接收器實參是類`*T`這種情況下編譯器會隱式地爲我們取變量的地址:
或者接收器形参是类型T但接收器实参是类`*T`这种情况下编译器会隐式地为我们取变量的地址:
```go
p.ScaleBy(2) // implicit (&p)
```
或者接收器形參是類`*T`實參是類型T。編譯器會隱式地爲我們解引用,取到指指向的實際變量:
或者接收器形参是类`*T`实参是类型T。编译器会隐式地为我们解引用,取到指指向的实际变量:
```go
pptr.Distance(q) // implicit (*pptr)
```
如果型T的所有方法都是用T型自己做接收器(而不是`*T`),那麽拷貝這種類型的例就是安全的;調用他的任何一方法也就會産生一值的拷。比如time.Duration的這個類型,在調用其方法時就會被全部拷一份,包括在作爲參數傳入函數的時候。但是如果一方法使用指針作爲接收器,你需要避免對其進行拷,因爲這樣可能會破壞掉該類型內部的不性。比如你bytes.Buffer對象進行了拷,那可能引起原始象和拷貝對象隻是别名而已,但實際上其指向的象是一致的。接着對拷貝後的變量進行脩改可能會有讓你意外的果。
如果型T的所有方法都是用T型自己做接收器(而不是`*T`),那么拷贝这种类型的例就是安全的;用他的任何一方法也就会产生一值的拷。比如time.Duration的这个类型,在用其方法时就会被全部拷一份,包括在作为参数传入函数的时候。但是如果一方法使用指针作为接收器,你需要避免对其进行拷,因为这样可能会破坏掉该类型内部的不性。比如你bytes.Buffer对象进行了拷,那可能引起原始象和拷贝对象只是别名而已,但实际上其指向的象是一致的。接着对拷贝后的变量进行修改可能会有让你意外的果。
**譯註** 作者這里説的比較繞,其實有兩點
**译注** 作者这里说的比较绕,其实有两点
1. 不管你的method的receiver是指針類型還是非指針類型,都是可以通過指針/非指針類型進行調用的,編譯器會幫你做類型轉換
2.明一method的receiver是指針還是非指針類型時,你需要考慮兩方面的部,第一方面是這個對象本身是不是特别大,如果聲明爲非指針變量時,調用會産生一次拷;第二方面是如果你用指針類型作receiver你一定要意,這種指針類型指向的始是一塊內存地址,就算你對其進行了拷。熟悉C或者C艹的人這里應該很快能明白。
1. 不管你的method的receiver是指针类型还是非指针类型,都是可以通过指针/非指针类型进行调用的,编译器会帮你做类型转换
2.明一method的receiver是指针还是非指针类型时,你需要考虑两方面的部,第一方面是这个对象本身是不是特别大,如果声明为非指针变量时,调用会产生一次拷;第二方面是如果你用指针类型作receiver你一定要意,这种指针类型指向的始是一块内存地址,就算你对其进行了拷。熟悉C或者C艹的人这里应该很快能明白。
{% include "./ch6-02-1.md" %}