啊哈!算法中用深度优先搜索(dfs)实现全排列

#include<stdio.h>
int a[10],book[10],n;
void dfs(int step){
	int i;
	if(step==n+1){
		for(i=1;i<=n;i++)
		printf("%d",a[i]);
		printf("\n");
		return;
	}
	for(i=1;i<=n;i++){
		if(book[i]==0){
			a[step]=i;
			book[i]=1;
			dfs(step+1);
			book[i]=0;
		}
	}
	return;
}
int main(){
	scanf("%d",&n);
	dfs(1);
	return 0;
}

深度优先搜索的基本模型

void dfs(int step)
{
	判断边界
	尝试每种可能 for(i=1;i<=n;i++)
	{
		继续下一步dfs(step+1); 
	 } 
	 返回 
}

理解深度优先搜索的关键在于解决“当下如何做”。至于“下一步如何做”是和“当下如何做”是一样的(用函数的递归即自己调用自己来实现的)。

代码说明

  • step用来表示盒子的编号,a[]用来表示盒子,book[]用来标记手中的牌,n表示从1到n这n个数进行全排列
  • 假设n=3,step=1,调用函数。step=1不满足n+1=4,那就进入for循环中放牌:开始的时候i=1,book[1]=0,说明手里有1号牌,在a[1]盒子里放入1号牌,然后标记book[1]=1,说明手中已经没有了1号牌。接着调用dfs(step+1)
  • 此时step=2,仍然不满足n+1=4,那就进入循环:i=1的时候book[1]=1,说明手里没有1号牌了。就进入第二层循环:i=2,book[2]=0,说明手里有2号牌,然后将2号牌放入a[2]号盒子中,并标记book[2]=1,说明此时手里已经没有2号牌了。接着调用dfs(step+1)
  • 此时step=3,仍然不满足n+1=4,那就进入循环:i=1的时候book[1]=1,说明手里没有1号牌了。就进入第二层循环:i=2,book[2]=1,说明手里没有2号牌。进入第三层循环i=3,book[3]=0,说明手里有三号牌,然后将3号牌放入a[3]号盒子中,并标记book[3]=1,说明此时手里已经没有3号牌了。接着调用dfs(step+1)
  • 此时step=4,满足n+1=4,这个时候遍历三个盒子(a),将三个盒子中的数字输出为123。然后return,返回到上一次调用dfs函数的地方,即dfs(step+1)(此时step=3)的地方,直接进行下一句book[i]=0,因为在step=3的时候循环i已经进行到了3,所以此时i=3,表示将3号牌从三号盒子中抽回。此时整个循环结束。然后继续执行下一句return,即返回step=3,时的上一个调用dfs函数的地方,即step=2
  • 直接执行下一句book[i]=0,因为在step=2时循环i只进行到了2,即i=2,所以将book[2]=0即将2号牌抽出,因为此时循环还没有进行完,所以将进行第三次循环i=3,此时book[3]=0,所以进入if语句,将3号牌放入a[2]号盒子中,然后标记book[3]=1,表示三号牌已不在手里,然后调用dfs(step+1)函数。
  • 此时step=3,此时step=3,不满足n+1=4,那就进入循环:i=1的时候book[1]=1,说明手里没有1号牌了。就进入第二层循环:i=2,book[2]=0,说明手里有2号牌,然后将2号牌放入a[3]盒子中,并标记2号牌已经不在手中。接着调用dfs(step+1)。
  • 此时step=4,满足n+1=4,这个时候遍历三个盒子(a),将三个盒子中的数字输出为132。
  • 接着依次如上,最终输出1,2,3这三个数字的全排列为123,132,213,231,312,321
  • 1
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值