开发者

Which one is faster and why?

(n >= 3 ) && (n <= 99)

O开发者_如何学JAVAR

 n `elem` [3..99]

Which one is faster and why?


The first one is faster

 (n >= 3) && (n <= 99)

it is doing 3 operations

 n >= 3
 n <= 99
 and

Where as the elem is looking up the item in the array, so is doing upto (99 - 3) * 2 operations.

index = 0
isFound = false
array[] = { 3, 4, 5, 6, ... 98, 99 }

while isFound == false
   isFound = (n == array[index++])


(n >= 3) && (n <= 99) is faster as it involves only two trivial comparisons. If the compiler/interpreter does not do any kind of real black magic optimization it has to construct the list ([3..99]) because lazy evaluation cannot be used (normally "pulling" the next value until you're done, which would have a complexity of O(n/2) in this case).


These two expressions don't mean the same thing. A subtle difference is that one relies on Ord and the other on Enum:

> :t \n -> (n >= 3) && (n <= 99)
\n -> (n >= 3) && (n <= 99) :: (Num a, Ord a) => a -> Bool

> :t \n -> n `elem` [3..99]
\n -> n `elem` [3..99] :: (Num a, Enum a) => a -> Bool

So, for example, if n is 3.14159, then the first test will pass, but the second won't:

> (pi >= 3) && (pi <= 99)
True

> pi `elem` [3..99]
False

Further, while the four Prelude Num instances (Int, Integer, Float, and Double) are all instances of both Ord and Enum, it is possible to imagine a numeric type that is an instance of Ord but not Enum. In such a case, then the second test wouldn't even be legal.

Hence, in general, the compiler can't optomize the second to be as fast as the first unless it knows for a given type, that it is Ord and that all ordered values in the range are also in the list enumeration created by enumFromTo. For Float and Double this isn't true, and for Int and Integer there is no way for the compiler to derive it, the compiler and library programmers would have to hand code it and ensure it held in all cases.


Depends on the machine and compiler (or interpreter).

0

上一篇:

下一篇:

精彩评论

暂无评论...
验证码 换一张
取 消

最新问答

问答排行榜