Sudoku solvers are old hat, of course, but this one is a bit different: it's a regexp. I suppose someone may have done this before, but if so, I haven't seen it. It's grossly inefficient, of course, solving the puzzle entirely by guesswork and backtracking, but that's not really the point. (It could be made quite a bit faster by hardcoding the givens in the regexp, but that would make the regexp non-generic.)

#!/usr/bin/perl -w
use strict;

my $sudoku = ("^.*" .
              (join ".*\n.*", "(.)",
               map {
                   my $n = $_;
                   ("(?!" .
                    (join "|",
                     (map "\\" . ($_+1),
                      (grep $_ < $n &&
                       (int($_ % 9) == int($n % 9) ||
                        int($_ / 9) == int($n / 9) ||
                        int($_ / 27) == int($n / 27) &&
                        int($_ / 3 % 3) == int($n / 3 % 3)),
                       0..80))) .
                    ")(.)");
               } 1..80) .
              ".*\$"
             );

while (<>) {
    chomp;
    length != 81 || tr/1-9_//c and (print("Bad input.\n"), next);
    s/(.)/$1 eq "_" ? "123456789\n" : "$1\n"/eg;
    print((join("", /$sudoku/) || "No solution."), "\n");
}


__END__

Input, in case it's not obvious from the code, is an 81-character string listing the numbers in left-right, top-down order, with "_" for blanks. The while loop mangles it into the proper format for the regexp (81 lines, each listing the allowed numbers for the corresponding cell) and outputs the solution in the same format as the input.

--
Ilmari Karonen

Reply via email to