并查集

/*
* 并查集
* 解决动态连通性一类问题的一种算法
* 在一些有N个元素的集合应用问题中,我们通常是在开始时让每个元素构成一个单元素的集合,
* 然后按一定顺序将属于同一组的元素所在的集合合并,其间要反复查找一个元素在哪个集合中。
* 这里使用并查集解决元素的集合归属的问题。
* 我们让每一个集合成为一个组织,每个组织有一个头目,下面可以分成若干等级。
* 如果两个元素向上找到的头目相同,那么他们就属于同一个组织。
* 就是说每个元素只要知道自己的上级是谁,然后根据上级继续找到头目就能判断是不是属于一个组织。
* 我们规定,当两个人满足同一组织规定的时候,两个人所在的组织也就是同一组织,可以合并起来。
* 这个时候,只需要使得其中一个的组织头目臣服于另一个组织头目就可以了
* 最好的情况下,每一个人都能一步找到头目,这样查找两个人的组织关系的效率是很高的
* 在查找的时候,顺便将查找路径上的人的上级直接定义为组织头目,来压缩路径是很好的办法
* 这里在图的连通的应用中很有价值,比如krustal算法判断点的连通性。
* */
public class _Union_Set_Implement {
    private int size;
    
    //father[i]==i的时候说明 i就是这个组织的头目
    private int[] father;
    private int OriginationNum;

    public _Union_Set_Implement(int size) {
        this.size = size;
        this.OriginationNum = size;
        this.father = new int[this.size];

        for (int i = 0; i < this.size; this.father[i] = i++);

    }

    private int findOrigination(int v) {
        int t;
        for (t = v; this.father[t] != t; t = this.father[t]);

        while (this.father[v] != t) {
            int tt = this.father[v];
            this.father[v] = t;
            v = tt;
        }

        return t;
    }

    public void Merge(int x, int y) {
        int fx = this.findOrigination(x);
        int fy = this.findOrigination(y);
        if (fx != fy) {
            --this.OriginationNum;
            this.father[fx] = fy;
        }
    }
    public boolean isSameOrigination(int x,int y){
        return findOrigination(x)==findOrigination(y);
    }
    public int getOriginationNum() {
        return this.OriginationNum;
    }
}

这里利用并查集来解决一个问题

/*
* 题目详情
* 在地图上给你若干个城镇,这些城镇都可以看作点,然后告诉你哪些对城镇之间是有道路直接相连的。
* 最后要解决的是整幅图的连通性问题。
* 比如随意给你两个点,让你判断它们是否连通,
* 或者问你整幅图一共有几个连通分支,也就是被分成了几个互相独立的块。像畅通工程这题,问还需要修几条路
* 4 2
* 1 3
* 4 3
* */

public class _UnionSet {
    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        _Union_Set_Implement union=new _Union_Set_Implement(scanner.nextInt());
        int routenum=scanner.nextInt();
        for (int i = 0; i < routenum; i++) {
            int a=scanner.nextInt()-1;
            int b=scanner.nextInt()-1;
            union.Merge(a,b);
        }
        System.out.printf("还需要修建%d条路",union.getOriginationNum()-1);
    }
}

 

posted @ 2017-04-24 19:23  Zview  阅读(178)  评论(0编辑  收藏  举报