| // run |
| |
| // Copyright 2009 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| // Test the 'for range' construct. |
| |
| package main |
| |
| // test range over channels |
| |
| func gen(c chan int, lo, hi int) { |
| for i := lo; i <= hi; i++ { |
| c <- i |
| } |
| close(c) |
| } |
| |
| func seq(lo, hi int) chan int { |
| c := make(chan int) |
| go gen(c, lo, hi) |
| return c |
| } |
| |
| func testchan() { |
| s := "" |
| for i := range seq('a', 'z') { |
| s += string(i) |
| } |
| if s != "abcdefghijklmnopqrstuvwxyz" { |
| println("Wanted lowercase alphabet; got", s) |
| panic("fail") |
| } |
| n := 0 |
| for range seq('a', 'z') { |
| n++ |
| } |
| if n != 26 { |
| println("testchan wrong count", n, "want 26") |
| } |
| } |
| |
| // test that range over slice only evaluates |
| // the expression after "range" once. |
| |
| var nmake = 0 |
| |
| func makeslice() []int { |
| nmake++ |
| return []int{1, 2, 3, 4, 5} |
| } |
| |
| func testslice() { |
| s := 0 |
| nmake = 0 |
| for _, v := range makeslice() { |
| s += v |
| } |
| if nmake != 1 { |
| println("range called makeslice", nmake, "times") |
| panic("fail") |
| } |
| if s != 15 { |
| println("wrong sum ranging over makeslice", s) |
| panic("fail") |
| } |
| |
| x := []int{10, 20} |
| y := []int{99} |
| i := 1 |
| for i, x[i] = range y { |
| break |
| } |
| if i != 0 || x[0] != 10 || x[1] != 99 { |
| println("wrong parallel assignment", i, x[0], x[1]) |
| panic("fail") |
| } |
| } |
| |
| func testslice1() { |
| s := 0 |
| nmake = 0 |
| for i := range makeslice() { |
| s += i |
| } |
| if nmake != 1 { |
| println("range called makeslice", nmake, "times") |
| panic("fail") |
| } |
| if s != 10 { |
| println("wrong sum ranging over makeslice", s) |
| panic("fail") |
| } |
| } |
| |
| func testslice2() { |
| n := 0 |
| nmake = 0 |
| for range makeslice() { |
| n++ |
| } |
| if nmake != 1 { |
| println("range called makeslice", nmake, "times") |
| panic("fail") |
| } |
| if n != 5 { |
| println("wrong count ranging over makeslice", n) |
| panic("fail") |
| } |
| } |
| |
| // test that range over []byte(string) only evaluates |
| // the expression after "range" once. |
| |
| func makenumstring() string { |
| nmake++ |
| return "\x01\x02\x03\x04\x05" |
| } |
| |
| func testslice3() { |
| s := byte(0) |
| nmake = 0 |
| for _, v := range []byte(makenumstring()) { |
| s += v |
| } |
| if nmake != 1 { |
| println("range called makenumstring", nmake, "times") |
| panic("fail") |
| } |
| if s != 15 { |
| println("wrong sum ranging over []byte(makenumstring)", s) |
| panic("fail") |
| } |
| } |
| |
| // test that range over array only evaluates |
| // the expression after "range" once. |
| |
| func makearray() [5]int { |
| nmake++ |
| return [5]int{1, 2, 3, 4, 5} |
| } |
| |
| func testarray() { |
| s := 0 |
| nmake = 0 |
| for _, v := range makearray() { |
| s += v |
| } |
| if nmake != 1 { |
| println("range called makearray", nmake, "times") |
| panic("fail") |
| } |
| if s != 15 { |
| println("wrong sum ranging over makearray", s) |
| panic("fail") |
| } |
| } |
| |
| func testarray1() { |
| s := 0 |
| nmake = 0 |
| for i := range makearray() { |
| s += i |
| } |
| if nmake != 1 { |
| println("range called makearray", nmake, "times") |
| panic("fail") |
| } |
| if s != 10 { |
| println("wrong sum ranging over makearray", s) |
| panic("fail") |
| } |
| } |
| |
| func testarray2() { |
| n := 0 |
| nmake = 0 |
| for range makearray() { |
| n++ |
| } |
| if nmake != 1 { |
| println("range called makearray", nmake, "times") |
| panic("fail") |
| } |
| if n != 5 { |
| println("wrong count ranging over makearray", n) |
| panic("fail") |
| } |
| } |
| |
| func makearrayptr() *[5]int { |
| nmake++ |
| return &[5]int{1, 2, 3, 4, 5} |
| } |
| |
| func testarrayptr() { |
| nmake = 0 |
| x := len(makearrayptr()) |
| if x != 5 || nmake != 1 { |
| println("len called makearrayptr", nmake, "times and got len", x) |
| panic("fail") |
| } |
| nmake = 0 |
| x = cap(makearrayptr()) |
| if x != 5 || nmake != 1 { |
| println("cap called makearrayptr", nmake, "times and got len", x) |
| panic("fail") |
| } |
| s := 0 |
| nmake = 0 |
| for _, v := range makearrayptr() { |
| s += v |
| } |
| if nmake != 1 { |
| println("range called makearrayptr", nmake, "times") |
| panic("fail") |
| } |
| if s != 15 { |
| println("wrong sum ranging over makearrayptr", s) |
| panic("fail") |
| } |
| } |
| |
| func testarrayptr1() { |
| s := 0 |
| nmake = 0 |
| for i := range makearrayptr() { |
| s += i |
| } |
| if nmake != 1 { |
| println("range called makearrayptr", nmake, "times") |
| panic("fail") |
| } |
| if s != 10 { |
| println("wrong sum ranging over makearrayptr", s) |
| panic("fail") |
| } |
| } |
| |
| func testarrayptr2() { |
| n := 0 |
| nmake = 0 |
| for range makearrayptr() { |
| n++ |
| } |
| if nmake != 1 { |
| println("range called makearrayptr", nmake, "times") |
| panic("fail") |
| } |
| if n != 5 { |
| println("wrong count ranging over makearrayptr", n) |
| panic("fail") |
| } |
| } |
| |
| // test that range over string only evaluates |
| // the expression after "range" once. |
| |
| func makestring() string { |
| nmake++ |
| return "abcd☺" |
| } |
| |
| func teststring() { |
| var s rune |
| nmake = 0 |
| for _, v := range makestring() { |
| s += v |
| } |
| if nmake != 1 { |
| println("range called makestring", nmake, "times") |
| panic("fail") |
| } |
| if s != 'a'+'b'+'c'+'d'+'☺' { |
| println("wrong sum ranging over makestring", s) |
| panic("fail") |
| } |
| } |
| |
| func teststring1() { |
| s := 0 |
| nmake = 0 |
| for i := range makestring() { |
| s += i |
| } |
| if nmake != 1 { |
| println("range called makestring", nmake, "times") |
| panic("fail") |
| } |
| if s != 10 { |
| println("wrong sum ranging over makestring", s) |
| panic("fail") |
| } |
| } |
| |
| func teststring2() { |
| n := 0 |
| nmake = 0 |
| for range makestring() { |
| n++ |
| } |
| if nmake != 1 { |
| println("range called makestring", nmake, "times") |
| panic("fail") |
| } |
| if n != 5 { |
| println("wrong count ranging over makestring", n) |
| panic("fail") |
| } |
| } |
| |
| // test that range over map only evaluates |
| // the expression after "range" once. |
| |
| func makemap() map[int]int { |
| nmake++ |
| return map[int]int{0: 'a', 1: 'b', 2: 'c', 3: 'd', 4: '☺'} |
| } |
| |
| func testmap() { |
| s := 0 |
| nmake = 0 |
| for _, v := range makemap() { |
| s += v |
| } |
| if nmake != 1 { |
| println("range called makemap", nmake, "times") |
| panic("fail") |
| } |
| if s != 'a'+'b'+'c'+'d'+'☺' { |
| println("wrong sum ranging over makemap", s) |
| panic("fail") |
| } |
| } |
| |
| func testmap1() { |
| s := 0 |
| nmake = 0 |
| for i := range makemap() { |
| s += i |
| } |
| if nmake != 1 { |
| println("range called makemap", nmake, "times") |
| panic("fail") |
| } |
| if s != 10 { |
| println("wrong sum ranging over makemap", s) |
| panic("fail") |
| } |
| } |
| |
| func testmap2() { |
| n := 0 |
| nmake = 0 |
| for range makemap() { |
| n++ |
| } |
| if nmake != 1 { |
| println("range called makemap", nmake, "times") |
| panic("fail") |
| } |
| if n != 5 { |
| println("wrong count ranging over makemap", n) |
| panic("fail") |
| } |
| } |
| |
| // test that range evaluates the index and value expressions |
| // exactly once per iteration. |
| |
| var ncalls = 0 |
| |
| func getvar(p *int) *int { |
| ncalls++ |
| return p |
| } |
| |
| func testcalls() { |
| var i, v int |
| si := 0 |
| sv := 0 |
| for *getvar(&i), *getvar(&v) = range [2]int{1, 2} { |
| si += i |
| sv += v |
| } |
| if ncalls != 4 { |
| println("wrong number of calls:", ncalls, "!= 4") |
| panic("fail") |
| } |
| if si != 1 || sv != 3 { |
| println("wrong sum in testcalls", si, sv) |
| panic("fail") |
| } |
| |
| ncalls = 0 |
| for *getvar(&i), *getvar(&v) = range [0]int{} { |
| println("loop ran on empty array") |
| panic("fail") |
| } |
| if ncalls != 0 { |
| println("wrong number of calls:", ncalls, "!= 0") |
| panic("fail") |
| } |
| } |
| |
| func main() { |
| testchan() |
| testarray() |
| testarray1() |
| testarray2() |
| testarrayptr() |
| testarrayptr1() |
| testarrayptr2() |
| testslice() |
| testslice1() |
| testslice2() |
| testslice3() |
| teststring() |
| teststring1() |
| teststring2() |
| testmap() |
| testmap1() |
| testmap2() |
| testcalls() |
| } |