Project

General

Profile

Feature #8748

Updated by nobu (Nobuyoshi Nakada) 11 months ago

How about adding `Integer#popcoun` t Integer#popcount method? 
 (actually `Fixnum#popcount` Fixnum#popcount and `Bignum#popcount` ) Bignum#popcount) 

   ``` 
   0.popcount                #=> 0 
   1.popcount                #=> 1 
   255.popcount              #=> 8 
   256.popcount              #=> 1 
   (10**100).popcount        #=> 105 
   (257**257).popcount       #=> 999 
   ``` 

 It counts the number of one bits in the integer. 
 If the integer is negative, the one bits in the absolute number is counted. 

 popcount has various applications. 
 Hamming distance, rank/select for succinct data structure, 
 brightness of monochrome image, etc. 

 In general, popcount is useful when an array is encoded as an integer. 

 Several lower layers provides this feature. 
 gcc and clang has `__builtin_popcount`. __builtin_popcount. 
 Intel and AMD provides popcnt instruction. 

 Several languages and libraries provides this feature: 

 absolute number: Mathmatica(`DigitCount`) Mathmatica(DigitCount) 
 two's complement: Java(`java.math.BigInteger#bitCount`), Scala(`bitCount`), CommonLisp(`logcount`), CLN(`logcount`) Java(java.math.BigInteger#bitCount), Scala(bitCount), CommonLisp(logcount), CLN(logcount) 
 other behavior: GMP(`mpz_popcount`), Haskell(`popCount`), Scheme(`bitwise-bit-count`) GMP(mpz_popcount), Haskell(popCount), Scheme(bitwise-bit-count) 
 fixed size: gcc (`__builtin_popcount`), Intel/AMD(`popcnt`), Java(`java.lang.Integer.bitCount`) (__builtin_popcount), Intel/AMD(popcnt), Java(java.lang.Integer.bitCount) 

 For negative numbers, my implementation counts bits in `abs(n)`. abs(n). 
 I think this is easy to understand, at least. 
 However many software counts bits in two's complement representation. 

 There are several names. 
 I think `popcount` popcount is popular but `bitcount` bitcount is also a possible name. 
 I don't like `logcount`. logcount. 

 Any comments? 

 Details of the other software: 

 Mathmatica has `DigitCount` DigitCount which can be used as `popcount`. popcount. 
 `n.popcount` n.popcount can be implemented as `DigitCount[n, DigitCount[n, 2, 1]`. 1]. 
 It seems work for `abs(n)`. abs(n).    (I tested with Wolfram Alpha.) 
 http://reference.wolfram.com/mathematica/ref/DigitCount.html 

 Java has `bitCount` bitCount method in java.lang.Integer and java.math.BigInteger. 
 java.lang.Integer counts one-bits in two's complement representation 
 (so it is not applicable for infinite precision integer). 
 java.math.BigInteger counts bits which is different to sign bit in 
 two's complement representation. 
 http://docs.oracle.com/javase/7/docs/api/java/lang/Integer.html#bitCount(int) 
 http://docs.oracle.com/javase/7/docs/api/java/math/BigInteger.html#bitCount() 

 Scala has `bitCount` bitCount method too. 
 It works as Java. 
 http://www.scala-lang.org/api/current/index.html#scala.math.BigInt 

 CommonLisp has `logcount` logcount function. 
 http://www.lispworks.com/documentation/HyperSpec/Body/f_logcou.htm 

 CLN has `logcount` logcount function. 
 http://www.ginac.de/CLN/cln.html#Exact-numbers 

 GMP has `mpz_popcount`. mpz_popcount. 
 It returns a some constant for negative values. 
 http://gmplib.org/manual/Integer-Logic-and-Bit-Fiddling.html#Integer-Logic-and-Bit-Fiddling 

 Haskell has `popCount`. popCount. 
 It seems hang for negative values. 
 http://www.haskell.org/ghc/docs/7.6.2/html/libraries/base/Data-Bits.html#t:Bits 

 Scheme has `bitwise-bit-count`. bitwise-bit-count. 
 It returns negative result for negative values. 
 http://www.r6rs.org/final/html/r6rs-lib/r6rs-lib-Z-H-12.html#node_sec_11.1 

Back