Bitwise operations equivalent of greater than operator

后端 未结 8 1569
花落未央
花落未央 2020-12-05 10:30

I am working on a function that will essentially see which of two ints is larger. The parameters that are passed are 2 32-bit ints. The trick is the only operat

相关标签:
8条回答
  • 2020-12-05 10:52

    As much as I don't want to do someone else's homework I couldn't resist this one.. :) I am sure others can think of a more compact one..but here is mine..works well, including negative numbers..

    Edit: there are couple of bugs though. I will leave it to the OP to find it and fix it.

        #include<unistd.h>
        #include<stdio.h>
        int a, b, i, ma, mb, a_neg, b_neg, stop;
    
        int flipnum(int *num, int *is_neg) {
            *num = ~(*num) + 1;
            *is_neg = 1;
    
            return 0;
        }
    
        int print_num1() {
            return ((a_neg && printf("bigger number %d\n", mb)) ||
                 printf("bigger number %d\n", ma));
        }
    
        int print_num2() {
            return ((b_neg && printf("bigger number %d\n", ma)) ||
                 printf("bigger number %d\n", mb));
        }
    
        int check_num1(int j) {
            return ((a & j) && print_num1());
        }
    
        int check_num2(int j) {
            return ((b & j) && print_num2());
        }
    
        int recursive_check (int j) {
            ((a & j) ^ (b & j)) && (check_num1(j) || check_num2(j))  && (stop = 1, j = 0);
    
            return(!stop && (j = j >> 1) && recursive_check(j));
        }
    
        int main() {
            int j;
            scanf("%d%d", &a, &b);
            ma = a; mb = b;
    
            i = (sizeof (int) * 8) - 1;
            j = 1 << i;
    
            ((a & j) && flipnum(&a, &a_neg));
    
            ((b & j) && flipnum(&b, &b_neg));
    
            j = 1 << (i - 1);
    
            recursive_check(j);
    
            (!stop && printf("numbers are same..\n"));
        }
    
    0 讨论(0)
  • 2020-12-05 10:53

    EDIT:

    Okay, there were some issues with the code, but I revised it and the following works.

    This auxiliary function compares the numbers' n'th significant digit:

    int compare ( int a, int b, int n )
    {
        int digit = (0x1 << n-1);
        if ( (a & digit) && (b & digit) )
           return 0; //the digit is the same
    
        if ( (a & digit) && !(b & digit) )
           return 1; //a is greater than b
    
        if ( !(a & digit) && (b & digit) )
           return -1; //b is greater than a
    }
    

    The following should recursively return the larger number:

    int larger ( int a, int b )
    {
        for ( int i = 8*sizeof(a) - 1 ; i >= 0 ; i-- )
        {
           if ( int k = compare ( a, b, i ) )
           {
               return (k == 1) ? a : b;
           }
        }
        return 0; //equal
    }
    
    0 讨论(0)
  • 2020-12-05 11:02

    An unsigned variant given that one can use logical (&&, ||) and comparison (!=, ==).

    int u_isgt(unsigned int a, unsigned int b)
    {
        return a != b && (    /* If a == b then a !> b and a !< b.             */
                   b == 0 ||  /* Else if b == 0 a has to be > b (as a != 0).   */
                   (a / b)    /* Else divide; integer division always truncate */
               );             /*              towards zero. Giving 0 if a < b. */
    }
    

    != and == can easily be eliminated., i.e.:

    int u_isgt(unsigned int a, unsigned int b)
    {
        return a ^ b && (
                   !(b ^ 0) ||
                   (a / b)
               );
    }
    

    For signed one could then expand to something like:

    int isgt(int a, int b)
    {
        return
        (a != b) &&
        (
            (!(0x80000000 & a) && 0x80000000 & b) ||  /* if a >= 0 && b < 0  */
            (!(0x80000000 & a) && b == 0) ||
            /* Two more lines, can add them if you like, but as it is homework
             * I'll leave it up to you to decide. 
             * Hint: check on "both negative" and "both not negative". */
        )
        ;
    }
    

    Can be more compact / eliminate ops. (at least one) but put it like this for clarity.

    Instead of 0x80000000 one could say ie:

    #include <limits.h>
    static const int INT_NEG = (1 << ((sizeof(int) * CHAR_BIT) - 1));
    

    Using this to test:

    void test_isgt(int a, int b)
    {
        fprintf(stdout,
            "%11d > %11d = %d : %d %s\n",
            a, b,
            isgt(a, b), (a > b),
            isgt(a, b) != (a>b) ? "BAD!" : "OK!");
    }
    

    Result:

             33 >           0 = 1 : 1 OK!
            -33 >           0 = 0 : 0 OK!
              0 >          33 = 0 : 0 OK!
              0 >         -33 = 1 : 1 OK!
              0 >           0 = 0 : 0 OK!
             33 >          33 = 0 : 0 OK!
            -33 >         -33 = 0 : 0 OK!
             -5 >         -33 = 1 : 1 OK!
            -33 >          -5 = 0 : 0 OK!
    -2147483647 >  2147483647 = 0 : 0 OK!
     2147483647 > -2147483647 = 1 : 1 OK!
     2147483647 >  2147483647 = 0 : 0 OK!
     2147483647 >           0 = 1 : 1 OK!
              0 >  2147483647 = 0 : 0 OK!
    
    0 讨论(0)
  • 2020-12-05 11:04

    Here's a loop-free version which compares unsigned integers in O(lg b) operations where b is the word size of the machine. Note the OP states no other data types than signed int, so it seems likely the top part of this answer does not meet the OP's specifications. (Spoiler version as at the bottom.)

    Note that the behavior we want to capture is when the most significant bit mismatch is 1 for a and 0 for b. Another way of thinking about this is any bit in a being larger than the corresponding bit in b means a is greater than b, so long as there wasn't an earlier bit in a that was less than the corresponding bit in b.

    To that end, we compute all the bits in a greater than the corresponding bits in b, and likewise compute all the bits in a less than the corresponding bits in b. We now want to mask out all the 'greater than' bits that are below any 'less than' bits, so we take all the 'less than' bits and smear them all to the right making a mask: the most significant bit set all the way down to the least significant bit are now 1.

    Now all we have to do is remove the 'greater than' bits set by using simple bit masking logic.

    The resulting value is 0 if a <= b and nonzero if a > b. If we want it to be 1 in the latter case we can do a similar smearing trick and just take a look at the least significant bit.

    #include <stdio.h>
    
    // Works for unsigned ints.
    // Scroll down to the "actual algorithm" to see the interesting code.
    
    // Utility function for displaying binary representation of an unsigned integer
    void printBin(unsigned int x) {
        for (int i = 31; i >= 0; i--) printf("%i", (x >> i) & 1);
        printf("\n");
    }
    // Utility function to print out a separator
    void printSep() {
        for (int i = 31; i>= 0; i--) printf("-");
        printf("\n");
    }
    
    int main()
    {
        while (1)
        {
            unsigned int a, b;
    
            printf("Enter two unsigned integers separated by spaces: ");
            scanf("%u %u", &a, &b);
            getchar();
    
            printBin(a);
            printBin(b);
            printSep();
    
                /************ The actual algorithm starts here ************/
    
            // These are all the bits in a that are less than their corresponding bits in b.
            unsigned int ltb = ~a & b;
    
            // These are all the bits in a that are greater than their corresponding bits in b.
            unsigned int gtb = a & ~b;
    
            ltb |= ltb >> 1;
            ltb |= ltb >> 2;
            ltb |= ltb >> 4;
            ltb |= ltb >> 8;
            ltb |= ltb >> 16;
    
            // Nonzero if a > b
            // Zero if a <= b
            unsigned int isGt = gtb & ~ltb;
    
            // If you want to make this exactly '1' when nonzero do this part:
            isGt |= isGt >> 1;
            isGt |= isGt >> 2;
            isGt |= isGt >> 4;
            isGt |= isGt >> 8;
            isGt |= isGt >> 16;
            isGt &= 1;
    
                /************ The actual algorithm ends here ************/
    
            // Print out the results.
            printBin(ltb); // Debug info
            printBin(gtb); // Debug info
            printSep();
            printBin(isGt); // The actual result
        }
    }
    

    Note: This should work for signed integers as well if you flip the top bit on both of the inputs, e.g. a ^= 0x80000000.

    Spoiler

    If you want an answer that meets all of the requirements (including 25 operators or less):

    int isGt(int a, int b)
    {
        int diff = a ^ b;
        diff |= diff >> 1;
        diff |= diff >> 2;
        diff |= diff >> 4;
        diff |= diff >> 8;
        diff |= diff >> 16;
    
        diff &= ~(diff >> 1) | 0x80000000;
        diff &= (a ^ 0x80000000) & (b ^ 0x7fffffff);
    
        return !!diff;
    }
    

    I'll leave explaining why it works up to you.

    0 讨论(0)
  • 2020-12-05 11:07

    A fully branchless version of Kaganar's smaller isGt function might look like so:

    int isGt(int a, int b)
    {
        int diff = a ^ b;
        diff |= diff >> 1;
        diff |= diff >> 2;
        diff |= diff >> 4;
        diff |= diff >> 8;
        diff |= diff >> 16;
    
        //1+ on GT, 0 otherwise.
        diff &= ~(diff >> 1) | 0x80000000;
        diff &= (a ^ 0x80000000) & (b ^ 0x7fffffff);
    
        //flatten back to range of 0 or 1.
        diff |= diff >> 1;
        diff |= diff >> 2;
        diff |= diff >> 4;
        diff |= diff >> 8;
        diff |= diff >> 16;
        diff &= 1;
    
        return diff;
    }
    

    This clocks in at around 60 instructions for the actual computation (MSVC 2010 compiler, on an x86 arch), plus an extra 10 stack ops or so for the function's prolog/epilog.

    0 讨论(0)
  • 2020-12-05 11:10

    To convert 001xxxxx to 00100000, you first execute:

    x |= x >> 4;
    x |= x >> 2;
    x |= x >> 1;
    

    (this is for 8 bits; to extend it to 32, add shifts by 8 and 16 at the start of the sequence).

    This leaves us with 00111111 (this technique is sometimes called "bit-smearing"). We can then chop off all but the first 1 bit:

    x ^= x >> 1;
    

    leaving us with 00100000.

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