Source Code
#include <fstream>
#include <iostream>
#include <string>
#include <sstream>
#include <iomanip>
#include <cmath>
using namespace std;
void sudokuSolver(int**, int&); //Function help solving sudoku
void horizontal(int**); //eliminate posibilities in rows
void vertical(int**); //eliminate posibilities in columns
void threeByThree(int**); //eliminate posibilities in 3 by 3 section
void readData(int**, ifstream&); //read data from "sudoku.txt"
void eliminationRule(int**); //if a single number can only be in that row, column
// and 3 by 3, then that cell must be that number
void check(int**); //if one posibility, that cell is that number
string output(int**); //outputing table
bool done(int**); //checks if it's done
void error(); //error message
int main()
{
int** table; //double pointer!!! fun fun....
int counter = 0;
int choice;
//allocation of double pointer
table = new int*[9];
for(int count = 0; count < 9; count++)
{
table[count] = new int[9];
}
//end allocation
cout << " ************************************************n"
<< " * Sudoku Solver v2.0 *n"
<< " * *n"
<< " * Author : Alka G Vulcan *n"
<< " * *n"
<< " * Contact : yake_2002 (at) Hotmail (dot) com *n"
<< " * *n"
<< " * Details : Reads data from "sudoku.txt" *n"
<< " * and solve sudoku. have 0 *n"
<< " * for unknow cell and seperate each *n"
<< " * cell by a space or any single char. *n"
<< " * *n"
<< " * Comment : let me know if there is any way *n"
<< " * to improve this code or any question *n"
<< " * about anything.... *n"
<< " * *n"
<< " * 0 indicates empty spot in sudoku *n"
<< " * *n"
<< " * Changes from previous : *n"
<< " * Implemented multi sudoku data capacity*n"
<< " * in "sudoku.txt" *n"
<< " * fixed sevral minor bugs where 1 wasn't*n"
<< " * correctly captured as possibile*n"
<< " * number *n"
<< " * changed data file to "sudoku.txt" *n"
<< " * *n"
<< " ************************************************nn";
cout << "Would you like to see sample of "sudoku.txt"?n1 for yes, 0 for non";
cin >> choice;
if(choice)
{
cout << "1. space between each numbers n"
<< "2. unknown number represented by 0 n"
<< "3. each 9 by 9 grid ceperated by a empty linenn"
<< "A sample table with 2 puzzles nn"
<< "0 0 0 0 4 0 0 5 2n"
<< "0 6 9 7 0 0 0 0 8n"
<< "0 3 0 8 0 0 0 0 0n"
<< "0 0 0 0 0 0 1 0 3n"
<< "0 7 5 0 3 0 2 4 0n"
<< "3 0 6 0 0 0 0 0 0n"
<< "0 0 0 0 0 7 0 8 0n"
<< "9 0 0 0 0 4 3 1 0n"
<< "7 8 0 0 9 0 0 0 0nn"
<< "0 0 0 1 9 0 3 0 0n"
<< "0 0 0 0 6 7 5 0 8n"
<< "7 9 5 0 0 8 0 0 0n"
<< "0 0 0 6 0 0 0 1 7n"
<< "0 7 1 0 2 0 4 3 0n"
<< "9 4 0 0 0 1 0 0 0n"
<< "0 0 0 8 0 0 7 5 3n"
<< "2 0 7 9 5 0 0 0 0n"
<< "0 0 8 0 7 3 0 0 0n";
}
cout << endl << endl
<< "How many sudoku tables in "sudoku.txt"?n";
cin >> choice;
cout << endl;
ifstream inData;
inData.open("sudoku.txt"); //open "sudoku.dat"
if(inData.fail())
error();
else
{
if(choice > 0)
{
cout << "Results......nn";
readData(table, inData);
sudokuSolver(table, counter);
cout << "1st table was done by " << counter << "th cycle.n"
<< output(table);
for(int n = 0; n < choice - 1; n++)
{
counter = 0;
inData.get();
readData(table, inData);
sudokuSolver(table, counter);
if( n == 0)
{
cout << "2nd table was done by " << counter << "th cycle.n"
<< output(table);
}
else
{
cout << n + 2 << "th table was done by " << counter << "th cycle.n"
<< output(table);
}
}
}
}
inData.close();
inData.clear();
cin.get();
for(int x = 0; x < 9; x++) //freeing allocated memories
{
delete [] table[x];
}
delete [] table;
cin.get();
}
void sudokuSolver(int** table, int&counter)
{
counter++;
for(int x = 0; x < 9; x++)
{
for (int y = 0; y < 9; y++)
{
if(table[x][y] == 0 || table[x][y] > 9) //if a cell is 0 or bigger then 9
{
table[x][y] = 987654321; //putting entire 9 possibilities in that cell
}
}
}
//calling actual sudoku solving functions
vertical(table);
horizontal(table);
threeByThree(table);
check(table);
eliminationRule(table);
//end of sudoku solving functions
//cout << output(table) << endl;
if(!done(table))
sudokuSolver(table, counter);
}
void vertical(int** table)
{
for(int x = 0; x < 9; x++)
{
for(int y = 0; y < 9; y++)
{
if(table[x][y] > 9) //if number is bigger then 9
{
for(int newX = 0; newX < 9; newX++)
{
int number = 0;
if(newX != x) //if it's not same cell
{
if(table[newX][y] == 9)
number = 900000000;
if(table[newX][y] == 8)
number = 80000000;
if(table[newX][y] == 7)
number = 7000000;
if(table[newX][y] == 6)
number = 600000;
if(table[newX][y] == 5)
number = 50000;
if(table[newX][y] == 4)
number = 4000;
if(table[newX][y] == 3)
number = 300;
if(table[newX][y] == 2)
number = 20;
if(table[newX][y] == 1)
number = 1;
}
table[x][y] -= number; //eliminate such possibility
}
}
}
}
}
void horizontal(int** table)
{
for(int x = 0; x < 9; x++)
{
for(int y = 0; y < 9; y++)
{
if(table[x][y] > 9)
{
for(int newY = 0; newY < 9; newY++)
{
int number = 0;
if(newY != y)
{
if(table[x][newY] == 9)
number = 900000000;
if(table[x][newY] == 8)
number = 80000000;
if(table[x][newY] == 7)
number = 7000000;
if(table[x][newY] == 6)
number = 600000;
if(table[x][newY] == 5)
number = 50000;
if(table[x][newY] == 4)
number = 4000;
if(table[x][newY] == 3)
number = 300;
if(table[x][newY] == 2)
number = 20;
if(table[x][newY] == 1)
number = 1;
}
//if number was eliminated previously by horizontal
for(int count = 0; count < 9; count++)
{
if(table[x][newY] == table[count][y])
number = 0;
}
//end
table[x][y] -= number;
}
}
}
}
}
void threeByThree(int** table)
{
for(int x = 0; x < 9; x++)
{
for(int y = 0; y < 9; y++)
{
if(table[x][y] > 9)
{
int newX, newY, checkY;
//setting 3x3 grid cells
if(x<9)
newX=6;
if(x<6)
newX=3;
if(x<3)
newX=0;
if(y<9)
newY=6;
if(y<6)
newY=3;
if(y<3)
newY=0;
//end of setting
checkY = newY; //to know when to move to next row
for(int count = 0; count < 9; count++, newY++)
{
int number = 0;
if(newY != y && newX !=x) //if not same cell
{
if(table[newX][newY] == 9)
number = 900000000;
if(table[newX][newY] == 8)
number = 80000000;
if(table[newX][newY] == 7)
number = 7000000;
if(table[newX][newY] == 6)
number = 600000;
if(table[newX][newY] == 5)
number = 50000;
if(table[newX][newY] == 4)
number = 4000;
if(table[newX][newY] == 3)
number = 300;
if(table[newX][newY] == 2)
number = 20;
if(table[newX][newY] == 1)
number = 1;
}
//if number was eliminated previously by horizontal
for(int count = 0; count < 9; count++)
{
if(count != newY)
{
if(table[newX][newY] == table[x][count])
number = 0;
}
}
//end
//if number was eliminated previously by vertical
for(int count = 0; count < 9; count++)
{
if(count != newX )
{
if(table[newX][newY] == table[count][y])
number = 0;
}
}
//end
table[x][y] -= number;
if(newY == checkY + 2) //if end column of 3x3 grid
{
newX++; //increment row
newY = checkY -1; //set column to -1 since it will get incrment soon
}
}
}
}
}
}
void check(int** table)
{
for(int x = 0; x < 9; x++)
{
for(int y = 0; y < 9; y++)
{
//if there is one posibility, then such number is that cell
if(table[x][y] == 900000000)
table[x][y] = 9;
if(table[x][y] == 80000000)
table[x][y] = 8;
if(table[x][y] == 7000000)
table[x][y] = 7;
if(table[x][y] == 600000)
table[x][y] = 6;
if(table[x][y] == 50000)
table[x][y] = 5;
if(table[x][y] == 4000)
table[x][y] = 4;
if(table[x][y] == 300)
table[x][y] = 3;
if(table[x][y] == 20)
table[x][y] = 2;
if(table[x][y] == 1)
table[x][y] = 1;
}
}
}
bool done(int** table)
{
bool flag = true;
for(int x = 0; x < 9; x ++)
{
for (int y = 0; y < 9; y++)
{
if(table[x][y] > 10 || table[x][y] == 0) //if cell is bigger then 10 or equal to 0
{
flag = false;
}
}
}
return flag;
}
string output(int** table)
{
ostringstream out; //using sstream, since it's more convineant
for(int x = 0; x < 9; x ++)
{
for (int y = 0; y < 9; y++)
{
out << setw(7) << table[x][y] << " ";
}
out << endl;
}
out << endl;
return out.str(); //returning sstream
}
void eliminationRule(int** table)
{
int test[9][9][9]; //first two digit for cell, last digit for possibility
//so if test[2][3][5] == 1, then 5 can be at row 2, column 3
int **temp;
temp = new int*[9];
for(int count = 0; count < 9; count++)
{
temp[count] = new int[9];
}
for(int x = 0; x < 9; x++)
{
for(int y = 0; y < 9; y++)
{
temp[x][y] = table[x][y];
}
}
for(int n = 9; n > 0; n--)
{
for(int x = 0; x < 9; x ++)
{
for(int y = 0; y < 9; y++)
{
test[x][y][n] = 0;
if(table[x][y] > 9)
{
int number=999999999;
//n * 10^(n-1)
if(n > 1)
number = n * (pow(10,static_cast<double>(n-1)));
else
number = 1;
//end
//I think because double to int conversion is imperfect
if((number % 2) != 0 && number != 1)
number+=1;
//end
if(temp[x][y] - number >= 0 || table[x][y] == n)
{
temp[x][y] -= number;
test[x][y][n] = 1; //number n is the possibility at such cell
}
}
}
}
}
for(int x = 0; x < 9; x++)
{
for (int y = 0; y < 9; y++)
{
for (int n = 9; n > 0; n--)
{
if(test[x][y][n] && table[x][y] > 9)
{
int newX, newY, checkY;
bool solo = true;
if(x<3)
newX=0;
else if(x<6)
newX=3;
else if(x<9)
newX=6;
if(y<3)
newY=0;
else if(y<6)
newY=3;
else if(y<9)
newY=6;
checkY = newY;
for(int m = 0; m < 9; m++) //check for vertical
{
if(test[m][y][n] && m != x)
solo = false;
}
for(int m = 0; m < 9; m++) //check for horizontal
{
if(test[x][m][n] && m != y)
solo = false;
}
for(int m = 0; m < 9; m++) //check for 3 by 3
{
if(test[newX][newY][n] && newX != x && newY != y)
{
solo = false;
}
if(newY == checkY + 2)
{
newX++;
newY = checkY - 1;
}
}
//if only that number can be at that cell
if(solo)
{
table[x][y] = n;
}
//end if
}
}
}
}
for(int x = 0; x < 9; x++)
{
for (int y = 0; y < 9; y++)
{
for (int n = 0; n < 9; n++)
{
if(test[x][y][n] && table[x][y] > 9)//if number is possibility in a cell and
//that cell is unknown number
{
bool solo = true;
for(int m = 0; m < 9; m++) //check for vertical
{
//if there is possible number present in that column
//or number it self is present in that column
if(test[m][y][n] && m != x || table[m][y] == n)
solo = false;
}
//if only that number can be at that cell
if(solo)
{
table[x][y] = n;
}
//end if
}
}
}
}
for(int x = 0; x < 9; x++)
{
for (int y = 0; y < 9; y++)
{
for (int n = 1; n < 10; n++)
{
if(test[x][y][n] && table[x][y] > 9) //if number is possibility in a cell and
//that cell is unknown number
{
bool solo = true;
for(int m = 0; m < 9; m++) //check for horizontal
{
//if there is possible number present in that row
//or number it self is present in that row
if(test[x][m][n] && m != y || table[x][m] == n)
solo = false;
}
//if only that number can be at that cell
if(solo)
{
table[x][y] = n;
}
//end if
}
}
}
}
for(int x = 0; x < 9; x ++)
{
delete [] temp[x];
}
delete [] temp;
}
void readData(int** table, ifstream& inData)
{
if(inData.fail()) //if failed to open
error();
int x = 0, y = 0;
for(int z = 0; z < 81; z++) //reading 81 char (9 x 9 sudoku table)
{
table[x][y] = inData.get()-48; //char - 48 is intiger... (look up ASCII table if need assistant)
inData.ignore(); //ignoring a space
if (y == 8) //if it's 9th column
{
x++; //increment row
y = -1; //reset column to -1 since it will get increment soon
}
y++; //increment column
}
}
void error()
{
cout << "ERROR... nn";
}
No comments:
Post a Comment