01-JAVA基础—>赏金任务—>三色旗问题

记录一下三色旗问题 对应LeetCode的 75 题
友情链接1
友情链接2
友情链接3
在这里插入图片描述

背景:三色旗的问题最早由E.W.Dijkstra所提出,一条绳子上有红、白、蓝三种颜色的旗子,旗子颜色并没有顺序,将之分类,并排列为B蓝、W白、R红的顺序,要如何移动次数才会最少,注意您只能在绳子上进行这个动作(相当于对一个数组or集合操作),而且一次只能调换两个旗子
在这里插入图片描述

最快的就是 从左边开始 遇到B蓝放到左边去,遇到W白不动,遇到R红放到右边去

但是我们可以先这样

	public void sortColors(int[] A) {
		int n = A.length;
		int i = 0;// 0的个数
		int j = 0;// 1的个数
		int k = 0;// 2的个数
		for (int p = 0; p < n; p++) {
			if (A[p] == 0) {	i++;
			} else if (A[p] == 1) { j++;
			} else   k++;
		}
 
		for (int p = 0; p < n; p++) {
			if (p < i)  A[p] = 0;
			else if (p >= i && p < i + j)  A[p] = 1;
			else   A[p] = 2;
		}}

这也是一个清奇的想法,先记录BWR的各个数量然后直接改变数组里面的数据。

但这样就还没有实现 只扫一遍,左边存放B&&W,右边存放R.两边往中间靠。

为了实现我们的梦想,我们可以找来两个指针
left记录第一个1的位置,left左边为0,
right记录第一个非2的位置,right右边为2.然后使用i从头到尾扫一遍,直到与right相遇。
i 遇到0就换到左边去,遇到2就换到右边去,遇到1就跳过。
在这里插入图片描述
实现代码

	public void sortColors(int[] A) {
 
		int left = 0;
		int right = A.length - 1;
		int i = 0;
		while (i <= right) {
			if (A[i] == 0) {
				swap(A, left, i);
				left++;
				i++;
			} else if (A[i] == 1) {
				i++;
			} else {
				swap(A, i, right);
				right--;
			}
		}
	}
 
 
	private void swap(int[] nums, int i, int j) {
		int t = nums[i];
		nums[i] = nums[j];
		nums[j] = t;
	}

代码注意点 思考为什么 if (A[i] == 0) 要 i++;
这是因为left记录第一个1的位置,因此A[left]与A[i]交换后,A[left]为0,A[i]为1,因此i++;
再来 为什么 A[i] 和 right 交换后不++呢??
这是因为right记录第一个非2的位置,可能为0或1,因此A[right]与A[i]交换后,A[right]为2,A[i]为0或1,i不能前进,要后续判断。

做一下优化就可以实现我们的梦想了

public void sortColors(int[] nums) {
		int j = 0, k = nums.length - 1;
		for (int i = 0; i <= k; i++) {
			// 遇到0和前面的交换
			if (nums[i] == 0)
				swap(nums, i, j++);
			// 遇到2和后面的交换
			else if (nums[i] == 2)
				swap(nums, i--, k--);
		}
	}
 
	private void swap(int[] nums, int i, int j) {
		int t = nums[i];
		nums[i] = nums[j];
		nums[j] = t;
	}

运行结果
在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

高高飞起的勇敢麦当

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值