文章:[不借助if、switch等语句求两个数较大的一个](http://blog.csdn.net/u013074465/article/details/42684559)
交换两个数在排序算法中用的很多:[冒泡排序中](http://blog.csdn.net/u013074465/article/details/44339187) 、[插入排序中](http://blog.csdn.net/u013074465/article/details/42043665)等等。正常的交换两个数是借助一个变量tmp:
~~~
void Swap(int &a, int &b) {
int tmp = a;
a = b;
b = tmp;
}
~~~
在面试题中有这样的题目“不借助第三个变量,交换两个数”
~~~
//A方法
void Swap(int &a, int &b) {
a = a + b;
b = a - b;
a = a - b;
}
~~~
~~~
//B方法
void Swap(int &a, int &b) {
a ^= b; b ^= a; a ^= b;
}
~~~
上述的两种方式中,A方法缺点:中当数a或b较大时 a = a + b会发生越界。
方法B的原理:
~~~
a = a ^ b;
b = a ^ b; //那么 b = a ^ b = (a ^ b) ^ b = a ^ (b ^ b) = a ^ 0 = a,即将a赋值给了b
a = a ^ b; //那么 a = a ^ b = a ^ (a ^ b) = (a ^ a) ^ b = 0 ^ b = b,即实现了将b赋值为a
~~~
以上两种方法的共有bug:
当a、b指向的是同一个数时,即Swap(a, a)时,会发生错误,将a置为了0。但是a等于b的情况下(即a和b不是指向同一个地址时)是不会错误的。
解决方法:
为Swap函数增加判断语句:当a和b相等的时候不交换。
~~~
//A方法
void test(int &a, int &b) {
if (a != b) {
a = a + b;
b = a - b;
a = a - b;
cout << "OK " << endl;
}
}
//B方法
void Swap(int &a, int &b) {
if (a != b) {
a ^= b;
b ^= a;
a ^= b;
}}
~~~
- 前言
- Josephus约瑟夫问题及其变种
- 链表的常见实现
- 二叉树遍历、插入、删除等常见操作
- 二叉堆的插入删除等操作C++实现
- 插入排序和希尔排序
- 堆排序
- 归并排序及其空间复杂度的思考
- 快速排序的几种常见实现及其性能对比
- 红黑树操作及实现
- 整数的二进制表示中1的个数
- 位操作实现加减乘除四则运算
- 冒泡排序的改进
- 直接选择排序
- 不借助变量交换两个数
- 基础排序算法总结
- AVL树(Adelson-Velskii-Landis tree)
- avl树的C++实现
- 动态规划之钢条分割
- hash函数的基本知识
- 动态规划:求最长公共子串/最长公共子序列
- 最长递增子序列
- 称砝码问题
- 汽水瓶
- 字符串合并处理(二进制位的倒序)
- 动态规划:计算字符串相似度
- m个苹果放入n个盘子
- 生成k个小于n的互不相同的随机数
- 栈和队列的相互模拟
- 字符串的排列/组合
- KMP(Knuth-Morris-Pratt)算法
- n个骰子的点数
- 位运算的常见操作和题目