百度&必应权4, 日IP1w+ 查看详情
自助收录

2023年华为OD机考真题:计算至少需要多少个快递主站点

算法刷题2年前 (2023)更新 江南白衣
405 0 0
2023年华为OD机考真题:计算至少需要多少个快递主站点

全网最全面的华为OD机试真题汇总,100%原题题库,不需要开会员即可查看全部内容,更多考题请查看真题库。

真题库:https://www.yuque.com/codernav.com/od

题目:计算至少需要多少个快递主站点
时间限制:1s 空间限制:256MB 限定语言:不限
题目描述:
快递业务范围有N个站点,A站点与B站点可以中转快递,则认为A-B站可达,如果A-B可达,B-C可达,则A-C可达。现在给N个站点编号0、1、…n-1,用s[i][j]表示i-j是否可达,s[i][j]=1表示i-j可达,s[i][j]=0表示i-j不可达。
现用二维数组给定N个站点的可达关系,请计算至少选择从几个主站点出发,才能可达所有站点(覆盖所有站点业务)。
说明:s[i][j]与s[j][i]取值相同。
输入描述:
第一行输入为N,N表示站点个数。
之后N行表示站点之间的可达关系,第i行第j个数值表示编号为i和j之间是否可达。
输出描述:
输出站点个数,表示至少需要多少个主站点。
补充说明:
1<N<10000
示例1
输入:
4
1 1 1 1
1 1 1 0
1 1 1 0
1 0 0 1
输出:
1
说明:
选择0号站点作为主站点,0站点可达其他所有站点,所以至少选择1个站点作为主站才能覆盖所有站点业务。
示例2
输入:
4
1 1 0 0
1 1 0 0
0 0 1 0
0 0 0 1
输出:
3
说明:
选择0号站点可以覆盖0、1站点,选择2号站点可以覆盖2号站点,选择3号站点可以覆盖3号站点,所以至少选择3个站点作为主站才能覆盖所有站点业务。
解题思路:
1. 通过回溯法将互通的快递站点放在一个集合中。
2. 步骤1执行的次数即快递主站点的个数。

代码实现一:

package com.codernav.demo.hwod.exam;

import java.util.HashSet;
import java.util.Scanner;
import java.util.Set;

/**
 * @title 计算至少需要多少个快递主站点
 * @Description 快递业务范围有N个站点,A站点与B站点可以中转快递,则认为A-B站可达,如果A-B可达,B-C可达,则A-C可达。
 * 现在给N个站点编号0、1、…n-1,用s[i][j]表示i-j是否可达,s[i][j]=1表示i-j可达,s[i][j]=0表示i-j不可达。
 * 现用二维数组给定N个站点的可达关系,请计算至少选择从几个主站点出发,才能可达所有站点(覆盖所有站点业务)。
 * 说明:s[i][j]与s[j][i]取值相同。
 * @Author 开发者导航
 * @website https://codernav.com
 * @date 2023/5/18
 */
public class Main {
    public static int[][] s;

    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        int N = sc.nextInt();
        s = new int[N][N];
        for (int i = 0; i < N; i++) {
            for (int j = 0; j < N; j++) {
                s[i][j] = sc.nextInt();
            }
        }

        Set<Integer> set = new HashSet<>();     //已经有连通的站点
        int res = 0;    //主站点数
        for (int i = 0; i < N; i++) {
            if (set.contains(i)) {    //站点已经有连通,无需再处理
                continue;
            }
            Set<Integer> temp = new HashSet<>();    //相互连通的站点集合
            temp.add(i);
            handle(temp, i);
            set.addAll(temp);   //已经连通的站点放在set里面用来判断后续是否需要处理
            res++;
        }
        System.out.println(res);
    }

    /**
     * 寻找跟 n 站点相连通的站点
     *
     * @param set 相互连通的站点集合
     * @param n   站点编号
     */
    public static void handle(Set<Integer> set, int n) {
        for (int i = 0; i < s.length; i++) {
            if (set.contains(i)) {    //已经连通的站点无需再判断
                continue;
            }
            if (n != i && s[n][i] == 1) {     //值等于1说明两站点连通(n==i就是本站点)
                set.add(i);
                handle(set, i);
            }
        }
    }
}

代码实现二:满分Java题解

package com.codernav.demo.hwod.exam;

import java.util.HashSet;
import java.util.Scanner;
import java.util.Set;

/**
 * @title 计算至少需要多少个快递主站点
 * @Description 快递业务范围有N个站点,A站点与B站点可以中转快递,则认为A-B站可达,如果A-B可达,B-C可达,则A-C可达。
 * 现在给N个站点编号0、1、…n-1,用s[i][j]表示i-j是否可达,s[i][j]=1表示i-j可达,s[i][j]=0表示i-j不可达。
 * 现用二维数组给定N个站点的可达关系,请计算至少选择从几个主站点出发,才能可达所有站点(覆盖所有站点业务)。
 * 说明:s[i][j]与s[j][i]取值相同。
 * @Author 开发者导航
 * @website https://codernav.com
 * @date 2023/5/18
 */
public class Main {
    public static int[][] square;

    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        int num = scanner.nextInt();
        square = new int[num][num];
        int row = 0;
        int column = 0;
        while (row < num) {
            column = 0;
            while (column < num) {
                square[row][column] = scanner.nextInt();
                column++;
            }
            row++;
        }
        Set<Integer> set = new HashSet<>();
        int res = 0;
        int i = 0;
        while (i < num) {
            if (set.contains(i)) {
                i++;
                continue;
            }
            Set<Integer> temporary = new HashSet<>();
            temporary.add(i);
            trace(temporary, i);
            set.addAll(temporary);
            res++;
            i++;
        }
        System.out.println(res);
    }

    private static void trace(Set<Integer> temporary, int n) {
        int i = 0;
        while (i < square.length) {
            if (temporary.contains(i)) {
                i++;
                continue;
            }
            if (n != i && square[n][i] == 1) {
                temporary.add(i);
                trace(temporary, i);
            }
            i++;
        }
    }
}

 

© 版权声明
开发者导航

相关文章

开发者导航

暂无评论

暂无评论...