mirror of
				https://github.com/golang/go.git
				synced 2025-10-31 16:50:58 +00:00 
			
		
		
		
	 8655f04d8e
			
		
	
	
		8655f04d8e
		
	
	
	
	
		
			
			Related to issue #9228 Change-Id: I0819e657f6393788754d1412f9c2126a170d4cf1 Reviewed-on: https://go-review.googlesource.com/1577 Reviewed-by: Rob Pike <r@golang.org>
		
			
				
	
	
		
			632 lines
		
	
	
	
		
			16 KiB
		
	
	
	
		
			HTML
		
	
	
	
	
	
			
		
		
	
	
			632 lines
		
	
	
	
		
			16 KiB
		
	
	
	
		
			HTML
		
	
	
	
	
	
| <!--{
 | |
| 	"Title": "How to Write Go Code"
 | |
| }-->
 | |
| 
 | |
| <h2 id="Introduction">Introduction</h2>
 | |
| 
 | |
| <p>
 | |
| This document demonstrates the development of a simple Go package and
 | |
| introduces the <a href="/cmd/go/">go tool</a>, the standard way to fetch,
 | |
| build, and install Go packages and commands.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| The <code>go</code> tool requires you to organize your code in a specific
 | |
| way. Please read this document carefully.
 | |
| It explains the simplest way to get up and running with your Go installation.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| A similar explanation is available as a
 | |
| <a href="//www.youtube.com/watch?v=XCsL89YtqCs">screencast</a>.
 | |
| </p>
 | |
| 
 | |
| 
 | |
| <h2 id="Organization">Code organization</h2>
 | |
| 
 | |
| <h3 id="Workspaces">Workspaces</h3>
 | |
| 
 | |
| <p>
 | |
| The <code>go</code> tool is designed to work with open source code maintained
 | |
| in public repositories. Although you don't need to publish your code, the model
 | |
| for how the environment is set up works the same whether you do or not.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| Go code must be kept inside a <i>workspace</i>.
 | |
| A workspace is a directory hierarchy with three directories at its root:
 | |
| </p>
 | |
| 
 | |
| <ul>
 | |
| <li><code>src</code> contains Go source files organized into packages (one package per directory),
 | |
| <li><code>pkg</code> contains package objects, and
 | |
| <li><code>bin</code> contains executable commands.
 | |
| </ul>
 | |
| 
 | |
| <p>
 | |
| The <code>go</code> tool builds source packages and installs the resulting
 | |
| binaries to the <code>pkg</code> and <code>bin</code> directories.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| The <code>src</code> subdirectory typically contains multiple version control
 | |
| repositories (such as for Git or Mercurial) that track the development of one
 | |
| or more source packages.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| To give you an idea of how a workspace looks in practice, here's an example:
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| bin/
 | |
|     hello                          # command executable
 | |
|     outyet                         # command executable
 | |
| pkg/
 | |
|     linux_amd64/
 | |
|         github.com/golang/example/
 | |
|             stringutil.a           # package object
 | |
| src/
 | |
|     <a href="https://github.com/golang/example/">github.com/golang/example/</a>
 | |
|         .git/                      # Git repository metadata
 | |
| 	hello/
 | |
| 	    hello.go               # command source
 | |
| 	outyet/
 | |
| 	    main.go                # command source
 | |
| 	    main_test.go           # test source
 | |
| 	stringutil/
 | |
| 	    reverse.go             # package source
 | |
| 	    reverse_test.go        # test source
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| This workspace contains one repository (<code>example</code>)
 | |
| comprising two commands (<code>hello</code> and <code>outyet</code>)
 | |
| and one library (<code>stringutil</code>).
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| A typical workspace would contain many source repositories containing many
 | |
| packages and commands. Most Go programmers keep <i>all</i> their Go source code
 | |
| and dependencies in a single workspace.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| Commands and libraries are built from different kinds of source packages.
 | |
| We will discuss the distinction <a href="#PackageNames">later</a>.
 | |
| </p>
 | |
| 
 | |
| 
 | |
| <h3 id="GOPATH">The <code>GOPATH</code> environment variable</h3>
 | |
| 
 | |
| <p>
 | |
| The <code>GOPATH</code> environment variable specifies the location of your
 | |
| workspace. It is likely the only environment variable you'll need to set
 | |
| when developing Go code.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| To get started, create a workspace directory and set <code>GOPATH</code>
 | |
| accordingly. Your workspace can be located wherever you like, but we'll use
 | |
| <code>$HOME/work</code> in this document. Note that this must <b>not</b> be the
 | |
| same path as your Go installation.
 | |
| (Another common setup is to set <code>GOPATH=$HOME</code>.)
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| $ <b>mkdir $HOME/work</b>
 | |
| $ <b>export GOPATH=$HOME/work</b>
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| For convenience, add the workspace's <code>bin</code> subdirectory
 | |
| to your <code>PATH</code>:
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| $ <b>export PATH=$PATH:$GOPATH/bin</b>
 | |
| </pre>
 | |
| 
 | |
| 
 | |
| <h3 id="PackagePaths">Package paths</h3>
 | |
| 
 | |
| <p>
 | |
| The packages from the standard library are given short paths such as
 | |
| <code>"fmt"</code> and <code>"net/http"</code>.
 | |
| For your own packages, you must choose a base path that is unlikely to
 | |
| collide with future additions to the standard library or other external
 | |
| libraries.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| If you keep your code in a source repository somewhere, then you should use the
 | |
| root of that source repository as your base path.
 | |
| For instance, if you have a <a href="https://github.com/">GitHub</a> account at
 | |
| <code>github.com/user</code>, that should be your base path.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| Note that you don't need to publish your code to a remote repository before you
 | |
| can build it. It's just a good habit to organize your code as if you will
 | |
| publish it someday. In practice you can choose any arbitrary path name,
 | |
| as long as it is unique to the standard library and greater Go ecosystem.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| We'll use <code>github.com/user</code> as our base path. Create a directory
 | |
| inside your workspace in which to keep source code:
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| $ <b>mkdir -p $GOPATH/src/github.com/user</b>
 | |
| </pre>
 | |
| 
 | |
| 
 | |
| <h3 id="Command">Your first program</h3>
 | |
| 
 | |
| <p>
 | |
| To compile and run a simple program, first choose a package path (we'll use
 | |
| <code>github.com/user/hello</code>) and create a corresponding package directory
 | |
| inside your workspace:
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| $ <b>mkdir $GOPATH/src/github.com/user/hello</b>
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| Next, create a file named <code>hello.go</code> inside that directory,
 | |
| containing the following Go code.
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| package main
 | |
| 
 | |
| import "fmt"
 | |
| 
 | |
| func main() {
 | |
| 	fmt.Printf("Hello, world.\n")
 | |
| }
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| Now you can build and install that program with the <code>go</code> tool:
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| $ <b>go install github.com/user/hello</b>
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| Note that you can run this command from anywhere on your system. The
 | |
| <code>go</code> tool finds the source code by looking for the
 | |
| <code>github.com/user/hello</code> package inside the workspace specified by
 | |
| <code>GOPATH</code>.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| You can also omit the package path if you run <code>go install</code> from the
 | |
| package directory:
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| $ <b>cd $GOPATH/src/github.com/user/hello</b>
 | |
| $ <b>go install</b>
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| This command builds the <code>hello</code> command, producing an executable
 | |
| binary. It then installs that binary to the workspace's <code>bin</code>
 | |
| directory as <code>hello</code> (or, under Windows, <code>hello.exe</code>).
 | |
| In our example, that will be <code>$GOPATH/bin/hello</code>, which is
 | |
| <code>$HOME/work/bin/hello</code>.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| The <code>go</code> tool will only print output when an error occurs, so if
 | |
| these commands produce no output they have executed successfully.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| You can now run the program by typing its full path at the command line:
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| $ <b>$GOPATH/bin/hello</b>
 | |
| Hello, world.
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| Or, as you have added <code>$GOPATH/bin</code> to your <code>PATH</code>,
 | |
| just type the binary name:
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| $ <b>hello</b>
 | |
| Hello, world.
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| If you're using a source control system, now would be a good time to initialize
 | |
| a repository, add the files, and commit your first change. Again, this step is
 | |
| optional: you do not need to use source control to write Go code.
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| $ <b>cd $GOPATH/src/github.com/user/hello</b>
 | |
| $ <b>git init</b>
 | |
| Initialized empty Git repository in /home/user/work/src/github.com/user/hello/.git/
 | |
| $ <b>git add hello.go</b>
 | |
| $ <b>git commit -m "initial commit"</b>
 | |
| [master (root-commit) 0b4507d] initial commit
 | |
|  1 file changed, 1 insertion(+)
 | |
|   create mode 100644 hello.go
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| Pushing the code to a remote repository is left as an exercise for the reader.
 | |
| </p>
 | |
| 
 | |
| 
 | |
| <h3 id="Library">Your first library</h3>
 | |
| 
 | |
| <p>
 | |
| Let's write a library and use it from the <code>hello</code> program.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| Again, the first step is to choose a package path (we'll use
 | |
| <code>github.com/user/stringutil</code>) and create the package directory:
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| $ <b>mkdir $GOPATH/src/github.com/user/stringutil</b>
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| Next, create a file named <code>reverse.go</code> in that directory with the
 | |
| following contents.
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| // Package stringutil contains utility functions for working with strings.
 | |
| package stringutil
 | |
| 
 | |
| // Reverse returns its argument string reversed rune-wise left to right.
 | |
| func Reverse(s string) string {
 | |
| 	r := []rune(s)
 | |
| 	for i, j := 0, len(r)-1; i < len(r)/2; i, j = i+1, j-1 {
 | |
| 		r[i], r[j] = r[j], r[i]
 | |
| 	}
 | |
| 	return string(r)
 | |
| }
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| Now, test that the package compiles with <code>go build</code>:
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| $ <b>go build github.com/user/stringutil</b>
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| Or, if you are working in the package's source directory, just:
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| $ <b>go build</b>
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| This won't produce an output file. To do that, you must use <code>go
 | |
| install</code>, which places the package object inside the <code>pkg</code>
 | |
| directory of the workspace.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| After confirming that the <code>stringutil</code> package builds,
 | |
| modify your original <code>hello.go</code> (which is in
 | |
| <code>$GOPATH/src/github.com/user/hello</code>) to use it:
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| package main
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 
 | |
| 	<b>"github.com/user/stringutil"</b>
 | |
| )
 | |
| 
 | |
| func main() {
 | |
| 	fmt.Printf(stringutil.Reverse("!oG ,olleH"))
 | |
| }
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| Whenever the <code>go</code> tool installs a package or binary, it also
 | |
| installs whatever dependencies it has.
 | |
| So when you install the <code>hello</code> program
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| $ <b>go install github.com/user/hello</b>
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| the <code>stringutil</code> package will be installed as well, automatically.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| Running the new version of the program, you should see a new, reversed message:
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| $ <b>hello</b>
 | |
| Hello, Go!
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| After the steps above, your workspace should look like this:
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| bin/
 | |
|     hello                 # command executable
 | |
| pkg/
 | |
|     linux_amd64/          # this will reflect your OS and architecture
 | |
|         github.com/user/
 | |
|             stringutil.a  # package object
 | |
| src/
 | |
|     github.com/user/
 | |
|         hello/
 | |
|             hello.go      # command source
 | |
|         stringutil/
 | |
|             reverse.go    # package source
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| Note that <code>go install</code> placed the <code>stringutil.a</code> object
 | |
| in a directory inside <code>pkg/linux_amd64</code> that mirrors its source
 | |
| directory.
 | |
| This is so that future invocations of the <code>go</code> tool can find the
 | |
| package object and avoid recompiling the package unnecessarily.
 | |
| The <code>linux_amd64</code> part is there to aid in cross-compilation,
 | |
| and will reflect the operating system and architecture of your system.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| Go command executables are statically linked; the package objects need not
 | |
| be present to run Go programs.
 | |
| </p>
 | |
| 
 | |
| 
 | |
| <h3 id="PackageNames">Package names</h3>
 | |
| 
 | |
| <p>
 | |
| The first statement in a Go source file must be
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| package <i>name</i>
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| where <code><i>name</i></code> is the package's default name for imports.
 | |
| (All files in a package must use the same <code><i>name</i></code>.)
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| Go's convention is that the package name is the last element of the
 | |
| import path: the package imported as "<code>crypto/rot13</code>"
 | |
| should be named <code>rot13</code>.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| Executable commands must always use <code>package main</code>.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| There is no requirement that package names be unique
 | |
| across all packages linked into a single binary,
 | |
| only that the import paths (their full file names) be unique.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| See <a href="/doc/effective_go.html#names">Effective Go</a> to learn more about
 | |
| Go's naming conventions.
 | |
| </p>
 | |
| 
 | |
| 
 | |
| <h2 id="Testing">Testing</h2>
 | |
| 
 | |
| <p>
 | |
| Go has a lightweight test framework composed of the <code>go test</code>
 | |
| command and the <code>testing</code> package.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| You write a test by creating a file with a name ending in <code>_test.go</code>
 | |
| that contains functions named <code>TestXXX</code> with signature
 | |
| <code>func (t *testing.T)</code>.
 | |
| The test framework runs each such function;
 | |
| if the function calls a failure function such as <code>t.Error</code> or
 | |
| <code>t.Fail</code>, the test is considered to have failed.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| Add a test to the <code>stringutil</code> package by creating the file
 | |
| <code>$GOPATH/src/github.com/user/stringutil/reverse_test.go</code> containing
 | |
| the following Go code.
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| package stringutil
 | |
| 
 | |
| import "testing"
 | |
| 
 | |
| func TestReverse(t *testing.T) {
 | |
| 	cases := []struct {
 | |
| 		in, want string
 | |
| 	}{
 | |
| 		{"Hello, world", "dlrow ,olleH"},
 | |
| 		{"Hello, 世界", "界世 ,olleH"},
 | |
| 		{"", ""},
 | |
| 	}
 | |
| 	for _, c := range cases {
 | |
| 		got := Reverse(c.in)
 | |
| 		if got != c.want {
 | |
| 			t.Errorf("Reverse(%q) == %q, want %q", c.in, got, c.want)
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| Then run the test with <code>go test</code>:
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| $ <b>go test github.com/user/stringutil</b>
 | |
| ok  	github.com/user/stringutil 0.165s
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| As always, if you are running the <code>go</code> tool from the package
 | |
| directory, you can omit the package path:
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| $ <b>go test</b>
 | |
| ok  	github.com/user/stringutil 0.165s
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| Run <code><a href="/cmd/go/#hdr-Test_packages">go help test</a></code> and see the
 | |
| <a href="/pkg/testing/">testing package documentation</a> for more detail.
 | |
| </p>
 | |
| 
 | |
| 
 | |
| <h2 id="remote">Remote packages</h2>
 | |
| 
 | |
| <p>
 | |
| An import path can describe how to obtain the package source code using a
 | |
| revision control system such as Git or Mercurial. The <code>go</code> tool uses
 | |
| this property to automatically fetch packages from remote repositories.
 | |
| For instance, the examples described in this document are also kept in a
 | |
| Git repository hosted at GitHub
 | |
| <code><a href="https://github.com/golang/example">github.com/golang/example</a></code>.
 | |
| If you include the repository URL in the package's import path,
 | |
| <code>go get</code> will fetch, build, and install it automatically:
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| $ <b>go get github.com/golang/example/hello</b>
 | |
| $ <b>$GOPATH/bin/hello</b>
 | |
| Hello, Go examples!
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| If the specified package is not present in a workspace, <code>go get</code>
 | |
| will place it inside the first workspace specified by <code>GOPATH</code>.
 | |
| (If the package does already exist, <code>go get</code> skips the remote
 | |
| fetch and behaves the same as <code>go install</code>.)
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| After issuing the above <code>go get</code> command, the workspace directory
 | |
| tree should now look like this:
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| bin/
 | |
|     hello                           # command executable
 | |
| pkg/
 | |
|     linux_amd64/
 | |
|         github.com/golang/example/
 | |
|             stringutil.a            # package object
 | |
|         github.com/user/
 | |
|             stringutil.a            # package object
 | |
| src/
 | |
|     github.com/golang/example/
 | |
| 	.git/                       # Git repository metadata
 | |
|         hello/
 | |
|             hello.go                # command source
 | |
|         stringutil/
 | |
|             reverse.go              # package source
 | |
|             reverse_test.go         # test source
 | |
|     github.com/user/
 | |
|         hello/
 | |
|             hello.go                # command source
 | |
|         stringutil/
 | |
|             reverse.go              # package source
 | |
|             reverse_test.go         # test source
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| The <code>hello</code> command hosted at GitHub depends on the
 | |
| <code>stringutil</code> package within the same repository. The imports in
 | |
| <code>hello.go</code> file use the same import path convention, so the
 | |
| <code>go get</code> command is able to locate and install the dependent
 | |
| package, too.
 | |
| </p>
 | |
| 
 | |
| <pre>
 | |
| import "github.com/golang/example/stringutil"
 | |
| </pre>
 | |
| 
 | |
| <p>
 | |
| This convention is the easiest way to make your Go packages available for
 | |
| others to use.
 | |
| The <a href="//golang.org/wiki/Projects">Go Wiki</a>
 | |
| and <a href="//godoc.org/">godoc.org</a>
 | |
| provide lists of external Go projects.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| For more information on using remote repositories with the <code>go</code> tool, see
 | |
| <code><a href="/cmd/go/#hdr-Remote_import_paths">go help importpath</a></code>.
 | |
| </p>
 | |
| 
 | |
| 
 | |
| <h2 id="next">What's next</h2>
 | |
| 
 | |
| <p>
 | |
| Subscribe to the
 | |
| <a href="//groups.google.com/group/golang-announce">golang-announce</a>
 | |
| mailing list to be notified when a new stable version of Go is released.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| See <a href="/doc/effective_go.html">Effective Go</a> for tips on writing
 | |
| clear, idiomatic Go code.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| Take <a href="//tour.golang.org/">A Tour of Go</a> to learn the language
 | |
| proper.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| Visit the <a href="/doc/#articles">documentation page</a> for a set of in-depth
 | |
| articles about the Go language and its libraries and tools.
 | |
| </p>
 | |
| 
 | |
| 
 | |
| <h2 id="help">Getting help</h2>
 | |
| 
 | |
| <p>
 | |
| For real-time help, ask the helpful gophers in <code>#go-nuts</code> on the
 | |
| <a href="http://freenode.net/">Freenode</a> IRC server.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| The official mailing list for discussion of the Go language is
 | |
| <a href="//groups.google.com/group/golang-nuts">Go Nuts</a>.
 | |
| </p>
 | |
| 
 | |
| <p>
 | |
| Report bugs using the
 | |
| <a href="//golang.org/issue">Go issue tracker</a>.
 | |
| </p>
 |