目录
排序算法总结分析(一)——开篇
排序算法总结分析(二)——常见八大排序算法
排序算法总结分析(三)——吃货排序之烙饼排序
今天先来个好玩点的,呃,确切说是好吃的点的问题。哈哈,就是如标题表明的烙饼排序。程序猿果然思维跟普通人就不一样,连吃个饼都想的这么多。问题描述是这样的:把一摞饼按照大小次序摆好,要求是小的在上面,大的在下面,只能通过翻转一摞饼进行排序,就像用铲子插入某个位置,把这个位置之上的所有饼进行翻转。那假设有N块大小不一的烙饼,最少要翻转几次才能达到最终有序排列呢?
翻转演示图
与传统排序不同的是,不能一张张抽出来,然后插入进去;也不能任意交换两块饼。说明基本的排序算法都不太好用。是不是有点意思呢?想当年比尔·盖茨也研究过这个问题~
目前这个问题的答案只有一个范围,没有确切的值,完成排序需要的最少次数在(15/14)N与(18/11)N之间。2011年的时候这个问题被定义为NP-Hard。什么是NP-Hard?NP(Non-Deterministic)官方定义是非确定性多项式。而非确定性是指,可用一定数量的运算去解决多项式时间内可解决的问题。例如,著名的推销员旅行问题(Travel Saleman Problem or TSP):假设一个推销员需要从香港出发,经过广州,北京,上海,…,等 n 个城市,最后返回香港。任意两个城市之间都有飞机直达,但票价不等。假设公司只给报销C元钱,问是否存在一个行程安排,使得他能遍历所有城市,而且总的路费小于C?推销员旅行问题显然是NP的。因为如果你任意给出一个行程安排,可以很容易算出旅行总开销。但是,要想知道一条总路费小于C的行程是否存在,在最坏情况下,必须检查所有可能的旅行安排!这将是个天文数字。
好了介绍性的东西说的差不多了,下面主要讲一下怎么个排法~
由于每次操作都是针对最上面的饼,如果最底层的饼已经排好序,然后就只需要处理上面的N-1个饼了。
翻转图
首先,经过两次翻转,最大的饼已经在最下面了。接着次大的饼也需要两次翻转,最后剩两张饼的时候只需要1次就可以,所以总的至少翻转次数为2(n-1)-1即2n-3。
当然这是这个问题解的一种上限,非最优。
那么下限呢?这里也只说一种简单的优化,非最佳。每一次翻转最多使得一个烙饼与大小跟它相邻的烙饼排到一起。如果当前状态N个烙饼中,有M对相邻的烙饼它们不相邻,那么至少需要M次才能排好。
下面稍微介绍下优化方法及算法的实现。
假如这堆烙饼中有好几个不同的部分相对有序,就可以先把小一些的烙饼翻转使其有序。这样就会减少翻转次数。可以考虑每次翻转的时候,把两个本来应该相邻的烙饼尽可能换到一起。这样,当所有的烙饼都换到一起之后。实际上就完成了排序。这样的话就会想到使用动态规划或者递归的方法来实现。可以从不同的翻转策略开始,递归所有可能性。这样,肯定能找到最优解。
代码如下:
#include
/************************************************************************/
/* 烙饼排序实现——By Sin_Geek 2014.04.12 */
/************************************************************************/
class CPancakeSorting
{
public:
CPancakeSorting()
{
m_nCakeCnt = 0;
m_nMaxSwap = 0;
}
//计算烙饼翻转信息,pCakeArray 存储烙饼索引数组,nCakeCnt烙饼个数
void Run(int* pCakeArray, int nCakeCnt)
{
Init(pCakeArray, nCakeCnt);
m_nSearch = 0;
Search(0);
}
//输出烙饼具体翻转的次数
void Output()
{
for (int i = 0; i < m_nMaxSwap; i++)
{
printf("%d", m_arrSwap[i]);
}
printf("\nSearch Times : %d\n", m_nSearch);
printf("Total Swap times = %d\n", m_nMaxSwap);
}
private:
//初始化数组信息,pCakeArray 存储烙饼索引数组,nCakeCnt烙饼个数
void Init(int* pCakeArray, int nCakeCnt)
{
m_nCakeCnt = nCakeCnt;
//初始化
m_CakeArray = new int[m_nCakeCnt];
for (int i = 0; i < m_nCakeCnt; i++)
{
m_CakeArray[i] = pCakeArray[i];
}
//设置最多交换次数信息
m_nMaxSwap = UpBound(m_nCakeCnt);
//初始化交换结果数组
m_SwapArray = new int[m_nMaxSwap];
//初始化中间交换结果信息
m_ReverseCakeArray = new int[m_nCakeCnt];
for (i = 0; i < m_nCakeCnt; i++)
{
m_ReverseCakeArray[i] = m_CakeArray[i];
}
m_ReverseCakeArraySwap = new int[m_nMaxSwap];
}
//寻找当前翻转的上界
int UpBound(int nCakeCnt)
{
return nCakeCnt*2 - 3;
}
//寻找当前翻转的下界
int LowerBound(int* pCakeArray, int nCakeCnt)
{
int t,ret = 0;
//根据当前数组的排序信息情况判断最少需要交换多少次
for (int i = 1; i < nCakeCnt; i++)
{
//判断位置相邻的两个烙饼是否为尺寸排序上相邻的
t = pCakeArray[i] - pCakeArray[i-1];
if ((t == 1) || (t == -1))
{
}
else
{
ret++;
}
}
return ret;
}
//排序的主函数
void Search(int step)
{
int i,nEstimate;
m_nSearch++;
//估算这次搜素所需的最小交换次数
nEstimate = LowerBound(m_ReverseCakeArray, m_nCakeCnt);
if (step + nEstimate > m_nMaxSwap)
return;
//如果已经排好序,输出结果
if (IsSorted(m_ReverseCakeArray, m_nCakeCnt))
{
if (step < m_nMaxSwap)
{
m_nMaxSwap = step;
for (i = 0; i < m_nMaxSwap; i++)
m_arrSwap[i] = m_ReverseCakeArraySwap[i];
}
return;
}
//递归翻转
for (i = 1; i < m_nCakeCnt; i++)
{
Revert(0,i);
m_ReverseCakeArraySwap[step] = i;
Search(step + 1);
Revert(0,i);
}
}
bool IsSorted(int* pCakeArray, int nCakeCnt)
{
for (int i = 1; i < m_nCakeCnt; i++)
{
if(pCakeArray[i-1] > pCakeArray[i])
return false;
}
return true;
}
//翻转烙饼
void Revert(int nBegin, int nEnd)
{
//ASSERT(nEnd > nBegin);
int i,j,t;
for (i = nBegin, j = nEnd; i < j; i++,j--)
{
t = m_ReverseCakeArray[i];
m_ReverseCakeArray[i] = m_ReverseCakeArray[j];
m_ReverseCakeArray[j] = t;
}
}
private:
int m_nCakeCnt; //烙饼个数
int m_nMaxSwap; //最多交换次数
int m_nSearch; //当前搜索次数
int* m_CakeArray; //烙饼信息数组
int* m_SwapArray; //交换结果数组
int* m_ReverseCakeArray;//当前翻转烙饼信息数组
int* m_ReverseCakeArraySwap;//当前翻转烙饼交换结果数组
int m_arrSwap[10];
};
void main()
{
int nCakeCnt = 10;
int arrSwap[10] = {3,2,1,6,5,4,9,8,7,0} ;
CPancakeSorting pan;
pan.Run(arrSwap, nCakeCnt);
pan.Output();
}
当然还可以把这个问题更复杂化一点,假定每个烙饼都有一面是烤过的,在原来排序的结果上附加一个条件,就是让所有烤过的那一面都朝下~有兴趣的可以思考一下哦~~