-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathn_queens.pl
103 lines (88 loc) · 2.68 KB
/
n_queens.pl
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
:- use_module(library(clpb)).
:- use_module(library(clpz)).
:- use_module(library(lists)).
:- use_module(library(dcgs)).
:- use_module(library(time)).
:- use_module(library(format)).
%?- run.
%@ |queens(0)| = 1 after 0.00s
%@ |queens(1)| = 1 after 0.00s
%@ |queens(2)| = 0 after 0.00s
%@ |queens(3)| = 0 after 0.01s
%@ |queens(4)| = 2 after 0.08s
%@ |queens(5)| = 10 after 0.51s
%@ |queens(6)| = 4 after 2.10s
%@ |queens(7)| = 40 after 10.34s
%@ |queens(8)| = 92 after 48.73s
%@ etc.
%?- n_queens(4, Qs, _, Sat), sat(Sat), append(Qs, Vs), labeling(Vs), maplist(portray_clause, Qs).
%@ [0,0,1,0]
%@ [1,0,0,0]
%@ [0,0,0,1]
%@ [0,1,0,0]
%@ etc.
run :-
length(_, N),
statistics(runtime, [T0,_]),
n_queens(N, _, _, Sat),
sat_count(Sat, C),
statistics(runtime, [T1|_]),
Time is T1 - T0,
format("|queens(~w)| = ~w ~t~25| after ~2fs\n", [N,C,Time/1000]),
false.
n_queens(N, Qs, Ands, *(Ands)) :-
length(Qs, N),
maplist(length_(N), Qs),
transpose(Qs, TQs),
phrase((rows(Qs),rows(TQs),
diagonals(Qs, 1, 1, N)), Ands).
rows([]) --> [].
rows([Row|Rows]) -->
[+Row],
not_same_row(Row),
rows(Rows).
not_same_row([]) --> [].
not_same_row([Q|Qs]) -->
not_same_row_(Qs, Q),
not_same_row(Qs).
not_same_row_([], _) --> [].
not_same_row_([L|Ls], Q) -->
[~Q + ~L],
not_same_row_(Ls, Q).
length_(L, Ls) :- length(Ls, L).
diagonals(Qs, Row, Col, N) -->
( { Row #> N } -> []
; { Col #> N } ->
{ Row1 #= Row + 1 },
diagonals(Qs, Row1, 1, N)
; { queen_at(Qs, Row, Col, Q),
DRow #= Row + 1,
DCol #= Col + 1 },
diagonal_down(Qs, DRow, DCol, N, Q),
{ URow #= Row - 1,
UCol #= Col + 1 },
diagonal_up(Qs, URow, UCol, N, Q),
{ Col1 #= Col + 1 },
diagonals(Qs, Row, Col1, N)
).
diagonal_down(Qs, Row, Col, N,Q) -->
( { Row #> N } -> []
; { Col #> N } -> []
; { queen_at(Qs, Row, Col, Q0) },
[~Q + ~Q0],
{ Row1 #= Row + 1,
Col1 #= Col + 1 },
diagonal_down(Qs, Row1, Col1, N, Q)
).
diagonal_up(Qs, Row, Col, N, Q) -->
( { Row #< 1 } -> []
; { Col #> N } -> []
; { queen_at(Qs, Row, Col, Q0) },
[~Q + ~Q0],
{ Row1 #= Row - 1,
Col1 #= Col + 1 },
diagonal_up(Qs, Row1, Col1, N, Q)
).
queen_at(Qs, NRow, NCol, Q) :-
nth1(NRow, Qs, Row),
nth1(NCol, Row, Q).