Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

spec: (non?)evaluation of operand is unclear when ranging over pointer-to-array type #4644

Closed
adonovan opened this issue Jan 11, 2013 · 3 comments
Milestone

Comments

@adonovan
Copy link
Member

My understanding of the range statement is that if the type of the range operand is
*[n]T (i.e. a pointer-to-array type) and if only one iteration variable is present, then
the operand is not necessarily evaluated and the loop degenerates to:

  for i := 0; i < n; i++ { ... }

However the wording of the spec is somewhat ambiguous as to whether implementations may
or must perform this optimisation:

"The range expression is evaluated once before beginning the loop except if the
expression is an array, in which case, depending on the expression, it might not be
evaluated (see below). ... As a special case, if only the first iteration variable is
present, the range loop produces iteration values from 0 up to len(a) and does not index
into the array or slice itself."

But the spec doesn't say how evaluation or non-evaluation depends on the expression. 
Here are two plausible explanations:
1. It's unspecified and implementation-dependent.
2. Side effects such as function calls and channel ops must be observed, but
implementations may optimise away pure expressions such as composite literals, etc.

The test function testarrayptr1 at $GOROOT/test/range.go seems to require that the
implementation not perform this optimisation, and it appears that gc does not:

http://play.golang.org/p/Oi3pmY08s8

Can you clarify the intended behaviour?
@rsc
Copy link
Contributor

rsc commented Jan 30, 2013

Comment 1:

Function calls and channel ops must run. However, 
var x **[10]int
for i = range *x {
    println(i)
}
does not crash, because len(*x) is a constant.

Labels changed: added priority-later, removed priority-triage.

@rsc
Copy link
Contributor

rsc commented Feb 9, 2013

Comment 2:

Owner changed to ---.

Status changed to Started.

@rsc
Copy link
Contributor

rsc commented Feb 15, 2013

Comment 3:

This issue was closed by revision 61e02ee.

Status changed to Fixed.

@rsc rsc added this to the Go1.1 milestone Apr 14, 2015
@rsc rsc removed the go1.1 label Apr 14, 2015
@golang golang locked and limited conversation to collaborators Jun 24, 2016
This issue was closed.
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Projects
None yet
Development

No branches or pull requests

3 participants