Можно ли передавать функции в качестве параметров?

158

В Java я могу сделать что-то вроде

derp(new Runnable { public void run () { /* run this sometime later */ } })

и «запустить» код в методе позже. Это трудная задача (анонимный внутренний класс), но это можно сделать.

Есть ли в Go что-то, что может облегчить передачу функции / обратного вызова в качестве параметра?

Саад
источник
7
Заметка / пояснение для читателей: в Java «функции» не проходимы (на самом деле, все «функции» в Java более точно называются методами). Runnable (и анонимные внутренние классы, которые вытекают из этого) - это просто: тип, для которого создаются объекты, который подписан на требуемый интерфейс ..
2
(Шесть лет спустя ...) У Java теперь есть способ передавать методы (например containingObject::instanceMethodName): docs.oracle.com/javase/tutorial/java/javaOO/…
vazor

Ответы:

226

Да, рассмотрим некоторые из этих примеров:

package main

import "fmt"

// convert types take an int and return a string value.
type convert func(int) string

// value implements convert, returning x as string.
func value(x int) string {
    return fmt.Sprintf("%v", x)
}

// quote123 passes 123 to convert func and returns quoted string.
func quote123(fn convert) string {
    return fmt.Sprintf("%q", fn(123))
}

func main() {
    var result string

    result = value(123)
    fmt.Println(result)
    // Output: 123

    result = quote123(value)
    fmt.Println(result)
    // Output: "123"

    result = quote123(func(x int) string { return fmt.Sprintf("%b", x) })
    fmt.Println(result)
    // Output: "1111011"

    foo := func(x int) string { return "foo" }
    result = quote123(foo)
    fmt.Println(result)
    // Output: "foo"

    _ = convert(foo) // confirm foo satisfies convert at runtime

    // fails due to argument type
    // _ = convert(func(x float64) string { return "" })
}

Play: http://play.golang.org/p/XNMtrDUDS0

Тур: https://tour.golang.org/moretypes/25 (Закрытие функций)

dskinner
источник
Можно ли передать параметр в функцию, которая сама является параметром? В приведенных выше примерах печатные объекты были жестко запрограммированы: печать 123. Могут ли быть внесены какие-либо изменения, чтобы мы могли напечатать что-то другое, а не 123? Без объявления глобальных переменных.
Сати
1
Если я правильно понимаю ваш вопрос, я думаю, что вы ищете функцию, которая возвращает функцию, смотрите здесь, где я заменяю жестко закодированную функцию «quote123» на функцию «quote», которая достигает того же результата после того, как вы передадите ей некоторый ввод: play.golang.org/p/52ahWAI2xsG
dskinner,
34

Вы можете передать функцию в качестве параметра функции Go. Вот пример передачи функции в качестве параметра другой функции Go:

package main

import "fmt"

type fn func(int) 

func myfn1(i int) {
    fmt.Printf("\ni is %v", i)
}
func myfn2(i int) {
    fmt.Printf("\ni is %v", i)
}
func test(f fn, val int) {
    f(val)
}
func main() {
    test(myfn1, 123)
    test(myfn2, 321)
}

Вы можете попробовать это по адресу: https://play.golang.org/p/9mAOUWGp0k

SeattleOrBayArea
источник
2
Спасибо! Это был действительно четкий пример того, как лучше всего использовать эту идею! Я воссоздал его, используя таблицу поиска структур, в которой хранится информация, включая указатель на функцию, которую вы хотите выполнить. Идеально для этого!
Джеймс О'Тул
15

Вот пример реализации «Карта» в Go. Надеюсь это поможет!!

func square(num int) int {
    return num * num
}

func mapper(f func(int) int, alist []int) []int {
    var a = make([]int, len(alist), len(alist))
    for index, val := range alist {

        a[index] = f(val)
    }
    return a
}

func main() {
    alist := []int{4, 5, 6, 7}
    result := mapper(square, alist)
    fmt.Println(result)

}
Робус Гаули
источник
8

Вот простой пример:

    package main

    import "fmt"

    func plusTwo() (func(v int) (int)) {
        return func(v int) (int) {
            return v+2
        }
    }

    func plusX(x int) (func(v int) (int)) {
       return func(v int) (int) {
           return v+x
       }
    }

    func main() {
        p := plusTwo()
        fmt.Printf("3+2: %d\n", p(3))

        px := plusX(3)
        fmt.Printf("3+3: %d\n", px(3))
    }
foamdino
источник
4
это возвращает функцию, не передавая функцию
Джон Ла Бардж
2

Я надеюсь, что приведенный ниже пример обеспечит большую ясность.

package main

type EmployeeManager struct{
    category            string
    city                string
    calculateSalary     func() int64
}


func NewEmployeeManager() (*EmployeeManager,error){

    return &EmployeeManager{
        category : "MANAGEMENT",
        city : "NY",
        calculateSalary: func() int64 {
            var calculatedSalary int64
            // some formula
            return calculatedSalary
        },
    },nil
}

func (self *EmployeeManager) emWithSalaryCalculation(){
    self.calculateSalary = func() int64 {
        var calculatedSalary int64
        // some new formula
        return calculatedSalary
    }
}

func updateEmployeeInfo(em EmployeeManager){
    // Some code
}

func processEmployee(){
    updateEmployeeInfo(struct {
        category        string
        city            string
        calculateSalary func() int64
    }{category: "", city: "", calculateSalary: func() int64 {
        var calculatedSalary int64
        // some new formula
        return calculatedSalary
    }})
}
CodeAdocate
источник
-2

Да, Go принимает первоклассные функции.

Смотрите статью «Функции первого класса в Go» для полезных ссылок.

Абдул Фаттах Попула
источник
4
Пожалуйста, опишите этот ответ немного;
3
фактически на этой странице 0 информации, только ссылка на глупый пример по исходному коду.
OZ_ 20.02.15