阿里笔试题2021.8.16(模拟小修改的并查集)

Posted Zephyr丶J

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了阿里笔试题2021.8.16(模拟小修改的并查集)相关的知识,希望对你有一定的参考价值。

阿里笔试题2021.8.16

也是帮同学做的,给自己做的笔试题就猿辅导,还一面就挂,等会总结下面试问的几个问题

第一题:计算变量的大小

题目描述


思路

简单的模拟思路,把字符串分割以后计算长度就行了

import java.util.*;
public class Main{

    public static void main(String[] args){
        Scanner sc = new Scanner(System.in);
        String s = sc.nextLine();
        //String s = "long a[10][10][10],b,xyz";
        Map<String, Integer> map = new HashMap<>();
        map.put("int", 4);
        map.put("char", 2);
        map.put("long", 8);

        String[] ss = s.split(" ");
        int len = map.get(ss[0]);
        String[] strs = ss[1].split(",");
        //System.out.println(strs[0]);
        int l = strs.length;
        int res = 0;
        for(int i = 0; i < l; i++){
            String str = strs[i];
            int ls = str.length();
            int idx = 0;
            List<Integer> list = new ArrayList<>();
            while(idx < ls){
                if(str.charAt(idx) == '[') {
                    int num = 0;
                    idx++;
                    while (str.charAt(idx) >= '0' && str.charAt(idx) <= '9'){
                        num = num * 10 + str.charAt(idx) - '0';
                        idx++;
                    }
                    list.add(num);
                }
                idx++;
            }
            if(list == null){
                res += len;
                continue;
            }
            int t = 1;
            for(int n : list){
                t *= n;
            }
            res += t * len;
        }

        System.out.println(res);

    }
}

第二道:部队合并

题目描述


思路

思路就是合并两个部队用并查集来合并,与普通并查集不同的一点是,在并查集中多加一个变量count,存储每个部队(这个编号的联盟)的人数,合并的话就是把两个部队的人加到小编号的那个部队中。
不过这里花费的计算我没看懂到底是算联盟的还是算部队的,这里根据例子可以改一下,但是又没拍例子,只能写成差不多这样了
查询就是正常并查集的查询

不知道对不对,

import java.util.*;
public class Main{


    public static void main(String[] args){
        Scanner sc = new Scanner(System.in);
        int n = sc.nextInt();
        int m = sc.nextInt();
        int[] people = new int[n];
        for(int i = 0; i < n; i++){
            people[i] = sc.nextInt();
        }
        int[][] cmd = new int[m][3];
        for(int i = 0; i < m; i++){
            cmd[i][0] = sc.nextInt();
            cmd[i][1] = sc.nextInt();
            cmd[i][2] = sc.nextInt();
        }

        UnionFind uf = new UnionFind(n, people);
        for(int i = 0; i < m; i++){
            if(cmd[i][0] == 1){
                int px = uf.find(cmd[i][1]);
                int py = uf.find(cmd[i][2]);
                if(px == py)
                    continue;
                int countx = uf.count[px];
                int county = uf.count[py];
                int cost = (cmd[i][0] + cmd[i][1]) ^ Math.abs(countx - county);
                System.out.println(cost);
                uf.union(cmd[i][1], cmd[i][2]);
            }
            if(cmd[i][0] == 2)
                System.out.println(uf.query(cmd[i][1], cmd[i][2]) ? "YES" : "NO");
        }

    }
}

class UnionFind{
    int[] parents;
    int n;
    int[] count;
    //构造器
    public UnionFind(int n, int[] people){
        this.n = n;
        parents = new int[n];
        count = people;
        for(int i = 0; i < n; i++){
            parents[i] = i;
        }
    }

    public void union(int x, int y) {
        int px = find(x);
        int py = find(y);
        if(px == py)
            return;
        //如果父节点的编号小,将部队给了小的编号
        if(px < py){
            parents[py] = px;
            count[px] += count[py];
        }else{
            parents[px] = py;
            count[py] += count[px];
        }
    }

    public int find(int x){
        if(x != parents[x]){
            parents[x] = find(parents[x]);
        }
        return parents[x];
    }
    //查询x和y是否连通
    public boolean query(int x, int y){
        return find(x) == find(y);
    }
}

以上是关于阿里笔试题2021.8.16(模拟小修改的并查集)的主要内容,如果未能解决你的问题,请参考以下文章

阿里一次面试:周芷若和韩小昭的并查集

238. 银河英雄传说边带权的并查集

并查集-----好忧伤的并查集

带偏移量的并查集

hdu 3938 (离线并查集)

一道神奇的并查集