提问人:user1877442 提问时间:2/27/2013 最后编辑:Mateen Ulhaquser1877442 更新时间:5/20/2023 访问量:543300
如何测试多个变量与单个值的相等性?
How to test multiple variables for equality against a single value?
问:
我正在尝试制作一个函数,该函数将多个变量与整数进行比较并输出三个字母的字符串。我想知道是否有办法将其翻译成 Python。所以说:
x = 0
y = 1
z = 3
mylist = []
if x or y or z == 0:
mylist.append("c")
if x or y or z == 1:
mylist.append("d")
if x or y or z == 2:
mylist.append("e")
if x or y or z == 3:
mylist.append("f")
这将返回以下列表:
["c", "d", "f"]
答:
您误解了布尔表达式的工作原理;它们不像英语句子那样工作,猜猜你在这里谈论的是所有名字的相同比较。您正在寻找:
if x == 1 or y == 1 or z == 1:
x
并以其他方式自行评估(如果,否则)。y
False
0
True
您可以使用针对元组的包含测试来缩短它:
if 1 in (x, y, z):
或者更好的是:
if 1 in {x, y, z}:
使用集合
来利用恒定成本成员测试(即 无论左操作数是什么,都需要固定的时间)。in
解释
当您使用 时,python 将运算符的每一侧视为单独的表达式。该表达式首先被视为 的布尔测试,如果为 False,则测试该表达式。or
x or y == 1
x
y == 1
这是由于运算符优先级。运算符的优先级低于测试,因此首先评估后者。or
==
但是,即使情况并非如此,并且表达式实际上被解释为相反,这仍然不会执行您期望它执行的操作。x or y or z == 1
(x or y or z) == 1
x or y or z
将计算为第一个“真实”的参数,例如 not 、数字 0 或空(有关 Python 在布尔上下文中认为 false 的详细信息,请参阅布尔表达式)。False
因此,对于值,将解析为 ,因为这是参数中的第一个类似 true 的值。然后是 ,即使会是 。x = 2; y = 1; z = 0
x or y or z
2
2 == 1
False
y == 1
True
反之亦然;针对单个变量测试多个值; 会因为同样的原因而失败。使用 或 .x == 1 or 2 or 3
x == 1 or x == 2 or x == 3
x in {1, 2, 3}
评论
set
timeit.timeit('0 in {seq}'.format(seq=tuple(range(9, -1, -1))))
)
set
set
if 1 in {x, y, z}:
set
x
y
z
tuple
set
set
in [...]
in {...}
直接的写作方式是x or y or z == 0
if any(map((lambda value: value == 0), (x,y,z))):
pass # write your logic.
但我不认为,你喜欢它。:) 这种方式很丑陋。
另一种方式(更好)是:
0 in (x, y, z)
顺便说一句,很多 s 可以写成这样的东西if
my_cases = {
0: Mylist.append("c"),
1: Mylist.append("d")
# ..
}
for key in my_cases:
if key in (x,y,z):
my_cases[key]()
break
评论
dict
.append
None
None
AttributeError
filter
map
any(v == 0 for v in (x, y, z))
使用字典结构可以更轻松地解决您的问题,例如:
x = 0
y = 1
z = 3
d = {0: 'c', 1:'d', 2:'e', 3:'f'}
mylist = [d[k] for k in [x, y, z]]
评论
d = "cdef"
MyList = ["cdef"[k] for k in [x, y, z]]
map(lambda i: 'cdef'[i], [x, y, z])
要检查某个值是否包含在一组变量中,您可以使用内置模块和 .itertools
operator
例如:
进口:
from itertools import repeat
from operator import contains
声明变量:
x = 0
y = 1
z = 3
创建值的映射(按要检查的顺序):
check_values = (0, 1, 3)
用于允许重复变量:itertools
check_vars = repeat((x, y, z))
最后,使用该函数创建一个迭代器:map
checker = map(contains, check_vars, check_values)
然后,在检查值(按原始顺序)时,使用:next()
if next(checker) # Checks for 0
# Do something
pass
elif next(checker) # Checks for 1
# Do something
pass
等。。。
这比 因为是一个内置模块,并且比使用必须创建自定义就地函数更快、更高效。lambda x: x in (variables)
operator
lambda
用于检查列表中是否存在非零(或 False)值的另一个选项:
not (x and y and z)
等效:
not all((x, y, z))
评论
我认为这会更好地处理它:
my_dict = {0: "c", 1: "d", 2: "e", 3: "f"}
def validate(x, y, z):
for ele in [x, y, z]:
if ele in my_dict.keys():
return my_dict[ele]
输出:
print validate(0, 8, 9)
c
print validate(9, 8, 9)
None
print validate(9, 8, 2)
e
d = {0:'c', 1:'d', 2:'e', 3: 'f'}
x, y, z = (0, 1, 3)
print [v for (k,v) in d.items() if x==k or y==k or z==k]
如果要使用 if、else 语句,以下是另一种解决方案:
myList = []
aList = [0, 1, 3]
for l in aList:
if l==0: myList.append('c')
elif l==1: myList.append('d')
elif l==2: myList.append('e')
elif l==3: myList.append('f')
print(myList)
如果你非常非常懒惰,你可以把这些值放在一个数组中。如
list = []
list.append(x)
list.append(y)
list.append(z)
nums = [add numbers here]
letters = [add corresponding letters here]
for index in range(len(nums)):
for obj in list:
if obj == num[index]:
MyList.append(letters[index])
break
您也可以将数字和字母放入字典中并执行此操作,但这可能比简单的 if 语句复杂得多。这就是你试图成为格外懒惰:)所得到的
还有一件事,你的
if x or y or z == 0:
将编译,但不是以您想要的方式编译。当您简单地将变量放在 if 语句中时(示例)
if b
程序将检查变量是否不为 null。写上述陈述的另一种方法(更有意义)是
if bool(b)
Bool 是 python 中的一个内置函数,它基本上执行验证布尔语句的命令(如果您不知道那是什么,那就是您现在:)在 if 语句中尝试的内容)
我发现的另一种懒惰方式是:
if any([x==0, y==0, z==0])
评论
list
xyz
xyz = [x, y, z]
any(v == 0 for v in (x, y, z))
Set 是这里的好方法,因为它对变量进行排序,这似乎是你在这里的目标。 无论参数的顺序如何。{z,y,x}
{0,1,3}
>>> ["cdef"[i] for i in {z,x,y}]
['c', 'd', 'f']
这样,整个解决方案是 O(n)。
此代码可能会有所帮助
L ={x, y, z}
T= ((0,"c"),(1,"d"),(2,"e"),(3,"f"),)
List2=[]
for t in T :
if t[0] in L :
List2.append(t[1])
break;
正如Martijn Pieters所说,正确且最快的格式是:
if 1 in {x, y, z}:
根据他的建议,您现在将拥有单独的 if 语句,以便 Python 将读取每个语句,无论前者是 OR 。如:True
False
if 0 in {x, y, z}:
mylist.append("c")
if 1 in {x, y, z}:
mylist.append("d")
if 2 in {x, y, z}:
mylist.append("e")
...
这将起作用,但是如果您习惯使用字典(请参阅我在那里所做的),您可以通过制作一个初始字典将数字映射到您想要的字母来清理它,然后仅使用 for 循环:
num_to_letters = {0: "c", 1: "d", 2: "e", 3: "f"}
for number in num_to_letters:
if number in {x, y, z}:
mylist.append(num_to_letters[number])
评论
for number in num_to_letters
.keys()
for i, c in enumerate('cdef'): if i in {x, y, z}: mylist.append(c)
s = 'cdef'; mylist = [s[i] for i in [x, y, z]]
这里提供的所有优秀答案都集中在原始海报的具体要求上,并集中在Martijn Pieters提出的解决方案上。
他们忽略了这个问题的更广泛含义:
如何根据多个值测试一个变量?
如果使用字符串,则提供的解决方案不适用于部分命中,例如:
测试字符串“Wild”是否包含多个值if 1 in {x,y,z}
>>> x = "Wild things"
>>> y = "throttle it back"
>>> z = "in the beginning"
>>> if "Wild" in {x, y, z}: print (True)
...
或
>>> x = "Wild things"
>>> y = "throttle it back"
>>> z = "in the beginning"
>>> if "Wild" in [x, y, z]: print (True)
...
对于此方案,最简单的方法是转换为字符串
>>> [x, y, z]
['Wild things', 'throttle it back', 'in the beginning']
>>> {x, y, z}
{'in the beginning', 'throttle it back', 'Wild things'}
>>>
>>> if "Wild" in str([x, y, z]): print (True)
...
True
>>> if "Wild" in str({x, y, z}): print (True)
...
True
但是,应该注意的是,如前所述,这种方法会丢失单词边界,如下所示:@codeforester
>>> x=['Wild things', 'throttle it back', 'in the beginning']
>>> if "rot" in str(x): print(True)
...
True
这 3 个字母确实在列表中组合存在,但不是作为一个单独的单词。测试“腐烂”会失败,但如果其中一个列表项是“地狱中的腐烂”,那也会失败。
结果是,如果使用此方法,请小心您的搜索条件,并注意它确实有此限制。rot
一线解决方案:
mylist = [{0: 'c', 1: 'd', 2: 'e', 3: 'f'}[i] for i in [0, 1, 2, 3] if i in (x, y, z)]
艺术
mylist = ['cdef'[i] for i in range(4) if i in (x, y, z)]
您可以使用字典:
x = 0
y = 1
z = 3
list=[]
dict = {0: 'c', 1: 'd', 2: 'e', 3: 'f'}
if x in dict:
list.append(dict[x])
else:
pass
if y in dict:
list.append(dict[y])
else:
pass
if z in dict:
list.append(dict[z])
else:
pass
print list
评论
在 Python 中表示伪代码的最 python 方式是:
x = 0
y = 1
z = 3
mylist = []
if any(v == 0 for v in (x, y, z)):
mylist.append("c")
if any(v == 1 for v in (x, y, z)):
mylist.append("d")
if any(v == 2 for v in (x, y, z)):
mylist.append("e")
if any(v == 3 for v in (x, y, z)):
mylist.append("f")
评论
if any(v >= 42 for v in (x, y, z)):
2 in {x,y,z}
2 in (x,y,z)
any(_v == 2 for _v in (x,y,z))
)
看起来你正在构建某种凯撒密码。
一个更通用的方法是:
input_values = (0, 1, 3)
origo = ord('c')
[chr(val + origo) for val in inputs]
输出
['c', 'd', 'f']
不确定这是否是代码的预期副作用,但输出的顺序将始终排序。
如果这是您想要的,可以将最后一行更改为:
sorted([chr(val + origo) for val in inputs])
要使用单个值测试多个变量,请执行以下操作:if 1 in {a,b,c}:
要使用一个变量测试多个值:if a in {1, 2, 3}:
它可以很容易地完成
for value in [var1,var2,var3]:
li.append("targetValue")
您可以尝试如下所示的方法。在此方法中,您可以自由指定/输入要输入的变量数量。
mydict = {0:"c", 1:"d", 2:"e", 3:"f"}
mylist= []
num_var = int(raw_input("How many variables? ")) #Enter 3 when asked for input.
for i in range(num_var):
''' Enter 0 as first input, 1 as second input and 3 as third input.'''
globals()['var'+str('i').zfill(3)] = int(raw_input("Enter an integer between 0 and 3 "))
mylist += mydict[globals()['var'+str('i').zfill(3)]]
print mylist
>>> ['c', 'd', 'f']
也许您需要直接公式来设置输出位。
x=0 or y=0 or z=0 is equivalent to x*y*z = 0
x=1 or y=1 or z=1 is equivalent to (x-1)*(y-1)*(z-1)=0
x=2 or y=2 or z=2 is equivalent to (x-2)*(y-2)*(z-2)=0
让我们映射到位:'c':1 'd':0xb10 'e':0xb100 'f':0xb1000
isc 的关系(是 'c'):
if xyz=0 then isc=1 else isc=0
如果公式 https://youtu.be/KAdKCgBGK0k?list=PLnI9xbPdZUAmUL8htSl6vToPQRRN3hhFp&t=315,则使用数学
[三]:(xyz=0 and isc=1) or (((xyz=0 and isc=1) or (isc=0)) and (isc=0))
[d]:((x-1)(y-1)(z-1)=0 and isc=2) or (((xyz=0 and isd=2) or (isc=0)) and (isc=0))
...
按以下逻辑连接这些公式:
- 逻辑是方程的平方和
and
- 逻辑是方程式的乘积
or
你会得到一个总方程 表示总和,你有总和公式
则 sum&1 是 c,sum&2 是 d,sum&4 是 e,sum&5 是 f
在此之后,您可以形成预定义的数组,其中字符串元素的索引将对应于就绪字符串。
array[sum]
给你字符串。
这将对你有所帮助。
def test_fun(val):
x = 0
y = 1
z = 2
myList = []
if val in (x, y, z) and val == 0:
myList.append("C")
if val in (x, y, z) and val == 1:
myList.append("D")
if val in (x, y, z) and val == 2:
myList.append("E")
test_fun(2);
你可以把这个联合起来
x = 0
y = 1
z = 3
在一个变量中。
In [1]: xyz = (0,1,3,)
In [2]: mylist = []
将我们的条件更改为:
In [3]: if 0 in xyz:
...: mylist.append("c")
...: if 1 in xyz:
...: mylist.append("d")
...: if 2 in xyz:
...: mylist.append("e")
...: if 3 in xyz:
...: mylist.append("f")
输出:
In [21]: mylist
Out[21]: ['c', 'd', 'f']
如果没有 dict,请尝试以下解决方案:
x, y, z = 0, 1, 3
offset = ord('c')
[chr(i + offset) for i in (x,y,z)]
并提供:
['c', 'd', 'f']
问题
虽然用于测试多个值的模式
>>> 2 in {1, 2, 3}
True
>>> 5 in {1, 2, 3}
False
可读性很强,在很多情况下都能工作,有一个陷阱:
>>> 0 in {True, False}
True
但我们希望有
>>> (0 is True) or (0 is False)
False
溶液
前面表达式的一个概括是基于 ytpillai 的答案:
>>> any([0 is True, 0 is False])
False
可以写成
>>> any(0 is item for item in (True, False))
False
虽然此表达式返回正确的结果,但它的可读性不如第一个表达式:-(
您可以通过两种方式开发它
def compareVariables(x,y,z):
mylist = []
if x==0 or y==0 or z==0:
mylist.append('c')
if x==1 or y==1 or z==1:
mylist.append('d')
if x==2 or y==2 or z==2:
mylist.append('e')
if x==3 or y==3 or z==3:
mylist.append('f')
else:
print("wrong input value!")
print('first:',mylist)
compareVariables(1, 3, 2)
或
def compareVariables(x,y,z):
mylist = []
if 0 in (x,y,z):
mylist.append('c')
if 1 in (x,y,z):
mylist.append('d')
if 2 in (x,y,z):
mylist.append('e')
if 3 in (x,y,z):
mylist.append('f')
else:
print("wrong input value!")
print('second:',mylist)
compareVariables(1, 3, 2)
这是另一种方法:
x = 0
y = 1
z = 3
mylist = []
if any(i in [0] for i in[x,y,z]):
mylist.append("c")
if any(i in [1] for i in[x,y,z]):
mylist.append("d")
if any(i in [2] for i in[x,y,z]):
mylist.append("e")
if any(i in [3] for i in[x,y,z]):
mylist.append("f")
它是列表理解和任何关键字的混合体。
评论
i in [0]
i == 0
正如这个答案所解释的那样,它不是那样工作的。or
虽然通用答案是使用
if 0 in (x, y, z):
...
对于特定问题来说,这不是最好的方法。在你的例子中,你正在做重复测试,因此值得组成一组这些变量:
values = {x, y, z}
if 0 in values:
mylist.append("c")
if 1 in values:
mylist.append("d")
我们可以使用字典来简化这一点 - 这将产生相同的值:
mappings = {0: "c", 1: "d", ...}
for k in mappings:
if k in values:
mylist.append(mappings[k])
或者,如果 的顺序是任意的,则可以循环访问这些值并将它们与映射匹配:mylist
mappings = {0: "c", 1: "d", ...}
for v in (x, y, z):
if v in mappings:
mylist.append(mappings[v])
不带 if 的用法示例:
x,y,z = 0,1,3
values = {0:"c",1:"d",2:"e",3:"f"} # => as if usage
my_list = [values[i] for i in (x,y,z)]
print(my_list)
首先,对条件的更正:OR
你需要说:
if x == 0 or y == 0 or z == 0:
原因是“or”将条件拆分为单独的逻辑部分。你原来的陈述是这样写的,这些部分是:
x
y
z == 0 // or 1, 2, 3 depending on the if statement
例如,最后一部分在检查 z == 0 --- --- 时很好,但前两部分只是说基本上和 .由于整数的计算结果总是为 0,除非它们为 0,这意味着条件的第一部分总是 when 或不等于 0(在 y 的情况下总是,因为你有 ,导致你的整个条件(因为工作方式)总是 。if x
if y
True
True
x
y
y = 1
OR
True
为了避免这种情况,你需要确保你的条件的所有部分(每一面)都有意义(你可以通过假装陈述的另一面不存在来做到这一点)。这就是您确认您的条件是否正确定义的方法。OR
OR
OR
您可以像这样单独编写语句:
if x == 0
if y == 0
if z == 0
这意味着带有关键字的正确 mergin 是:OR
if x == 0 or y == 0 or z == 0
二、如何解决问题:
您基本上想要检查是否有任何变量与给定整数匹配,如果是,则在一对一映射中为其分配一个与它匹配的字母。您希望对某个整数列表执行此操作,以便输出是字母列表。你可以这样做:
def func(x, y, z):
result = []
for integer, letter in zip([0, 1, 2, 3], ['c', 'd', 'e', 'f']):
if x == integer or y == integer or z == integer:
result.append(letter)
return result
同样,您可以使用 LIST COMPREHENSION 更快地获得相同的结果:
def func(x, y, z):
return [
letter
for integer, letter in zip([0, 1, 2, 3], ['c', 'd', 'e', 'f'])
if x == integer or y == integer or z == integer
]
#selection
: a=np.array([0,1,3])
#options
: np.diag(['c','d','e','f'])
array([['c', '', '', ''],
['', 'd', '', ''],
['', '', 'e', ''],
['', '', '', 'f']], dtype='<U1')
现在我们可以使用 AS [ROW,COL] 选择器,它的作用类似于 any(...) 条件:
#list of options[sel,sel]
: np.diag(['c','d','e','f'])[a,a]
array(['c', 'd', 'f'], dtype='<U1')
评论
要针对单个值测试多个变量,请执行以下操作:
将变量包装在一个集合对象中,例如 {a, b, c}。 使用 in 运算符测试值是否存储在任何变量中。 如果值存储在至少一个变量中,则 in 运算符将返回 True。
# ✅ test multiple variables against single value using tuple
if 'a' in (a, b, c):
print('value is stored in at least one of the variables')
# ---------------------------------------------------------
# ✅ test multiple variables against single value using tuple
if 'a' in {a, b, c}:
print('value is stored in at least one of the variables')
# ---------------------------------------------------------
# ✅ test multiple variables against single value (OR operator chaining)
if a == 'a' or b == 'a' or c == 'a':
print('value is stored in at least one of the variables')
评论
1
any
all
all([1, 2, 3, 4, False])
all([True, 1, 2, 3])
any([False, 0, 0, False])
any([False, 0, True, False])
x == 0 或 1
不起作用?”问题的数量之多,我宁愿将这个问题用作这些问题的规范重复目标。if x == 0 or 1:
if x or y == 0:
0
0.0
False