C++ 实现冒泡排序法

 冒泡排序法

这里使用C++来实现冒泡排序法

冒泡排序法采用不停地交换彼此位置来实现,故而形象地称之为冒泡

大致一个气泡从水底一直冒到水面

一小段排序程序如下

        1、外层循环:主要是用来轮询

        2、内层循环:主要用来交换位置(前提是满足if条件)

        3、外层循环的停止条件 i < len 这应该能理解,内层循环的停止条件 j < len -i -1 来说明一下: 首先 len - 1 这个为什么要减掉 1, 因为下面 if 中有使用 buf [j+1], 所以这里要减掉1,不然 buf [j+1] 就会发生越界行为(超出了数组最大数组下标),接着看 len-i,为什么又要减掉 i 呢?因为内层循环每完成一次,最大那个值(或最小那个值),就已经交换到了最后的位置,所以下次交换的时候我们就要减少一次交换已经排序好的元素,依次类推,所以就有 len - i 这样的条件,最后合起来就是 j < len - i - 1

void sort(int *buf, int len)
{
	for (int i = 0; i < len; ++i) { // 外层循环
		for (int j = 0; j < len - i - 1; ++j) { // 内层循环
			if (buf[j] > buf[j + 1]) { // 如果满足条件就交换位置
				int temp = buf[j];
				buf[j] = buf[j + 1];
				buf[j + 1] = temp;
			}
		}
	}
}

接着我们可以看看,这个冒泡排序一共要执行多少次才完成了,对程序进行修改一下,添加了 cnt 变量

void sort1(int *buf, int len)
{
	int cnt = 0; // 用来计数,看看执行了多少次

	for (int i = 0; i < len; ++i) {
		for (int j = 0; j < len - i - 1; ++j) {
			if (buf[j] > buf[j + 1]) { 
				int temp = buf[j];
				buf[j] = buf[j + 1];
				buf[j + 1] = temp;
			}
		}

		++cnt; // 在这里记录外层循环的次数
	}

	cout << "一共执行了:" << cnt << endl;
}

int main(int argc, char* argv[])
{
	int buf1[] = { 5,0,1,2,3,4 };
	sort1(buf1, sizeof(buf1) / sizeof(int));
	return 0;
}

运行结果可以看到,外层循环是全部执行完才能完成排序

 但上面主函数的 buf1 数组中,只有 5 是需要排序的,其他的都不用排序,好像就是直接把 5 放在最后面即可完成排序了,也就是进行排序 1 次就行了,现在在小优化一下

void sort2(int* buf, int len)
{
	int cnt = 0;
	bool finish = true; // 添加了停止条件

	for (int i = 0; i < len; ++i) {
		finish = true;
		for (int j = 0; j < len - i - 1; ++j) {
			if (buf[j] > buf[j + 1]) {
				int temp = buf[j];
				buf[j] = buf[j + 1];
				buf[j + 1] = temp;
				finish = false;	// 发生交换了,不停止循环
			}
			
		}

		if (finish) // 如果没有发生交换位置,就结束循环
			break;

		++cnt;
	}

	cout << "一共执行了:" << cnt << endl;
}

int main(int argc, char* argv[])
{
	int buf1[] = { 5,0,1,2,3,4 };
	sort2(buf1, sizeof(buf2) / sizeof(int));

	return 0;
}

 运行结果,符合我们的预想

完整代码如下

#include <iostream>
using namespace std;

/* 冒泡排序法2 */
void sort1(int *buf, int len)
{
	int cnt = 0; // 用来计数,看看执行了多少次

	for (int i = 0; i < len; ++i) {
		for (int j = 0; j < len - i - 1; ++j) {
			if (buf[j] > buf[j + 1]) { 
				int temp = buf[j];
				buf[j] = buf[j + 1];
				buf[j + 1] = temp;
			}
		}

		++cnt; // 在这里记录外层循环的次数
	}

	cout << "一共执行了:" << cnt << endl;
}

/* 冒泡排序法2 */
void sort2(int* buf, int len)
{
	int cnt = 0;
	bool finish = true; // 添加了停止条件

	for (int i = 0; i < len; ++i) {
		finish = true;
		for (int j = 0; j < len - i - 1; ++j) {
			if (buf[j] > buf[j + 1]) {
				int temp = buf[j];
				buf[j] = buf[j + 1];
				buf[j + 1] = temp;
				finish = false;	// 发生交换了,不停止循环
			}
			
		}

		if (finish) // 如果没有发生交换位置,就结束循环
			break;

		++cnt;
	}

	cout << "一共执行了:" << cnt << endl;
}

int main(int argc, char* argv[])
{
	// 2 个元素相同的数组
	int buf1[] = { 5,0,1,2,3,4 };
	int buf2[] = { 5,0,1,2,3,4 };

	sort1(buf1, sizeof(buf1) / sizeof(int)); /* 冒泡排序法1 */
	sort2(buf2, sizeof(buf2) / sizeof(int)); /* 冒泡排序法2 */

	return 0;
}

当然,这不是万能的哈,比如这样,它就不适用了,把 buf1 和把 buf2 的元素是完全倒序

#include <iostream>
using namespace std;

/* 冒泡排序法2 */
void sort1(int *buf, int len)
{
	int cnt = 0; // 用来计数,看看执行了多少次

	for (int i = 0; i < len; ++i) {
		for (int j = 0; j < len - i - 1; ++j) {
			if (buf[j] > buf[j + 1]) { 
				int temp = buf[j];
				buf[j] = buf[j + 1];
				buf[j + 1] = temp;
			}
		}

		++cnt; // 在这里记录外层循环的次数
	}

	cout << "一共执行了:" << cnt << endl;
}

/* 冒泡排序法2 */
void sort2(int* buf, int len)
{
	int cnt = 0;
	bool finish = true; // 添加了停止条件

	for (int i = 0; i < len; ++i) {
		finish = true;
		for (int j = 0; j < len - i - 1; ++j) {
			if (buf[j] > buf[j + 1]) {
				int temp = buf[j];
				buf[j] = buf[j + 1];
				buf[j + 1] = temp;
				finish = false;	// 发生交换了,不停止循环
			}
			
		}

		if (finish) // 如果没有发生交换位置,就结束循环
			break;

		++cnt;
	}

	cout << "一共执行了:" << cnt << endl;
}

int main(int argc, char* argv[])
{
	// 2 个元素相同的数组
	int buf1[] = { 5,4,3,2,1,0 };
	int buf2[] = { 5,4,3,2,1,0 };

	sort1(buf1, sizeof(buf1) / sizeof(int)); /* 冒泡排序法1 */
	sort2(buf2, sizeof(buf2) / sizeof(int)); /* 冒泡排序法2 */

	return 0;
}

 基本上,sort1 和 sort2 的执行次是一样的同样的 6 次,那上面结果为什么是 sort16 而 sort25,原因是在 if 那里退出了,cnt 少加了 1 次,不过这个影响不大。

结果说明,某种的优化只能说针对某一个方面,最后还是得见招拆招。

  • 9
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值