Минимальное расстояние в графе

121
03 марта 2021, 06:00

Дано связный невзвешенный граф с N вершинами N-1 ребром. И также подграф етого графа P. Найти вершину от которой расстояние до каждой вершины подграфа P будёт минимальна и вывести само расстояние.

Буду рад любой идеи к решению, а сейчас немного про ограничения:

TIME : 2 сек
MEMORY : 256 МБ
N, K <= 10^5, где N количество вершин графа, а K количество вершин подграфа P

Так же наведу пример для наглядности:

N K
| |
8 4
Далее задано N - 1 ребер:
1 2
1 3
1 4
4 6
4 5
5 7
5 8
Далее K чисел через пробел, вершины подграфа P:
2 5 6 7
Ответ: 
6

Моя попытка, которая идёт на лимит времени:

#include <bits/stdc++.h>
using namespace std;
int n;
int BFS(vector<vector<int>> graph, int from, int to)
{
    vector <bool> mark(1 + n, true);
    vector <int> dist(1 + n, 0);
    queue <int> Q;
    Q.push(from);
    mark[from] = false;
    while (!Q.empty())
    {
        int v = Q.front();
        Q.pop();
        for (int i = 0; i < graph[v].size(); ++i)
        {
            int vert = graph[v][i];
            if (mark[vert])
            {
                mark[vert] = false;
                dist[vert] = 1 + dist[v];
                Q.push(vert);
            }
        }
    }
    return dist[to];
}
int main()
{
    int m;
    cin >> n >> m;
    vector <vector<int>> graph(1 + n);
    for (int i = 0; i < (n - 1); ++i)
    {
        int u, v;
        cin >> u >> v;
        graph[u].push_back(v);
        graph[v].push_back(u);
    }
    vector <int> P(m);
    for (int i = 0; i < m; ++i)
    {
        cin >> P[i];
    }
    int ans = INT_MAX;
    for (int from = 1; from <= n; from++)
    {
        int cur = 0;
        for (auto to : P)
        {
            int dist = BFS(graph, from, to);
            cur += dist;
        }
        ans = min(ans, cur);
    }
    cout << ans << endl;
    return 0;
}
Answer 1

Решение, которое работает линейно от числа вершин в графе.

vector< vector<int> > G;
vector<char> isP;
int K;
int dfs2(int v, int p, int l){
    int res = 0;
    for (auto next : G[v])
        if (next !=p)
            res+=dfs2(next,v, l+1);
    if (isP[v])
        res+=l;
    return res;
}

int dfs(int v, int p){
    int res = isP[v];
    for (auto next : G[v])
        if (next !=p)
            res+=dfs(next,v);
    if (res > K/2){
        cout << dfs2(v,-1,0)<<" "<<v<<endl;
        exit(0);
    }
    return res;
}

Запускаемый пример https://ideone.com/eIbkNs

Мы находим первую вершину, в поддереве которой не меньше половины всех отмеченных вершин. И от неё уже считаем расстояния.

Почему это оптимальная вершина. Давайте возьмём любую вершину (X), вычислим от неё все сумму расстояний. Если в каком-то её потомке больше половины вершин, то переместив X в эту вершину, мы уменьшим сумму. Т.к. больше половины путей станут на 1 короче, а меньше половины на 1 длиннее. Повторив это пока возможно, мы получим вершину, в поддереве которой не меньше половины всех отмеченных, но для каждого потомка это не выполняется. Почему минимум расстояний не зависит от выбора исходной вершины X - доказывается аналогично.

Доказательство использует ацикличность графа (упражнение - где именно). Напомню, что связный граф с N-1 ребром и N вершинами - дерево и, естественно, не содержит циклов.

Answer 2

Для начала, я бы упросил задачу. Например, посчитал бы сумму расстояний от одной произвольной вершины до P. Тут можно поспользоваться тем фактом, что граф связный, то есть из любой его вершины можно дойти до любой. Таким образом, используя поиск в ширину, можно нвйти минимальные расстояния от вершины до P за линейное время (кол-во вершин + кол-во ребер). Код нв Java

public  static int GetDistSum(ArrayList<Integer>[] graph, HashSet<Integer> p, int from)
{
    Queue<Integer> q = new LinkedList<>();
    boolean[] visited = new boolean[graph.length];
    int steps = 0;
    int sumSteps = 0;
    int pVisited = 0;
    q.add(from);
    while (q.isEmpty() == false)
    {
        int levelCount = q.size();
        for(int i=0; i<levelCount; i++)
        {
            int v = q.poll();
            if (visited[v]) continue;
            visited[v] = true;
            if (p.contains(v)) {
                sumSteps+=steps;
                pVisited++;
                if (pVisited == p.size()) return sumSteps;
            }
            for(Integer c : graph[v])
            {
                if (visited[c]) continue;
                q.add(c);
            }
        }
        steps++;
    }
    return sumSteps;
}

Ну и теперь осталось пройти по всем вершинам и найти минимум.

public static int GetMinDist(ArrayList<Integer>[] graph, int[] p)
{
    HashSet<Integer> pset = new HashSet<>();
    for(int i : p) pset.add(i);
    int min = Integer.MAX_VALUE;
    for(int i=1; i<graph.length; i++)
        min = Math.min(min, GetDistSum(graph, pset, i));
    return min;
}

То есть в результате сложность алгоритма будет N*(N+G), где N - колв вершин, G - колво ребер. Но мы и так значем, что ребер будет N-1, то есть сложность становится N(N+N-1) = 2N^2, то есть квадратичная.

Пример вызова для случая из вопроса:

public static void main(String[] args) {    
    ArrayList<Integer>[] graph = new ArrayList[9];
    for(int i=0; i<graph.length; i++) graph[i] = new ArrayList<>();
    graph[1].add(2);
    graph[2].add(1);
    graph[1].add(3);
    graph[3].add(1);
    graph[1].add(4);
    graph[4].add(1);
    graph[4].add(5);
    graph[5].add(4);
    graph[4].add(6);
    graph[6].add(4);
    graph[5].add(7);
    graph[7].add(5);
    graph[5].add(8);
    graph[8].add(5);
    int[] p = {2, 5, 6, 7};
    System.out.println(GetMinDist(graph, p));
}

Так как граф ненаправленный, то каждое ребро добавляется в него дважды.

Результат:

6

У автора же проблема в том, что он вызывает BFS для каждой вершины P, что делает сложность в итоге кубической (если P=N).

READ ALSO
Сортировка пузырьком на С++

Сортировка пузырьком на С++

Есть задача: отсортировать пузырьковым методом массивЕсть код

132
Итерацию по вектору объектов

Итерацию по вектору объектов

QtCreator, cmakeНе даёт проитерироваться по вектору объектов и передать их в функцию

103
Как избавится от дублирования кода в шаблоне template?

Как избавится от дублирования кода в шаблоне template?

Всем привет, мне нужно построить код внутри шаблона который тоже будет шаблоном Вот мой код

124
Error: $scope.$watch is not a function Angular 1.6

Error: $scope.$watch is not a function Angular 1.6

Объявляю $watch на переменную, но вылетает эта ошибка:

127