Bit reversal of an integer, ignoring integer size and endianness

后端 未结 12 2211
既然无缘
既然无缘 2021-01-03 00:18

Given an integer typedef:

typedef unsigned int TYPE;

or

typedef unsigned long TYPE;

I have the following

相关标签:
12条回答
  • 2021-01-03 00:30

    The following program serves to demonstrate a leaner algorithm for reversing bits, which can be easily extended to handle 64bit numbers.

    #include <stdio.h>
    #include <stdint.h>
    int main(int argc, char**argv)
    {
            int32_t x;
            if ( argc != 2 ) 
            {
                    printf("Usage: %s hexadecimal\n", argv[0]);
                    return 1;
            }
    
            sscanf(argv[1],"%x", &x);
            /* swap every neigbouring bit */
            x = (x&0xAAAAAAAA)>>1 | (x&0x55555555)<<1;
            /* swap every 2 neighbouring bits */
            x = (x&0xCCCCCCCC)>>2 | (x&0x33333333)<<2;
            /* swap every 4 neighbouring bits */
            x = (x&0xF0F0F0F0)>>4 | (x&0x0F0F0F0F)<<4;
            /* swap every 8 neighbouring bits */
            x = (x&0xFF00FF00)>>8 | (x&0x00FF00FF)<<8;
            /* and so forth, for say, 32 bit int */
            x = (x&0xFFFF0000)>>16 | (x&0x0000FFFF)<<16;
            printf("0x%x\n",x);
            return 0;
    }
    

    This code should not contain errors, and was tested using 0x12345678 to produce 0x1e6a2c48 which is the correct answer.

    0 讨论(0)
  • 2021-01-03 00:37

    We can store the results of reversing all possible 1 byte sequences in an array (256 distinct entries), then use a combination of lookups into this table and some oring logic to get the reverse of integer.

    0 讨论(0)
  • 2021-01-03 00:37

    Here is a variation and correction to TK's solution which might be clearer than the solutions by sundar. It takes single bits from t and pushes them into return_val:

    typedef unsigned long TYPE;
    #define TYPE_BITS sizeof(TYPE)*8
    
    TYPE reverser(TYPE t)
    {
        unsigned int i;
        TYPE return_val = 0
        for(i = 0; i < TYPE_BITS; i++)
        {/*foreach bit in TYPE*/
            /* shift the value of return_val to the left and add the rightmost bit from t */
            return_val = (return_val << 1) + (t & 1);
            /* shift off the rightmost bit of t */
            t = t >> 1;
        }
        return(return_val);
    }
    
    0 讨论(0)
  • 2021-01-03 00:39

    Here's my generalization of freespace's solution (in case we one day get 128-bit machines). It results in jump-free code when compiled with gcc -O3, and is obviously insensitive to the definition of foo_t on sane machines. Unfortunately it does depend on shift being a power of 2!

    #include <limits.h>
    #include <stdio.h>
    
    typedef unsigned long foo_t;
    
    foo_t reverse(foo_t x)
    {
            int shift = sizeof (x) * CHAR_BIT / 2;
            foo_t mask = (1 << shift) - 1;
            int i;
    
            for (i = 0; shift; i++) {
                    x = ((x & mask) << shift) | ((x & ~mask) >> shift);
                    shift >>= 1;
                    mask ^= (mask << shift);
            }
    
            return x;
    }       
    
    int main() {
            printf("reverse = 0x%08lx\n", reverse(0x12345678L));
    }
    
    0 讨论(0)
  • 2021-01-03 00:42

    The generic approach hat would work for objects of any type of any size would be to reverse the of bytes of the object, and the reverse the order of bits in each byte. In this case the bit-level algorithm is tied to a concrete number of bits (a byte), while the "variable" logic (with regard to size) is lifted to the level of whole bytes.

    0 讨论(0)
  • 2021-01-03 00:43

    How about:

    long temp = 0;
    int counter = 0;
    int number_of_bits = sizeof(value) * 8; // get the number of bits that represent value (assuming that it is aligned to a byte boundary)
    
    while(value > 0)            // loop until value is empty
    {
        temp <<= 1;             // shift whatever was in temp left to create room for the next bit
        temp |= (value & 0x01); // get the lsb from value and set as lsb in temp
        value >>= 1;            // shift value right by one to look at next lsb
    
        counter++;
    }
    
    value = temp;
    
    if (counter < number_of_bits)
    {
        value <<= counter-number_of_bits;
    }
    

    (I'm assuming that you know how many bits value holds and it is stored in number_of_bits)

    Obviously temp needs to be the longest imaginable data type and when you copy temp back into value, all the extraneous bits in temp should magically vanish (I think!).

    Or, the 'c' way would be to say :

    while(value)
    

    your choice

    0 讨论(0)
提交回复
热议问题