Editorial for COCI '21 Contest 1 #4 Set


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.

For the first subtask, it is sufficient to try out every triplet of cards and check whether it forms a set in O(n3k).

For the second subtask, one should notice that if we choose two cards, the third card needed for a set is uniquely determined. Namely, for each position, if the corresponding characters in the first two cards are the same, then the third character has to be equal to them, and if they are different, the third character will be the remaining one. With time complexity O(n2k), we can now try out every pair of cards and check whether there exists a third card which will form a set with them. Of course, we should first record in an array of size 3k for each type of card if it shows up in the input.

From now on, we will assume that the characters in question are 0,1,2 instead of 1,2,3, so that we can view each card as a number in base 3. Let's try to come up with a simple rule that associates a pair of characters with the third character needed for a set. In other words, we are looking for a rule that acts in the following way:

(0,0)0(1,1)1(2,2)2(0,1)2(0,2)1(1,2)0

We can notice that (a,b)(a+b)mod3, or equivalently, the characters a,b,c form a set if and only if (a+b+c)mod3=0. Now let's make an analogy with the bitwise xor operation. This operation is denoted by and represents addition modulo 2 with the digits in base 2. In this problem, we will consider addition modulo 3 with the digits in base 3, which we will denote by . Having in mind the things mentioned above, three cards a,b,c form a set if and only if abc=0, where the cards are thought of as numbers in base 3.

Let's fix some card c and try to figure out how many pairs of cards a and b exist so that ab=c. For each individual c, we can calculate this in O(nk) so the total complexity is O(n2k), but we will show a way to find the answer for all cards c simultaneously in the complexity O(3kk). If instead of the operation we had the operation +, the problem could be solved by calculating the desired +-convolution with fast multiplication of polynomials using FFT. In this problem, we will therefore try to modify this idea to calculate the -convolution.

The operations and are very similar and the -convolution is calculated in the same manner as the xor-convolution. Thus, what follows is a description of the modification of the 'fast Walsh–Hadamard transformation' (FWHT) to work modulo 3. More about this can be found on this Codeforces blog. (P.S. the day before the contest, another great blog on the topic appeared on Codeforces: link)

At a high level, the idea is the following:

  • The given deck of cards is represented by a polynomial.
  • Each term in the polynomial represents a certain type of card.
  • The coefficients of the polynomial represent the number of times a card appears in the deck. In the beginning, all of the coefficients are either 0 or 1.
  • We will square the polynomial to get new coefficients which represent the result of the desired convolution. (For now, this corresponds to the + operation).
  • Before multiplying, we will convert the polynomial from coefficient form to point value form as a sequence of calculated values (xi,P(xi)), which is more desirable for multiplication.
  • The result of the multiplication should be converted back to coefficient form.

Regular multiplication of polynomials corresponds to the operation +, that is (xa,xb)xa+b. We would like to make a modification so that (xa,xb)xab. We need to make two modifications:

  1. Addition should be done separately for each digit.
  2. Addition should be done modulo 3.

Problem 1 can be solved by introducing a polynomial with k variables x0,x1,,xk1. For example, a pair of cards 1123 and 2321 (that is 0012 and 1210) is represented by a polynomial

P(x3,x2,x1,x0)=x30x20x11x02+x31x22x11x00

Multiplication now corresponds to addition of the digits separately. Looking at each of the variables separately, the polynomial is of degree at most 2, but when squaring, the degree might grow larger.

To convert a polynomial (which remember has 3k coefficients) to point-value form, we will calculate the value of the polynomial at 3k different points. We will choose three values v0,v1,v2 and calculate P(xk1,,x0) for each possible combination where xi{v0,v1,v2}, of which there are also 3k. In the implementation, we will therefore have a transformation F that converts a sequence of 3k coefficients to a sequence of 3k calculated values.

Since the product of two polynomials has more coefficients than the original polynomials, if we wanted to calculate the true product, we would have had to extend the polynomials to bigger powers, making the new coefficients 0. However, to solve problem 2, that is precisely what we will not do. When doing the inverse transformation F1 which returns the coefficient form, we will purposefully demand that the result has 3k coefficients. Additionally, for the values v0,v1,v2, we will choose the third roots of unity (both real and complex), that is the numbers 1, 1+3i2 and 13i2, so that v03=v13=v23=1. The effect of these two things is that the coefficients of larger powers in the resulting product (x3,x4,) will get added to the coefficients of the smaller powers - precisely with the smallest power that is the same modulo 3 (because of the choice of vi). Thus, the powers will reduce modulo 3 and we will get exactly the desired 3k coefficients of the -convolution.

Let us illustrate this on an example with k=2. For the polynomial, we take

P(x1,x0)=a00x10x00+a01x10x01+a02x10x02+a10x11x00+a11x11x01+a12x11x02+a20x12x00+a21x12x01+a22x12x02

which can also be written as

P(x1,x0)=Q0(x0)x10+Q1(x0)x11+Q2(x0)x12

where

Qi(x0)=ai0x00+ai1x01+ai2x02

First, we will apply the transformation F on each of the polynomials Qi separately. Using the label w=12+32i, we have:

(ai0,ai1,ai2)(ai0+ai1+ai2,ai0+wai1+w2ai2,ai0+w2ai1+wai2)

In this way, the sequence of coefficients

(a00,a01,a02,a10,a11,a12,a20,a21,a22)

turns into a new list of coefficients, which we will label with

(a00,a01,a02,a10,a11,a12,a20,a21,a22)

We would like to obtain a list that has the following values in order

P(1,1),P(1,w),P(1,w2),P(w,1),P(w,w),P(w,w2),P(w2,1),P(w2,w),P(w2,w2)

What remains is to replace the triplets (a0i,a1i,a2i) with new values, in the same manner as we did when calculating the values for Qi.

For bigger values of k, the process is analogous. In each of the k iterations, we gradually transform the coefficients in the described way, each time jumping by a larger power of 3.

The inverse transformation F1 is almost identical to the original, having the formula

(a,b,c)13(a+b+c,a+w2b+wc,a+wb+w2c)

It should be noted that in the implementation, there is no need to make calculations with complex numbers, whose real and imaginary parts are stored with a floating point type. Instead, we can notice that at each moment every number will be of the form a+bw, where a and b are whole numbers which fit in long long int. When calculating, it is useful to keep in mind that w3=1 and w2=1w.


Comments

There are no comments at the moment.