Editorial for COCI '15 Contest 6 #6 San


Remember to use this editorial only when stuck, and not to copy-paste code from it. Please be respectful to the problem author and editorialist.
Submitting an official solution before solving the problem yourself is a bannable offence.

Let tab[y] denote the number of occurrences of the number y in the table. We can (not in an efficient way) calculate it using the following algorithm for each y from 1 to N:

Copy
for x from 1 to N:
    tab[x] = tab[x] + 1
    tab[x + rev(x)] = tab[x + rev(x)] + tab[x]

This algorithm is linear, but N can be up to 1010 so it is not efficient enough. It is crucial to notice that the amount of numbers for which tab[y]>1 is very small. Let's denote the number of different numbers x such that x+rev(x)=y as cnt[y]. Notice that cnt[y]>0 if and only if tab[y]>1. Let's denote the array of numbers consisting of all the numbers for which cnt[y]>0 as S and sort it ascendingly. The values of tab[y] for these numbers can be calculated using the following algorithm:

Copy
for each x in S:
    tab[x] = tab[x] + cnt[x]
    tab[x + rev(x)] = tab[x + rev(x)] + tab[x]

This algorithm consists of |S| loop iterations. It is crucial to notice that the amount of numbers in set S is relatively small (only a couple of millions) so the upper algorithm is efficient enough in order to calculate the values tab[y] for each y for which the value is larger than 1. For each other y it holds tab[y]=1, so now we actually have the value tab[y] calculated for each y, which enables us to efficiently answer queries (using partial sums of the array tab). Notice that the indices in the array tab are very big, but we don't need the whole array so we will actually store it in a map.

We are left to calculate the values of cnt[y] for all numbers. We will recursively find all numbers y and the values cnt[y] for each y for which the value is larger than 0. Let's see what happens when we add a 6-digit number with itself, only in reverse:

Copy
    a1     a2     a3     a4     a5     a6
 +  a6     a5     a4     a3     a2     a1
-----------------------------------------
c0  b1+c1  b2+c2  b3+c3  b3+c4  b2+c5  b1

Here c0,,c5 are 0 or 1 and denote the carrying digits, b1=(a1+a6)mod10, b2=(a2+a5)mod10 and b3=(a3+a4)mod10.

The values cnt[y] could be determined so that we fix the digits a1,,a6 in all possible ways, but there are a lot of combinations for that. Another approach would be to determine b1,b2,b3 and c0,,c5 in all possible ways (therefore the sum is uniquely determined) and calculate how many different selections of a1,,a6 results in exactly b1,b2,b3 and c0,,c5. How can we do this? We will use a recursive function gen which takes 4 parameters, their meanings described in the following list:

  • pos means we are currently determining bpos
  • c1 means that cpos1 is equal to c1
  • c2 means that c6pos+1 is equal to c2
  • num is the value of the current part of the sum a1a6+a6a1

Here is the pseudocode for function gen:

Copy
gen(pos, c1, c2, num):

  if pos = 4:
    (we have determined the whole sum and the number of ways to obtain it)
    if c1 != c2 return
    (c1 and c2 must be equal because they both represent c3)
    increase cnt[num] by 1 and return

  for pairs of digits (x1, x2):
    (x1 and x2 determine b_pos, we are determining digits pos and 6-pos+1 of num)
    if c1 = 1 and x1+x2 < 10 continue
    if c1 = 0 and x1+x2 >= 10 continue
    bb = (x1 + x2) % 10

    nnum = num
    set digit 6-pos+1 of nnum to bb + c2
    set digit pos of nnum to bb + 1 and call
      gen(pos+1, 1, x1+x2 >= 10, nnum)
      (c1 = 1 denotes that we want the carry digit)

    set digit pos of nnum to bb and call
      gen(pos+1, 0, x1+x2 >= 10, nnum)
      (c1 = 0 denotes that there is no carry)

The upper approach can be generalized for an arbitrary number of digits. The upper recursion for a number of length L performs approximately 81L/2 operations, which is too slow for all points. It is crucial to note that we don't need to iterate over all pairs of digits, because it is sufficient to iterate over all possible sums and for each sum know how many ways there is to obtain it. Then we perform approximately 19L/2 operations, which is fast enough for all points. We leave the details as an exercise to the reader.


Comments

There are no comments at the moment.