算法学习——求有重复元素的全排列(递归)

算法学习——求有重复元素的全排列(递归)

思路:看到这个题目首先能想到的一点就是:①我们要求元素的所有全排列②我们要对求出的全排列去重

第一步:求全排列,这里先讨论对不含重复元素的数组元素进行全排列,用递归方法怎么实现叻

思考这样一种方法:假如我们要求1,2,3,4的全排列,我们可以把1放到前面来,求2,3,4的全排列,之后把2放到前面,求1,3,4的全排列,之后把3放到前面,求1,2,4的全排列,然后把4放到前面来求1,2,3的全排列;这是外层循环,就是遍历每个数,将该数与第一个位置的数交换然后求得余下所有数的全排列,而余下的数的全排列也可以用该种方法求得,当递归到只剩一个数,无法再求其全排列时,递归就有了出口,此时就可以输出了。注意:当我们把1放到前面来,求得剩下所有数的全排列后输出结束,我们需要将1还放回原位置,然后继续将2放到前面来......

void perm(char list[],int k,int m)
{
	if(k==m)    //当只剩下一个元素时则输出
	{
		count++;
		for(int i=0;i<=m;i++)
			cout<<list[i];
		cout<<endl;
	}
	for(int i=k;i<=m;i++)  //还有多个元素待排列,递归产生排列
	{
		swap(list[k],list[i]);
		perm(list,k+1,m);
		swap(list[k],list[i]);
	}
}

第一步完成后,思考第二步,如何给得到的结果去重呢?

我们求全排列时,是往前提一个数,求余下的数的全排列,假如我们有这样一组数:1,2,3,1,4,如何求全排列呢?实际上只要稍加思考就可以看出,依照上面的方法,两个1分别移到前面求剩下的数的全排列的效果一样,这就产生了重复,所以当1已经在之前求过全排列的话,后面那个1就不用再求了,这样就达到了去重的目的。

我们添加一个条件判断该数是否在之前使用过。

int finish(char list[],int k,int i)
{//第i个元素是否在前面元素[k...i-1]中出现过
	if(i>k)
	{
		for(int j=k;j<i;j++)
			if(list[j]==list[i])
				return 0;
	}
	return 1;
}
void perm(char list[],int k,int m)
{
	if(k==m)    //当只剩下一个元素时则输出
	{
		count++;
		for(int i=0;i<=m;i++)
			cout<<list[i];
		cout<<endl;
	}
	for(int i=k;i<=m;i++)  //还有多个元素待排列,递归产生排列
	{
		if(finish(list,k,i)) //没有在前面出现过,处理该数
		{
			swap(list[k],list[i]);
			perm(list,k+1,m);
			swap(list[k],list[i]);
		}
	}
}

这样就可以实现带有重复元素的数组求全排列了!

完整代码如下

#include <iostream>
using namespace std;
int count=0;
void swap(char &a,char &b)
{
	char temp;
	temp=a;
	a=b;
	b=temp;
}
int finish(char list[],int k,int i)
{//第i个元素是否在前面元素[k...i-1]中出现过
	if(i>k)
	{
		for(int j=k;j<i;j++)
			if(list[j]==list[i])
				return 0;
	}
	return 1;
}
void perm(char list[],int k,int m)
{
	if(k==m)    //当只剩下一个元素时则输出
	{
		count++;
		for(int i=0;i<=m;i++)
			cout<<list[i];
		cout<<endl;
	}
	for(int i=k;i<=m;i++)  //还有多个元素待排列,递归产生排列
	{
		if(finish(list,k,i))
		{
			swap(list[k],list[i]);
			perm(list,k+1,m);
			swap(list[k],list[i]);
		}
	}
}
int main()
{
	int i,n;
	cout<<"请输入元素个数: "<<endl;
	cin>>n;
	cout<<"请输入待排列的元素: "<<endl;
	//getchar();
	char *a=new char[n];

	for(i=0;i<n;i++)
		cin>>a[i];
	cout<<"所有不同排列为: "<<endl;
	perm(a,0,n-1);
	cout<<"排列总数为: "<<count<<endl;
	return 0;
}

最后,加一个容易理解的小视频(这个up讲的太好了,我一个笨蛋都听懂了,供大家学习交流):https://www.bilibili.com/video/BV1dx411S7WR

wish you have a good day!

  • 40
    点赞
  • 146
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
全排列递归算法的C语言实现可以通过递归和交换元素来实现。在这个算法中,我们使用了一个perm函数来进行全排列操作。 引用中的代码实现了一个全排列的C语言算法。这个算法使用了递归和交换元素的技巧来生成全排列。首先,我们从数组的第一个元素开始,通过for循环选择一个元素与第一个元素交换位置,然后递归求解第二个元素到最后一个元素全排列。然后,我们再次交换回来,以便进行下一次循环。当递归到最后一个元素时,我们已经得到了一个完整的排列,可以将其输出。这样,通过递归的方式,我们可以得到数组的所有全排列。 另外,引用中的代码实现了另一种全排列的C语言算法。这个算法使用了递归和布尔类型的数组来表示每个元素是否被使用过。通过循环遍历每个元素,并将其标记为已使用,然后递归求解剩余元素全排列。在递归结束后,我们需要将元素的状态恢复为未使用,以便进行下一次循环。这样,通过递归的方式,我们也可以得到数组的所有全排列。 总之,无论是哪种方法,全排列算法的核心思想都是通过递归和交换元素来生成所有可能的排列。你可以选择其中一种方法来使用,具体取决于你的需求和编程习惯。123 #### 引用[.reference_title] - *1* [全排列--【C语言】递归](https://blog.csdn.net/unseven/article/details/105219139)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}} ] [.reference_item] - *2* [排列组合之——全排列(c语言)](https://blog.csdn.net/m0_74820906/article/details/127779230)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}} ] [.reference_item] - *3* [C语言通过递归实现全排列](https://blog.csdn.net/weixin_43394832/article/details/105313758)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}} ] [.reference_item] [ .reference_list ]
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值