在VB。And和AndAlso的区别是什么?我应该用哪一种?


And运算符求两边的值,而AndAlso运算符求右边的值当且仅当左边为真。

一个例子:

If mystring IsNot Nothing And mystring.Contains("Foo") Then
  ' bla bla
End If

如果mystring = Nothing,则会抛出异常

If mystring IsNot Nothing AndAlso mystring.Contains("Foo") Then
  ' bla bla
End If

这个不抛出异常。

所以如果你来自c#世界,你应该像使用&&一样使用AndAlso。

更多信息请访问:http://www.panopticoncentral.net/2003/08/18/the-ballad-of-andalso-and-orelse/


If Bool1 And Bool2 Then

同时计算Bool1和Bool2

If Bool1 AndAlso Bool2 Then

当且仅当Bool1为真时计算Bool2。


And操作符将在继续执行之前检查语句中的所有条件,而Andalso操作符将在知道条件为false时停止。例如:

if x = 5 And y = 7

检查x是否等于5,y是否等于7,如果两者都为真,则继续。

if x = 5 AndAlso y = 7

检查x是否等于5。如果不是,它不会检查y是否为7,因为它知道条件已经为假。(这叫做短路。)

通常人们使用短路方法,如果有理由显式地不检查第二部分,如果第一部分不为真,比如如果检查它会抛出异常。例如:

If Not Object Is Nothing AndAlso Object.Load()

如果使用And而不是AndAlso,它仍然会尝试Object.Load(),即使它什么都不是,这将抛出异常。


AndAlso很像And,除了它像c#和c++中的&&一样工作。

区别在于,如果第一个子句(AndAlso之前的子句)为真,那么第二个子句永远不会被求值——复合逻辑表达式是“短路的”。

这有时是非常有用的,例如在这样的表达式中:

If Not IsNull(myObj) AndAlso myObj.SomeProperty = 3 Then
   ...
End If

如果myObj为null,则在上面的表达式中使用旧的And将抛出NullReferenceException。


另见堆栈溢出问题:我应该总是使用AndAlso和OrElse操作符吗?

另外:对那些提到使用And的人的评论,如果表达式的右侧有副作用,你需要:

如果右侧有你需要的副作用,只需将其移到左侧,而不是使用“And”。只有在双方都有副作用的情况下你才需要"并且"如果你有那么多的副作用你可能做错了其他事情。一般来说,你应该更喜欢AndAlso。


对于所有那些说副作用是邪恶的人来说:在一种情况下有两个副作用是好的,那就是同时读取两个文件对象。

While File1.Seek_Next_Row() And File2.Seek_Next_Row()
    Str1 = File1.GetRow()
    Str2 = File2.GetRow()
End While

使用And可以确保每次检查条件时都使用一行。而AndAlso可能会读取File1的最后一行,而不消耗File2的行。

当然,上面的代码不会起作用,但我一直在使用这样的副作用,不会像有些人让你相信的那样,认为它是“坏”或“邪恶”的代码。它易于阅读和高效。


一种简单的思考方法是使用更简单的英语

If Bool1 And Bool2 Then
If [both are true] Then


If Bool1 AndAlso Bool2 Then
If [first is true then evaluate the second] Then

除了上面的答案,AndAlso还提供了一个称为短路的调节过程。许多编程语言都像vb.net一样内置了这种功能,并且可以通过删除不必要的计算来大幅提高长条件语句的性能。

另一个类似的条件是OrElse条件,它只在左侧条件为假的情况下检查右侧条件,从而在发现真条件后省去不必要的条件检查。

我会建议你总是使用短路过程,并以最能从中受益的方式组织你的条件语句。例如,首先测试最有效和最快的条件,这样您只在绝对必须时运行长条件,而在其他时间运行短路。


有趣的是,没有一个答案提到VB中的And和Or。NET是位操作符,而OrElse和AndAlso是严格的布尔操作符。

Dim a = 3 OR 5 ' Will set a to the value 7, 011 or 101 = 111
Dim a = 3 And 5 ' Will set a to the value 1, 011 and 101 = 001
Dim b = 3 OrElse 5 ' Will set b to the value true and not evaluate the 5
Dim b = 3 AndAlso 5 ' Will set b to the value true after evaluating the 5
Dim c = 0 AndAlso 5 ' Will set c to the value false and not evaluate the 5

注意:非零整数被认为是真;Dim e = not 0将把e设为-1,说明not也是位运算符。

||和&& (OrElse和AndAlso的c#版本)返回最后求值的表达式,分别为3和5。这让你可以使用c#中的习语v || 5,当v为null或(0和一个整数)时,将5作为表达式的值,否则将v的值作为表达式的值。语义上的差异可以让c#程序员接触VB。NET猝不及防,因为这个“默认值习语”在VB.NET中不起作用。

因此,要回答这个问题:对于位操作(整数或布尔)使用或和。使用OrElse和AndAlso来“短路”操作以节省时间,或者在评估之前测试评估的有效性。如果是有效的(评估),那么也是评估;如果不是(不安全的(评估),或者其他(不是评估)),那么

附赠:以下内容的价值是什么?

Dim e = Not 0 And 3

对于我们大多数人来说,除了少数令人困惑的例外(不到1%的情况下,我们可能不得不使用Or和and), OrElse和AndAlso可以做到这一点。

尽量不要被那些炫耀布尔逻辑的人带着走,让它看起来像火箭科学。

这是非常简单和直接的,偶尔你的系统可能不会像预期的那样工作,因为它首先不喜欢你的逻辑。然而,你的大脑一直告诉你,他的逻辑是100%经过测试和证明的,它应该是有效的。在那一刻,不要再相信你的大脑,让他重新思考,或者(不是OrElse,也可能是OrElse)你强迫自己去找另一份不需要太多逻辑的工作。


理解:用语言而不是用密码来理解:

用例:使用“And”,编译器将检查所有条件,所以如果你检查一个对象可能是“Nothing”,然后你检查它的一个属性,你将会有一个运行时错误。 但是对于AndAlso,如果条件中第一个“false”,它会检查下一个,这样就不会出现错误。


使用And和Or进行逻辑位操作,例如x% = y%或3

AndAlso和OrElse是If语句:

如果x > 3 AndAlso x <= 5那么

If (x > 3) And (x <= 5) Then

在我看来…

同样值得注意的是,(我)建议你在If语句中包含逻辑运算符的方程,这样它们就不会被编译器误解,例如:

If x = (y And 3) And also…


And / AndAlso Or / OrElse实际上非常有用。考虑下面的代码,如果函数DoSomethingWihtAandB返回False,它可能不会设置A和/或B:

Dim A,B,C,D As Object
If DoSomethingWithAandB(A,B)=True And A=1 And B=2 Then C=3

如果DoSomethingWithAandB返回False,它将崩溃,因为它将在And之后继续计算,A和B将等于Nothing

Dim A,B,C,D As Object
If DoSomethingWithAandB(A,B)=True AndAlso A=1 AndAlso B=2 Then C=3

如果DoSomethingWithAandB返回False,它不会崩溃,因为它将在DoSomethingWithAandB(A,B)=True时停止求值,因为返回的是False。AndAlso阻止对条件进行任何进一步的计算(因为第一个条件失败了)。OrElse的工作原理与此相同。逻辑链中求值的第一个True将停止任何进一步求值。