记录一下三色旗问题 对应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;
}
运行结果