Sunday, 3 January 2021

AtCoder - 167D. Teleporter

You can practice the problem here.

Problem statement

The Kingdom of Takahashi has N towns, numbered 1 through N.

There is one teleporter in each town. The teleporter in Town i (1 <= i <= N) sends you to Town A[i].

Takahashi, the king, loves the positive integer K. The selfish king wonders what town he will be in if he starts at Town 1 and uses a teleporter exactly K times from there.

Help the king by writing a program that answers this question.

Sample Input 1

4 5
3 2 4 1

Sample Output 1

4

Sample Input 2

6 727202214173249351
6 5 2 5 3 2

Sample Output 2

2

Solution 1: Math

If you take a look at the sample input 2, we can see that there is no way to traverse k times by listing out the travel. We can easily obverse the pattern the loop starts at a certain point.

Example:

6 5 2 5 3 2
1 -> 6 -> 2 -> 5 -> 3 -> 2 -> 5 -> 3 -> 2 -> 5 ....  

Starting from the third step, it starts the loop 2 -> 5 -> 3.

We can first create a map to store the index (0-based) and the value.

m[i] = a[i]

Then we can find how many steps we need to make before entering the loop, let's say c[i], and how many steps to reach the first duplicate element, let's say step.

bool duplicate = false;
ll i = 0, step = 0, kk = k;
while(!duplicate && kk){
    step++; // step is used to find out when a town has been visited
    if(d[i] == 1){
        // if it is visited before, that is the start of the loop
        duplicate = true;
        continue;
    }
    d[i] = 1; // set the current i to 1, in other word, this has been visited
    c[i] = step; // we need to store the step to another map because we need to calculate from where the looping starts
    i = m[i] - 1; // -1 because it s 0 based
    kk--; // monitor if it s out of boundary
}

We can calculate how many times we need to teleport from the repeating pattern as we know the length of the repeating pattern till k k - c[i], and the length of the repeating pattern step - c[i].

6 5 2 5 3 2
1 -> 6 -> 2 -> 5 -> 3 -> 2 -> 5 -> 3 -> 2 -> 5 ....
          x              x 
|-----------step---------|
          |--------------k-c[i]---------------- ....
|---c[i]--|                         
|------------------------k--------------------- ....

It performs only if it is not out of boundary.

if(kk){
    ll r = (k - c[i]) % (step - c[i]);
    while(r >= 0){
        i = m[i] - 1;
        r--;
    } 
}
OUT(i + 1);

Solution 2: Binary Lifting

After half a year, I came up with another solution using Binary Lifting which is a technique used to find the k-th ancestor of any node in a tree in O(logn). You can use it to find out Lowest Common Ancestor(LCA) as well.

We know the k can go up to 10 ^ 18, which means we only need around 60 nodes (log2(10 ^ 18)).

const int mxN = 200005;
const int mxNode = 60; // ~ log2(k) 
int up[mxN][mxNode];

First we need to preprecess the tree in O(nlogn) using dynamic programming.

void binary_lifting() {
  for(int i = 1; i < mxNode; i++) {
    for(int u = 0; u < mxN; u++) {
      up[u][i] = up[up[u][i - 1]][i - 1];
    }
  }
}

Then we can easily find the k-th ancestor of any node in a tree in O(logn). In this case, we start from node 0.

int kth_ancestor(ll u, ll k) {
  for(int i = 0; i < mxNode; i++) {
    if(k & (1LL << i)) u = up[u][i];
  }
  return u;
}

The answer would be kth_ancestor(0, k) + 1.

Source Code

The full solution is avilable here.

No comments:

Post a Comment

A Fun Problem - Math

# Problem Statement JATC's math teacher always gives the class some interesting math problems so that they don't get bored. Today t...