Recently I was recommended the problem CERC'10 J — Justice for All. I really liked the solution I came up with and it's different from the editorial, so here's a blogpost talking about it.
Problem
You are given a number $$$n$$$ and you want to create a bipartite graph with $$$n$$$ perfect matchings, $$$1 \leq n \leq 10^6$$$.
The number of nodes in the graph shouldn't exceed $$$400$$$.
Main Idea
We want to build a kind of graph whose number of perfect matchings can be expanded.
If the graph has $$$x$$$ perfect matchings, we want to be able to expand it so that the new graph now has either $$$2x$$$ or $$$2x+1$$$.
If it's possible to do this, then we can use the binary representation of $$$n$$$ to create the graph.
Solution
Locking Mechanism
The idea for doubling is simple, we just need to add a graph with two matchings in such a way that it doesn't disturb the configurations that were previously there. Such graph with two matchings can simply be a square:
data:image/s3,"s3://crabby-images/8d72a/8d72a2de96b6f6d5c744eb968ca9cb5da6108431" alt=""
Now for doubling and adding one, we can keep the idea of adding a square, but after adding it we need some kind of locking mechanism that either forces a specific configuration of the whole graph or doesn't disturb the configurations that were already there.
We can see the idea of a locking mechanism on a small graph.
Without the locking edge, it has $$$2$$$ perfect matchings.
data:image/s3,"s3://crabby-images/94c56/94c56b1c821fe74a5b8bd5129edc546edebf6337" alt=""
With the locking edge, it has $$$3$$$ perfect matchings. When the locking edge is selected the configuration of the whole graph is forced, otherwise it's the same as if there was no locking edge.
data:image/s3,"s3://crabby-images/cfff1/cfff1763036a185b19b908a8cc15ca0f129e06d6" alt=""
This idea for one square can be extended to lock a chain of squares:
data:image/s3,"s3://crabby-images/a3c48/a3c485701ad4193cb9d3f372f730493bce730049" alt=""
If the locking edge is selected, the configuration of the whole chain is locked. If it's not selected, each square is independent.
data:image/s3,"s3://crabby-images/67d58/67d58f0161dd6766a51406c960c0c1102846e8a3" alt=""
Extending $$$x \rightarrow 2x+1$$$
Now we can see that these chains can be extended.
data:image/s3,"s3://crabby-images/53231/5323180586881752dd4c8fa24f8165dcda772a10" alt=""
By extending and adding a locking edge, the same previous scenarios can happen:
data:image/s3,"s3://crabby-images/83ba4/83ba4493180b6676c8d9b9f229411957ec1194ae" alt=""
data:image/s3,"s3://crabby-images/cc0b1/cc0b17a7f12f927a002bd7f7e9bdcf89755dc13c" alt=""
So if the previous chain had $$$x$$$ perfect matchings, this extension now has $$$2x+1$$$.
Extending $$$x \rightarrow 2x$$$
By extending and not adding the locking edge, we just don't have the locked configuration:
data:image/s3,"s3://crabby-images/d53f0/d53f0b495cf520023bee98f258194a37f0c67693" alt=""
If the chain had $$$x$$$ perfect matchings, this extension now has $$$2x$$$.
So this is it. We found a procedure that lets us extend a chain in the manner we wanted.
Construction
We start with a simple edge with $$$1$$$ perfect matching:
data:image/s3,"s3://crabby-images/c6cc3/c6cc3644177ba325b081aa7c3ed9f3bdce3cbd35" alt=""
If we have a graph with $$$x$$$ perfect matchings and we want to make it $$$2x$$$, we add a square without a locking edge:
data:image/s3,"s3://crabby-images/deb45/deb4558f1de0c15be94a0b6323c5d0f38e92829c" alt=""
If we have a graph with $$$x$$$ perfect matchings and we want to make it $$$2x+1$$$, we add a square with a locking edge:
data:image/s3,"s3://crabby-images/3f08c/3f08cbbdbcc0309fdfb211f83aa7f098d1e28a9d" alt=""
Analysis
We want the number of nodes to be less or equal to $$$400$$$.
This construction starts with $$$2$$$ nodes and adds $$$4$$$ for each bit in $$$n$$$ below the most significative.
So this solution has at most $$$4\lfloor \log_2(10^6) \rfloor+2 = 78$$$ nodes, which fits very comfortably below the limit.
Note: the official solution follows a similar idea, but adds 6 nodes for increasing by one and for doubling, resulting in $$$214$$$ nodes in the worst case
Code
The code for this is surprisingly short.
Here is an accepted submission (you may need coach mode to access it).
OTZ