| 1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
 | #include "SudokuGen.h"
#include <ctime>
#include <cstdlib>
#include <vector>
using namespace std;
SudokuGen::SudokuGen() // 建構子(Constructor)
{
    srand(time(NULL)); //以物件初始化時間設定為亂數種子
    generator(); //以亂數方法產生數獨棋盤
}
void SudokuGen::generator() //亂數法產生新棋盤
{
    int tmp; //置換用暫存變數
    int numtmp[9] = {1,2,3,4,5,6,7,8,9}; //9宮格內的數字
    vector <int> board(81); //棋盤
    bool check = false; //表示棋盤是否成功產生
    for (i=0;i<81;i++){board[i] = 0;} //初始化棋盤(填入0)
    for (i=0;i<9;i++) //先填入左上角9宮格
//    1 2 3
//    4 5 6
//    7 8 9
    {
        board[i/3*9+i%3] = numtmp[i];
    }
    while(!check) //不斷嘗試直到符合規則的棋盤產生
    {
//        填入數字函式,嘗試失敗超過1000次會回傳false值
        check = board_put(board);
    }
    for (i=0;i<9;i++) //將1-9的數字順序打亂
    {
        j = rand() % 9;
        tmp = numtmp[i];
        numtmp[i] = numtmp[j];
        numtmp[j] = tmp;
    }
    for(i=0;i<81;i++) //將棋盤的數字序置換為新產生的數字序
    {
        board[i] = numtmp[board[i]-1];
    }
    for (i=0;i<SIZE;i++) //將結果寫入new_board變數
        for (j=0;j<SIZE;j++)
        {
            new_board[i][j] = board[i*SIZE+j];
        }
}
vector <int > SudokuGen::get_board() //輸出棋盤至vector變數
{
    vector < int > out_board (SIZE*SIZE);
    for (i=0;i<SIZE;i++)
        for (j=0;j<SIZE;j++)
        {
            out_board[i*SIZE+j] = new_board[i][j];
        }
    return out_board;
}
bool SudokuGen::check_rep(vector<int> board,int pos,int n)
{
    int i;
    for (i=0;i<9;i++)
    {
        if(board[pos/9*9+i]==n) //check橫排上數字是否有重複
        {
            return true; //有重複,回傳true
        }
        if(board[i*9+pos%9]==n) //check直排上數字是否有重複
        {
            return true; //有重複,回傳true
        }
    }
    return false;  //無重複,回傳false
}
bool SudokuGen::board_put(vector<int > &board) //填入數字函式
{
    int pos;
    int count1,count2; //計數器
    bool check = true; //check數字是否重複
    for (j=0;j<9;j++) //填入的數字1-9 (j+1的結果)
    {
        count2 = 0; //計算單一數字填入總共甞試錯誤次數
        for (i=1;i<9;i++) //依序將數字填入其餘八個小九宮格(左上角已經產生)
        {
            pos = i/3*27+i%3*3; //每個小九宮格的起始index
            count1 = 0; //計算單一數字填入甞試錯誤次數
            while (check==true)
            {
                k = rand() % 9;
                check = check_rep(board,pos+k/3*9+k%3,j+1);
                ++count1;
                ++count2;
                if (board[pos+k/3*9+k%3]==0 && check==false) //數字沒有重複,成功填入結束while loop
                {
                    board[pos+k/3*9+k%3] = j+1;
                    check = true;
                    break;
                }
                else
                {
                    check = true;
                }
                if (count1 > 100) // 錯誤超過100次,數字重填
                {
                    for (l=0;l<81;l++)
                    {
                        if (board[l]==j+1)
                        {
                            board[l]=0;
                        }
                    }
                    board[j/3*9+j%3]=j+1;
                    i = 0;
                    break;
                }
                if (count2 > 1000) // 錯誤超過1000次,回傳false並結束函式
                {
                    return false;
                }
            }
        }
    }
    return true; //成功填滿棋盤,回傳true
}
 | 
沒有留言:
張貼留言