Games::Sudoku::CPSearch - Solve Sudoku problems quickly.


Games-Sudoku-CPSearch documentation Contained in the Games-Sudoku-CPSearch distribution.

Index


Code Index:

NAME

Top

Games::Sudoku::CPSearch - Solve Sudoku problems quickly.

VERSION

Top

Version 1.00

SYNOPSIS

Top

    use Games::Sudoku::CPSearch;

    my $puzzle = <<PUZZLE;
    4.....8.5
    .3.......
    ...7.....
    .2.....6.
    ....8.4..
    ....1....
    ...6.3.7.
    5..2.....
    1.4......
    PUZZLE

    open FH, ">example.txt";
    print FH $puzzle;
    close FH;

    my $sudoku = Games::Sudoku::CPSearch->new("example.txt");
    print $sudoku->solve(), "\n";

DESCRIPTION

Top

This module solves a Sudoku puzzle using the same constraint propagation technique/algorithm explained on Peter Norvig's website (http://norvig.com/sudoku.html), and implemented there in Python.

METHODS

Top

$o = Games::Sudoku::CPSearch->new()

Initializes the sudoku solving framework.

$o->solve()

Solves the puzzle. Returns the solution as a flat 81 character string.

$o->set_puzzle($puzzle)

Sets the puzzle to be solved. The only parameter is the 81 character string representing the puzzle. The only characters allowed are [0-9\.\-]. Sets the puzzle to be solved. You can then reuse the object:

    my $o = Games::Sudoku::CPSearch->set_puzzle($puzzle);
    print $o->solve(), "\n";
    $o->set_puzzle($another_puzzle);
    print $o->solve(), "\n";

$o->solution()

Returns the solution string, or the empty string if there is no solution.

INTERNAL METHODS

Top

These methods are exposed but are not intended to be used.

$o->_fullgrid()

Returns a hash with squares as keys and "123456789" as each value.

$o->_puzzle()

Returns the object's puzzle as an 81 character string.

$o->_unitlist($square)

Returns an list of sudoku "units": rows, columns, boxes for a given square.

$o->_propagate()

Perform the constraint propagation on the Sudoku grid.

$o->_eliminate($grid, $square, $digit)

Eliminate digit from the square in the grid.

$o->_assign($grid, $square, $digit)

Assign digit to square in grid. Mutually recursive with eliminate().

$o->_rows()

Returns array of row values: A-I

$o->_cols()

Returns array of column values: 1-9

$o->_squares()

Return list of all the squares in a Sudoku grid: A1, A2, ..., A9, B1, ..., I1, ..., I9

$o->_units($square)

Return list of all the units for a given square.

$o->_peers($square)

Return list of all the peers for a given square.

Perform search for a given grid after constraint propagation.

$o->_cross()

Return "cross product" of 2 arrays.

$o->_verify($solution)

Returns undef if the sudoku solution is not valid. Returns 1 if it is.

AUTHOR

Top

Martin-Louis Bright, <mlbright at gmail.com>

BUGS

Top

Please report any bugs or feature requests to bug-games-sudoku-cpsearch at rt.cpan.org, or through the web interface at http://rt.cpan.org/NoAuth/ReportBug.html?Queue=Games-Sudoku-CPSearch. I will be notified, and then you'll automatically be notified of progress on your bug as I make changes.

SUPPORT

Top

You can find documentation for this module with the perldoc command.

    perldoc Games::Sudoku::CPSearch




You can also look for information at:

* RT: CPAN's request tracker

http://rt.cpan.org/NoAuth/Bugs.html?Dist=Games-Sudoku-CPSearch

* AnnoCPAN: Annotated CPAN documentation

http://annocpan.org/dist/Games-Sudoku-CPSearch

* CPAN Ratings

http://cpanratings.perl.org/d/Games-Sudoku-CPSearch

* Search CPAN

http://search.cpan.org/dist/Games-Sudoku-CPSearch

ACKNOWLEDGEMENTS

Top

Peter Norvig, for the explanation/tutorial and python code at http://www.norvig.com/sudoku.html.

COPYRIGHT & LICENSE

Top


Games-Sudoku-CPSearch documentation Contained in the Games-Sudoku-CPSearch distribution.
package Games::Sudoku::CPSearch;

use warnings;
use strict;
use 5.008;
use List::MoreUtils qw(all mesh);

our $VERSION = '1.00';

# Public methods

sub new {
    my ($class, $file) = @_;
    my $puzzle;
    if (defined $file) {
        undef $/;
        open FH, $file or die "could not open puzzle file\n";
        my $puzzle = <FH>;
        close FH;
        $puzzle =~ s/\s+//;
    }

    my $rows = [qw(A B C D E F G H I)];
    my $cols = [qw(1 2 3 4 5 6 7 8 9)];
    my $squares = $class->_cross($rows, $cols);

    my @unitlist = ();
    push @unitlist, $class->_cross($rows, [$_]) for @$cols;
    push @unitlist, $class->_cross([$_], $cols) for @$rows;
    foreach my $r ([qw(A B C)],[qw(D E F)],[qw(G H I)]) {
        foreach my $c ([qw(1 2 3)],[qw(4 5 6)],[qw(7 8 9)]) {
            push @unitlist, $class->_cross($r, $c); 
        }
    }

    my %units;
    foreach my $s (@$squares) {
        $units{$s} = [];
        foreach my $unit (@unitlist) {
            foreach my $s2 (@$unit) {
                if ($s eq $s2) {
                    push @{$units{$s}}, $unit;
                    last;
                }
            }
        }
    }

    my %peers;
    foreach my $s (@$squares) {
        $peers{$s} = [];
        foreach my $u (@{$units{$s}}) {
            foreach my $s2 (@$u) {
                push(@{$peers{$s}}, $s2) if ($s2 ne $s);
            }
        }
    }

    my $self = {
        _unitlist => \@unitlist,
        _rows => $rows,
        _cols => $cols,
        _squares => $squares,
        _units => \%units,
        _peers => \%peers,
        _puzzle => undef,
        _solution => "",
    };

    bless $self, $class;
    $self->set_puzzle($puzzle) if defined $puzzle;
    return $self;
}

sub solution {
    my ($self) = @_;
    return $self->{_solution};
}

sub solve {
    my ($self) = @_;
    my $solution = $self->_search($self->_propagate());
    return undef unless (defined $solution);
    $self->{_solution} = "";
    $self->{_solution} .= $solution->{$_} for ($self->_squares());
    return $self->{_solution};
}

sub set_puzzle {
    my ($self, $puzzle) = @_;
    return undef
        unless ((length($puzzle) == 81) && ($puzzle =~ /^[\d\.\-]+$/)); 
    $puzzle =~ s/0/\./g; # 0 is a digit, which makes things hairy.
    $self->{_puzzle} = $puzzle;
    return $self->{_puzzle};
}

# internal methods

sub _unitlist {
    my ($self) = @_;
    return @{$self->{_unitlist}};
}

sub _rows {
    my ($self) = @_;
    return $self->{_rows};
}

sub _cols {
    my ($self) = @_;
    return $self->{_cols};
}

sub _units {
    my ($self, $s) = @_;
    return @{$self->{_units}{$s}};
}

sub _peers {
    my ($self, $s) = @_;
    return @{$self->{_peers}{$s}};
}

sub _squares {
    my ($self) = @_;
    return @{$self->{_squares}};
}

sub _cross {
    my ($class, $a, $b) = @_;
    my @cross = ();
    foreach my $x (@$a) {
        foreach my $y (@$b) {
            push @cross, "$x$y";
        }
    }
    return \@cross; 
}

sub _fullgrid {
    my ($self) = @_;
    my %grid;
    $grid{$_} = "123456789" for ($self->_squares());
    return \%grid;
}

sub _propagate {
    my ($self) = @_;
    return undef unless defined $self->_puzzle();
    my @d = split(//, $self->_puzzle());
    my @s = $self->_squares();
    my @z = mesh @s, @d;
    my $grid = $self->_fullgrid();
    while (scalar(@z) > 0) {
        my ($s, $d) = splice(@z,0,2);
        next unless ($d =~ /^\d$/);
        return undef unless defined $self->_assign($grid, $s, $d);
    }
    return $grid;
}

sub _assign {
    my ($self, $grid, $s, $d) = @_;
    my @delete = grep {$_ ne $d} split(//, $grid->{$s});
    return $grid if (scalar(@delete) == 0);
    my @results;
    foreach my $del (@delete) { 
        $grid = $self->_eliminate($grid, $s, $del);
        push @results, $grid;
    }
    return $grid if all { defined($_) } @results;
    return undef;
}

sub _eliminate {
    my ($self, $grid, $s, $d) = @_;
    return $grid
        unless ((defined $grid->{$s}) && ($grid->{$s} =~ /$d/));
    $grid->{$s} =~ s/$d//;
    my $len = length($grid->{$s});
    return undef if ($len == 0);
    if ($len == 1) {
        foreach my $peer ($self->_peers($s)) {
            $grid = $self->_eliminate($grid, $peer, $grid->{$s});
            return undef unless defined $grid;
        }
    }

    foreach my $unit ($self->_units($s)) {
        my @dplaces = grep { $grid->{$_} =~ /$d/ } @$unit;
        my $locations = scalar @dplaces;
        return undef if ($locations == 0);
        if ($locations == 1) {
            $grid = $self->_assign($grid, $dplaces[0], $d);
            return undef unless defined $grid;
        }
    }   
    return $grid;
}

sub _search {
    my ($self, $grid) = @_;
    return undef unless defined $grid;
    return $grid if (all {length($grid->{$_}) == 1} $self->_squares());
    # solved!
    my @sorted = sort {length($grid->{$a}) <=> length($grid->{$b})}
        grep {length($grid->{$_}) > 1} $self->_squares();
    my $fewest_digits = shift @sorted;
    my $result = undef;
    foreach my $d (split(//, $grid->{$fewest_digits})) {
        my %grid_copy = %$grid; 
        $result = $self->_search($self->_assign(\%grid_copy, $fewest_digits, $d));
        return $result if defined $result;
    }
    return $result;
}

sub _puzzle {
    my ($self) = @_;
    return $self->{_puzzle};
}

sub _verify {
    my ($self, $puzzle) = @_;
    for (1..9) {
        my $count = () = $puzzle =~ /$_/g;
        return undef unless $count == 9;
    }
    return 1;
}

1; # End of Games::Sudoku::CPSearch