eval [was Re: dict to boolean expression, how to?]

Peter Otten __peter__ at web.de
Sat Aug 2 03:43:54 EDT 2014


Steven D'Aprano wrote:

> On Fri, 01 Aug 2014 17:44:27 +0200, Peter Otten wrote:
> [...]
>>> bool = ((df['a'] == 1) & (df['A'] == 0) |
>>>          (df['b'] == 1) & (df['B'] == 0) |
>>>          (df['c'] == 1) & (df['C'] == 0))
>>  
>> This is how it might look without eval():
>> 
>> #untested
>> result = functools.reduce(operator.or_, ((v == 1) & (df[k.upper()] == 0)
>> for k, v in df.items() if k.islower()))
> 
> For those who agree with Guido that reduce makes code unreadable:
> 
> result = True
> for key in df:
>     if key.islower():
>         result = result or (df[key] == 1 and df[key.upper()] == 0)

I cheated a bit and gave the solution that the OP was unlikely to come up 
with ;)


> Or if you insist on a single expression:
> 
> result = any(df[k] == 1 and df[k.upper()] == 0 for k in df if k.islower())
> 
> 
>> And here is an eval-based solution:
>> 
>> # untested
>> expr = "|".join(
>>     "((df[{}] == 1) | (df[{}] == 0))".format(c, c.upper()) for c in df
>>     is c.islower())
>> result = eval(expr)
> 
> I really don't believe that there is any benefit to that in readability,
> power, flexibility, or performance. 

You can put in a print(expr) to verify that it's identical to the original 
spelt out expression.

> Also, you're using bitwise operators
> instead of shortcut bool operators. Any reason why?

Since he started the thread Alex has moved the goal posts a bit and added 
that he is using a pandas DataFrame. That works much like a numpy array. So:

>>> df
   a  A    b     B   c   C
0  1  4  100  1000   7  14
1  2  1  200  2000  21  28
2  1  0  300  3000  35  42
>>> df["a"] == 1
0     True
1    False
2     True
Name: a, dtype: bool
>>> (df["a"] == 1) and (df["A"] == 0)
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
ValueError: The truth value of an array with more than one element is 
ambiguous. Use a.any() or a.all()
>>> (df["a"] == 1) & (df["A"] == 0)
0    False
1    False
2     True
dtype: bool





More information about the Python-list mailing list