Commit 46dfa5f4 authored by Skip Montanaro's avatar Skip Montanaro

require list comprehensions to start with a for clause

parent 2823f03a
...@@ -153,7 +153,7 @@ square brackets: ...@@ -153,7 +153,7 @@ square brackets:
\begin{verbatim} \begin{verbatim}
list_display: "[" [listmaker] "]" list_display: "[" [listmaker] "]"
listmaker: expression ( list_iter | ( "," expression)* [","] ) listmaker: expression ( list_for | ( "," expression)* [","] )
list_iter: list_for | list_if list_iter: list_for | list_if
list_for: "for" expression_list "in" testlist [list_iter] list_for: "for" expression_list "in" testlist [list_iter]
list_if: "if" test [list_iter] list_if: "if" test [list_iter]
...@@ -164,7 +164,8 @@ by providing either a list of expressions or a list comprehension. ...@@ -164,7 +164,8 @@ by providing either a list of expressions or a list comprehension.
When a comma-separated list of expressions is supplied, its elements are When a comma-separated list of expressions is supplied, its elements are
evaluated from left to right and placed into the list object in that evaluated from left to right and placed into the list object in that
order. When a list comprehension is supplied, it consists of a order. When a list comprehension is supplied, it consists of a
single expression followed by one or more "for" or "if" clauses. In this single expression followed by at least one "for" clause and zero or more
"for" or "if" clauses. In this
case, the elements of the new list are those that would be produced case, the elements of the new list are those that would be produced
by considering each of the "for" or "if" clauses a block, nesting from by considering each of the "for" or "if" clauses a block, nesting from
left to right, and evaluating the expression to produce a list element left to right, and evaluating the expression to produce a list element
......
...@@ -1755,10 +1755,15 @@ item, then to the result and the next item, and so on. For example, ...@@ -1755,10 +1755,15 @@ item, then to the result and the next item, and so on. For example,
\subsection{List Comprehensions} \subsection{List Comprehensions}
List comprehensions provide a concise way to create lists without List comprehensions provide a concise way to create lists without resorting
resorting to use of the \function{map()} or \function{filter()} to use of \function{map()}, \function{filter()} and/or \keyword{lambda}.
functions. The resulting construct tends often to be clearer than use The resulting list definition tends often to be clearer than lists built
of those functions. using those constructs. Each list comprehension consists of an expression
following by a \keyword{for} clause, then zero or more \keyword{for} or
\keyword{if} clauses. The result will be a list resulting from evaluating
the expression in the context of the \keyword{for} and \keyword{if} clauses
which follow it. If the expression would evaluate to a tuple, it must be
parenthesized.
\begin{verbatim} \begin{verbatim}
>>> freshfruit = [' banana', ' loganberry ', 'passion fruit '] >>> freshfruit = [' banana', ' loganberry ', 'passion fruit ']
...@@ -1771,6 +1776,17 @@ of those functions. ...@@ -1771,6 +1776,17 @@ of those functions.
[12, 18] [12, 18]
>>> [3*x for x in vec if x < 2] >>> [3*x for x in vec if x < 2]
[] []
>>> [{x: x**2} for x in vec]
[{2: 4}, {4: 16}, {6: 36}]
>>> [[x,x**2] for x in vec]
[[2, 4], [4, 16], [6, 36]]
>>> [x, x**2 for x in vec] # error - parens required for tuples
File "<stdin>", line 1
[x, x**2 for x in vec]
^
SyntaxError: invalid syntax
>>> [(x, x**2) for x in vec]
[(2, 4), (4, 16), (6, 36)]
>>> vec1 = [2, 4, 6] >>> vec1 = [2, 4, 6]
>>> vec2 = [4, 3, -9] >>> vec2 = [4, 3, -9]
>>> [x*y for x in vec1 for y in vec2] >>> [x*y for x in vec1 for y in vec2]
......
...@@ -77,7 +77,7 @@ term: factor (('*'|'/'|'%') factor)* ...@@ -77,7 +77,7 @@ term: factor (('*'|'/'|'%') factor)*
factor: ('+'|'-'|'~') factor | power factor: ('+'|'-'|'~') factor | power
power: atom trailer* ('**' factor)* power: atom trailer* ('**' factor)*
atom: '(' [testlist] ')' | '[' [listmaker] ']' | '{' [dictmaker] '}' | '`' testlist '`' | NAME | NUMBER | STRING+ atom: '(' [testlist] ')' | '[' [listmaker] ']' | '{' [dictmaker] '}' | '`' testlist '`' | NAME | NUMBER | STRING+
listmaker: test ( list_iter | (',' test)* [','] ) listmaker: test ( list_for | (',' test)* [','] )
lambdef: 'lambda' [varargslist] ':' test lambdef: 'lambda' [varargslist] ':' test
trailer: '(' [arglist] ')' | '[' subscriptlist ']' | '.' NAME trailer: '(' [arglist] ')' | '[' subscriptlist ']' | '.' NAME
subscriptlist: subscript (',' subscript)* [','] subscriptlist: subscript (',' subscript)* [',']
......
...@@ -55,4 +55,5 @@ classdef ...@@ -55,4 +55,5 @@ classdef
[(1, 'Apple'), (1, 'Banana'), (1, 'Coconut'), (2, 'Apple'), (2, 'Banana'), (2, 'Coconut'), (3, 'Apple'), (3, 'Banana'), (3, 'Coconut'), (4, 'Apple'), (4, 'Banana'), (4, 'Coconut'), (5, 'Apple'), (5, 'Banana'), (5, 'Coconut')] [(1, 'Apple'), (1, 'Banana'), (1, 'Coconut'), (2, 'Apple'), (2, 'Banana'), (2, 'Coconut'), (3, 'Apple'), (3, 'Banana'), (3, 'Coconut'), (4, 'Apple'), (4, 'Banana'), (4, 'Coconut'), (5, 'Apple'), (5, 'Banana'), (5, 'Coconut')]
[(1, 'Banana'), (1, 'Coconut'), (2, 'Banana'), (2, 'Coconut'), (3, 'Banana'), (3, 'Coconut'), (4, 'Banana'), (4, 'Coconut'), (5, 'Banana'), (5, 'Coconut')] [(1, 'Banana'), (1, 'Coconut'), (2, 'Banana'), (2, 'Coconut'), (3, 'Banana'), (3, 'Coconut'), (4, 'Banana'), (4, 'Coconut'), (5, 'Banana'), (5, 'Coconut')]
good: got a SyntaxError as expected good: got a SyntaxError as expected
good: got a SyntaxError as expected
[('Boeing', 'Airliner'), ('Boeing', 'Engine'), ('Ford', 'Engine'), ('Macdonalds', 'Cheeseburger')] [('Boeing', 'Airliner'), ('Boeing', 'Engine'), ('Ford', 'Engine'), ('Macdonalds', 'Cheeseburger')]
...@@ -578,6 +578,12 @@ try: ...@@ -578,6 +578,12 @@ try:
except SyntaxError: except SyntaxError:
print "good: got a SyntaxError as expected" print "good: got a SyntaxError as expected"
try:
eval("[x if y]")
print "FAIL: should have raised a SyntaxError!"
except SyntaxError:
print "good: got a SyntaxError as expected"
suppliers = [ suppliers = [
(1, "Boeing"), (1, "Boeing"),
(2, "Ford"), (2, "Ford"),
......
...@@ -1043,7 +1043,7 @@ com_list_iter(struct compiling *c, ...@@ -1043,7 +1043,7 @@ com_list_iter(struct compiling *c,
static void static void
com_list_comprehension(struct compiling *c, node *n) com_list_comprehension(struct compiling *c, node *n)
{ {
/* listmaker: test list_iter */ /* listmaker: test list_for */
char tmpname[12]; char tmpname[12];
sprintf(tmpname, "__%d__", ++c->c_tmpname); sprintf(tmpname, "__%d__", ++c->c_tmpname);
com_addoparg(c, BUILD_LIST, 0); com_addoparg(c, BUILD_LIST, 0);
...@@ -1052,7 +1052,7 @@ com_list_comprehension(struct compiling *c, node *n) ...@@ -1052,7 +1052,7 @@ com_list_comprehension(struct compiling *c, node *n)
com_addopnamestr(c, LOAD_ATTR, "append"); com_addopnamestr(c, LOAD_ATTR, "append");
com_addopnamestr(c, STORE_NAME, tmpname); com_addopnamestr(c, STORE_NAME, tmpname);
com_pop(c, 1); com_pop(c, 1);
com_list_iter(c, n, CHILD(n, 0), tmpname); com_list_for(c, CHILD(n, 1), CHILD(n, 0), tmpname);
com_addopnamestr(c, DELETE_NAME, tmpname); com_addopnamestr(c, DELETE_NAME, tmpname);
--c->c_tmpname; --c->c_tmpname;
} }
...@@ -1060,8 +1060,8 @@ com_list_comprehension(struct compiling *c, node *n) ...@@ -1060,8 +1060,8 @@ com_list_comprehension(struct compiling *c, node *n)
static void static void
com_listmaker(struct compiling *c, node *n) com_listmaker(struct compiling *c, node *n)
{ {
/* listmaker: test ( list_iter | (',' test)* [','] ) */ /* listmaker: test ( list_for | (',' test)* [','] ) */
if (NCH(n) > 1 && TYPE(CHILD(n, 1)) == list_iter) if (NCH(n) > 1 && TYPE(CHILD(n, 1)) == list_for)
com_list_comprehension(c, n); com_list_comprehension(c, n);
else { else {
int len = 0; int len = 0;
......
...@@ -1317,7 +1317,7 @@ static state states_59[4] = { ...@@ -1317,7 +1317,7 @@ static state states_59[4] = {
{1, arcs_59_3}, {1, arcs_59_3},
}; };
static arc arcs_60_0[2] = { static arc arcs_60_0[2] = {
{128, 1}, {120, 1},
{129, 1}, {129, 1},
}; };
static arc arcs_60_1[1] = { static arc arcs_60_1[1] = {
...@@ -1340,7 +1340,7 @@ static arc arcs_61_3[1] = { ...@@ -1340,7 +1340,7 @@ static arc arcs_61_3[1] = {
{9, 4}, {9, 4},
}; };
static arc arcs_61_4[2] = { static arc arcs_61_4[2] = {
{120, 5}, {128, 5},
{0, 4}, {0, 4},
}; };
static arc arcs_61_5[1] = { static arc arcs_61_5[1] = {
...@@ -1361,7 +1361,7 @@ static arc arcs_62_1[1] = { ...@@ -1361,7 +1361,7 @@ static arc arcs_62_1[1] = {
{21, 2}, {21, 2},
}; };
static arc arcs_62_2[2] = { static arc arcs_62_2[2] = {
{120, 3}, {128, 3},
{0, 2}, {0, 2},
}; };
static arc arcs_62_3[1] = { static arc arcs_62_3[1] = {
...@@ -1622,7 +1622,7 @@ static label labels[130] = { ...@@ -1622,7 +1622,7 @@ static label labels[130] = {
{25, 0}, {25, 0},
{2, 0}, {2, 0},
{3, 0}, {3, 0},
{316, 0}, {317, 0},
{1, "lambda"}, {1, "lambda"},
{314, 0}, {314, 0},
{307, 0}, {307, 0},
...@@ -1630,7 +1630,7 @@ static label labels[130] = { ...@@ -1630,7 +1630,7 @@ static label labels[130] = {
{309, 0}, {309, 0},
{1, "class"}, {1, "class"},
{315, 0}, {315, 0},
{317, 0}, {316, 0},
{318, 0}, {318, 0},
}; };
grammar _PyParser_Grammar = { grammar _PyParser_Grammar = {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment