Блог пользователя BekzhanKassenov

Автор BekzhanKassenov, 10 лет назад, перевод, По-русски

519A - A и B и шахматы

Автор: BekzhanKassenov

В этой задаче необходимо было найти чья шахматная позиция сильнее.

Решение: Пробегаемся по всей доске, складывая веса фигур игроков, и в конце выводим ответ.

Асимптотика: O(n2), где n — длина стороны доски (в этой задаче — 8)

Код: 10083191

519B - A и B и ошибки компиляции

Автор: ADJA

В этой задаче вам было дано 3 массива. Второй содержал все элементы первого, за исключением одного, третий массив содержал все элементы второго, за исключением одного. Необходимо было вывести удаленные элементы.

Решение: Я опишу простейшее решение этой задачи: Обозначим сумму элементов в первом, втором и третьем массиве как a, b и c соответственно. Ответом будут числа a - b и b - c

Также существуют множество других решений, использующих map, xor и прочее.

Асимптотика: O(N)

Код: 10083248

519C - A и B и командная тренировка

Автор: ADJA

В этой задаче вам необходимо было поделить n опытных участников и m новичков на команды.

Решение: Обозначим команды с 2 опытными участниками и 1 новичком как type1 и команды с 1 опытным участником и 2 новичками как type2. Зафиксируем количество команд type1 как i. Их количество не превышает m. Тогда количество команд type2 равно min(m - 2 * i, n - i). Осталось проверить все возможные варианты i и выбрать оптимальный.

Асимптотика: O(N)

Код: 10083265.

519D - A и B и интересные подстроки

Автор: ADJA

В этой задаче необходимо было найти количество подстрок данной строки таких, что каждая подстрока начинается и заканчивается на одну и ту же букву и сумма весов букв в этой подстроке (кроме начала и конца) равно нулю.

Решение: Обозначим sum[i] как сумму весов первых i букв. Создадим 26 map < longlong, int > -ов, 1 на каждую букву. Пусть в данный момент мы находимся на позиции i и map текущего символа m. Тогда необходимо добавить к ответу m[sum[i — 1]], и sum[i] в m.

Асимптотика: O(Nlog(N)), где N — длина исходной строки.

Код: 10083293.

519E - A и B и аудитории

Автор: BekzhanKassenov

В задаче необходимо было отвечать на следующие запросы на дереве: для заданных пар вершин найти количество вершин, равноудаленных от них.

Принятые обозначения:

dist(a, b) — расстояние между вершинами a и b.

LCA(a, b)наименьший общий предок вершин a и b.

depth[a] — расстояние от корня дерева до вершины a.

size[a] — размер поддерева, корнем которой является вершина a.

На каждой картинке вершины, покрашенные в зеленый цвет — вершины из ответа, вершины, покрашенные в синий цвет — вершины из запросов.

Препроцессинг: Для начала необходимо считать ребра дерева и построить по ним структуру данных для нахождения LCA (удобнее всего использовать двоичный подъем, так как в будущем он понадобится для других целей). Асимптотика: O(NlogN)

Ответы на запросы:

Для каждого запроса необходимо рассмотреть несколько случаев:

1) a = b. В этом случае ответ равен n.

2) dist(a, b) нечетно. Тогда ответом будет 0.

3) dist(a, l) = dist(b, l), где l = LCA(a, b).

Двоичным подъемом найдем сыновей вершины l, которые являются предками вершин a и b (обозначим их как aa и bb соответственно). Ответом будет n - size[aa] - size[bb].

4) Все остальные случаи.

Предположим, что depth[a] > depth[b]. Тогда найдем с помощью двоичного подъема dist(a, b) / 2-ого предка вершины a (обозначим его как p1), dist(a, b) / 2 - 1-ого предка вершины a (обозначим его как p2). Ответом будет size[p1] - size[p2].

Асимптотика: O(logN) на запрос, O(MlogN) на все запросы.

Итоговая асимптотика: O(MlogN + NlogN)

Код: 10083310.

  • Проголосовать: нравится
  • +99
  • Проголосовать: не нравится

»
10 лет назад, # |
  Проголосовать: нравится +1 Проголосовать: не нравится

А можно подробнее решение D? Совсем не понятно, в чем идея и как к ней прийти.

  • »
    »
    10 лет назад, # ^ |
      Проголосовать: нравится +4 Проголосовать: не нравится

    Для каждой буквы нужно найти сколько таких же букв слева от нее, которые имеют такую же префиксную сумму как и буква, слева от текущей. Когда мы отнимем эти префиксные суммы мы получим сумму на подотрезке, которая будет равна нулю. Префиксные суммы слева удобно хранить в map-е и добавлять текущую в него же.

»
10 лет назад, # |
Rev. 2   Проголосовать: нравится +17 Проголосовать: не нравится

Только в задаче С можно проще за счёт жадности. Min(n, m, (n+m)/3). Асимптотика О(1). Удивлён что эталонное решение не такое. Думаю что так решали почти все. Берём или 2 опытных, или новичков — кого больше. Так пока не кончатся новички, либо опытные, либо люди вообще, отсюда формула.

  • »
    »
    10 лет назад, # ^ |
      Проголосовать: нравится +1 Проголосовать: не нравится

    Да, это было одним из авторских решений. Мы не стали поднимать ограничения, чтобы задачу можно было решить как и формулой, так и простым линейным решением.

  • »
    »
    10 лет назад, # ^ |
      Проголосовать: нравится +15 Проголосовать: не нравится

    Ох уж эти споры, что проще: подумать или написать цикл.

»
10 лет назад, # |
  Проголосовать: нравится +24 Проголосовать: не нравится

Очень крутое решение B, ни сам ни догадался, ни одного такого в комнате не заметил.

»
10 лет назад, # |
  Проголосовать: нравится -8 Проголосовать: не нравится

Problem C can be solved in O(1) time. Just 1 line of code! cout<<min(m,min(n,(m+n)/3))<<endl;

Check my solution

  • »
    »
    9 лет назад, # ^ |
      Проголосовать: нравится 0 Проголосовать: не нравится

    Why was the above solution downvoted. It is very good!

  • »
    »
    4 года назад, # ^ |
      Проголосовать: нравится 0 Проголосовать: не нравится

    I did the same couldn't figure out the need for dynamic programming in this specific case

  • »
    »
    3 года назад, # ^ |
      Проголосовать: нравится 0 Проголосовать: не нравится

    Can u explain your solution ?

    • »
      »
      »
      2 года назад, # ^ |
      Rev. 2   Проголосовать: нравится 0 Проголосовать: не нравится

      The secret is here

      (m+n)/3 means that you take 1 type and 2 type for 2/3 teams and reverse type for the other 1/3 teams.

      And that cuz of the two values n and m is close to each other so there are no precious type.

      if n is minimum means that you take 1 type n and 2 of m for all teams
      that happens if you m is greater than m by (2*n) which means n is so precious so you must use all of it.

      and the same thing for m when n is greater than 2*m by m then m is precious and you must use all of it.

           if( n-(m*2) > 0) cout<<m;
           else if( m-(n*2) > 0) cout<<n;
           else cout<<(n+m)/3;
      
  • »
    »
    2 года назад, # ^ |
      Проголосовать: нравится 0 Проголосовать: не нравится

    yeah it's good one. I upvoted

    Actually I thought it's wrong cuz of downvotes So thanks.

»
10 лет назад, # |
  Проголосовать: нравится +3 Проголосовать: не нравится

519C — A and B and Team Training my codes got a AC,but do not know why,can somebody prove that?

int main (){ LL a,b; cin >> a >> b; cout << min((a+b)/3, min(a,b)) << endl; return 0; }

  • »
    »
    10 лет назад, # ^ |
      Проголосовать: нравится +1 Проголосовать: не нравится

    Think of creating a matrix with maximum number of rows and 3 columns such that each row should contain at least 1 newbie and 1 experienced player and fill the rest with remaining players. If the number of newbie players is more than 2 times the number of experienced players then the number of experienced players itself is the answer since each team should contain at least one experienced player. Just interchange experienced and newbie in the previous sentence.

    If you have enough players, the maximum number of teams that can be formed is floor((m+n)/3). Fill the newbies and experienced players in the grid vertically such that each row satisfies the conditions mentioned in the question.

    • »
      »
      »
      10 лет назад, # ^ |
        Проголосовать: нравится +7 Проголосовать: не нравится

      Thanks. Through your thoughts.I prove that by maths.

      The number of teams is x, the newbie is a, the experienced player is b. Each team have a newbie and a experienced player at least. After that, the rest of newbie is (a-x), the rest of experienced player is (b-x). After complete team, the left player is (a-x) + (b-x) — x.

      if 0 <= (a-x) + (b-x) — x < 3,then x <= (a+b)/3 < x+1, so x = (a+b)/3. The left player may have newbie and experienced player, so x <= a or x <= b, so x <= min(a,b), so (a+b)/3 <= min(a,b). else if (a-x) + (b-x) — x >= 3, the left player is all newbie or all experienced player. The other is in team. so we get x <= (a+b)/3-1 and x = min(a,b), so x = min(a,b) < (a+b)/3.

      Through discuss, if 0 <= (a-x) + (b-x) — x < 3. x = (a+b)/3 <= min(a,b) => x = min((a+b)/3, min(a,b)). else if (a-x) + (b-x) — x >= 3 x = min(a,b) < (a+b)/3 => x = min((a+b)/3, min(a,b)).

      so. the number of team x = min((a+b)/3, min(a,b)).

      My English is so bad, excuse me.

      • »
        »
        »
        »
        10 лет назад, # ^ |
          Проголосовать: нравится +6 Проголосовать: не нравится

        why (a-x) + (b-x) — x is less than 3 ??

        • »
          »
          »
          »
          »
          10 лет назад, # ^ |
            Проголосовать: нравится 0 Проголосовать: не нравится

          Because while (a-x) + (b-x) — x is less than 3, the left players do not form a team. And (a-x) + (b-x) — x is equal or greater than 3, we can sure that the left player is all newbie or all experienced player.

    • »
      »
      »
      10 лет назад, # ^ |
        Проголосовать: нравится +3 Проголосовать: не нравится

      But How you will handle the case when newbie players are greater than the experienced players but are less than the 2*experienced players and vice versa.

»
10 лет назад, # |
  Проголосовать: нравится 0 Проголосовать: не нравится

Oh.. The prob. D is map<longlong,int> which I used map<int,int> .. pretest is ok then I failed system test. Sad story. QAQ

  • »
    »
    10 лет назад, # ^ |
      Проголосовать: нравится 0 Проголосовать: не нравится

    same here :(

    • »
      »
      »
      10 лет назад, # ^ |
        Проголосовать: нравится 0 Проголосовать: не нравится

      hello the sum could exceding int 32 bits because in the case a = 100000 b = 100000 ..... z = 100000

      n * 100000 = 10^10 > 32 bits int overflow....

»
10 лет назад, # |
  Проголосовать: нравится +1 Проголосовать: не нравится

Can anybody share some problem links related to the problem E :) . It will be a great help :).

»
10 лет назад, # |
  Проголосовать: нравится 0 Проголосовать: не нравится

Can you upload the pictures some where else. I can't see them :(

»
10 лет назад, # |
  Проголосовать: нравится 0 Проголосовать: не нравится

B solution is great!

»
10 лет назад, # |
  Проголосовать: нравится 0 Проголосовать: не нравится

Could anybody explain more problem D? How is that map working? What is stored there?

  • »
    »
    10 лет назад, # ^ |
      Проголосовать: нравится 0 Проголосовать: не нравится

    map<long long, int> m[26]; m[i] stores <sum, num of prefixes ending with char 'i'>

  • »
    »
    10 лет назад, # ^ |
      Проголосовать: нравится 0 Проголосовать: не нравится

    What you do with that map, is store what was the sum for the previous characters. If you sum 6 for an 'a', and then find another 'a' that the sum for previous characters is 6, that means that the sum between the first 'a', and the second 'a' is 0, because they have the same sum range.

    • »
      »
      »
      9 лет назад, # ^ |
        Проголосовать: нравится 0 Проголосовать: не нравится

      I used this approach too, and got an AC despite expecting a TLE. Can you tell me why 10^5 element insertion in a map is passing? I've found it to give a TLE many times. Test cases not strong?

      • »
        »
        »
        »
        9 лет назад, # ^ |
        Rev. 2   Проголосовать: нравится 0 Проголосовать: не нравится

        A simple C++ map has O(log(n)) insertion, deletion, and searching operations. With n = 10^5, think that your algorithm is doing just 10^5 * log(10^5) = 10^5 * 17 = 1.700.000 operatios (in average), and you can do aprox. 30.000.000 operations per second in a normal judge (or more), and for this problem you have 2 seconds. So it would never give you TLE.

        • »
          »
          »
          »
          »
          9 лет назад, # ^ |
            Проголосовать: нравится 0 Проголосовать: не нравится

          Sadly I've still encountered problems where 10^5 is a problem. But now I realise, this is CF we're talking of here, and unlike SPOJ, its servers are fast.

          • »
            »
            »
            »
            »
            »
            9 лет назад, # ^ |
              Проголосовать: нравится 0 Проголосовать: не нравится

            I don't think an O(n log n) approach can give you TLE with 1 second in any judge. If you have some examples we can check them.

            • »
              »
              »
              »
              »
              »
              »
              9 лет назад, # ^ |
                Проголосовать: нравится 0 Проголосовать: не нравится

              I can't recall right now, but I'll definitely discuss them with you when I encounter another one. Thanks though! :)

»
10 лет назад, # |
  Проголосовать: нравится +4 Проголосовать: не нравится

The same code submitted after the contest gave AC but failed on Test 36 during system testing.

Problem: A Failed code: 10066549 Passed code; 10085855

These are the same piece of code. Please look into this.

»
10 лет назад, # |
  Проголосовать: нравится +8 Проголосовать: не нравится

Can someone share a good tutorial related to the binary raise technique ?

»
10 лет назад, # |
  Проголосовать: нравится 0 Проголосовать: не нравится

It was good contest & good problems

»
10 лет назад, # |
  Проголосовать: нравится +13 Проголосовать: не нравится

Задачи классные (особенно понравились D и E), авторы молодцы :)

»
10 лет назад, # |
  Проголосовать: нравится +1 Проголосовать: не нравится

Could someone explain problem D? What is the map doing there?

  • »
    »
    10 лет назад, # ^ |
      Проголосовать: нравится +6 Проголосовать: не нравится

    The problem asks to compute the number of substrings that begin and end with the same letter, and whose sum of weights equals 0 (ignoring the endpoints).

    There is a very simple, correct, and inefficient solution. For each pair (i, j), i < j of occurrences of letter c calculate in O(1) the sum of the inner sub-interval (i + 1, j - 1). Given that this would take O(N^2), we have to discard this solution as N can be as big as 105.

    We can do better. Assume we take each occurrence of c in a right-to-left fashion. The occurrence at index j has prefixsum(j) weight. Assume at index i (i < j) we have another occurrence of c. Under what circumstances could S[i... j] be interesting? Well, if prefixsum(i) + weight(c) = prefixsum(j), we have a guarantee that the sum of the elements between i and j is zero-sum.

    So, the map is used as a sort of cache that indicates whether we have previously achieved certain prefix sums.

»
10 лет назад, # |
Rev. 3   Проголосовать: нравится 0 Проголосовать: не нравится

My solution for prob.C was binary searching ans with L=1 and R=(m+n)/3

If pair of N experienced participants and M newbies (N,M) can form (L+R)/2 teams then L=mid+1 else R=mid-1;

To check if (N,M) can form x teams I did this: - Assume that (N,M) can form x teams , so we have x1 teams of type1 (XP,XP,NB) and (x-x1) teams of type2 (XP,NB,NB) with x1 in [0..x]. So there will be (x+x1) XP and (2x-x1) NB in total. - Hence the conditions are ((x+x1 <= N) AND (2x-x1) <= M)) OR ((x+x1 <= M) AND (2x-x1) <= N)) with x1 in [0..x]

So the complexity is O(nlogn) i think?

»
10 лет назад, # |
Rev. 2   Проголосовать: нравится +3 Проголосовать: не нравится

Эх. Чтобы решить задачу D, человеку пишущему на Pascal-e, придётся писать сбалансированное дерево. Хочется узнать, решил ли её кто-нибудь без map-a?

  • »
    »
    10 лет назад, # ^ |
      Проголосовать: нравится +4 Проголосовать: не нравится

    Я решил. И я так и не понял, что у них за решение такое с map-ом. 10086155

    • »
      »
      »
      10 лет назад, # ^ |
        Проголосовать: нравится 0 Проголосовать: не нравится

      По сути у нас очень похожие решения (наш map заменен на массив+сортировку+бинарный поиск).

  • »
    »
    10 лет назад, # ^ |
      Проголосовать: нравится 0 Проголосовать: не нравится

    10087483 Мое решение без мапа, думаю один из худших решении.

»
10 лет назад, # |
  Проголосовать: нравится +1 Проголосовать: не нравится

Could problem C be solved using integer linear programming?

»
10 лет назад, # |
  Проголосовать: нравится +4 Проголосовать: не нравится

E can also be solved using Heavy-Light Decomposition!!!

»
10 лет назад, # |
  Проголосовать: нравится 0 Проголосовать: не нравится

solution for C :

int res = 0;
while (n > 0 and m > 0 and n + m >= 3) {
	if (n < m)
		n--, m -= 2;
	else m--, n -= 2;
	res++;
}
cout << res << "\n";

can anyone explain why this solution works ?

  • »
    »
    10 лет назад, # ^ |
      Проголосовать: нравится 0 Проголосовать: не нравится

    guess this is a solution based on greedy theory.

    I doubted this idea during the contest,but now I feel it's quite right to think so.

    if(n < m) then it's better to use more m; otherwise it's better to use more n. Am i getting it right?

    BTW::another solution is fairly unbelievable the ans is min(min(n,m),(n+m)/3)

    • »
      »
      »
      10 лет назад, # ^ |
        Проголосовать: нравится 0 Проголосовать: не нравится

      But in fact, min(min(n,m),(n+m)/3) is also based on greedy theory. Suppose that n<m, if m>=2*n, we must run out of n, and at the same time, use more m. so we choose 1n2m and the answer become n; if m<2*n, 1<m/n<2 , so we can choose 1n2m for a times and 2n1m for b times, and let a+2b=n and 2a+b=m ,here the answer is a+b=(n+m)/3;

      at last, we can combine them together to get the expression:min(n,(n+m)/3), here n=min(n,m)

»
10 лет назад, # |
  Проголосовать: нравится 0 Проголосовать: не нравится

Можно ли E решить через BFS, без lca и двоичных подъемов? То есть, я их не знаю и пытался решить по другому но это дело вылетело по времени. можно ли это доработать?посылка

  • »
    »
    10 лет назад, # ^ |
      Проголосовать: нравится +2 Проголосовать: не нравится

    Ну представь, что будет, если дадут граф-цепочку, и каждым запросом будут давать концы этой цепочки. Учи двоичные подъемы. Это вроде не требует никаких специальных знаний, так что не должно вызывать особых трудностей.

    • »
      »
      »
      10 лет назад, # ^ |
        Проголосовать: нравится 0 Проголосовать: не нравится

      не совсем понял, к чему вы клоните. то есть >>Ну представь, что будет, если дадут граф-цепочку, и каждым запросом будут давать концы этой цепочки.>> ...ээ, и что будет?

      • »
        »
        »
        »
        10 лет назад, # ^ |
          Проголосовать: нравится +6 Проголосовать: не нравится

        Будешь середину искать за линию — выйдет O(N^2) => TL.

  • »
    »
    10 лет назад, # ^ |
      Проголосовать: нравится 0 Проголосовать: не нравится

    Выше писали, что можно с помощью HLD :)

»
10 лет назад, # |
  Проголосовать: нравится 0 Проголосовать: не нравится

image not visible!!!

»
10 лет назад, # |
  Проголосовать: нравится 0 Проголосовать: не нравится

Nice problems. I really appreciate them.

»
10 лет назад, # |
Rev. 4   Проголосовать: нравится 0 Проголосовать: не нравится

Thx for nice problems!

»
10 лет назад, # |
Rev. 5   Проголосовать: нравится +18 Проголосовать: не нравится

As for problem C, I solved it using linear programming graphically.

Let A = Number of experienced people that will join the contest. B = Number of newbies joining the contest. X = Number of all experienced people. Y = Number of all newbies.

The objective function is to maximize Z = A + B. And the answer will be floor(Z/3).

The constraints are: A <= 2*B , B <= 2*A , A <= X , B <= Y You can deduce these constraints yourself.

If you draw the graph, you can deduce that there are only three cases:

(1) Point (X,Y) is feasible and then it is the answer because it the optimal corner point.

(2) Point (X,Y) is in the upper left of the feasible region, meaning that X < Y / 2. Then the optimal corner point will be (X,2*X).

(3) Point (X,Y) is in the bottom right of the feasible region, meaning that Y < X /2. Then the optimal corner point will be (Y,2*Y).

So you can easily check by conditions which case it is and print Z / 3.

There is another solution which is minimum between (X,Y,(X+Y)/3). Actually it is the same. Because the right case of these three cases will have the minimum value and the other two will be unfeasible solutions, which means that the feasible solution across the three cases is the minimum across the three solutions. And you can check that by drawing.

If you have any question, feel free to ask.

My Submission : 10099801

»
10 лет назад, # |
  Проголосовать: нравится 0 Проголосовать: не нравится

How safe is it to assume that the variables are initialized to zero by default? The code shown here immediately uses timer++ (problem E), as well as ans += and Map[sum[p]]++ in problem D.

  • »
    »
    10 лет назад, # ^ |
      Проголосовать: нравится 0 Проголосовать: не нравится

    depends.. global variables are zero initialized as they are static. whereas local variables dont have any default values

»
10 лет назад, # |
  Проголосовать: нравится 0 Проголосовать: не нравится

Solution C: In the solution leftn = n-i is divided by 2. Why?

  • »
    »
    10 лет назад, # ^ |
      Проголосовать: нравится 0 Проголосовать: не нравится

    You group them into 2 of the leftn : 1 of m (or leftm), so you only have leftn/2 of those teams.

»
10 лет назад, # |
  Проголосовать: нравится 0 Проголосовать: не нравится

In E, what does anc[i][j] store, and how does go_up() work? Are these routines from a standard algorithm?

»
10 лет назад, # |
  Проголосовать: нравится 0 Проголосовать: не нравится

Another solution for problem C in O(N): If there's more EX than NB, then create a 2EX, 1NB team, else create a 1EX, 2NB team. Stop until it's impossible to create new team.

My solution: 10189473

»
10 лет назад, # |
  Проголосовать: нравится 0 Проголосовать: не нравится

I have written LCA finding algorithm in <O(N logN, O(logN)> time complexity, but I still get TLE on test 12 on java. Can anybody help me?? here is my solution 10217373

  • »
    »
    10 лет назад, # ^ |
      Проголосовать: нравится 0 Проголосовать: не нравится

    To make it absolutely clear, I mean Problem E.

    • »
      »
      »
      10 лет назад, # ^ |
      Rev. 2   Проголосовать: нравится +3 Проголосовать: не нравится

      It seems that the slowest part is following lines:

      for(int k = 0; k < dist / 2 - 1; k++)
          p = T[p];
      

      They make your solution to answer in O(N) because dist can be n - 1 in worst case.

      • »
        »
        »
        »
        10 лет назад, # ^ |
          Проголосовать: нравится 0 Проголосовать: не нравится

        Thanks, you are right! Although, I rewrote the code in C++ and got AC, but I will remake that piece of code to make it even faster. Thank you again!

»
7 лет назад, # |
Rev. 2   Проголосовать: нравится 0 Проголосовать: не нравится

http://codeforces.net/contest/519/submission/28950258

Can anyone help me find why my solution of problem E is giving TLE in testcase#20???

  • »
    »
    7 лет назад, # ^ |
      Проголосовать: нравится 0 Проголосовать: не нравится

    Use another sparse table or use binary raise technique to add the appropriate subtrees, instead of the for loop you are using.

»
5 лет назад, # |
  Проголосовать: нравится 0 Проголосовать: не нравится

For problem E, why is the max height is 30?

»
4 года назад, # |
  Проголосовать: нравится 0 Проголосовать: не нравится

Div 2 C can be done in a more easily understandable way like 83759343

»
4 года назад, # |
  Проголосовать: нравится 0 Проголосовать: не нравится

shortcut for problem C:

ans = min(m, m, (m+n)/3)

Explanation can be found by plotting lines in 2D space