Special case: Multi-value arguments to functions

December 21, 2023

You may recall that when we started the section on function calls, we were told about a “special case”… well here it is!

Calls

As a special case, if the return values of a function or method g are equal in number and individually assignable to the parameters of another function or method f, then the call f(g(parameters_of_g)) will invoke f after binding the return values of g to the parameters of f in order. The call of f must contain no parameters other than the call of g, and g must have at least one return value. If f has a final ... parameter, it is assigned the return values of g that remain after assignment of regular parameters.

func Split(s string, pos int) (string, string) {
	return s[0:pos], s[pos:]
}

func Join(s, t string) string {
	return s + t
}

if Join(Split(value, len(value)/2)) != value {
	log.Panic("test fails")
}

Alright, so let’s illustrate this with examples, just in case it’s not completely clear. I’ll start with two examples of this exception in action, then I’ll talk about when it doesn’t work.

func g() (int, string) {
	return 13, "carrots"
}

func f(count int, item string) {
	fmt.Printf("Will buy %d %s\n", count, item)
}

f(g()) // Outputs "Will buy 13 carrots"

It can also work with a variadic f:

func f(count int, items ...string) {
  switch len(items) {
	case 0:
		fmt.Printf("Will buy %d of nothing\n", count)
	case 1:
		fmt.Printf("Will buy %d %s\n", count, items[0])
	default:
		fmt.Printf("Will buy %d of each %s\n", count, strings.Join(items, ", "))
	}
}

f(g()) // Still works as before, and outputs: "Will by 13 carrots"

func h() (int, string, string, string) {
	return 8, "apples", "pears", "oranges"
}

func i() int {
	return 9854
}

f(h()) // Outputs: Will buy 8 of each apples, pears, oranges
f(i()) // Outputs: Will buy 9854 of nothing

So that’s pretty much it for where this exception works. Let’s look at some examples where it doesn’t.

func j() (uint, string) {
	return 12, "eggs"
}

f(j()) // cannot use j() (value of type uint) as int value in argument to f

This one is probably obvious: It fails because the return types of j are not individually assignable to the paramaters of f. Specifically, the first return value of type uint is not assignable to f’s first parameter of type int.

f(g(), "onions") // multiple-value g() (value of type (int, string)) in single-value context

This illustrates the rule “the call of f must contain no parameters other than the call of g”. By adding another argument ("onions"), we’ve told the compiler that we expect g to return a single value, thus explaining the particular error we receive. If we really want this to work, we can do it in two steps:

count, item := g()
f(count, item, "onions")

Quotes from The Go Programming Language Specification Version of August 2, 2023


Share this

Direct to your inbox, daily. I respect your privacy .

Unsure? Browse the archive .

Related Content


Refresher on methods

In yesterday’s email, I failed to include the example included in the spec, which doesn’t make sense to include otherwise. I’ve updated the web version of the email in case you’re super curious to see what I left out. We’ve already covered these details, but they’re logically mentioned again in the section about function calls, so we’ll cover them here: Calls … A method call x.m() is valid if the method set of (the type of) x contains m and the argument list can be assigned to the parameter list of m.


nil functions

Calls … Calling a nil function value causes a run-time panic. So, calling a nil function causes a panic. Sensible enough. But when would you ever run across a nil function? Let’s look at some examples. This is probably the most obvious example, and perhaps one that popped to your mind: var fn func(int) // fn is of type func(int), but uninitialized, so nil fn(3) // run-time panic But let’s consider another example, which can be a bit more confusing.


Evaluation of function parameters

Calls If f denotes a generic function, it must be instantiated before it can be called or used as a function value. We’ve seen this rule applied to methods and other types. It should be standard by now, so I won’t dwell on it. In a function call, the function value and arguments are evaluated in the usual order. After they are evaluated, the parameters of the call are passed by value to the function and the called function begins execution.

Get daily content like this in your inbox!

Subscribe