Tic-tac-toe con solo croci il più velocemente possibile


10

Secondo la richiesta di Luke e l'aggiunta di Peter Taylor a questa sfida.

introduzione

Tutti conoscono il gioco tic-tac-toe, ma in questa sfida introdurremo una piccola svolta. Useremo solo croci . La prima persona che piazza tre croci di fila perde. Un fatto interessante è che la quantità massima di croci prima che qualcuno perda, è pari a 6 :

X X -
X - X
- X X

Ciò significa che per una scheda 3 x 3, l'importo massimo è 6 . Quindi per N = 3, dobbiamo produrre 6.

Un altro esempio, per N = 4, o una scheda 4 x 4:

X X - X
X X - X
- - - -
X X - X

Questa è una soluzione ottimale, puoi vedere che la quantità massima di croci è pari a 9 . Una soluzione ottimale per una scheda 12 x 12 è:

X - X - X - X X - X X -
X X - X X - - - X X - X
- X - X - X X - - - X X
X - - - X X - X X - X -
- X X - - - X - - - - X
X X - X X - X - X X - -
- - X X - X - X X - X X
X - - - - X - - - X X -
- X - X X - X X - - - X
X X - - - X X - X - X -
X - X X - - - X X - X X
- X X - X X - X - X - X

Ciò si traduce in 74 .

L'obiettivo

Il tuo compito è calcolare i risultati il ​​più velocemente possibile. Eseguirò il tuo codice sul test case 13. Questo verrà fatto 5 volte e poi verrà presa la media dei tempi di esecuzione. Questo è il tuo punteggio finale. Più basso è, meglio è.

Casi test

N     Output
1       1
2       4
3       6
4       9
5       16
6       20
7       26
8       36
9       42
10      52
11      64
12      74
13      86
14      100
15      114

Ulteriori informazioni sono disponibili all'indirizzo https://oeis.org/A181018 .

Regole

  • Questo è , quindi vince l'invio più veloce!
  • È necessario fornire un programma completo .
  • Fornisci anche come devo eseguire il programma. Non ho familiarità con tutti i linguaggi di programmazione e come funzionano, quindi ho bisogno di un po 'di aiuto qui.
  • Naturalmente, il tuo codice deve calcolare il risultato corretto per ogni caso di test.

osservazioni:

  • feersum (C ++ 11): 28s
  • Peter Taylor (Java): 14m 31s


Non è solo una copia della seconda domanda , per quanto posso dire, hai appena cambiato le condizioni vincenti?
Blue

1
@muddyfish Anche se la sfida in sé sembra la stessa, posso assicurarti che l'approccio per questa sfida è molto diverso dall'altra mia sfida.
Adnan,

3
@muddyfish Meta discussione pertinente. "Basta cambiare la condizione vincente" può essere un cambiamento sostanziale per una sfida. Sebbene non abbia senso pubblicare un code-golf , un algoritmo più veloce e un codice più veloce per ogni possibile sfida, ci sono alcuni casi in cui esplorare un problema da due angolazioni può aggiungere molto valore al sito. Penso che sia un caso del genere.
Martin Ender,

1
Sfida meravigliosa! (+1)

Risposte:


5

C ++ 11, 28s

Questo utilizza anche un approccio di programmazione dinamica basato su righe. Ci sono voluti 28 secondi per eseguire l'argomento 13 per me. Il mio trucco preferito è la nextfunzione che usa un po 'di bashing per trovare la disposizione della riga successiva lessicograficamente che soddisfi una maschera e la regola no-in-a-row.

Istruzioni

  1. Installa l'ultimo MinGW-w64 con thread SEH e Posix
  2. Compilare il programma con g++ -std=c++11 -march=native -O3 <filename>.cpp -o <executable name>
  3. Corri con <executable name> <n>
#include <vector>
#include <stddef.h>
#include <iostream>
#include <string>

#ifdef _MSC_VER
#include <intrin.h>
#define popcount32 _mm_popcnt_u32
#else
#define popcount32 __builtin_popcount
#endif


using std::vector;

using row = uint32_t;
using xcount = uint8_t;

uint16_t rev16(uint16_t x) { // slow
    static const uint8_t revbyte[] {0,128,64,192,32,160,96,224,16,144,80,208,48,176,112,240,8,136,72,200,40,168,104,232,24,152,88,216,56,184,120,248,4,132,68,196,36,164,100,228,20,148,84,212,52,180,116,244,12,140,76,204,44,172,108,236,28,156,92,220,60,188,124,252,2,130,66,194,34,162,98,226,18,146,82,210,50,178,114,242,10,138,74,202,42,170,106,234,26,154,90,218,58,186,122,250,6,134,70,198,38,166,102,230,22,150,86,214,54,182,118,246,14,142,78,206,46,174,110,238,30,158,94,222,62,190,126,254,1,129,65,193,33,161,97,225,17,145,81,209,49,177,113,241,9,137,73,201,41,169,105,233,25,153,89,217,57,185,121,249,5,133,69,197,37,165,101,229,21,149,85,213,53,181,117,245,13,141,77,205,45,173,109,237,29,157,93,221,61,189,125,253,3,131,67,195,35,163,99,227,19,147,83,211,51,179,115,243,11,139,75,203,43,171,107,235,27,155,91,219,59,187,123,251,7,135,71,199,39,167,103,231,23,151,87,215,55,183,119,247,15,143,79,207,47,175,111,239,31,159,95,223,63,191,127,255};
    return uint16_t(revbyte[x >> 8]) | uint16_t(revbyte[x & 0xFF]) << 8;
}

// returns the next number after r that does not overlap the mask or have three 1's in a row
row next(row r, uint32_t m) {
    m |= r >> 1 & r >> 2;
    uint32_t x = (r | m) + 1;
    uint32_t carry = x & -x;
    return (r | carry) & -carry;
}

template<typename T, typename U> void maxequals(T& m, U v) {
    if (v > m)
        m = v;
}

struct tictac {
    const int n;
    vector<row> rows;
    size_t nonpal, nrows_c;
    vector<int> irow;
    vector<row> revrows;

    tictac(int n) : n(n) { }

    row reverse(row r) {
        return rev16(r) >> (16 - n);
    }

    vector<int> sols_1row() {
        vector<int> v(1 << n);
        for (uint32_t m = 0; !(m >> n); m++) {
            auto m2 = m;
            int n0 = 0;
            int score = 0;
            for (int i = n; i--; m2 >>= 1) {
                if (m2 & 1) {
                    n0 = 0;
                } else {
                    if (++n0 % 3)
                        score++;
                }
            }
            v[m] = score;
        }
        return v;
    }

    void gen_rows() {
        vector<row> pals;
        for (row r = 0; !(r >> n); r = next(r, 0)) {
            row rrev = reverse(r);
            if (r < rrev) {
                rows.push_back(r);
            } else if (r == rrev) {
                pals.push_back(r);
            }
        }
        nonpal = rows.size();
        for (row r : pals) {
            rows.push_back(r);
        }
        nrows_c = rows.size();
        for (int i = 0; i < nonpal; i++) {
            rows.push_back(reverse(rows[i]));
        }
        irow.resize(1 << n);
        for (int i = 0; i < rows.size(); i++) {
            irow[rows[i]] = i;
        }
        revrows.resize(1 << n);
        for (row r = 0; !(r >> n); r++) {
            revrows[r] = reverse(r);
        }
    }

    // find banned locations for 1's given 2 above rows
    uint32_t mask(row a, row b) {
        return ((a & b) | (a >> 1 & b) >> 1 | (a << 1 & b) << 1) /*& ((1 << n) - 1)*/;
    }

    int calc() {
        if (n < 3) {
            return n * n;
        }
        gen_rows();
        int tdim = n < 5 ? n : (n + 3) / 2;
        size_t nrows = rows.size();
        xcount* t = new xcount[2 * nrows * nrows_c]{};
#define tb(nr, i, j) t[nrows * (nrows_c * ((nr) & 1) + (i)) + (j)]

        // find optimal solutions given 2 rows for n x k grids where 3 <= k <= ceil(n/2) + 1

        {
            auto s1 = sols_1row();
            for (int i = 0; i < nrows_c; i++) {
                row a = rows[i];
                for (int j = 0; j < nrows; j++) {
                    row b = rows[j];
                    uint32_t m = mask(b, a) & ~(1 << n);
                    tb(3, i, j) = s1[m] + popcount32(a << 16 | b);
                }
            }
        }
        for (int r = 4; r <= tdim; r++) {
            for (int i = 0; i < nrows_c; i++) {
                row a = rows[i];
                for (int j = 0; j < nrows; j++) {
                    row b = rows[j];
                    bool rev = j >= nrows_c;
                    auto cj = rev ? j - nrows_c : j;
                    uint32_t m = mask(a, b);
                    for (row c = 0; !(c >> n); c = next(c, m)) {
                        row cc = rev ? revrows[c] : c;
                        int count = tb(r - 1, i, j) + popcount32(c);
                        maxequals(tb(r, cj, irow[cc]), count);
                    }
                }
            }
        }
        int ans = 0;
        if (tdim == n) { // small sizes
            for (int i = 0; i < nrows_c; i++) {
                for (int j = 0; j < nrows; j++) {
                    maxequals(ans, tb(n, i, j));
                }
            }
        } else {
            int tdim2 = n + 2 - tdim;
            // get final answer by joining two halves' solutions down the middle
            for (int i = 0; i < nrows_c; i++) {
                int apc = popcount32(rows[i]);
                for (int j = 0; j < nrows; j++) {
                    row b = rows[j];
                    int top = tb(tdim2, i, j);
                    int bottom = j < nrows_c ? tb(tdim, j, i) : tb(tdim, j - nrows_c, i < nonpal ? i + nrows_c : i);
                    maxequals(ans, top + bottom - apc - popcount32(b));
                }
            }
        }
        delete[] t;
        return ans;
    }
};


int main(int argc, char** argv) {
    int n;
    if (argc < 2 || (n = std::stoi(argv[1])) < 0 || n > 16) {
        return 1;
    }
    std::cout << tictac{ n }.calc() << '\n';
    return 0;
}

7

Java, 14m 31s

Questo è essenzialmente il programma che ho pubblicato su OEIS dopo averlo usato per estendere la sequenza, quindi è un buon riferimento per le altre persone da battere. L'ho modificato per prendere la dimensione della scheda come primo argomento della riga di comando.

public class A181018 {
    public static void main(String[] args) {
        int n = Integer.parseInt(args[0]);
        System.out.println(calc(n));
    }

    private static int calc(int n) {
        if (n < 0) throw new IllegalArgumentException("n");
        if (n < 3) return n * n;

        // Dynamic programming approach: given two rows, we can enumerate the possible third row.
        // sc[i + rows.length * j] is the greatest score achievable with a board ending in rows[i], rows[j].
        int[] rows = buildRows(n);
        byte[] sc = new byte[rows.length * rows.length];
        for (int j = 0, k = 0; j < rows.length; j++) {
            int qsc = Integer.bitCount(rows[j]);
            for (int i = 0; i < rows.length; i++) sc[k++] = (byte)(qsc + Integer.bitCount(rows[i]));
        }

        int max = 0;
        for (int h = 2; h < n; h++) {
            byte[] nsc = new byte[rows.length * rows.length];
            for (int i = 0; i < rows.length; i++) {
                int p = rows[i];
                for (int j = 0; j < rows.length; j++) {
                    int q = rows[j];
                    // The rows which follow p,q cannot intersect with a certain mask.
                    int mask = (p & q) | ((p << 2) & (q << 1)) | ((p >> 2) & (q >> 1));
                    for (int k = 0; k < rows.length; k++) {
                        int r = rows[k];
                        if ((r & mask) != 0) continue;

                        int pqrsc = (sc[i + rows.length * j] & 0xff) + Integer.bitCount(r);
                        int off = j + rows.length * k;
                        if (pqrsc > nsc[off]) nsc[off] = (byte)pqrsc;
                        if (pqrsc > max) max = pqrsc;
                    }
                }
            }

            sc = nsc;
        }

        return max;
    }

    private static int[] buildRows(int n) {
        // Array length is a tribonacci number.
        int c = 1;
        for (int a = 0, b = 1, i = 0; i < n; i++) c = a + (a = b) + (b = c);

        int[] rows = new int[c];
        int i = 0, j = 1, val;
        while ((val = rows[i]) < (1 << (n - 1))) {
            if (val > 0) rows[j++] = val * 2;
            if ((val & 3) != 3) rows[j++] = val * 2 + 1;
            i++;
        }

        return rows;
    }
}

Salva in A181018.java; compilare come javac A181018.java; corri come java A181018 13. Sul mio computer sono necessari circa 20 minuti per eseguire questo input. Probabilmente varrebbe la pena parallelizzarlo.


Da quanto tempo lo gestisci? Sto ancora aggiungendo termini all'OEIS, vedo.
mbomb007,

1
@ mbomb007, non lo sto ancora eseguendo. Come puoi vedere dalle date dell'OEIS, ci sono voluti diversi giorni per correre n=16; Ho estrapolato che ci sarebbe voluto circa un mese n=17, quindi non ho provato a eseguirlo per quello. Anche l'utilizzo della memoria stava diventando un grosso fastidio. (PS Attualmente sto usando 2 dei miei 4 core per una sfida di programmazione non PPCG: azspcs.com/Contest/Tetrahedra/Standings )
Peter Taylor
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.