[USACO 6.5.1]All Latin Squares

题解

  搜索剪枝.

  置换圈个数相同及对应的置换圈内元素个数相同即视为相同方案.

代码

/*
TASK:latin
LANG:C++
*/
#include <cstdio>
#include <cstring>
#include <iostream>

using namespace std;

int n, s[10][10], fact[] = {1, 1, 2, 6, 24, 120, 720, 5040};
bool row[10][10], col[10][10];
long long f[10][100];

long long dfs(int x, int y)
{
    if (x == n) return 1;
    int totn = 0, totlen = 1;
    if (x == 3 && y == 2)
    {
        int v[10];
        memset(v, true, sizeof(v));
        for (int i = 1; i <= n; ++i)
            if (v[i])
            {
                int len = 0, st = i;
                totn++;
                for (;;)
                {
                    len++;
                    v[st] = false;
                    st = s[2][st];
                    if (!v[st]) break;
                }
                totlen *= len;
            }
        if (f[totn][totlen] != -1) return f[totn][totlen];
    }
    long long ans = 0;
    for (int i = 1; i <= n; ++i)
        if (row[x][i] && col[y][i])
        {
            row[x][i] = col[y][i] = false;
            s[x][y] = i;
            if (y == n) ans += dfs(x + 1, 2);
            else ans += dfs(x, y + 1);
            row[x][i] = col[y][i] = true;
        }
    if (x == 3 && y == 2) return f[totn][totlen] = ans;
    else return ans;
}

int main()
{
    freopen("latin.in", "r", stdin);
    freopen("latin.out", "w", stdout);
    scanf("%d", &n);
    memset(row, true, sizeof(row));
    memset(col, true, sizeof(col));
    for (int i = 1; i <= n; ++i)
    {
        row[i][i] = false;
        col[i][i] = false;
        s[1][i] = s[i][1] = i;
    }
    memset(f, -1, sizeof(f));
    cout << dfs(2, 2) * fact[n-1] << endl;
    return 0;
}

 

posted @ 2016-10-05 19:59  albertxwz  阅读(313)  评论(0编辑  收藏  举报