我正在寻找一种快速的函数(不涉及字符串)
leading_ones(0b11101) # =>3
leading_ones(0b1111000110) # =>4
感谢您的努力!
我正在寻找一种快速的函数(不涉及字符串)
leading_ones(0b11101) # =>3
leading_ones(0b1111000110) # =>4
感谢您的努力!
def leading_ones(n)
nbr = 0
(n.bit_length-1).downto(0) do |i|
return nbr if n[i].zero?
nbr += 1
end
nbr
end
leading_ones(6)
#=> 2
注意:6.to_s(2) #=> "110"
。这里使用了方法Fixnum#bit_length和Fixnum#[]。
def leading_ones(num)
counter = 0
while num > 0
if num % 2 == 0
counter = 0
else
counter += 1
end
num = num / 2
end
counter
end
leading_ones(0b111) # => 3
leading_ones(0b11101) # => 3
leading_ones(0b111101) # => 4
leading_ones(0b1000) # => 1
leading_ones(0b01000) # => 1
您使用循环的版本实际上略微更快,但是无论如何,这里是一个不使用循环的版本:
def leading_ones(n)
# Number of bits needed to hold `n` as an unsigned integer
bits = n.bit_length
# `digits` bits, all on
max_possible = (1 << bits) - 1
# Flips all of `n`'s bits
flipped = n ^ max_possible
# First right-index of a zero in `n`, or the first index of a 1 in `flipped`
first_zero_rindex = flipped.bit_length
# Left-index of the first zero
first_zero_index = bits - first_zero_rindex
first_zero_index
end
bit_length
基本上是一个循环。顺便说一下,你可以使用 (1 << bits) - 1
来避免昂贵的指数运算。 - Stefan2 ** n
优化为1 << n
,其中n是正整数。不过,我编辑了我的答案,使用位移来计算max_possible
。 - nloveladyallen这是我目前的方法:
def leading_ones(arg)
c=0
log=Math.log2(arg).to_i
max_index.downto(0){|i|
if arg[i]==1
c+=1
else
return c
end
}
return c
end
有没有不用循环的想法?我认为可以在不迭代的情况下从另一个数字计算出一个数字。
添加:
好的,它已经非常快了,我想我会采用这个解决方案。但是另一个不需要循环的解决方案将是完美的 :-)
欢迎提出想法!
添加:
不要使用我的方法。它会产生舍入误差。请参阅所选答案以获取正确版本。