本帖最后由 iooops 于 2016-4-6 04:18 编辑
【老掉牙系列之四】如果作为一个程序猿,你连这些都还不会,你就等着被 叭!
【说明】
三色旗的问题最早由E.W.Dijkstra所提出,他所使用的用语为Dutch Nation Flag(Dijkstra为荷兰
人),而多数的作者则使用Three-Color Flag来称之。
假设有一条绳子,上面有红、白、蓝三种颜色的旗子,起初绳子上的旗子颜色并没有顺序,您
希望将之分类,并排列为红、白、蓝的顺序,要如何移动次数才会最少,注意您只能在绳子上
进行这个动作,而且一次只能调换两个旗子。
其实这是一个快速排序的算法。
好吧不理解题意的筒子打开下面这个链接:
http://www.csse.monash.edu.au/~lloyd/tildeAlgDS/Sort/Flag/
然后到
Three Colours 版块
点go
然后就会出一个随机数,排好序啦。
觉得脑洞够大的筒子可以不看下面自行解决了。
好吧然后……
解决思路其实已经在链接里面被说明得很清楚了……
先来想办法把色列排成这样:
a[1..Lo-1] zeroes (red)
a[Lo..Mid-] ones (white)
a[Mid..Hi] unknown
a[Hi+1..N] twos (blue)
然后再进行下面一步:
Lo := 1; Mid := 1; Hi := N;
while Mid <= Hi do
Invariant: a[1..Lo-1]=0 and a[Lo..Mid-1]=1 and a[Hi+1..N]=2; a[Mid..Hi] are unknown.
case a[Mid] in
0: swap a[Lo] and a[Mid]; Lo++; Mid++
1: Mid++
2: swap a[Mid] and a[Hi]; Hi--
--- Dutch National Flag Algorithm, or 3-way Partitioning ---
至于配套的图嘛,看链接里的图好啦。
楼主反正是看完图才终于理解了该算法的精髓的。
有兴趣的筒子可以把上述0、1、2排序的结果用C语言实现一下。对于大神理应是轻而易举。对于楼主小白要去好好在墙角思索一下敲出代码来 。
那么三色旗的C语言实现代码如下(假设排列为BWR哈,至于排列为RWB,筒子们可以稍微改动一下哈):
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define SWAP_FLAGS(x, y) { char temp; \
temp = flags[x]; \
flags[x] = flags[y]; \
flags[y] = temp; }
void printFlags(char* flags) {
int i;
for(i = 0; i < strlen(flags); i++) {
printf("%c ", flags[i]);
}
printf("\n");
}
void adjust(char* flags) {
int w = 0;
int b = 0;
int r = strlen(flags) - 1;
while(flags[w] == 'B' && w < strlen(flags)) { b++; w++; }
while(flags[r] == 'R' && r > 0) { r--; }
while(w <= r) switch(flags[w]) {
case 'W':
w++;
break;
case 'B':
SWAP_FLAGS(b, w);
b++; w++;
break;
case 'R':
SWAP_FLAGS(r, w);
r--;
}
}
int main() {
char flags[] = {'R', 'W', 'B', 'W', 'W',
'B', 'R', 'B', 'W', 'R', '\0'};
printFlags(flags);
adjust(flags);
printFlags(flags);
return 0;
}
复制代码
输出结果如下:
R W B W W B R B W R
B B B W W W W R R R
好吧容我做一下代码的搬运工 - -
论数学的重要性 = =
刚刚提到排序0、1、2
楼主小白花了大概4个小时终于把上面的0、1、2能排出来了…………虽然还是参考了很多上述的代码,啊感觉还是要多多思索深入理解一下啊。
#include <stdio.h>
#include <stdlib.h>
void adjust(int *a, int n) {
int x = 0;
int y = 0;
int z = n - 1;
printf("x %d y %d z %d\n", x, y, z);
printf("%d\n", a[x]);
while(a[y] == 0 && y < n) {
x++;
y++;
printf("x %d y %d z %d\n", x, y, z);
} //如果最左边是0的话,把0保留在最左边
while(a[z] == 2 && z > 0) {
z--;
printf("x %d y %d z %d\n", x, y, z);
} //如果最右边是2的话,把2保留在最右边。
//至此第一步完成,减少第二步的工作量,提高算法效率
while(y <= z) {
switch(a[y]) { //第二步开始,按照上面的思想
case 0:
a[y] = a[y] + a[x] - (a[x] = a[y]);
x++;
y++;
break;
printf("x %d y %d z %d\n", x, y, z);
case 1:
y++;
break;
printf("x %d y %d z %d\n", x, y, z);
case 2:
a[y] = a[y] + a[z] - (a[z] = a[y]);
z--;
printf("x %d y %d z %d\n", x, y, z);
}
} //至此第二步完成
int i;
for(i = 0; i < 6; i++) {
printf("%d ", a[i]);
}
}
int main() {
int numbers[6] = {0, 2, 1, 0, 2, 1}; //我这里懒得生成随机数了 - -
int i;
adjust(numbers, 6);
printf("\nsorted numbers are \n");
for(i = 0; i < 6; i++) {
printf("%d ", numbers[i]);
}
return 0;
} 复制代码
好吧虽然楼主觉得对于数字而言,会有比这更高效的排序算法的。
啊终于可以去睡觉啦!!!!!
参考:
http://www.openhome.cc/Gossip/AlgorithmGossip/ThreeColorsFlags.htm
http://www.csse.monash.edu.au/~lloyd/tildeAlgDS/Sort/Flag/