# Fungsi

Sebuah fungsi dapat memiliki dan tidak memiliki argumen. Pada contoh di bawah ini sebuah fungsi `add` yang memiliki `2` parameter bertipe `int` dan mengembalikan nilai `int`&#x20;

```go
package main

import "fmt"

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

func main() {
	fmt.Println(add(5, 5))
}
```

![Output program menampilkan kembalian nilai dari fungsi add()](https://2969676661-files.gitbook.io/~/files/v0/b/gitbook-legacy-files/o/assets%2F-MQaVepPFjuLoNjbs6f8%2F-MSSzl2v73RQtvIS-LmE%2F-MST1AmlvaD_f44uX1cf%2FScreen%20Shot%202021-02-01%20at%2021.39.11.png?alt=media\&token=7e3b22d3-9108-4354-9f49-d6bd2e982a5e)

Jika dua atau lebih variabel dari parameter fungsi memiliki tipe data yang sama, maka kita bisa menggunakan cara yang lebih singkat seperti berikut ini:

```
x int, y int
```

to

```
x, y int
```

```go
package main

import "fmt"

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

func main() {
	fmt.Println(add(5, 5))
}
```

### Multiple Results

Biasanya sebuah fungsi hanya dapat mengembalikan 1 buah nilai namun pada Go, fungsi dapat mengembalikan lebih dari 1 nilai.

```go
package main

import "fmt"

func swap(x, y string) (string, string) {
	return y, x
}

func main() {
	a, b := swap("hello", "world")
	fmt.Println(a, b)
}
```

![Fungsi swap mengembalikan dua buah nilai](https://2969676661-files.gitbook.io/~/files/v0/b/gitbook-legacy-files/o/assets%2F-MQaVepPFjuLoNjbs6f8%2F-MSSzl2v73RQtvIS-LmE%2F-MST46V7s5BuDXBprxOn%2FScreen%20Shot%202021-02-01%20at%2021.52.06.png?alt=media\&token=67aece9e-a4f3-490c-a808-99f6ca135878)

Pada kode di atas kita melihat jika kita ingin mengembalikan nilai lebih dari 1, maka harus dideklarasikan juga tipe datanya.

### Fungsi Variadic

Go memiliki konsep **variadic function** atau pembuatan fungsi dengan parameter sejenis yang tak terbatas. Maksud **tak terbatas** disini adalah jumlah argumen yang dimasukkan ketika pemanggilan fungsi bisa berapa saja.

Parameter variadic memiliki sifat yang mirip dengan slice. Nilai dari parameter-parameter yang digunakan bertipe data sama, dan ditampung oleh sebuah variabel saja. Cara pengaksesan tiap datanya juga sama, dengan menggunakan index.

```go
package main

import "fmt"

func calculate(numbers ...int) (float64, int) {
	var total int = 0
	for _, number := range numbers {
		total += number
	}

	var avg = float64(total) / float64(len(numbers))
	return avg, total
}

func main() {
	var avg, total = calculate(5, 5, 6, 4)
	fmt.Println("Total dari hasil kalkulasi adalah", total)
	fmt.Printf("Rata-rata : %.2f\n", avg)
}
```

![Output dari fungsi calculate()](https://2969676661-files.gitbook.io/~/files/v0/b/gitbook-legacy-files/o/assets%2F-MQaVepPFjuLoNjbs6f8%2F-MST6Uilf705OQmTZqJn%2F-MST82eZD9eiuBMMJJzc%2FScreen%20Shot%202021-02-01%20at%2022.09.18.png?alt=media\&token=033ccf04-2315-436f-a887-c575c0940411)

Kita bisa melihat bahwa pada fungsi `calcluate` , kita dapat menggunakan banyak argumen untuk diolah di dalam fungsi tersebut.

{% hint style="info" %}
Kita bisa mengkombinasikan antara fungsi biasa dan fungsi variadic
{% endhint %}

### Pengisian Parameter Fungsi Variadic Menggunakan Data Slice <a href="#a202-pengisian-parameter-fungsi-variadic-menggunakan-data-slice" id="a202-pengisian-parameter-fungsi-variadic-menggunakan-data-slice"></a>

Kita dapat menggunakan data `slice` pada argumen fungsi variadic dengan cara menambahkan `...` di belakang nama variabel dari `slice` .

```go
package main

import "fmt"

func calculate(numbers ...int) (float64, int) {
	var total int = 0
	for _, number := range numbers {
		total += number
	}

	var avg = float64(total) / float64(len(numbers))
	return avg, total
}

func main() {
	var score = []int{2, 4, 6}
	var avg, total = calculate(score...)
	fmt.Println("Total dari hasil kalkulasi adalah", total)
	fmt.Printf("Rata-rata : %.2f\n", avg)
}
```

![Output program menampilkan hasil dari fungsi variadic berisi argumen bertipe slice](https://2969676661-files.gitbook.io/~/files/v0/b/gitbook-legacy-files/o/assets%2F-MQaVepPFjuLoNjbs6f8%2F-MST875I2igtKCy5s-EB%2F-MSTA8V5-k3tFqXY7y0D%2FScreen%20Shot%202021-02-01%20at%2022.18.25.png?alt=media\&token=25106267-1931-4e21-9aa2-71177833f512)

### Fungsi Closure

Fungsi **Closure** adalah sebuah fungsi yang bisa disimpan dalam variabel. Dengan menerapkan konsep tersebut, kita bisa membuat fungsi didalam fungsi, atau bahkan membuat fungsi yang mengembalikan fungsi.

Closure merupakan *anonymous function* atau fungsi tanpa nama. Biasa dimanfaatkan untuk membungkus suatu proses yang hanya dipakai sekali atau dipakai pada blok tertentu saja.

#### Fungsi di simpan pada Variabel

Pada kode di bawah ini kita akan membuat fungsi yang akan di simpan pada sebuah variabel.

```go
package main

import "fmt"

func main() {
	var getTotal = func(numbers ...int) int {
		var total int = 0
		for _, number := range numbers {
			total += number
		}
		return total
	}

	result := getTotal(5, 5, 5)
	fmt.Println(result)
}
```

![Output dari fungsi closure](https://2969676661-files.gitbook.io/~/files/v0/b/gitbook-legacy-files/o/assets%2F-MQaVepPFjuLoNjbs6f8%2F-MSTBLjzUsFiCR7N6Z4J%2F-MSTDsaw6ClFf20F02Da%2FScreen%20Shot%202021-02-01%20at%2022.34.45.png?alt=media\&token=80ca10ea-3998-4c32-8d82-a5454b20290f)

Pada kode di atas kita membuat sebuah fungsi yang disimpan ke dalam variabel `getTotal` .

### Immediately-Invoked Function Expression (IIFE) <a href="#a212-immediately-invoked-function-expression-iife" id="a212-immediately-invoked-function-expression-iife"></a>

Closure jenis ini dieksekusi langsung pada saat deklarasinya. Biasa digunakan untuk membungkus proses yang hanya dilakukan sekali, bisa mengembalikan nilai, bisa juga tidak.

```go
package main

import "fmt"

func main() {
	var getTotal = func(numbers ...int) int {
		var total int = 0
		for _, number := range numbers {
			total += number
		}
		return total
	}(5, 5, 5)

	fmt.Println(getTotal)
}
```

Pada contoh di atas IIFE menghasilkan nilai balik yang kemudian ditampung `getTotal`. Perlu diperhatikan bahwa yang ditampung adalah **nilai kembaliannya** bukan body fungsi atau **closure**.

![Output program dari penggunaan IIFE](https://2969676661-files.gitbook.io/~/files/v0/b/gitbook-legacy-files/o/assets%2F-MQaVepPFjuLoNjbs6f8%2F-MSTBLjzUsFiCR7N6Z4J%2F-MSTDsaw6ClFf20F02Da%2FScreen%20Shot%202021-02-01%20at%2022.34.45.png?alt=media\&token=80ca10ea-3998-4c32-8d82-a5454b20290f)


---

# Agent Instructions: Querying This Documentation

If you need additional information that is not directly available in this page, you can query the documentation dynamically by asking a question.

Perform an HTTP GET request on the current page URL with the `ask` query parameter:

```
GET https://davidwinalda94.gitbook.io/mastering-golang/fungsi.md?ask=<question>
```

The question should be specific, self-contained, and written in natural language.
The response will contain a direct answer to the question and relevant excerpts and sources from the documentation.

Use this mechanism when the answer is not explicitly present in the current page, you need clarification or additional context, or you want to retrieve related documentation sections.
