Hero Image of content
세 번째 배우는 Go - 파트 2: Tour of Go를 학습하며 남기는 기록과 예제

패키지

모든 프로그램은 main 패키지에서 실행을 시작한다.

패키지를 import하고 import한 패키지 이름(경로)의 마지막 부분을 네임스페이스처럼 그 패키지 안에 있는 메서드를 호출할 때 사용된다.
아래 예제 코드에서 import한 math/rand 패키지 이름에서는 rand 부분을 사용한다.

package main

import (
  "fmt"
  "math/rand"
)

func main() {
  rand.Seed(100)
  fmt.Println("내가 제일 좋아하는 숫자는 ", rand.Intn(10))
}

import할 때 괄호로 그룹을 짓는 형태를 factored import라고 하며, 이 스타일을 사용하는 것을 추천한다.
import된 패키지에서는 대문자로 시작하는 이름을 가진 함수와 변수만 접근할 수 있다.

함수

함수의 매개 변수가 2개 이상이며 같은 Type인 경우 마지막 매개변수 뒤에만 Type을 선언하면 된다.

func add(x, y int) int {
  return x + y
}

그리고 함수는 여러개의 결과를 반환할 수 있다.

func swap(x, y string) (string, string, string) {
  return y, x, "good"
}
func split(sum int) (x, y int) {
  x = sum * 4 / 9
  y = sum - x
  return
}

변수

변수를 선언할 때는 var키워드를 먼저 사용하고, 변수명 다음에 Type을 명시한다.
초기값을 지정할 경우 Type을 생략할 수 있다.

var i, j int = 1, 2
var c, python, java = true, false, "no!"
func main() {
  var i, j int = 1, 2
  k := 3
  c, python, java := true, false, "no!"
}
  • 숫자 Type: 0
  • Boolean Type: false
  • String Type: “"(빈 문자열)

기존 Type을 다른 Type으로 변환하려면 변수를 Type의 이름으로 감싸면 된다.

var i int = 42
var f float64 = float64(i)

숫자를 := 또는 지정한 값에 의해 Type이 유추될 경우 그 정확도에 따라 int, float64, complex128이 된다.

func main() {
  i := 42
  f := 3.142
  g := 0.867 + 0.5i
  
  fmt.Printf("i의 타입은 %T\n", i)
  fmt.Printf("f의 타입은 %T\n", f)
  fmt.Printf("g의 타입은 %T\n", g)
}

상수는 var 대신 const 키워드로 선언하며 :=은 사용할 수 없다.

For

Go에서 반복문은 오직 for 하나만 존재한다.
for문의 구조는 ;으로 구분되는 세 가지 구성 요소를 갖는다.

  1. 초기화 구문;
  2. ;반복 조건 표현식;
  3. ;반복 후 구문

‘초기화 구문’에는 짧은 변수 형태로 선언할 수 있는데 이 변수는 for문안에서만 사용할 수 있다.
‘반복 조건 표현식’의 평가 결과가 false가 되면 반복은 종료된다.

func main() {
  sum := 0
  for i := 0; i < 10; i++ {
    sum += i
  }
  fmt.Println(sum)
}

for문의 각 구성 요소는 필수가 아니다.
그래서 ‘반복 조건 표현식’만 사용한다면, while처럼 사용될 수 있고 전부 생략할 경우 무한 루프를 의미한다.

for sum < 1000 {
  sum += sum
}

for {
  fmt.Println("무한도전")
}

If

Go의 if는 소괄호를 사용하지 않는다는 점에서 for와 비슷하다.
그리고 if문 조건식에도 짧은 변수를 선언할 수 있고, for문과 마찬가지로 이 변수도 if문안에서만 사용할 수 있다.

func pow(x, n, lim float64) float64 {
  if v := math.Pow(x, n); v < lim {
    return v
  }
  return lim
}

Switch

if-else 구문을 반복해서 사용할 경우, switch 구문을 사용할 수 있다.
Go의 switch는 다른 언어와 다르게 마지막 case까지 실행하지 않고, 먼저 선택된 case 조건만 실행하고 멈춘다. (자동 break)
즉, switch는 위에서 아래로 case를 평가해서 성공하는 case에서 바로 멈추는 방식이다.

today := time.Now().Weekday()

switch time.Saturday {
case today + 0:
  fmt.Println("오늘")
case today + 1:
  fmt.Println("내일")
case today + 2:
  fmt.Println("이틀 후")
default:
  fmt.Println("너무 나중이네")
}

switch의 조건문에 꼭 상수만 쓸 수 있는 것은 아니다. 아래와 같이 짧은 변수를 선언하고, 즉시 사용할 수 있다.

switch os := runtime.GOOS; os {
case "darwin":
  fmt.Println("MAC OS")
case "linux":
  fmt.Println("Linux")
default:
  fmt.Printf("%s \n", os)
}

switch의 조건문은 생략할 수도 있다.

switch {
case today + 0 == time.Saturday:
  fmt.Println("오늘")
case today + 1 == time.Saturday:
  fmt.Println("내일")
default:
  fmt.Println("너무 나중이네")
}

Defer

defer문은 자신이 선언된 함수가 종료할 때까지 실행을 연기한다.
함수 안에서 defer를 여러번 사용하면 스택에 쌓이고, 함수가 종료될 때 LIFO(Last In, First Out) 순서로 실행된다.

func main() {
  fmt.Println("counting")
  for i := 0; i < 10; i++ {
    defer fmt.Println(i)
  }
  fmt.Println("done")
}

Pointers

Go는 변수의 메모리 주소를 가리키는 포인터를 지원한다.
*T 라는 타입으로 변수를 선언하면 T 타입의 값을 가리키는 포인터가 된다.
포인터의 Zero value는 nil이다.

var p *int

기존 변수명에 &연산자를 앞에 붙이면 이 값에 대한 포인터를 생성한다.

i := 100
p = &i

반대로 포인터 변수에 대해 *연산자를 앞에 붙이면 이 포인터가 가리키는 실제 값을 나타낸다.

fmt.Println(*p)
*p = 101

Structs

Struct는 필드의 집합이며, 필드는 .(dot)으로 접근할 수 있다.
struct 타입을 선언할 때 필드의 이름을 직접 명시해서 값을 할당할 수 있다. 값을 지정하지 않은 필드는 Zero value로 초기화된다.
struct을 선언할 때부터 접두사 &을 붙이면 자동으로 struct 값의 포인터가 반환된다.

type Vertex struct {
  X, Y int
}

var (
  v1 = Vertex{1, 2}
  v1 = Vertex{X: 1}
  v3 = Vertex{}
  p = &Vertex{1, 2}
)

Array

배열을 선언할 때 [n]T 형태로 Type을 지정한다. 이것은 T라는 타입의 값이 n개 있는 배열이라는 의미를 갖는다.
배열 길이(length)는 그 타입의 일부라서 한번 선언된 배열의 길이는 조정할 수 없다.

var a [10]int

Slice

배열의 길이(length)는 선언할 때 고정되지만, 이 배열에 대한 슬라이스를 만들어서 배열의 요소와 길이를 가변적으로 다룰 수 있다.
[]T 타입같이 슬라이스를 선언하면, T라는 타입의 슬라이스를 의미한다.

슬라이스 표현식은 아래와 같은 형태를 가지는데, 첫번째 요소(low)는 슬라이스의 시작 요소이며, 두번째 요소(high)를 제외하는 범위에 해당한다.
만약 low, high를 생략하게 되면 low의 기본값은 0이고, high의 기본값은 배열의 길이가 된다.

a[low : high]
primes := [6]int{1, 2, 3, 4, 5, 6}
var s []int = primes[1:]
names := [4]string{
  "john",
  "paul",
  "george",
  "ringo",
}

a := names[0:2]
b := names[1:3]
fmt.Println(names)  // [john paul] [paul george]

b[0] = "jonnung"
fmt.Println(names)  // [john jonnung george ringo]

당연히(?) 슬라이스에 슬라이스를 포함 시킬 수 있다. (2차원 배열처럼)

board := [][]string{
  []string{"_", "_", "_"},
  []string{"_", "_", "_"},
  []string{"_", "_", "_"),
}

Slice literals

슬라이스 리터럴 표현식은 배열 리터럴과 거의 같고, 단지 길이를 지정하지 않는다.

[]bool{true, true, false}

Slice length & capacity

슬라이스는 길이(length)와 용량(capacity)를 갖는다.
슬라이스의 길이는 슬라이스가 가리키는 범위에 해당되는 요소의 개수이다.
슬라이스의 용량은 슬라이스가 가리키는 원본 배열 요소의 개수이다.

s := []int{1, 2, 3, 4, 5}

fmt.Printf("Length: %d\n", len(s))  // 5
fmt.Printf("Capacity: %d", cap(s))  // 5

슬라이스의 Zero Value는 nil이며, nil 슬라이스의 길이와 용량은 0이다.

make로 슬라이스 만들기

내장 함수 make로 동적 크기의 슬라이스를 만들 수 있다.
make함수의 첫번째 인자는 슬라이스, 두번째 인자는 길이(length), 세번째 인자는 용량(capacity)을 지정할 수 있다.

a := make([]int, 5, 5) // len(a)=5, cap(a)=5

make는 슬라이스 리터럴이나 원본 배열에서 슬라이스를 생성하는 경우가 아닌 길이(length)와 용량(capacity)을 예상하거나 결정된 상태에서 슬라이스를 만들어야 하는 경우 사용하면 좋을 것 같다.

슬라이스에 요소 추가하기

내장 함수 append로 슬라이스에 새로운 요소를 추가할 수 있다.
append의 첫번째 인자는 대상 슬라이스 변수, 두번째 인자부터 추가할 값이다.
append의 호출 결과는 주어진 슬라이스의 모든 요소와 추가한 값을 포함하는 새로운 슬라이스가 된다.

var s []int  // len(s)=0, cap(s)=0

s = append(s, 0)  // [0], len(s)=1, cap(s)=1

Map

맵은 Key와 Value의 쌍이다.
Map으로 선언된 변수는 nil 이고, Nil Map에는 값을 넣을 수 없다.
그래서 Nil Map을 쓰기 위해 make라는 내장함수로 초기화를 해야한다. 그래야 Empty Map 된다.

var m map[string]string
fmt.Println(m)  // map[]
// m["jonnung"] = "zzang" // panic: assignment to entry in nil map

m = make(map[string]string)
m["jonnung"] = "zzang"

fmt.Println(m)  // map[jonnung:zzang]

좀 더 간편하게 Map을 사용하기 위해 리터럴을 이용할 수 있다.
Map 리터럴은 Struct 리터럴과 비슷하지만 Key를 갖는다.

var m = map[string]string{
  "foo": "bar"
}  // map[foo:bar]

Map 다루기

// Map에 요소를 추가하거나 업데이트하기
m[key] = elem

// 요소 찾기
elem = m[key]

// 요소 제거하기
delete(m, key)

// 두 개의 변수에 Map 값을 할당해서 Key가 존재하는지 확인할 수 있다.
// Map에 key가 없다면, ok는 false이며, elem은 Map요소의 타입의 Zero value가 된다.
elem, ok = m[key]

Range

for문에 range를 붙이면 슬라이스와 맵을 순회할 수 있다.
range를 사용하면, 각 반복할 때마다 두 개의 값이 반환되는 데 첫 번째는 인덱스, 두 번째는 그 순서의 값이 복사된다.

var pow = []int{1, 2, 3, 4, 5}

for i, v := range pow {
  fmt.Printf("Index: %d, Value: %d \n", i, v)
}

// Index: 0, Value: 1 
// Index: 1, Value: 2 
// Index: 2, Value: 3 
// Index: 3, Value: 4 
// Index: 4, Value: 5 

range가 반환하는 인덱스와 값 중에 사용하지 않으려면 _에 할당해서 생략할 수 있다.

Closure

Go 함수는 함수의 인자가 될 수 있고, 함수의 반환값이 될 수 있다.
이 특징 때문에 함수 안에서 선언된 함수는 클로저가 될 수 있다.

Method

Go는 클래스가 없다. 하지만 Type(Struct 포함)에 메소드를 정의할 수 있다. 메소드는 단지 receiver 라는 인자가 있는 함수일 뿐이다.
이 receiver는 func키워드와 메소드명 사이에 추가된다.

type Vertex struct {
  X, Y float64
}

func (v Vertex) Abs() float64 {
  return math.Sqrt(v.X*v.X + v.Y*v.Y)
}
type MyFloat float64

func (f MyFloat) Abs() float64 {
  if f < 0 {
    return float64(-f)
  }
  return float64(f)
}

Pointer Receiver

포인터를 리시버로 사용할 수도 있다.
만약 리시버를 포인터로 선언하면 *Vertex같은 형식이 된다.

포인터 리시버가 있는 메소드는 리시버가 가리키는 값을 수정할 수 있다.
그렇기 때문에 포인터 리시버가 (그냥) 값 리시버보다 더 자주 쓰이는 편이다.
(값 리시버는 메소드 안에서 원본의 복사본이기 때문이다.)

type Vertex struct {
  X, Y float64
}

func (v *Vertex) Scale(f float64) {
  v.X = v.X * f
  v.Y = v.Y * f
}
var v Vertex
v.Scale(5)  // (&v).Scale(5) 하는 것과 같다.

p := &v
p.Scale(5)

포인터 리시버를 사용하는 이유는 두 가지가 있다.

  1. 메소드 안에서 리시버가 가리키는 값을 수정하기 위해
  2. 각 메소드가 호출될 때마다 값이 복사되는 문제를 피하기 위해

Interface

인터페이스는 메소드의 집합이다.
인터페이스의 메소드를 구현하는 모든 것들은 인터페이스 Type이 될 수 있다.
인터페이스 Type을 사용한다는 것은 명시적인 implementation 같은 키워드를 사용하지 않더라도 인터페이스 Type이 가진 메소드를 구현함으로써 이 인터페이스를 구현했다는 것을 의미한다.

인터페이스 Type의 값이 nil인 경우, 그 메소드는 nil 리시버로 호출된다.
아래 경우 변수inil이 아니지만, tnil이다.

type I interface {
  M()
}

type T struct {
  S string
}

func (t *T) M() {
  if t == nil {
    fmt.Println("<nil>")
    return
  }
  fmt.Println(t.S)
}

func main() {
	var i I  // 인터페이스 값
	// i.M()  // 런타임 에러

	t := &T{"It works"}
	i = t
	i.M()
}

Go에서 nil 리시버로 메소드가 호출되는 것은 예외를 발생시키지 않지만, 인터페이스의 값이 nil인 변수의 메소드를 호출하면 런타임 에러가 발생한다. (당연히 구현되지 않은 메소드이므로 호출할 수는 없음)

빈 인터페이스

메소드가 없는 인터페이스 유형을 empty interface라고 한다.

interface{}

빈 인터페이스는 모든 Type의 값을 받을 수 있어서 알 수 없는 값을 처리하는데 이용된다.

타입 체크

빈 인터페이스로 선언된 값의 타입을 체크할 때 아래와 같은 방법을 사용할 수 있다.

var i interface{} = "hello"

s1 := i.(string)
fmt.Println(s1)  // hello

s2, ok := i.(float64)
if ok == true {
  fmt.Println(s2)
} else {
  fmt.Println("errrr")  // errrr
}

타입 Switch 문

Type Switch문은 일반 Switch문에 Type을 명시해서 빈 인터페이스로 전달된 인자의 Type을 비교할 때 사용한다.

func do(i interface{}) {
  switch v := i.(type) {
  case int:
    fmt.Println("Int")
  case string:
    fmt.Println("String")
  default:
    fmt.Println("I don't know")
  }
}

Errors

error Type은 fmt.Stringer와 유사한 내장 인터페이스이다.

type error interface {
  Error() string
}
comments powered by Disqus