七段码问题(dfs以及并查集的应用)

该博客介绍了如何使用七段码数码管表示不同字符的问题,并通过深度优先搜索和并查集算法来判断亮灯是否连成一片。代码实现中,首先枚举所有可能的亮灯组合,然后利用并查集检测亮灯是否全部相连,最终计算出可表示的不同字符数。
摘要由CSDN通过智能技术生成

题目在这里插入图片描述

上图给出了七段码数码管的一个图示,数码管中一共有 77 段可以发光的二 极管,分别标记为 a, b, c, d, e, f, ga,b,c,d,e,f,g。
小蓝要选择一部分二极管(至少要有一个)发光来表达字符。在设计字符 的表达时,要求所有发光的二极管是连成一片的。
例如:bb 发光,其他二极管不发光可以用来表达一种字符。
例如 cc 发光,其他二极管不发光可以用来表达一种字符。这种方案与上 一行的方案可以用来表示不同的字符,尽管看上去比较相似。
例如:a, b, c, d, ea,b,c,d,e 发光,f, gf,g 不发光可以用来表达一种字符。
例如:b, fb,f 发光,其他二极管不发光则不能用来表达一种字符,因为发光 的二极管没有连成一片。
请问,小蓝可以用七段码数码管表达多少种不同的字符?

题解代码

#include<bits/stdc++.h>
using namespace std;
const int N=10;
int use[N],ans,e[N][N],fa[N];
void init(){
	/*	
		连边建图,e[i][j]==1表示i和j相邻
		a b c d e f g
		1 2 3 4 5 6 7
	*/
	e[1][2]=e[1][6]=1;
	e[2][1]=e[2][7]=e[2][3]=1;
	e[3][2]=e[3][4]=e[3][7]=1;
	e[4][3]=e[4][5]=1;
	e[5][4]=e[5][6]=e[5][7]=1;
	e[6][1]=e[6][5]=e[6][7]=1;
}
int find(int u){
	if(fa[u]==u)return u;
	fa[u]=find(fa[u]);
	return fa[u];
}//并查集
void dfs(int d){
	if(d>7){
		/* 并查集判是否在同一集合 */
		for(int i=1;i<=7;i++)fa[i]=i;//初始化父亲集合
		for(int i=1;i<=7;i++)//遍历所有边集
			for(int j=1;j<=7;j++)
				if(e[i][j]&&use[i]&&use[j]){
					int fx=find(i),fy=find(j);//fx!=fy就是两个不同的灯都亮的情况 
					if(fx!=fy)fa[fx]=fy;//如果不在同一集合,合并:此步将两个相邻的灯合并到同一个集合中 
				}
		int k=0;
		for(int i=1;i<=7;i++)
			if(use[i]&&fa[i]==i)k++;//如果一个灯都不亮则k=0,如果有多个灯亮而且有多组灯相连则k>1 
		if(k==1)ans++;//如果所有亮灯都属于同一个集合
		return;
	}
	use[d]=1;//打开d这个灯,继续开关下一个灯
	dfs(d+1);
	use[d]=0;//关闭d这个灯,继续开关下一个灯
	dfs(d+1);
}
int main(){
	init();
	dfs(1);
	cout<<ans;
}

解析

此题先是用深度优先搜索枚举了所有可能出现的亮灯情况
然后使用并查集来检测所有亮了的灯是否全部相连,其中并查集的合并操作确保了相邻的灯所属的集合相同,从而加下来的判断

for(int i=1;i<=7;i++)
			if(use[i]&&fa[i]==i)k++;//如果一个灯都不亮则k=0,如果有多个灯亮而且有多组灯相连则k>1 

只有当深度优先的目前情况下,只有一组相邻亮灯的情况下结果才加1.
巧妙地达成了所有亮灯都相连的条件!
妙啊!!!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值