mirror of
https://github.com/gopl-zh/gopl-zh.github.com.git
synced 2025-12-19 12:14:20 +08:00
rebuild
This commit is contained in:
142
ch6/ch6-02.html
142
ch6/ch6-02.html
@@ -5,7 +5,7 @@
|
||||
|
||||
<meta charset="UTF-8">
|
||||
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
|
||||
<title>基於指鍼對象的方法 | Go编程语言</title>
|
||||
<title>基於指針對象的方法 | Go编程语言</title>
|
||||
<meta content="text/html; charset=utf-8" http-equiv="Content-Type">
|
||||
<meta name="description" content="">
|
||||
<meta name="generator" content="GitBook 2.5.2">
|
||||
@@ -48,7 +48,7 @@
|
||||
<body>
|
||||
|
||||
|
||||
<div class="book" data-level="6.2" data-chapter-title="基於指鍼對象的方法" data-filepath="ch6/ch6-02.md" data-basepath=".." data-revision="Wed Dec 16 2015 10:54:29 GMT+0800 (中国标准时间)">
|
||||
<div class="book" data-level="6.2" data-chapter-title="基於指針對象的方法" data-filepath="ch6/ch6-02.md" data-basepath=".." data-revision="Mon Dec 21 2015 12:51:02 GMT+0800 (中国标准时间)">
|
||||
|
||||
|
||||
<div class="book-summary">
|
||||
@@ -146,7 +146,7 @@
|
||||
|
||||
<b>0.5.</b>
|
||||
|
||||
緻謝
|
||||
致謝
|
||||
</a>
|
||||
|
||||
|
||||
@@ -212,7 +212,7 @@
|
||||
|
||||
<b>1.3.</b>
|
||||
|
||||
査找重復的行
|
||||
査找重複的行
|
||||
</a>
|
||||
|
||||
|
||||
@@ -227,7 +227,7 @@
|
||||
|
||||
<b>1.4.</b>
|
||||
|
||||
GIF動畫
|
||||
GIF動畵
|
||||
</a>
|
||||
|
||||
|
||||
@@ -257,7 +257,7 @@
|
||||
|
||||
<b>1.6.</b>
|
||||
|
||||
併髮穫取多個URL
|
||||
併發穫取多個URL
|
||||
</a>
|
||||
|
||||
|
||||
@@ -479,7 +479,7 @@
|
||||
|
||||
<b>3.3.</b>
|
||||
|
||||
復數
|
||||
複數
|
||||
</a>
|
||||
|
||||
|
||||
@@ -494,7 +494,7 @@
|
||||
|
||||
<b>3.4.</b>
|
||||
|
||||
佈爾型
|
||||
布爾型
|
||||
</a>
|
||||
|
||||
|
||||
@@ -544,7 +544,7 @@
|
||||
|
||||
<b>4.</b>
|
||||
|
||||
復閤數據類型
|
||||
複合數據類型
|
||||
</a>
|
||||
|
||||
|
||||
@@ -857,7 +857,7 @@
|
||||
|
||||
<b>6.2.</b>
|
||||
|
||||
基於指鍼對象的方法
|
||||
基於指針對象的方法
|
||||
</a>
|
||||
|
||||
|
||||
@@ -887,7 +887,7 @@
|
||||
|
||||
<b>6.4.</b>
|
||||
|
||||
方法值和方法錶達式
|
||||
方法值和方法表達式
|
||||
</a>
|
||||
|
||||
|
||||
@@ -953,7 +953,7 @@
|
||||
|
||||
<b>7.1.</b>
|
||||
|
||||
接口是閤約
|
||||
接口是合約
|
||||
</a>
|
||||
|
||||
|
||||
@@ -1073,7 +1073,7 @@
|
||||
|
||||
<b>7.9.</b>
|
||||
|
||||
示例: 錶達式求值
|
||||
示例: 表達式求值
|
||||
</a>
|
||||
|
||||
|
||||
@@ -1103,7 +1103,7 @@
|
||||
|
||||
<b>7.11.</b>
|
||||
|
||||
基於類型斷言識彆錯誤類型
|
||||
基於類型斷言識别錯誤類型
|
||||
</a>
|
||||
|
||||
|
||||
@@ -1214,7 +1214,7 @@
|
||||
|
||||
<b>8.2.</b>
|
||||
|
||||
示例: 併髮的Clock服務
|
||||
示例: 併發的Clock服務
|
||||
</a>
|
||||
|
||||
|
||||
@@ -1229,7 +1229,7 @@
|
||||
|
||||
<b>8.3.</b>
|
||||
|
||||
示例: 併髮的Echo服務
|
||||
示例: 併發的Echo服務
|
||||
</a>
|
||||
|
||||
|
||||
@@ -1274,7 +1274,7 @@
|
||||
|
||||
<b>8.6.</b>
|
||||
|
||||
示例: 併髮的Web爬蟲
|
||||
示例: 併發的Web爬蟲
|
||||
</a>
|
||||
|
||||
|
||||
@@ -1289,7 +1289,7 @@
|
||||
|
||||
<b>8.7.</b>
|
||||
|
||||
基於select的多路復用
|
||||
基於select的多路複用
|
||||
</a>
|
||||
|
||||
|
||||
@@ -1304,7 +1304,7 @@
|
||||
|
||||
<b>8.8.</b>
|
||||
|
||||
示例: 併髮的字典遍歷
|
||||
示例: 併發的字典遍歷
|
||||
</a>
|
||||
|
||||
|
||||
@@ -1319,7 +1319,7 @@
|
||||
|
||||
<b>8.9.</b>
|
||||
|
||||
併髮的退齣
|
||||
併發的退齣
|
||||
</a>
|
||||
|
||||
|
||||
@@ -1354,7 +1354,7 @@
|
||||
|
||||
<b>9.</b>
|
||||
|
||||
基於共享變量的併髮
|
||||
基於共享變量的併發
|
||||
</a>
|
||||
|
||||
|
||||
@@ -1415,7 +1415,7 @@
|
||||
|
||||
<b>9.4.</b>
|
||||
|
||||
內存衕步
|
||||
內存同步
|
||||
</a>
|
||||
|
||||
|
||||
@@ -1460,7 +1460,7 @@
|
||||
|
||||
<b>9.7.</b>
|
||||
|
||||
示例: 併髮的非阻塞緩存
|
||||
示例: 併發的非阻塞緩存
|
||||
</a>
|
||||
|
||||
|
||||
@@ -1475,7 +1475,7 @@
|
||||
|
||||
<b>9.8.</b>
|
||||
|
||||
Goroutines和綫程
|
||||
Goroutines和線程
|
||||
</a>
|
||||
|
||||
|
||||
@@ -1748,7 +1748,7 @@
|
||||
|
||||
<b>12.1.</b>
|
||||
|
||||
為何需要反射?
|
||||
爲何需要反射?
|
||||
</a>
|
||||
|
||||
|
||||
@@ -1793,7 +1793,7 @@
|
||||
|
||||
<b>12.4.</b>
|
||||
|
||||
示例: 編碼S錶達式
|
||||
示例: 編碼S表達式
|
||||
</a>
|
||||
|
||||
|
||||
@@ -1823,7 +1823,7 @@
|
||||
|
||||
<b>12.6.</b>
|
||||
|
||||
示例: 解碼S錶達式
|
||||
示例: 解碼S表達式
|
||||
</a>
|
||||
|
||||
|
||||
@@ -1975,50 +1975,14 @@
|
||||
|
||||
</li>
|
||||
|
||||
<li class="chapter " data-level="14" data-path="exercise/ex.html">
|
||||
|
||||
|
||||
<a href="../exercise/ex.html">
|
||||
|
||||
<i class="fa fa-check"></i>
|
||||
|
||||
<b>14.</b>
|
||||
|
||||
習題解答
|
||||
</a>
|
||||
|
||||
|
||||
<ul class="articles">
|
||||
|
||||
|
||||
<li class="chapter " data-level="14.1" data-path="exercise/ex-ch1.html">
|
||||
|
||||
|
||||
<a href="../exercise/ex-ch1.html">
|
||||
|
||||
<i class="fa fa-check"></i>
|
||||
|
||||
<b>14.1.</b>
|
||||
|
||||
第一章 入門
|
||||
</a>
|
||||
|
||||
|
||||
</li>
|
||||
|
||||
|
||||
</ul>
|
||||
|
||||
</li>
|
||||
|
||||
<li class="chapter " data-level="15" data-path="errata.html">
|
||||
<li class="chapter " data-level="14" data-path="errata.html">
|
||||
|
||||
|
||||
<a href="../errata.html">
|
||||
|
||||
<i class="fa fa-check"></i>
|
||||
|
||||
<b>15.</b>
|
||||
<b>14.</b>
|
||||
|
||||
勘誤
|
||||
</a>
|
||||
@@ -2059,20 +2023,20 @@
|
||||
|
||||
<section class="normal" id="section-">
|
||||
|
||||
<h2 id="62-基於指鍼對象的方法">6.2. 基於指鍼對象的方法</h2>
|
||||
<p>當調用一個函數時,會對其每一個參數值進行拷貝,如果一個函數需要更新一個變量,或者函數的其中一個參數實在太大我們希望能夠避免進行這種默認的拷貝,這種情況下我們就需要用到指鍼了。對應到我們這裏用來更新接收器的對象的方法,當這個接受者變量本身比較大時,我們就可以用其指鍼而不是對象來聲明方法,如下:</p>
|
||||
<h2 id="62-基於指針對象的方法">6.2. 基於指針對象的方法</h2>
|
||||
<p>當調用一個函數時,會對其每一個參數值進行拷貝,如果一個函數需要更新一個變量,或者函數的其中一個參數實在太大我們希望能夠避免進行這種默認的拷貝,這種情況下我們就需要用到指針了。對應到我們這里用來更新接收器的對象的方法,當這個接受者變量本身比較大時,我們就可以用其指針而不是對象來聲明方法,如下:</p>
|
||||
<pre><code class="lang-go"><span class="hljs-keyword">func</span> (p *Point) ScaleBy(factor <span class="hljs-typename">float64</span>) {
|
||||
p.X *= factor
|
||||
p.Y *= factor
|
||||
}
|
||||
</code></pre>
|
||||
<p>這個方法的名字是<code>(*Point).ScaleBy</code>。這裏的括號是必鬚的;沒有括號的話這個錶達式可能會被理解為<code>*(Point.ScaleBy)</code>。</p>
|
||||
<p>在現實的程序裏,一般會約定如果Point這個類有一個指鍼作為接收器的方法,那麼所有Point的方法都必鬚有一個指鍼接收器,卽使是那些併不需要這個指鍼接收器的函數。我們在這裏打破了這個約定隻是為了展示一下兩種方法的異衕而已。</p>
|
||||
<p>隻有類型(Point)和指曏他們的指鍼(*Point),纔是可能會齣現在接收器聲明裏的兩種接收器。此外,為了避免歧義,在聲明方法時,如果一個類型名本身是一個指鍼的話,是不允許其齣現在接收器中的,比如下麫這個例子:</p>
|
||||
<p>這個方法的名字是<code>(*Point).ScaleBy</code>。這里的括號是必鬚的;沒有括號的話這個表達式可能會被理解爲<code>*(Point.ScaleBy)</code>。</p>
|
||||
<p>在現實的程序里,一般會約定如果Point這個類有一個指針作爲接收器的方法,那麽所有Point的方法都必鬚有一個指針接收器,卽使是那些併不需要這個指針接收器的函數。我們在這里打破了這個約定隻是爲了展示一下兩種方法的異同而已。</p>
|
||||
<p>隻有類型(Point)和指向他們的指針(*Point),纔是可能會齣現在接收器聲明里的兩種接收器。此外,爲了避免歧義,在聲明方法時,如果一個類型名本身是一個指針的話,是不允許其齣現在接收器中的,比如下面這個例子:</p>
|
||||
<pre><code class="lang-go"><span class="hljs-keyword">type</span> P *<span class="hljs-typename">int</span>
|
||||
<span class="hljs-keyword">func</span> (P) f() { <span class="hljs-comment">/* ... */</span> } <span class="hljs-comment">// compile error: invalid receiver type</span>
|
||||
</code></pre>
|
||||
<p>想要調用指鍼類型方法<code>(*Point).ScaleBy</code>,隻要提供一個Point類型的指鍼卽可,像下麫這樣。</p>
|
||||
<p>想要調用指針類型方法<code>(*Point).ScaleBy</code>,隻要提供一個Point類型的指針卽可,像下面這樣。</p>
|
||||
<pre><code class="lang-go">r := &Point{<span class="hljs-number">1</span>, <span class="hljs-number">2</span>}
|
||||
r.ScaleBy(<span class="hljs-number">2</span>)
|
||||
fmt.Println(*r) <span class="hljs-comment">// "{2, 4}"</span>
|
||||
@@ -2088,34 +2052,34 @@ fmt.Println(p) <span class="hljs-comment">// "{2, 4}"</span>
|
||||
(&p).ScaleBy(<span class="hljs-number">2</span>)
|
||||
fmt.Println(p) <span class="hljs-comment">// "{2, 4}"</span>
|
||||
</code></pre>
|
||||
<p>不過後麫兩種方法有些笨拙。倖運的是,go語言本身在這種地方會幫到我們。如果接收器p是一個Point類型的變量,併且其方法需要一個Point指鍼作為接收器,我們可以用下麫這種簡短的寫法:</p>
|
||||
<p>不過後面兩種方法有些笨拙。幸運的是,go語言本身在這種地方會幫到我們。如果接收器p是一個Point類型的變量,併且其方法需要一個Point指針作爲接收器,我們可以用下面這種簡短的寫法:</p>
|
||||
<pre><code class="lang-go">p.ScaleBy(<span class="hljs-number">2</span>)
|
||||
</code></pre>
|
||||
<p>編譯器會隱式地幫我們用&p去調用ScaleBy這個方法。這種簡寫方法隻適用於“變量”,包括struct裏的字段比如p.X,以及array和slice內的元素比如perim[0]。我們不能通過一個無法取到地址的接收器來調用指鍼方法,比如臨時變量的內存地址就無法穫取得到:</p>
|
||||
<p>編譯器會隱式地幫我們用&p去調用ScaleBy這個方法。這種簡寫方法隻適用於“變量”,包括struct里的字段比如p.X,以及array和slice內的元素比如perim[0]。我們不能通過一個無法取到地址的接收器來調用指針方法,比如臨時變量的內存地址就無法穫取得到:</p>
|
||||
<pre><code class="lang-go">Point{<span class="hljs-number">1</span>, <span class="hljs-number">2</span>}.ScaleBy(<span class="hljs-number">2</span>) <span class="hljs-comment">// compile error: can't take address of Point literal</span>
|
||||
</code></pre>
|
||||
<p>但是我們可以用一個<code>*Point</code>這樣的接收器來調用Point的方法,因為我們可以通過地址來找到這個變量,隻要用解引用符號<code>*</code>來取到該變量卽可。編譯器在這裏也會給我們隱式地插入<code>*</code>這個操作符,所以下麫這兩種寫法等價的:</p>
|
||||
<p>但是我們可以用一個<code>*Point</code>這樣的接收器來調用Point的方法,因爲我們可以通過地址來找到這個變量,隻要用解引用符號<code>*</code>來取到該變量卽可。編譯器在這里也會給我們隱式地插入<code>*</code>這個操作符,所以下面這兩種寫法等價的:</p>
|
||||
<pre><code class="lang-Go">pptr.Distance(q)
|
||||
(*pptr).Distance(q)
|
||||
</code></pre>
|
||||
<p>Let’s summarize these three cases again, since they are a frequent point of confusion. In every valid method call expression, exactly one of these three statements is true.
|
||||
這裏的幾個例子可能讓你有些睏惑,所以我們總結一下:在每一個閤法的方法調用錶達式中,也就是下麫三種情況裏的任意一種情況都是可以的:</p>
|
||||
<p>不論是接收器的實際參數和其接收器的形式參數相衕,比如兩者都是類型T或者都是類型<code>*T</code>:</p>
|
||||
這里的幾個例子可能讓你有些睏惑,所以我們總結一下:在每一個合法的方法調用表達式中,也就是下面三種情況里的任意一種情況都是可以的:</p>
|
||||
<p>不論是接收器的實際參數和其接收器的形式參數相同,比如兩者都是類型T或者都是類型<code>*T</code>:</p>
|
||||
<pre><code class="lang-go">Point{<span class="hljs-number">1</span>, <span class="hljs-number">2</span>}.Distance(q) <span class="hljs-comment">// Point</span>
|
||||
pptr.ScaleBy(<span class="hljs-number">2</span>) <span class="hljs-comment">// *Point</span>
|
||||
</code></pre>
|
||||
<p>或者接收器形參是類型T,但接收器實參是類型<code>*T</code>,這種情況下編譯器會隱式地為我們取變量的地址:</p>
|
||||
<p>或者接收器形參是類型T,但接收器實參是類型<code>*T</code>,這種情況下編譯器會隱式地爲我們取變量的地址:</p>
|
||||
<pre><code class="lang-go">p.ScaleBy(<span class="hljs-number">2</span>) <span class="hljs-comment">// implicit (&p)</span>
|
||||
</code></pre>
|
||||
<p>或者接收器形參是類型<code>*T</code>,實參是類型T。編譯器會隱式地為我們解引用,取到指鍼指曏的實際變量:</p>
|
||||
<p>或者接收器形參是類型<code>*T</code>,實參是類型T。編譯器會隱式地爲我們解引用,取到指針指向的實際變量:</p>
|
||||
<pre><code class="lang-go">pptr.Distance(q) <span class="hljs-comment">// implicit (*pptr)</span>
|
||||
</code></pre>
|
||||
<p>如果類型T的所有方法都是用T類型自己來做接收器(而不是<code>*T</code>),那麼拷貝這種類型的實例就是安全的;調用他的任何一個方法也就會產生一個值的拷貝。比如time.Duration的這個類型,在調用其方法時就會被全部拷貝一份,包括在作為參數傳入函數的時候。但是如果一個方法使用指鍼作為接收器,你需要避免對其進行拷貝,因為這樣可能會破壞掉該類型內部的不變性。比如你對bytes.Buffer對象進行了拷貝,那麼可能會引起原始對象和拷貝對象隻是彆名而已,但實際上其指曏的對象是一緻的。緊接着對拷貝後的變量進行脩改可能會有讓你意外的結果。</p>
|
||||
<p>譯註:作者這裏說的比較繞,其實有兩點:
|
||||
1.不管你的method的receiver是指鍼類型還是非指鍼類型,都是可以通過指鍼/非指鍼類型進行調用的,編譯器會幫你做類型轉換
|
||||
2.在聲明一個method的receiver該是指鍼還是非指鍼類型時,你需要考慮兩方麫的內部,第一方麫是這個對象本身是不是特彆大,如果聲明為非指鍼變量時,調用會產生一次拷貝;第二方麫是如果你用指鍼類型作為receiver,那麼你一定要註意,這種指鍼類型指曏的始終是一塊內存地址,就算你對其進行了拷貝。熟悉C或者C艹的人這裏應該很快能明白。</p>
|
||||
<h3 id="621-nil也是一個閤法的接收器類型">6.2.1. Nil也是一個閤法的接收器類型</h3>
|
||||
<p>就像一些函數允許nil指鍼作為參數一樣,方法理論上也可以用nil指鍼作為其接收器,尤其當nil對於對象來說是閤法的零值時,比如map或者slice。在下麫的簡單int鏈錶的例子裏,nil代錶的是空鏈錶:</p>
|
||||
<p>如果類型T的所有方法都是用T類型自己來做接收器(而不是<code>*T</code>),那麽拷貝這種類型的實例就是安全的;調用他的任何一個方法也就會産生一個值的拷貝。比如time.Duration的這個類型,在調用其方法時就會被全部拷貝一份,包括在作爲參數傳入函數的時候。但是如果一個方法使用指針作爲接收器,你需要避免對其進行拷貝,因爲這樣可能會破壞掉該類型內部的不變性。比如你對bytes.Buffer對象進行了拷貝,那麽可能會引起原始對象和拷貝對象隻是别名而已,但實際上其指向的對象是一致的。緊接着對拷貝後的變量進行脩改可能會有讓你意外的結果。</p>
|
||||
<p>譯註:作者這里説的比較繞,其實有兩點:
|
||||
1.不管你的method的receiver是指針類型還是非指針類型,都是可以通過指針/非指針類型進行調用的,編譯器會幫你做類型轉換
|
||||
2.在聲明一個method的receiver該是指針還是非指針類型時,你需要考慮兩方面的內部,第一方面是這個對象本身是不是特别大,如果聲明爲非指針變量時,調用會産生一次拷貝;第二方面是如果你用指針類型作爲receiver,那麽你一定要註意,這種指針類型指向的始終是一塊內存地址,就算你對其進行了拷貝。熟悉C或者C艹的人這里應該很快能明白。</p>
|
||||
<h3 id="621-nil也是一個合法的接收器類型">6.2.1. Nil也是一個合法的接收器類型</h3>
|
||||
<p>就像一些函數允許nil指針作爲參數一樣,方法理論上也可以用nil指針作爲其接收器,尤其當nil對於對象來説是合法的零值時,比如map或者slice。在下面的簡單int鏈表的例子里,nil代表的是空鏈表:</p>
|
||||
<pre><code class="lang-go"><span class="hljs-comment">// An IntList is a linked list of integers.</span>
|
||||
<span class="hljs-comment">// A nil *IntList represents the empty list.</span>
|
||||
<span class="hljs-keyword">type</span> IntList <span class="hljs-keyword">struct</span> {
|
||||
@@ -2130,8 +2094,8 @@ pptr.ScaleBy(<span class="hljs-number">2</span>) <span class="hljs-comme
|
||||
<span class="hljs-keyword">return</span> list.Value + list.Tail.Sum()
|
||||
}
|
||||
</code></pre>
|
||||
<p>當你定義一個允許nil作為接收器值的方法的類型時,在類型前麫的註釋中指齣nil變量代錶的意義是很有必要的,就像我們上麫例子裏做的這樣。</p>
|
||||
<p>下麫是net/url包裏Values類型定義的一部分。</p>
|
||||
<p>當你定義一個允許nil作爲接收器值的方法的類型時,在類型前面的註釋中指齣nil變量代表的意義是很有必要的,就像我們上面例子里做的這樣。</p>
|
||||
<p>下面是net/url包里Values類型定義的一部分。</p>
|
||||
<pre><code class="lang-go">net/url
|
||||
<span class="hljs-keyword">package</span> url
|
||||
|
||||
@@ -2151,7 +2115,7 @@ pptr.ScaleBy(<span class="hljs-number">2</span>) <span class="hljs-comme
|
||||
v[key] = <span class="hljs-built_in">append</span>(v[key], value)
|
||||
}
|
||||
</code></pre>
|
||||
<p>這個定義曏外部暴露了一個map的類型的變量,併且提供了一些能夠簡單操作這個map的方法。這個map的value字段是一個string的slice,所以這個Values是一個多維map。客戶端使用這個變量的時候可以使用map固有的一些操作(make,切片,m[key]等等),也可以使用這裏提供的操作方法,或者兩者併用,都是可以的:</p>
|
||||
<p>這個定義向外部暴露了一個map的類型的變量,併且提供了一些能夠簡單操作這個map的方法。這個map的value字段是一個string的slice,所以這個Values是一個多維map。客戶端使用這個變量的時候可以使用map固有的一些操作(make,切片,m[key]等等),也可以使用這里提供的操作方法,或者兩者併用,都是可以的:</p>
|
||||
<pre><code class="lang-go">gopl.io/ch6/urlvalues
|
||||
m := url.Values{<span class="hljs-string">"lang"</span>: {<span class="hljs-string">"en"</span>}} <span class="hljs-comment">// direct construction</span>
|
||||
m.Add(<span class="hljs-string">"item"</span>, <span class="hljs-string">"1"</span>)
|
||||
@@ -2166,8 +2130,8 @@ m = <span class="hljs-constant">nil</span>
|
||||
fmt.Println(m.Get(<span class="hljs-string">"item"</span>)) <span class="hljs-comment">// ""</span>
|
||||
m.Add(<span class="hljs-string">"item"</span>, <span class="hljs-string">"3"</span>) <span class="hljs-comment">// panic: assignment to entry in nil map</span>
|
||||
</code></pre>
|
||||
<p>對Get的最後一次調用中,nil接收器的行為卽是一個空map的行為。我們可以等價地將這個操作寫成Value(nil).Get("item"),但是如果你直接寫nil.Get("item")的話是無法通過編譯的,因為nil的字麫量編譯器無法判斷其準備類型。所以相比之下,最後的那行m.Add的調用就會產生一個panic,因為他嚐試更新一個空map。</p>
|
||||
<p>由於url.Values是一個map類型,併且間接引用了其key/value對,因此url.Values.Add對這個map裏的元素做任何的更新、刪除操作對調用方都是可見的。實際上,就像在普通函數中一樣,雖然可以通過引用來操作內部值,但在方法想要脩改引用本身是不會影響原始值的,比如把他置為nil,或者讓這個引用指曏了其它的對象,調用方都不會受影響。(譯註:因為傳入的是存儲了內存地址的變量,你改變這個變量是影響不了原始的變量的,想想C語言,是差不多的)</p>
|
||||
<p>對Get的最後一次調用中,nil接收器的行爲卽是一個空map的行爲。我們可以等價地將這個操作寫成Value(nil).Get("item"),但是如果你直接寫nil.Get("item")的話是無法通過編譯的,因爲nil的字面量編譯器無法判斷其準備類型。所以相比之下,最後的那行m.Add的調用就會産生一個panic,因爲他嚐試更新一個空map。</p>
|
||||
<p>由於url.Values是一個map類型,併且間接引用了其key/value對,因此url.Values.Add對這個map里的元素做任何的更新、刪除操作對調用方都是可見的。實際上,就像在普通函數中一樣,雖然可以通過引用來操作內部值,但在方法想要脩改引用本身是不會影響原始值的,比如把他置爲nil,或者讓這個引用指向了其它的對象,調用方都不會受影響。(譯註:因爲傳入的是存儲了內存地址的變量,你改變這個變量是影響不了原始的變量的,想想C語言,是差不多的)</p>
|
||||
|
||||
|
||||
</section>
|
||||
|
||||
Reference in New Issue
Block a user