异或交换算法中的小坑

突然想起来挺久前的一件事,因为太琐碎就不放到「深夜学算法」系列里了。

「交换两数」大概是编程入门者紧接着Hello World写的程序,常用和知名到都有段子:

// 普通程序员
void normal_swap(int &x, int &y) {
    int temp = x;
    x = y;
    y = temp;
}

// 二逼程序员
void foolish_swap(int x, int y) {
    int temp = x;
    x = y;
    y = temp;
}

通常交换两数都会使用临时变量temp,但是也有不使用的方法,称为异或交换算法(XOR swap algorithm):

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

算法本身算是广为人知,原理也很简单。假设x = 3,y = 5,那么分别写成二进制形式就是x = 00000011,y = 00000101

  1. 经过 x ^= y
    x = 00000110,y = 00000101
  2. 经过y ^= x
    x = 00000110,y = 00000011
  3. 经过x ^= y
    x = 00000101,y = 00000011

此时x = 5,y = 3,交换成功。

看起来非常漂亮,但用C/C++实现时,有一个隐藏的小坑。


知道这个交换算法时我正在练习各种排序算法,就直接用在快速排序里了。写完代码一运行,别说排好序了简直比随机数还随机数。

当然作为C/C++学习者,我的第一反应是:

又搞错指针了耶…

但对着教材看了几遍快速排序的实现都没找到错误,用特殊输入试了几次后,我突然发现了问题所在。

交换两数有一个特例:「要交换的两个数指向同一片存储区域」。

什么意思呢?就是说可能会调用swap(x, x),这件事虽然很傻,但调用后应该保证x的值不变。然而上面实现的xor_swap就在这里出了问题。

还是假设x = 3,写成二进制就是00000011:

  1. 经过x ^= y
    x = 00000000
  2. 经过y ^= x
    x = 00000000
  3. 经过x ^= y
    x = 00000000

所以无论x原来的值是什么,调用swap(x, x)后都变成0了。

好一个すべてがゼロになる(全部成为0)!

解决起来很简单,判断一下两个数是否相等,相等不用交换,不相等肯定不指向同一片存储区域。

所以正确的异或交换算法实现应该是:

void xor_swap(int &x, int &y) {
    if(x != y) {
        x ^= y;
        y ^= x;
        x ^= y;
    }
}

这件事情的启示有两条:

  1. 遇事不决,先看wiki
  2. 数学算法和程序实现还是有差别的
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,923评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,154评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,775评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,960评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,976评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,972评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,893评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,709评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,159评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,400评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,552评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,265评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,876评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,528评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,701评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,552评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,451评论 2 352

推荐阅读更多精彩内容

  • 背景 一年多以前我在知乎上答了有关LeetCode的问题, 分享了一些自己做题目的经验。 张土汪:刷leetcod...
    土汪阅读 12,743评论 0 33
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,621评论 18 399
  • 总结一下常见的排序算法。 排序分内排序和外排序。内排序:指在排序期间数据对象全部存放在内存的排序。外排序:指在排序...
    jiangliang阅读 1,338评论 0 1
  • 首页 资讯 文章 资源 小组 相亲 登录 注册 首页 最新文章 IT 职场 前端 后端 移动端 数据库 运维 其他...
    Helen_Cat阅读 3,869评论 1 10
  • 医学证明,如果人体倾向酸性,体内细胞的作用就会变弱,废物就不易排出,肾脏、肝脏的负担就会加大,新陈代谢缓慢,各器官...
    刘祥168阅读 158评论 0 0