0. 前言

本文包括如下几个内容:

  • 异或算法
  • 异或加密
  • 两个整数交换问题


1. 异或算法

异或是数字逻辑中的基本概念,也是每种编程语言都支持的基本运算。基本原理就是,对于数字1和0有如下的运算公式:

1 ^ 1 = 0
0 ^ 0 = 0
1 ^ 0 = 1
0 ^ 1 = 1



很自然地,这个运算可以扩展到一长串码流上。



更常见的是字节一级或整数的异或运算,可以参考相应的编程语言的相关介绍。


2. 异或加密

异或加密是最简单的一种加密方法,简单来讲,假定有一串码流A,然后可以用key进行加密,把加密后的数据保存或发送出去。而在读取加密数据之后,再用key进行解密。

直接用代码更容易理解,如下示例:

#include <stdio.h>
#include <stdlib.h>

static void xor_encrypt(char *data, char *key, int len)
{
    int i = 0;
    for (i = 0; i < len; i++) {
        data[i] ^= key[i];
    }
}

static void dumphex(const unsigned char *data, int len)
{
    int i;
    for (i = 0; i < len; i++) {
        printf("%02X", data[i]);
    }
    printf("\n");
}

static unsigned char* get_random_key(int len)
{
    int i = 0;
    unsigned char* buffer = (unsigned char*)malloc(len);
    for (i = 0; i < len; i++) {
        buffer[i] = rand() % 10 + '0';
    }

    return buffer;
}

int main()
{
    unsigned char data[] = "hello, xor!";
    int len = sizeof(data);

    unsigned char *key;

    key = get_random_key(len);
    printf("original data: %s\n", data);
    printf("original data (hex): ");
    dumphex(data, len);

    xor_encrypt(data, key, len);
    printf("encrypted data: %s\n", data);
    printf("encrypted data (hex): ");
    dumphex(data, len);

    xor_encrypt(data, key, len);
    printf("original data: %s\n", data);
    printf("original data (hex): ");
    dumphex(data, len);

    free(key);

    return 0;
}



运行结果:

original data: hello, xor!
original data (hex): 68656C6C6F2C20786F722100
encrypted data: [S[Y\JVC7
encrypted data (hex): 5B535B595C19164A56431337
original data: hello, xor!
original data (hex): 68656C6C6F2C20786F722100



3. 交换两个整数

有些TX在面试的时候,可能会遇到这个问题,即:

有两个整数a和b,如何在不引入第三个变量的情况下,交换两个整数的值?



在了解了前面介绍的异或的概念之后,这个问题就迎刃而解。直接给出代码:

#include <stdio.h>

void swap(int &x, int &y)
{
    x ^= y;
    y ^= x;
    x ^= y;
}

int main()
{
    int x = 55555;
    int y = 33333;

    printf("first, x = %d, y = %d\n", x, y);
    swap(x, y);
    printf("and now, x = %d, y = %d\n", x, y);

    return 0;
}



执行结果:

flying-bird@flying-bird:~/docs/The_Road_to_Cpp/src$ g++ swap_ints.cc 
flying-bird@flying-bird:~/docs/The_Road_to_Cpp/src$ ./a.out 
first, x = 55555, y = 33333
and now, x = 33333, y = 55555
flying-bird@flying-bird:~/docs/The_Road_to_Cpp/src$



4. 再谈key

前面谈到异或加密的时候,提到的密钥key。如果是远程通信,发送方是把加密后的数据发出去的,那么接收方势必要用原始的密钥key才能解密。那么接收方如何才能拿到这个密钥呢?有关这方面,可以参考《Core Java》卷二关于加密一章的讨论。——老外的书总是深入浅出,通过一个个例子&故事,引出一系列高深的专题。推荐TXs能够好好阅读几遍。


这里再给出一种简单的数据&密钥发送发生,也是很多项目采用的方法。即:

  • 假定有原始数据data,长度为len;
  • 生成长度为len的一串密钥key(比如前面示例给出的随机数);
  • 发送数据规定为2*len长度,前面len部分是data^key,后面len部分是key;
  • 接收方收到数据之后,就用后面len部分异或到前面len部分,最后的结果就是原始数据。


当然了,可能这个也比较简单。那么可以在此基础上进一步引入其他的算法,比如:

  • 增加一个字节,引入奇偶校验码;
    将码流顺序按照一定的规律打乱,比如相邻的两个字节交换;
  • 等等


5. 参考资料

维基百科的异或加密介绍:http://zh.wikipedia.org/wiki/%E5%BC%82%E6%88%96%E5%8A%A0%E5%AF%86