不相交集ADT 你是和谁是一类人?
//不相交集ADT (抽象数据类型)
//一般用于集合运算
//用树,这种结构组成,有多个树(=森林)
//属于同一颗数的元素,表示处于同一个集合中
//主要支持2个操作.
//1. Find操作,找到给定元素所属的集合编号
//2. Union操作,给出2个元素,将他们纳入同一个集合中
//常见使用场景:元素a和b是否属于同一集合?
//find(a) == find(b) :true 表示a和b属于同一集合
//但注意:因为是不相交集,所以一个元素在同一时刻,只能最多属于一个集合中,不能存在同时属于两个集合中的情况(因为这是存在相交集的情况)
public class Disjoint {
// int setSize;//集合个数
//
//正数:存储下标作为所属于的集合,下标是元素编号,也是集合编号,某2个元素是否属于相同集合是指这2个元素所指向的最终集合编号是否一致
//负数:表示当前集合树的深度,-1表示树深度为1,该元素只属于自己这个集合,用于 按高度求并
int[] set; public Disjoint(int setSize) { //集合大小
// this.setSize = setSize;
set = new int[setSize];
for (int i = 0; i < setSize; i++)
set[i] = -1; //-1表示单独属于一个集合
} //查找元素所属集合的编号
public int find(int elementType) {
if (check(elementType)) return elementType; if (set[elementType] <= 0) //集合树根:该元素不和其他元素没有处于同一集合,返回该元素本身作为集合类型
return elementType;
//set[elementType] = find(set[elementType]) 是为了压缩路径(减少深度),每次查找时,将深度过深的树,顺着所属路径给扁平化,节点直接指向树根
return set[elementType] = find(set[elementType]); //集合子树:顺着该元素所属的集合类型 作为元素类型,递归的找到树根(最终所属的用作该集合统一标识的 集合类型)
} //对2个元素求并
// setRoot2 -> setRoot1
public void union(int setRoot1, int setRoot2) {
if (check(setRoot1) || check(setRoot2)) return;
//求并方式有3种
//1. 普通求并方式 ,缺点是树深可能会退化成单链表 N
// set[setRoot2] = setRoot1;
//2. 按大小求并
//3. 按高度求并 ,保证所有树的最大深度最多为 logN
//树根节点挂载原则:每次求并2个树的时候,把深度小的树接入深度大的树,当2个树深度一样的时候,作为根的树深度+1
if (set[setRoot1] < set[setRoot2]) //root1 更深
set[setRoot2] = setRoot1; //将浅树 root2的根指向root1
else {
if (set[setRoot1] == set[setRoot2])
set[setRoot2]--; //两树一样深,让root2 深度加1,接下来让root1指向root2
set[setRoot1] = setRoot2; //root2更深,将root1 指向 root2
}
} public boolean isSameUnion(int a, int b) {
if (check(a) || check(b)) return false;
return find(a) == find(b);
} private boolean check(int setRoot2) {
return setRoot2 > set.length || setRoot2 < 0;
} public static void main(String[] a) {
Disjoint disjoint = new Disjoint(5);
int 张三 = 0; //嗯,java默认支持Unicode作为源码中的字符集
int 李四 = 1;
int 赵家六 = 2;
int 热爱大自然一类人 = 3;
int 热爱老大哥一类人 = 4;
disjoint.union(热爱大自然一类人, 张三);
disjoint.union(热爱老大哥一类人, 赵家六);
disjoint.union(赵家六, 李四); System.out.println("张三 和 李四 是一类人?" + disjoint.isSameUnion(张三, 李四));
System.out.println("张三 是 热爱大自然一类人?" + disjoint.isSameUnion(张三, 热爱大自然一类人));
System.out.println("李四 是 热爱大自然一类人?" + disjoint.isSameUnion(李四, 热爱大自然一类人));
System.out.println("李四 是 热爱老大哥一类人?" + disjoint.isSameUnion(李四, 热爱老大哥一类人));
System.out.println("赵家六 是 热爱老大哥一类人?" + disjoint.isSameUnion(赵家六, 热爱老大哥一类人));
System.out.println("赵家六 和 李四 是一类人?" + disjoint.isSameUnion(赵家六, 李四));
}
}
输出
张三 和 李四 是一类人?false
张三 是 热爱大自然一类人?false
李四 是 热爱大自然一类人?false
李四 是 热爱老大哥一类人?true
赵家六 是 热爱老大哥一类人?true
赵家六 和 李四 是一类人?true
不相交集ADT 你是和谁是一类人?的更多相关文章
- 前阿里CEO卫哲谈阿里创业经验:如何找人、找钱、找方向?(不同的阶段分别有:时间优先、金额优先、比例优先,不要做平台,太难)
新浪科技李根 整理报道 卫哲现在是御嘉基金的创始合伙人,他另一个更加知名的身份是阿里巴巴(B2B)前CEO,在2006年到2011年的时间里,卫哲见证了阿里巴巴如何利用人才.资本和方向选择一路壮大. ...
- 北漂的IT人
北京的互联网人,是工作日完全没有个人生活的一类人,也是整个北漂大队伍中,下班时间最晚的那一波人,如果赶上周末还要加班,那毫不夸张地说,你的整个人生都在互联网上奋斗着. 虽说十点上班让多少行内外的人羡慕 ...
- 我的跟我学Ffmpeg 视频受众有哪些人
经常有人问我如何学习音视频以及如何学习Ffmpeg,问我有没有比较好的书的书推荐.比较好的音视频以及FFmpeg方面的 书,我了解到的比较全面又能深入浅出的还真没有.很多朋友都推荐雷神的博客,雷神的博 ...
- 连载《一个程序猿的生命周期》-《发展篇》 - 7.是什么阻碍了"程序猿"的发展?
有两件事想记录一下,具有普遍性和代表性."程序猿"加了引号,是泛指一类人,也并非局限于IT行业. 山东子公司的总经理是公司大股东之一,个子不高.有些秃顶.面容显老,但看 ...
- 浅谈我对DDD领域驱动设计的理解
从遇到问题开始 当人们要做一个软件系统时,一般总是因为遇到了什么问题,然后希望通过一个软件系统来解决. 比如,我是一家企业,然后我觉得我现在线下销售自己的产品还不够,我希望能够在线上也能销售自己的产品 ...
- JavaScript面向对象
理解对象 对象这个词如雷贯耳,同样出名的一句话:XXX语言中一切皆为对象! 对象究竟是什么?什么叫面向对象编程? 对象(object),台湾译作物件,是面向对象(Object Oriented)中的术 ...
- 了解JavaScript 面向对象基础 & 原型与对象
面向对象语言中的对象 老是能听到什么基于对象, 面向对象. 什么是对象, 如果有面向对象基础的人可以无视了, 下面举个简单的例子给大家讲讲面向对象中, 对象的定义, 这个是比较通用的, 不过对于JS来 ...
- 安装Ubuntu的那些事儿
这是博主第一次写博客,本人虽然目前就读的专业属计算机,但目前也是属于新手上路的那一类人.正好不久前解决了一个困扰了我很久的问题 ,现在拿出来给大家分享一下. 上个学期学校的工作室给大家集中普及linu ...
- HTML 学习笔记 JavaScript(面向对象)
现在让我们继续跟着大神的脚步前进 学习一下JavaScript中的面向对象的思想,其实作为一个iOS开发者,对面向对象还是比较熟悉的,但是昨晚看了一下Js中的面向对象,妈蛋 一脸萌比啊.还好有大神.让 ...
随机推荐
- Linux’s init system & systemd
一.init system 1.计算机是如何启动的 以早期 Fedora 系统为例. 1.开机 2.BIOS 和 GRUB(引导加载程序) 3.Linux 内核启动后,init 进程 是在 Fedor ...
- redis系列之------主从复制
什么是主从复制 Redis的主从复制机制是指可以让从服务器(slave)能精确复制主服务器(master)的数据,如下图所示: 或者 主从复制的方式和工作原理 工作方式: Redis主从复制主要 ...
- NLTK实现文本切分
之前已经了解了使用nltk库,将文本作为参数传入相应函数进行切分的方法,下面看看使用正则表达式如何来进行文本切分. 1. 使用正则表达式切分 1.1 通过RegexpTokenizer 进行切分.先导 ...
- Docker运行dotnetcore
windows下安装docker 参考: https://www.jianshu.com/p/502b4ac536ef https://docs.docker.com/ ...
- Windows性能监控监视器(perfmon使用)
一.在命令窗口或运行中执行perfmon.exe,打开性能监视器 二.在用户定义中,即可新建--数据收集器--性能计数器,步骤如下: 三.添加监控Windows服务器的资源类型,例如:内存(Avail ...
- 简单搭建docker registry
已知信息: 服务端IP:192.168.7.2xx 客户端IP:192.168.7.1xx 服务端: docker registry中镜像本地映射地址:/Users/dockergit/private ...
- 用dotnet core搭建web服务器(二)路由表与封装
https://gitee.com/lightsever/netcore_study/tree/master/server02_path 先上代码,首先我们把httpserver封装一下,以后用起来方 ...
- 深入浅出14个Java并发容器
前言 不考虑多线程并发的情况下,容器类一般使用ArrayList.HashMap等线程不安全的类,效率更高.在并发场景下,常会用到ConcurrentHashMap.ArrayBlockingQueu ...
- API收藏
1.百度图片api get 方法$word = '北海';//要搜索的词$num = 3;//要搜索的数量$url = 'https://image.baidu.com/search/acjson?t ...
- 上海街头灵魂摄影师:勤劳de小懒熊
上海中年大叔,街头摄影师,眼光比较独特,题材不限于: 酒吧晚上醉酒躺尸的.喝多亲嘴的.拉拉les的.流泪告别的.地铁露肉的.短裤露沟的. 尺度不大,但比较真实,艺术来源于生活,比那些摆拍的有意思. 大 ...