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

Автор Grzegorz, 8 лет назад, По-английски

It was enough interesting problems, I suggest discuss some of them. Who has proofed solutions of E and H?

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

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

Where can we find the problems?

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

H: Let's find a dfs search forest (from some vertices in each connected component). If it contains a path of length k + 1, print it. Otherwise, color each vertex with it's depth in the corresponding dfs tree.

Any ideas for problem I?

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

Who has proofed solution of H?

I do not know which solution did you mean, but my solution is proved by design.

Let's start coloring vertices using dfs. Each time we'll try to find such color C that is not used by neighbours, but color (C-1+K)%K is used. When paint vertex in this color C and remember neighbour with color (C-1+K)%K as previous for current vertex.

If we do not fail on such painting process, we have a graph's coloring. Otherwise, we failed on painting of some vertex v. That does mean, it neighbours uses all the colors, including color K-1 (0-based), so there is adjusted vertex u witch has color K-1. Here is some simple path from vertex u which is constructed by walking to previous vertices, and it's length is not less than K-1, so by adding edge u->v we have sought-for path with length K.

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

Ребята! Подскажите, пожалуйста, как A решать? (div1) Падает на 29ом хоть убей

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

    Нельзя бинарить ответ. Вроде это wa 29.

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

      Да, мы это поздно поняли) Но как решать по-умному?

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

        Dlya kazhdogo i nahodim samuiu levuiu j takuiu chto summa ot j do i men'she S. Eto delaesh derevom otrezkov.

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

      В смысле нельзя? Наше зашло на бинпоиске.

      1) Построим частичные суммы от правого края.
      2) Затем построим такие индексы, которые идут от правого края и если двигаться по ним, то сумма увеличивается.
      3) Потом переберём длину отрезка слева и бинпоиском по построенному на предыдущем шаге массиву найдём самую правую хорошую границу.

      Во втором пункте для массива [2,4,-2,1,1] массив индексов будет [4,3,1,0]

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

      Можно бинарить! Запомним максимумы частичных сумм. Тогда для длины LEN будем проверять есть ли последовательности >=LEN с суммой <=S. Как это проверить? Просто переберём: для [x, x+len) надо проверить PART[x+len] — MAX[x] <= S. Если это выполняется, то отрезок является суффиксом какой-то хорошей последовательности.

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

How to solve C?

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

    Problem C:

    Consider multigraph on n vertices (not 2n!).

    Disjunction ([not]x[i] | [not]x[j]) is edge between vertices i and j.

    The solution is recursion (backtracking with optimizations).

    1. If value of any variable may be deduced, deduce it.

    2. If there are to connected components, calculate it separately, multiply results.

    3. Fix unknown variable of maximum degree and do 2 recursive calls.

    4. Your may add memoization to speed up the solution.

    5. Your may handle the case of "maximum degree is two" in polynomial time.

    As I know, the first 3 ideas were enough to get OK.

    P.S. T(n) ≤ T(n - 1) + T(n - 1 - maxDegree)

    UPD: (4) implies (2) for components of size 1 (degree equal to 0)

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

E: Main idea: number of ways to divide subset only depends on count of vertices. We can use DP to calculate it for every size.

dp[0] = 1;
for (int i = 2; i <= n; i += 2)
{
	for (int j = 1; j <= i; j++)
	{
		for (int h = j + 1; h <= i; h += 2)
		{
			dp[i] += dp[h - j - 1] * dp[i - (h - j - 1) - 2];
		}
	}
}

Let's find first pair in subset. Sort all possible pairs and iterate over them. We need to know number of ways with this fixed pair in order to choose, take this or not. Number of ways depends only on size of left and right subset. Find this pair and recursively solve at first for left and then for right subsets.

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

What about I? We had 2^32*const solution, it was too slow, like 45 seconds instead of 5

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

I've seen several "read a paper" problems in Open Cup but I've never solved it. How do you solve it?

Search, read, understand, and implement a 10-page paper during a contest looks overwhelming to me. Is it important to find an easy-to-read article? Do you know the algorithm beforehand? Are you actually very good at reading papers?

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

    Could you please give a link to the paper?

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

      According to wikipedia the fastest algorithm is O(1.2377n), but we don't have an access and maybe this is an overkill.

      This paper looks simpler, but it says this is O(1.62n) but I'm not sure if this is fast enough.

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

        This paper is fine. The algorithm presented there is even harder than needed, what I've successfully submitted looks like:

        • if at any moment you can deduce value of some variable or remove unnecessary clauses, do it
        • if the graph induced by clauses on the variables (n vertices in total) is disconnected, solve each component separately and multiply the answers
        • otherwise, try two options for some of the variables: if mentioned graph has the articulation point, try it (and perform two recursive calls), otherwise try the vertex of the maximum degree

        But I think that using such classical problems on a competition is a completely bad idea. All I had to do was to know that this problem is called #SAT (or sharp-SAT). Also, this particular problem is very hard to prepare, I'm not sure that it is possible to come up with reasonable tests for it.

        By the way, it is one of the main counting problems that arise in CS, the #3SAT is actually a #P-complete problem, meaning that many counting problems may be reduced to it, like 0-1 matrix permanent, counting number of topological sorts of the DAG, counting number of correct k-colorings, and several others.

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

        This one

        http://codeforces.me/blog/entry/47641?#comment-319796

        is not very hard to invent without any papers.

        It works in 1.46n or 1.38n (if you handle the last case).

        In practice it works faster.

        To solve you have to know only one base idea "consider graph, take vertex of maximal degree".

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

          P.S It's interesting to know, how many people used papers from google to get OK.

          Please, "like" this comment,

          if you read special papers during the contest, and got OK.

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

          Please, "like" this comment,

          if you did not used any special papers during the contest, and got OK.

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

            173 people got OK?

            Anyway, if we believe the ratio, it seems we should ignore the paper and solve it as usual.

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

              Possible reasons: If you rating is high, your upvote gives more than +1. Also most teams consist of three members.

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

                Lol, your upvote gives +2 at maximum, there are 3 members at a team and 19 teams that solved this problem, giving us 114 votes at maximum.

                I believe there were lots of people who didn't even participate in opencup, but who like the idea of using or not using any papers in general, and who felt that it is fine to give their upvote for one of the options, even though they were not asked for it.

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

    Despite the fact I've never solved, let alone attempted such a problem, I suppose it's more about having an eye for a particular algorithm, which usually means you have already implemented the said algorithm in the past at least once or twice, and all you need during the contest is a quick(?) rehearsal.

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

Отвечаю про задачу I (shuffle-twice и упрощённый вариант shuffle-again для Div2). Отдельным комментарием, потому что вопросы на английском, а ответить я сейчас готов только на русском.


Обозначим исходное состояние генератора как x0, а все последующие как x1, x2, x3, ....

Заметим, что за один shuffle каждое число участвует в одном обязательном swap-е: в тот момент, когда на него указывает индекс цикла. При этом число перемещается на неотрицательное расстояние влево. После этого число может попасть в другие swap-ы, каждый из которых перемещает его на положительное расстояние вправо.


Сначала решим задачу shuffle-again (Div2-версию): как восстановить seed после одного вызова shuffle. Легко понять, что последнее число участвует ровно в одном swap-е. Поэтому его позиция позволяет нам установить x10000·10000 div 232. Более того, если мы обратим последний swap, то по аналогичным причинам предпоследнее число участвовало ровно в одном из оставшихся swap-ов, и мы можем узнать x9999·9999 div 232, и аналогично далее: x9998·9998 div 232, x9997·9997 div 232, ....

Научимся по следующему состоянию генератора находить предыдущее: если xk + 1 = xk·a + c, то xk = (xk + 1 - ca - 1 (оба равенства — сравнения по модулю 232). Для этого нужно всего лишь обратить число a по модулю 232.

Заметим, что, если мы знаем x10000·10000 div 232, это оставляет нам всего около 232 / 10000 возможных вариантов для истинного значения x10000. Переберём их все. Для каждого будем вычислять x9999, x9998, ... и сразу проверять, что числа x9999·9999 div 232, x9998·9998 div 232, ... такие, как у нас получились. Для каждого значения x10000 мы либо пройдём все эти проверки, либо очень быстро (за 1-2 шага) найдём проверку, которая выдаст ошибку. Поэтому суммарное время перебора будет близко к 232 / 10000. Как только мы найдём такое x10000, для которого пройдём все проверки, соответствующее ему x0 и будет искомым значением seed.


Решим теперь задачу shuffle-twice с двойным перемешиванием. Теперь каждое число участвует не менее чем в двух swap-ах. Откуда вообще мы можем получить информацию? Нам хотелось бы найти число, которое редко двигалось — тогда, может быть, мы перебором сможем найти, как именно оно двигалось. Поэтому будем искать такие числа, которые участвуют ровно в двух swap-ах, а не больше. Можно экспериментально проверить, что таких чисел примерно 1 / 6 от общего количества.

Будем рассматривать числа, начиная с конца перестановки. Пусть очередное число не меньше своего индекса, например, p9000 = 9876. Предположим, что число 9876 участвовало ровно в двух swap-ах: в момент 9876 оно переместилось на позицию mid, а в момент 10000 + mid — на позицию 9000. Переберём все возможные значения mid от 9000 до 9876 включительно.

При зафиксированном значении mid у нас есть два сравнения по модулю следующего вида (*):

x9876·9876 div 232 = mid,

x10000 + mid·mid div 232 = 9000.

Предположим, мы научимся быстро находить все решения такой системы - а при достаточно больших модулях 9876 и mid их обычно не больше пары сотен. Тогда все x0, соответствующие этим решениям, являются кандидатами на правильное значение seed. Проверять правильность seed для всех них может всё ещё быть долго, поэтому будем складывать x0 в set и проверять только для тех, которые попали в set во второй раз. Когда мы в переборе в первый раз найдём две верные гипотезы вида (*), проверка подтвердит правильность найденного x0, и мы сможем вывести ответ.

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

Возможно, если хорошо организовать перебор на этапе поиска чисел, участвовавших ровно в двух swap-ах, может хватить и медленного решения системы двух сравнений: перебора решений одного сравнения за 232 / mid и проверки, подходят ли они по второму сравнению.

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

    Разве на n-ом вызове range(k) не возвращает ?
    Тогда получаются уравнения вида

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

      Да, я тоже решаю эту систему переходом к отрезкам [l1, r1] и [l2, r2] по модулю 232 для двух некоторых xi. А потом ищу минимальные положительные шаги иксом первого отрезка, которые дают шаг на втором отрезке от  - (r2 - l2 + 1) до  - 1 и шаг от  + 1 до  + (r2 - l2 + 1).

      Вот код, этот кусок решения в строках 175-199.

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

      Я понял со второго раза. Действительно, не mod k, а ·232 div k везде. Спасибо, исправил.

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

    На контесте мы написали точно такое же решение за исключением решения системы — оно было написано втупую за и, естественно, не зашло.

    В итоге решение системы свелось к пересечению арифметической прогрессии по модулю и отрезка. Его можно найти за методом а-ля baby-step-giant-step(можно ли решить эту задачу быстрее?). После замены mapы для хранения кандидатов на хэштаблицу, решение со скрипом влезло в 3 секунды.

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

How to solve G?

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

    Just try to randomly generate some sets with sizes 10..1e7 and look at the min/max/avg value of distinct_values/total_asked_values (hint: it's better to ask 10_000 values).
    Then you'll able to find the answers and insert it to the main program.

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

    My solution used Birthday Paradox. That is, the number of questions until two cards repeat is somehow close to sqrt(n), where n is the size of the set.

    Let's do the following procedure: while I still have questions, ask questions until two cards are identical. Then, store in a vector number of question asked and repeat.

    All numbers from my vector should be close to a value of sqrt(n), where n is in the set {10, 100, ...., 10^7}. The only remaining part is to find n such as numbers from my vector match best sqrt(n).

    Let's iterae over powers of 10 and call X = sqrt(current number). We simply need to find an heuristic to see how close are number from the vector to X. We used sum {(vector[i] — X) ^ 2}

    Then, pick X such as the sum is minimized and output corresponding power of 10.

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

    I used Bayes formula and returned sets size with maximum probability (using uniform distribution over allowed values as prior).

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

Had solved stringy problem 'J' (about timer) with Perl, 127 lines (ideone).

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

Anyone know how to solve problem I?