Добавление без использования арифметических или побитовых операторов

Как я могу добавить 2 числа без использования + или побитовых операторов?

3 ответа

Решение

Здесь вы идете, сложение и вычитание в C++, используя только сравнение и логические операторы. Я даже не использовал индексирование массивов, потому что для этого потребовалась бы неявная арифметика с указателями - и разрешение арифметики с указателями могло бы упростить это.

#include <iostream>
#include <iomanip>
#include <ctime>
using namespace std;

typedef unsigned char byte;

enum {
    LITTLE_ENDIAN,
    BIG_ENDIAN,
} endianness;

union dword_union {
    long dword;
    struct {
        byte byte0;
        byte byte1;
        byte byte2;
        byte byte3;
    } bytes;
};

struct byte_bits {
    bool bit0; bool bit1; bool bit2; bool bit3; bool bit4; bool bit5; bool bit6; bool bit7;
};

struct dword_bytes {
    byte_bits byte0;
    byte_bits byte1;
    byte_bits byte2;
    byte_bits byte3;
};

void init_endianness()
{
    dword_union db;
    db.bytes.byte0=1;
    db.bytes.byte1=2;
    db.bytes.byte2=3;
    db.bytes.byte3=4;
    {{}} if (db.dword == 0x04030201) endianness = LITTLE_ENDIAN;
    else if (db.dword == 0x01020304) endianness = BIG_ENDIAN;
    else
        throw "Unknown endianness";
}

#define R(a,b) (x>=a && x<=b)
void byte_to_bits(byte_bits &bits, byte x)
{
    bits.bit0 = R(1,1)||R(3,3)||R(5,5)||R(7,7)||R(9,9)||R(11,11)||R(13,13)||R(15,15)||R(17,17)||R(19,19)||R(21,21)||R(23,23)||R(25,25)||R(27,27)||R(29,29)||R(31,31)||R(33,33)||R(35,35)||R(37,37)||R(39,39)||R(41,41)||R(43,43)||R(45,45)||R(47,47)||R(49,49)||R(51,51)||R(53,53)||R(55,55)||R(57,57)||R(59,59)||R(61,61)||R(63,63)||R(65,65)||R(67,67)||R(69,69)||R(71,71)||R(73,73)||R(75,75)||R(77,77)||R(79,79)||R(81,81)||R(83,83)||R(85,85)||R(87,87)||R(89,89)||R(91,91)||R(93,93)||R(95,95)||R(97,97)||R(99,99)||R(101,101)||R(103,103)||R(105,105)||R(107,107)||R(109,109)||R(111,111)||R(113,113)||R(115,115)||R(117,117)||R(119,119)||R(121,121)||R(123,123)||R(125,125)||R(127,127)||R(129,129)||R(131,131)||R(133,133)||R(135,135)||R(137,137)||R(139,139)||R(141,141)||R(143,143)||R(145,145)||R(147,147)||R(149,149)||R(151,151)||R(153,153)||R(155,155)||R(157,157)||R(159,159)||R(161,161)||R(163,163)||R(165,165)||R(167,167)||R(169,169)||R(171,171)||R(173,173)||R(175,175)||R(177,177)||R(179,179)||R(181,181)||R(183,183)||R(185,185)||R(187,187)||R(189,189)||R(191,191)||R(193,193)||R(195,195)||R(197,197)||R(199,199)||R(201,201)||R(203,203)||R(205,205)||R(207,207)||R(209,209)||R(211,211)||R(213,213)||R(215,215)||R(217,217)||R(219,219)||R(221,221)||R(223,223)||R(225,225)||R(227,227)||R(229,229)||R(231,231)||R(233,233)||R(235,235)||R(237,237)||R(239,239)||R(241,241)||R(243,243)||R(245,245)||R(247,247)||R(249,249)||R(251,251)||R(253,253)||R(255,255);
    bits.bit1 = R(2,3)||R(6,7)||R(10,11)||R(14,15)||R(18,19)||R(22,23)||R(26,27)||R(30,31)||R(34,35)||R(38,39)||R(42,43)||R(46,47)||R(50,51)||R(54,55)||R(58,59)||R(62,63)||R(66,67)||R(70,71)||R(74,75)||R(78,79)||R(82,83)||R(86,87)||R(90,91)||R(94,95)||R(98,99)||R(102,103)||R(106,107)||R(110,111)||R(114,115)||R(118,119)||R(122,123)||R(126,127)||R(130,131)||R(134,135)||R(138,139)||R(142,143)||R(146,147)||R(150,151)||R(154,155)||R(158,159)||R(162,163)||R(166,167)||R(170,171)||R(174,175)||R(178,179)||R(182,183)||R(186,187)||R(190,191)||R(194,195)||R(198,199)||R(202,203)||R(206,207)||R(210,211)||R(214,215)||R(218,219)||R(222,223)||R(226,227)||R(230,231)||R(234,235)||R(238,239)||R(242,243)||R(246,247)||R(250,251)||R(254,255);
    bits.bit2 = R(4,7)||R(12,15)||R(20,23)||R(28,31)||R(36,39)||R(44,47)||R(52,55)||R(60,63)||R(68,71)||R(76,79)||R(84,87)||R(92,95)||R(100,103)||R(108,111)||R(116,119)||R(124,127)||R(132,135)||R(140,143)||R(148,151)||R(156,159)||R(164,167)||R(172,175)||R(180,183)||R(188,191)||R(196,199)||R(204,207)||R(212,215)||R(220,223)||R(228,231)||R(236,239)||R(244,247)||R(252,255);
    bits.bit3 = R(8,15)||R(24,31)||R(40,47)||R(56,63)||R(72,79)||R(88,95)||R(104,111)||R(120,127)||R(136,143)||R(152,159)||R(168,175)||R(184,191)||R(200,207)||R(216,223)||R(232,239)||R(248,255);
    bits.bit4 = R(16,31)||R(48,63)||R(80,95)||R(112,127)||R(144,159)||R(176,191)||R(208,223)||R(240,255);
    bits.bit5 = R(32,63)||R(96,127)||R(160,191)||R(224,255);
    bits.bit6 = R(64,127)||R(192,255);
    bits.bit7 = R(128,255);
}

long random_dword()
{
    return rand() + (rand()<<15) + (rand()<<30);
}

void byte_not(byte_bits &a, const byte_bits &x)
{
    a.bit0 = !x.bit0;
    a.bit1 = !x.bit1;
    a.bit2 = !x.bit2;
    a.bit3 = !x.bit3;
    a.bit4 = !x.bit4;
    a.bit5 = !x.bit5;
    a.bit6 = !x.bit6;
    a.bit7 = !x.bit7;
}

void byte_and(byte_bits &a, const byte_bits &x, const byte_bits &y)
{
    a.bit0 = x.bit0 && y.bit0;
    a.bit1 = x.bit1 && y.bit1;
    a.bit2 = x.bit2 && y.bit2;
    a.bit3 = x.bit3 && y.bit3;
    a.bit4 = x.bit4 && y.bit4;
    a.bit5 = x.bit5 && y.bit5;
    a.bit6 = x.bit6 && y.bit6;
    a.bit7 = x.bit7 && y.bit7;
}

void byte_xor(byte_bits &a, const byte_bits &x, const byte_bits &y)
{
    a.bit0 = x.bit0 != y.bit0;
    a.bit1 = x.bit1 != y.bit1;
    a.bit2 = x.bit2 != y.bit2;
    a.bit3 = x.bit3 != y.bit3;
    a.bit4 = x.bit4 != y.bit4;
    a.bit5 = x.bit5 != y.bit5;
    a.bit6 = x.bit6 != y.bit6;
    a.bit7 = x.bit7 != y.bit7;
}

bool byte_nonzero(const byte_bits &x)
{
    return x.bit0 || x.bit1 || x.bit2 || x.bit3 || x.bit4 || x.bit5 || x.bit6 || x.bit7;
}

class bit_int {
public:
    bit_int() {}
    bit_int(const dword_bytes &x)
    {
        bytes = x;
    }
    bit_int(const bit_int &x)
    {
        bytes = x.bytes;
    }
    bit_int(long x)
    {
        dword_union db;
        db.dword = x;
        if (endianness==LITTLE_ENDIAN)
        {
            byte_to_bits(bytes.byte0, db.bytes.byte0);
            byte_to_bits(bytes.byte1, db.bytes.byte1);
            byte_to_bits(bytes.byte2, db.bytes.byte2);
            byte_to_bits(bytes.byte3, db.bytes.byte3);
        }
        else
        {
            byte_to_bits(bytes.byte0, db.bytes.byte3);
            byte_to_bits(bytes.byte1, db.bytes.byte2);
            byte_to_bits(bytes.byte2, db.bytes.byte1);
            byte_to_bits(bytes.byte3, db.bytes.byte0);
        }
    }
    bit_int operator~() const
    {
        bit_int copy;
        byte_not(copy.bytes.byte0, bytes.byte0);
        byte_not(copy.bytes.byte1, bytes.byte1);
        byte_not(copy.bytes.byte2, bytes.byte2);
        byte_not(copy.bytes.byte3, bytes.byte3);
        return copy;
    }
    bit_int operator&(const bit_int &y) const
    {
        bit_int copy;
        byte_and(copy.bytes.byte0, bytes.byte0, y.bytes.byte0);
        byte_and(copy.bytes.byte1, bytes.byte1, y.bytes.byte1);
        byte_and(copy.bytes.byte2, bytes.byte2, y.bytes.byte2);
        byte_and(copy.bytes.byte3, bytes.byte3, y.bytes.byte3);
        return copy;
    }
    bit_int operator^(const bit_int &y) const
    {
        bit_int copy;
        byte_xor(copy.bytes.byte0, bytes.byte0, y.bytes.byte0);
        byte_xor(copy.bytes.byte1, bytes.byte1, y.bytes.byte1);
        byte_xor(copy.bytes.byte2, bytes.byte2, y.bytes.byte2);
        byte_xor(copy.bytes.byte3, bytes.byte3, y.bytes.byte3);
        return copy;
    }
    bit_int lshift() const
    {
        bit_int copy;
        copy.shifter.shifted = bytes;
        copy.bytes.byte0.bit0 = 0;
        return copy;
    }
    operator bool() const
    {
        return byte_nonzero(bytes.byte0) || byte_nonzero(bytes.byte1) || byte_nonzero(bytes.byte2) || byte_nonzero(bytes.byte3);
        return true;
    }
    bit_int operator+(const bit_int &addend)
    {
        bit_int x = *this;
        bit_int y = addend;
        bit_int a, b;
        do {
            a = x & y;
            b = x ^ y;
            x = b;
            y = a.lshift();
        } while (a);
        return b;
    }
    bit_int operator-(const bit_int &addend)
    {
        bit_int x = *this;
        bit_int y = addend;
        bit_int a, b;
        do {
            a = ~x & y;
            b =  x ^ y;
            x = b;
            y = a.lshift();
        } while (a);
        return b;
    }
    friend ostream &operator<<(ostream &out, const bit_int &x)
    {
        out << x.nibbles.nibble7 << x.nibbles.nibble6 << x.nibbles.nibble5 << x.nibbles.nibble4;
        out << x.nibbles.nibble3 << x.nibbles.nibble2 << x.nibbles.nibble1 << x.nibbles.nibble0;
        return out;
    }

private:
    struct nibble {
        bool bit0;
        bool bit1;
        bool bit2;
        bool bit3;

        friend ostream &operator<<(ostream &out, const nibble &a)
        {
            return out << (a.bit3 ? a.bit2 ? a.bit1 ? a.bit0 ? 'F' :
                                                               'E' :
                                                      a.bit0 ? 'D' :
                                                               'C' :
                                             a.bit1 ? a.bit0 ? 'B' :
                                                               'A' :
                                                      a.bit0 ? '9' :
                                                               '8' :
                                    a.bit2 ? a.bit1 ? a.bit0 ? '7' :
                                                               '6' :
                                                      a.bit0 ? '5' :
                                                               '4' :
                                             a.bit1 ? a.bit0 ? '3' :
                                                               '2' :
                                                      a.bit0 ? '1' :
                                                               '0');
        }
    };

    union {
        dword_bytes bytes;
        struct {
            nibble nibble0;
            nibble nibble1;
            nibble nibble2;
            nibble nibble3;
            nibble nibble4;
            nibble nibble5;
            nibble nibble6;
            nibble nibble7;
        } nibbles;
        struct {
            bool highbit;
            dword_bytes shifted;
        } shifter;
    };
};

void main()
{
    try {
        init_endianness();
        srand((int)time(NULL));
        int x = random_dword();
        int y = random_dword();
        bit_int a(x);
        bit_int b(y);
        cout << "0x" << a << " + " << "0x" << b << " = " << "0x" << (a + b) << endl;
        cout << "0x" << a << " - " << "0x" << b << " = " << "0x" << (a - b) << endl;
    }
    catch (char *error) {
        printf("Error: %s\n", error);
    }
}

Индекс K&R показывает sizeof как оператора, плюс в этом фрагменте что-то мешает моей Zen Thing. Может кто-нибудь мне помочь?

/* sum: Add two numbers without using an operand. 

/* I assume limits.h CHARBIT is set to 8. 

/* I assume both input integers are positive. 

int main( int argc, char *argv[]) {

    return sum( (int) argv[1], (int) argv[2] );

};

int sum( int leftOperand, int rightOperand ) {

    static struct operands {

        operandOne char[ leftOperand ];

        operandTwo char[ rightOperand ];

    };

    return sizeof( operands );
};

Могу я спросить, зачем тебе это нужно? Вы ДОЛЖНЫ использовать какой-либо оператор сложения / побитовый оператор, насколько я знаю... или я думаю, вместо того, чтобы делать a+b, вы можете написать a - (b*-1) .. хотя это не имеет никакого смысла вообще.

Другие вопросы по тегам