Java (n = 8)
import java.util.*;
import java.util.concurrent.*;
public class HankelCombinatorics {
public static final int NUM_THREADS = 8;
private static final int[] FACT = new int[13];
static {
FACT[0] = 1;
for (int i = 1; i < FACT.length; i++) FACT[i] = i * FACT[i-1];
}
public static void main(String[] args) {
long prevElapsed = 0, start = System.nanoTime();
for (int i = 1; i < 12; i++) {
long count = count(i), elapsed = System.nanoTime() - start;
System.out.format("%d in %dms, total elapsed %dms\n", count, (elapsed - prevElapsed) / 1000000, elapsed / 1000000);
prevElapsed = elapsed;
}
}
@SuppressWarnings("unchecked")
private static long count(int n) {
int[][] perms = new int[FACT[n]][];
genPermsInner(0, 0, new int[n], perms, 0);
// We partition by canonical representation of the row sum multiset, discarding any with a density > 50%.
Map<CanonicalMatrix, Map<CanonicalMatrix, Integer>> part = new HashMap<CanonicalMatrix, Map<CanonicalMatrix, Integer>>();
for (int m = 0; m < 1 << (2*n-1); m++) {
int density = 0;
int[] key = new int[n];
for (int i = 0; i < n; i++) {
key[i] = Integer.bitCount((m >> i) & ((1 << n) - 1));
density += key[i];
}
if (2 * density <= n * n) {
CanonicalMatrix _key = new CanonicalMatrix(key);
Map<CanonicalMatrix, Integer> map = part.get(_key);
if (map == null) part.put(_key, map = new HashMap<CanonicalMatrix, Integer>());
map.put(new CanonicalMatrix(m, perms[0]), m);
}
}
List<Job> jobs = new ArrayList<Job>();
ExecutorService pool = Executors.newFixedThreadPool(NUM_THREADS);
for (Map.Entry<CanonicalMatrix, Map<CanonicalMatrix, Integer>> e : part.entrySet()) {
Job job = new Job(n, perms, e.getKey().sum() << 1 == n * n ? 0 : 1, e.getValue());
jobs.add(job);
pool.execute(job);
}
pool.shutdown();
try {
pool.awaitTermination(1, TimeUnit.DAYS); // i.e. until it's finished - inaccurate results are useless
}
catch (InterruptedException ie) {
throw new IllegalStateException(ie);
}
long total = 0;
for (Job job : jobs) total += job.subtotal;
return total;
}
private static int genPermsInner(int idx, int usedMask, int[] a, int[][] perms, int off) {
if (idx == a.length) perms[off++] = a.clone();
else for (int i = 0; i < a.length; i++) {
int m = 1 << (a[idx] = i);
if ((usedMask & m) == 0) off = genPermsInner(idx+1, usedMask | m, a, perms, off);
}
return off;
}
static class Job implements Runnable {
private volatile long subtotal = 0;
private final int n;
private final int[][] perms;
private final int shift;
private final Map<CanonicalMatrix, Integer> unseen;
public Job(int n, int[][] perms, int shift, Map<CanonicalMatrix, Integer> unseen) {
this.n = n;
this.perms = perms;
this.shift = shift;
this.unseen = unseen;
}
public void run() {
long result = 0;
int[][] perms = this.perms;
Map<CanonicalMatrix, Integer> unseen = this.unseen;
while (!unseen.isEmpty()) {
int m = unseen.values().iterator().next();
Set<CanonicalMatrix> equiv = new HashSet<CanonicalMatrix>();
for (int[] perm : perms) {
CanonicalMatrix canonical = new CanonicalMatrix(m, perm);
if (equiv.add(canonical)) {
result += canonical.weight() << shift;
unseen.remove(canonical);
}
}
}
subtotal = result;
}
}
static class CanonicalMatrix {
private final int[] a;
private final int hash;
public CanonicalMatrix(int m, int[] r) {
this(permuteRows(m, r));
}
public CanonicalMatrix(int[] a) {
this.a = a;
Arrays.sort(a);
int h = 0;
for (int i : a) h = h * 37 + i;
hash = h;
}
private static int[] permuteRows(int m, int[] perm) {
int[] cols = new int[perm.length];
for (int i = 0; i < perm.length; i++) {
for (int j = 0; j < cols.length; j++) cols[j] |= ((m >> (perm[i] + j)) & 1L) << i;
}
return cols;
}
public int sum() {
int sum = 0;
for (int i : a) sum += i;
return sum;
}
public int weight() {
int prev = -1, count = 0, weight = FACT[a.length];
for (int col : a) {
if (col == prev) weight /= ++count;
else {
prev = col;
count = 1;
}
}
return weight;
}
@Override public boolean equals(Object obj) {
// Deliberately unsuitable for general-purpose use, but helps catch bugs faster.
CanonicalMatrix that = (CanonicalMatrix)obj;
for (int i = 0; i < a.length; i++) {
if (a[i] != that.a[i]) return false;
}
return true;
}
@Override public int hashCode() {
return hash;
}
}
}
Speichern als HankelCombinatorics.java
, Kompilieren als javac HankelCombinatorics.java
, Ausführen als java -Xmx2G HankelCombinatorics
.
Mit NUM_THREADS = 4
auf meiner Quad-Core - Maschine wird es 20420819767436
für n=8
in 50 bis 55 Sekunden verstrichen, mit einem fairen Betrag von Variabilität zwischen den Läufen; Ich gehe davon aus, dass es auf Ihrem Octa-Core-Computer problemlos funktioniert, aber es dauert mindestens eine Stunde, bis es verfügbar istn=9
.
Wie es funktioniert
Vorausgesetzt n
, es gibt 2^(2n-1)
binäre n
x n
Hankel-Matrizen. Die Zeilen können auf verschiedene n!
Arten und die Spalten auf verschiedene n!
Arten permutiert werden. Alles was wir tun müssen, ist Doppelzählungen zu vermeiden ...
Wenn Sie die Summe jeder Zeile berechnen, ändert weder das Permutieren der Zeilen noch das Permutieren der Spalten die Summenmehrfachmenge. Z.B
0 1 1 0 1
1 1 0 1 0
1 0 1 0 0
0 1 0 0 1
1 0 0 1 0
hat Zeilensummen-Multiset {3, 3, 2, 2, 2}
und alle daraus abgeleiteten Hankelable-Matrizen. Dies bedeutet, dass wir die Hankel-Matrizen nach diesen Zeilensummen-Multisätzen gruppieren und dann jede Gruppe unabhängig behandeln können, wobei mehrere Prozessorkerne ausgenutzt werden.
Es gibt auch eine ausnutzbare Symmetrie: Die Matrizen mit mehr Nullen als Einsen stehen im Widerspruch zu den Matrizen mit mehr Einsen als Nullen.
Doppelzählung auftritt , wenn Hankel - Matrix M_1
mit Zeilen Permutation r_1
und Spaltenpermutation c_1
einstimmt Hankel - Matrix M_2
mit Zeilen Permutation r_2
und Spaltenpermutation c_2
(mit bis zu zwei , aber nicht alle drei von M_1 = M_2
, r_1 = r_2
, c_1 = c_2
). Die Zeilen- und Spaltenpermutationen sind unabhängig. Wenn Sie also die Zeilenpermutation r_1
auf M_1
und die Zeilenpermutation r_2
auf anwenden M_2
, müssen die Spalten als Multisets gleich sein. Daher berechne ich für jede Gruppe alle Spalten-Multisets, die durch Anwenden einer Zeilenpermutation auf eine Matrix in der Gruppe erhalten werden. Der einfache Weg, eine kanonische Darstellung der Multisets zu erhalten, besteht darin, die Spalten zu sortieren. Dies ist auch im nächsten Schritt nützlich.
Nachdem wir die verschiedenen Spalten-Multisets erhalten haben, müssen wir herausfinden, wie viele der n!
Permutationen von jeder einzigartig sind. Zu diesem Zeitpunkt kann eine Doppelzählung nur stattfinden, wenn ein gegebenes Spalten-Multiset doppelte Spalten enthält. Wir müssen lediglich die Anzahl der Vorkommen jeder einzelnen Spalte im Multiset zählen und dann den entsprechenden Multinomialkoeffizienten berechnen. Da die Spalten sortiert sind, ist es einfach, sie zu zählen.
Schließlich addieren wir sie alle.
Die asymptotische Komplexität ist nicht einfach mit voller Genauigkeit zu berechnen, da wir einige Annahmen über die Mengen treffen müssen. Wir bewerten die Reihenfolge der 2^(2n-2) n!
Spalten-Multisets, wobei wir uns jeweils n^2 ln n
Zeit nehmen (einschließlich der Sortierung). Wenn die Gruppierung nicht mehr als einen ln n
Faktor ausmacht, haben wir zeitliche Komplexität Theta(4^n n! n^2 ln n)
. Aber da die Exponentialfaktoren die Polynomfaktoren vollständig dominieren, ist dies der Fall Theta(4^n n!) = Theta((4n/e)^n)
.