#include <bits/stdc++.h>
#define rep(i, n) for (int i = 0; i < (n); i++)
#define repr(i, n) for (int i = (n) - 1; i >= 0; i--)
#define rep2(i, l, r) for (int i = (l); i < (r); i++)
#define rep2r(i, l, r) for (int i = (r) - 1; i >= (l); i--)
#define range(a) a.begin(), a.end()
using namespace std;
using ll = long long;
template<class A, class B>
ostream &operator<<(ostream &o, pair<A, B> a) {
return o << "(" << a.first << "," << a.second << ")";
}
template<class T>
ostream &operator<<(ostream &o, vector<T> a) {
o << '[';
for (size_t i = 0; i < a.size(); i++) {
if (i > 0) o << ',';
o << a[i];
}
o << ']';
return o;
}
template<class T>
struct fibonacci_heap {
int mn;
struct node {
int l; // left
int r; // right
int p; // parent
int c; // children
int d; // degree
T key; // value
bool mark;
};
int root;
vector<node> dat;
fibonacci_heap(int n, T inf) : dat(n) {
root = 0;
mn = 0;
for (int i = 0; i < n; i++) {
dat[i].l = (i + n - 1) % n;
dat[i].r = (i + 1) % n;
dat[i].p = -1;
dat[i].c = -1;
dat[i].d = 0;
dat[i].key = inf;
dat[i].mark = false;
}
}
void insert_list(int *z, int x, int p) {
dat[x].p = p;
if (p != -1) dat[p].d++;
if (*z == -1) {
*z = x;
dat[x].l = x;
dat[x].r = x;
return;
}
int y = dat[*z].r;
dat[*z].r = x;
dat[y].l = x;
dat[x].l = *z;
dat[x].r = y;
}
void remove_list(int *z, int x) {
int p = dat[x].p;
if (p != -1) dat[p].d--;
if (*z == x) {
*z = dat[x].r;
if (*z == x) {
*z = -1;
return;
}
}
int l = dat[x].l;
int r = dat[x].r;
dat[l].r = r;
dat[r].l = l;
}
vector<int> make_list(int x) {
if (x == -1) return {};
vector<int> res;
int y = x;
do {
res.push_back(y);
y = dat[y].r;
} while (y != x);
return res;
}
pair<T, int> extract_min() {
int z = mn;
pair<T, int> res(dat[mn].key, mn);
int c = dat[z].c;
vector<int> children = make_list(c);
for (int c : children) {
insert_list(&root, c, -1);
}
mn = dat[z].r;
remove_list(&root, z);
if (root == -1) mn = -1;
if (z == root) root = dat[z].r;
else consolidate();
return res;
}
void consolidate() {
vector<int> A(40, -1);
int w = root;
vector<int> children = make_list(root);
for (int w : children) {
if (dat[w].p != -1) continue;
int x = w;
int d = dat[x].d;
while (A[d] != -1) {
int y = A[d];
if (dat[x].key > dat[y].key) {
swap(x, y);
}
fib_heap_link(y, x);
A[d] = -1;
d++;
}
A[d] = x;
}
mn = -1;
root = -1;
for (int i = 0; i < A.size(); i++) {
if (A[i] != -1) {
insert_list(&root, A[i], -1);
if (mn == -1 || dat[A[i]].key < dat[mn].key) {
mn = A[i];
}
}
}
}
void fib_heap_link(int y, int x) {
remove_list(&root, y);
insert_list(&dat[x].c, y, x);
dat[y].mark = false;
}
void decrease_key(int x, T v) {
assert(dat[x].key >= v);
dat[x].key = v;
int y = dat[x].p;
if (y != -1 && dat[x].key < dat[y].key) {
cut(x, y);
cascading_cut(y);
}
if (dat[x].key < dat[mn].key) {
mn = x;
}
}
void cut(int x, int y) {
remove_list(&dat[y].c, x);
insert_list(&root, x, -1);
dat[x].mark = false;
}
void cascading_cut(int y) {
int z = dat[y].p;
if (z != -1) {
if (!dat[y].mark) {
dat[y].mark = true;
} else {
cut(y, z);
cascading_cut(z);
}
}
}
};
template<class T>
struct binary_heap {
vector<T> key;
vector<int> hv;
vector<int> vh;
binary_heap(int n, T inf) : key(n, inf), hv(n), vh(n) {
for (int i = 0; i < n; i++) {
hv[i] = i;
vh[i] = i;
}
}
void exchange(int x, int y) {
swap(key[x], key[y]);
int X = hv[x];
int Y = hv[y];
swap(vh[X], vh[Y]);
swap(hv[x], hv[y]);
}
pair<T, int> extract_min() {
pair<T, int> res(key[0], hv[0]);
exchange(0, key.size() - 1);
key.pop_back();
const int n = key.size();
int k = 0;
for (;;) {
int l = k;
if (k * 2 + 1 < n && key[k * 2 + 1] < key[l]) l = k * 2 + 1;
if (k * 2 + 2 < n && key[k * 2 + 2] < key[l]) l = k * 2 + 2;
if (l == k) break;
exchange(k, l);
k = l;
}
return res;
}
void decrease_key(int x, T v) {
x = vh[x];
assert(key[x] >= v);
key[x] = v;
while (x > 0) {
int p = (x - 1) >> 1;
if (key[p] > key[x]) {
exchange(p, x);
x = p;
} else break;
}
}
};
int main() {
cin.tie(nullptr);
ios::sync_with_stdio(false);
cout << fixed << setprecision(15);
const int n = 5000;
vector<vector<int>> G(n, vector<int>(n, 1e9));
mt19937 mt(123);
rep(i, n) rep2(j, i + 1, n) G[i][j] = (n - i) * 5000 + uniform_int_distribution<int>(0, 4999)(mt);
rep(i, n - 1) G[i][i + 1] = 0;
{
clock_t beg = clock();
constexpr int INF = 1e9;
fibonacci_heap<int> hp(n, INF);
vector<int> d(n, INF);
d[0] = 0;
hp.decrease_key(0, 0);
int cnt = 0;
for (int ii = 0; ii < n; ii++) {
int u = hp.extract_min().second;
if (d[u] == INF) break;
rep(v, n) {
if (d[v] > d[u] + G[u][v]) {
d[v] = d[u] + G[u][v];
hp.decrease_key(v, d[v]);
cnt++;
}
}
}
clock_t end = clock();
cout << "fibonacci heap : " << (double)(end - beg) / CLOCKS_PER_SEC * 1000 << endl;
cout << cnt << endl;
}
{
clock_t beg = clock();
constexpr int INF = 1e9;
binary_heap<int> hp(n, INF);
vector<int> d(n, INF);
d[0] = 0;
hp.decrease_key(0, 0);
int cnt = 0;
for (int ii = 0; ii < n; ii++) {
int u = hp.extract_min().second;
if (d[u] == INF) break;
rep(v, n) {
if (d[v] > d[u] + G[u][v]) {
d[v] = d[u] + G[u][v];
hp.decrease_key(v, d[v]);
cnt++;
}
}
}
clock_t end = clock();
cout << "binary heap with decrease key : " << (double)(end - beg) / CLOCKS_PER_SEC * 1000 << endl;
cout << cnt << endl;
}
{
clock_t beg = clock();
constexpr int INF = 1e9;
priority_queue<pair<int, int>> hp;
vector<int> dist(n, INF);
dist[0] = 0;
hp.emplace(0, 0);
int cnt = 0;
while (!hp.empty()) {
int d = -hp.top().first;
int u = hp.top().second;
hp.pop();
if (dist[u] < d) continue;
rep(v, n) {
if (dist[v] > dist[u] + G[u][v]) {
dist[v] = dist[u] + G[u][v];
hp.emplace(-dist[v], v);
cnt++;
}
}
}
clock_t end = clock();
cout << "binary heap without decrease key : " << (double)(end - beg) / CLOCKS_PER_SEC * 1000 << endl;
cout << cnt << endl;
}
{
clock_t beg = clock();
constexpr int INF = 1e9;
vector<int> dist(n, INF);
dist[0] = 0;
int cnt = 0;
vector<bool> used(n);
rep(ii, n) {
int u = min_element(range(dist)) - dist.begin();
if (dist[u] == INF) break;
used[u] = true;
rep(v, n) {
if (!used[v] && dist[v] > dist[u] + G[u][v]) {
dist[v] = dist[u] + G[u][v];
cnt++;
}
}
dist[u] = INF;
}
clock_t end = clock();
cout << "straight forward : " << (double)(end - beg) / CLOCKS_PER_SEC * 1000 << endl;
cout << cnt << endl;
}
}
#include <bits/stdc++.h>

#define rep(i, n) for (int i = 0; i < (n); i++)
#define repr(i, n) for (int i = (n) - 1; i >= 0; i--)
#define rep2(i, l, r) for (int i = (l); i < (r); i++)
#define rep2r(i, l, r) for (int i = (r) - 1; i >= (l); i--)
#define range(a) a.begin(), a.end()

using namespace std;
using ll = long long;

template<class A, class B>
ostream &operator<<(ostream &o, pair<A, B> a) {
  return o << "(" << a.first << "," << a.second << ")";
}

template<class T>
ostream &operator<<(ostream &o, vector<T> a) {
  o << '[';
  for (size_t i = 0; i < a.size(); i++) {
    if (i > 0) o << ',';
    o << a[i];
  }
  o << ']';
  return o;
}

template<class T>
struct fibonacci_heap {
  int mn;

  struct node {
    int l; // left
    int r; // right
    int p; // parent
    int c; // children
    int d; // degree
    T key; // value
    bool mark;
  };

  int root;
  vector<node> dat;

  fibonacci_heap(int n, T inf) : dat(n) {
    root = 0;
    mn = 0;
    for (int i = 0; i < n; i++) {
      dat[i].l = (i + n - 1) % n;
      dat[i].r = (i + 1) % n;
      dat[i].p = -1;
      dat[i].c = -1;
      dat[i].d = 0;
      dat[i].key = inf;
      dat[i].mark = false;
    }
  }

  void insert_list(int *z, int x, int p) {
    dat[x].p = p;
    if (p != -1) dat[p].d++;
    if (*z == -1) {
      *z = x;
      dat[x].l = x;
      dat[x].r = x;
      return;
    }
    int y = dat[*z].r;
    dat[*z].r = x;
    dat[y].l = x;
    dat[x].l = *z;
    dat[x].r = y;
  }

  void remove_list(int *z, int x) {
    int p = dat[x].p;
    if (p != -1) dat[p].d--;
    if (*z == x) {
      *z = dat[x].r;
      if (*z == x) {
        *z = -1;
        return;
      }
    }
    int l = dat[x].l;
    int r = dat[x].r;
    dat[l].r = r;
    dat[r].l = l;
  }

  vector<int> make_list(int x) {
    if (x == -1) return {};
    vector<int> res;
    int y = x;
    do {
      res.push_back(y);
      y = dat[y].r;
    } while (y != x);
    return res;
  }

  pair<T, int> extract_min() {
    int z = mn;
    pair<T, int> res(dat[mn].key, mn);
    int c = dat[z].c;
    vector<int> children = make_list(c);
    for (int c : children) {
      insert_list(&root, c, -1);
    }
    mn = dat[z].r;
    remove_list(&root, z);
    if (root == -1) mn = -1;
    if (z == root) root = dat[z].r;
    else consolidate();
    return res;
  }

  void consolidate() {
    vector<int> A(40, -1);
    int w = root;
    vector<int> children = make_list(root);
    for (int w : children) {
      if (dat[w].p != -1) continue;
      int x = w;
      int d = dat[x].d;
      while (A[d] != -1) {
        int y = A[d];
        if (dat[x].key > dat[y].key) {
          swap(x, y);
        }
        fib_heap_link(y, x);
        A[d] = -1;
        d++;
      }
      A[d] = x;
    }
    mn = -1;
    root = -1;
    for (int i = 0; i < A.size(); i++) {
      if (A[i] != -1) {
        insert_list(&root, A[i], -1);
        if (mn == -1 || dat[A[i]].key < dat[mn].key) {
          mn = A[i];
        }
      }
    }
  }

  void fib_heap_link(int y, int x) {
    remove_list(&root, y);
    insert_list(&dat[x].c, y, x);
    dat[y].mark = false;
  }

  void decrease_key(int x, T v) {
    assert(dat[x].key >= v);
    dat[x].key = v;
    int y = dat[x].p;
    if (y != -1 && dat[x].key < dat[y].key) {
      cut(x, y);
      cascading_cut(y);
    }
    if (dat[x].key < dat[mn].key) {
      mn = x;
    }
  }

  void cut(int x, int y) {
    remove_list(&dat[y].c, x);
    insert_list(&root, x, -1);
    dat[x].mark = false;
  }

  void cascading_cut(int y) {
    int z = dat[y].p;
    if (z != -1) {
      if (!dat[y].mark) {
        dat[y].mark = true;
      } else {
        cut(y, z);
        cascading_cut(z);
      }
    }
  }
};

template<class T>
struct binary_heap {
  vector<T> key;
  vector<int> hv;
  vector<int> vh;

  binary_heap(int n, T inf) : key(n, inf), hv(n), vh(n) {
    for (int i = 0; i < n; i++) {
      hv[i] = i;
      vh[i] = i;
    }
  }

  void exchange(int x, int y) {
    swap(key[x], key[y]);
    int X = hv[x];
    int Y = hv[y];
    swap(vh[X], vh[Y]);
    swap(hv[x], hv[y]);
  }

  pair<T, int> extract_min() {
    pair<T, int> res(key[0], hv[0]);
    exchange(0, key.size() - 1);
    key.pop_back();
    const int n = key.size();
    int k = 0;
    for (;;) {
      int l = k;
      if (k * 2 + 1 < n && key[k * 2 + 1] < key[l]) l = k * 2 + 1;
      if (k * 2 + 2 < n && key[k * 2 + 2] < key[l]) l = k * 2 + 2;
      if (l == k) break;
      exchange(k, l);
      k = l;
    }
    return res;
  }

  void decrease_key(int x, T v) {
    x = vh[x];
    assert(key[x] >= v);
    key[x] = v;
    while (x > 0) {
      int p = (x - 1) >> 1;
      if (key[p] > key[x]) {
        exchange(p, x);
        x = p;
      } else break;
    }
  }
};

int main() {
  cin.tie(nullptr);
  ios::sync_with_stdio(false);
  cout << fixed << setprecision(15);
  const int n = 5000;
  vector<vector<int>> G(n, vector<int>(n, 1e9));
  mt19937 mt(123);
  rep(i, n) rep2(j, i + 1, n) G[i][j] = (n - i) * 5000 + uniform_int_distribution<int>(0, 4999)(mt);
  rep(i, n - 1) G[i][i + 1] = 0;
{
  clock_t beg = clock();
  constexpr int INF = 1e9;
  fibonacci_heap<int> hp(n, INF);
  vector<int> d(n, INF);
  d[0] = 0;
  hp.decrease_key(0, 0);
  int cnt = 0;
  for (int ii = 0; ii < n; ii++) {
    int u = hp.extract_min().second;
    if (d[u] == INF) break;
    rep(v, n) {
      if (d[v] > d[u] + G[u][v]) {
        d[v] = d[u] + G[u][v];
        hp.decrease_key(v, d[v]);
        cnt++;
      }
    }
  }
  clock_t end = clock();
  cout << "fibonacci heap : " << (double)(end - beg) / CLOCKS_PER_SEC * 1000 << endl;
  cout << cnt << endl;
}
{
  clock_t beg = clock();
  constexpr int INF = 1e9;
  binary_heap<int> hp(n, INF);
  vector<int> d(n, INF);
  d[0] = 0;
  hp.decrease_key(0, 0);
  int cnt = 0;
  for (int ii = 0; ii < n; ii++) {
    int u = hp.extract_min().second;
    if (d[u] == INF) break;
    rep(v, n) {
      if (d[v] > d[u] + G[u][v]) {
        d[v] = d[u] + G[u][v];
        hp.decrease_key(v, d[v]);
        cnt++;
      }
    }
  }
  clock_t end = clock();
  cout << "binary heap with decrease key : " << (double)(end - beg) / CLOCKS_PER_SEC * 1000 << endl;
  cout << cnt << endl;
}
{
  clock_t beg = clock();
  constexpr int INF = 1e9;
  priority_queue<pair<int, int>> hp;
  vector<int> dist(n, INF);
  dist[0] = 0;
  hp.emplace(0, 0);
  int cnt = 0;
  while (!hp.empty()) {
    int d = -hp.top().first;
    int u = hp.top().second;
    hp.pop();
    if (dist[u] < d) continue;
    rep(v, n) {
      if (dist[v] > dist[u] + G[u][v]) {
        dist[v] = dist[u] + G[u][v];
        hp.emplace(-dist[v], v);
        cnt++;
      }
    }
  }
  clock_t end = clock();
  cout << "binary heap without decrease key : " << (double)(end - beg) / CLOCKS_PER_SEC * 1000 << endl;
  cout << cnt << endl;
}
{
  clock_t beg = clock();
  constexpr int INF = 1e9;
  vector<int> dist(n, INF);
  dist[0] = 0;
  int cnt = 0;
  vector<bool> used(n);
  rep(ii, n) {
    int u = min_element(range(dist)) - dist.begin();
    if (dist[u] == INF) break;
    used[u] = true;
    rep(v, n) {
      if (!used[v] && dist[v] > dist[u] + G[u][v]) {
        dist[v] = dist[u] + G[u][v];
        cnt++;
      }
    }
    dist[u] = INF;
  }
  clock_t end = clock();
  cout << "straight forward : " << (double)(end - beg) / CLOCKS_PER_SEC * 1000 << endl;
  cout << cnt << endl;
}
}
