-
Notifications
You must be signed in to change notification settings - Fork 0
/
AnalyticTableaux.cpp
241 lines (221 loc) · 6.55 KB
/
AnalyticTableaux.cpp
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
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
#include "AnalyticTableaux.h"
#include <iostream>
/**
* Método construtor. Inicia o tableau com um nó cuja expressão
* é expression e cujo valor-verdade é truthValue.
*
* Exemplo: Node tableau = Node("((~P) v P)", false)
*/
Node::Node(string expression, bool truthValue)
{
this->expression = expression;
this->truthValue = truthValue;
this->isAppliable = regex_match(expression, regex("\\(.*\\)"));
this->isContradictory = false;
this->leftChild = nullptr;
this->rightChild = nullptr;
this->frontChild = nullptr;
this->parent = nullptr;
}
/**
* Bifurca a árvore e insere dois nós em cada ramo, um com expressão expression1 e valor-verdade truthValue1,
* e outro com expressão expression2 e valor-verdade truthValue2, em todas as folhas alcançáveis
* a partir deste nó, e retorna todos os nós adicionados por essa operação.
*
* Exemplo: vector<Node*> insertedNodes = node.InsertSides("P", false, "Q", true)
*/
vector<Node *> Node::insertSides(string expression1, bool truthValue1, string expression2, bool truthValue2)
{
vector<Node *> insertedNodes;
vector<Node *> reachableLeafs = getReachableLeafs();
for (Node *leaf : reachableLeafs)
{
Node *leftChild = new Node(expression1, truthValue1);
Node *rightChild = new Node(expression2, truthValue2);
insertNode(leaf, leftChild, rightChild);
insertedNodes.push_back(leftChild);
insertedNodes.push_back(rightChild);
}
return insertedNodes;
}
/**
* Insere um nó com expressão expression e valor-verdade truthValue em todas as folhas alcançáveis
* a partir deste nó, e retorna todos os nós adicionados por essa operação.
* A outra versão permite inserir dois nós.
*
* Exemplo: vector<Node*> insertedNodes = node.InsertFront("(~P)", false)
*/
vector<Node *> Node::insertFront(string expression, bool truthValue)
{
vector<Node*> insertedNodes;
vector<Node *> reachableLeafs = getReachableLeafs();
for (Node *leaf : reachableLeafs)
{
Node *child = new Node(expression, truthValue);
insertNode(leaf, child);
insertedNodes.push_back(child);
}
return insertedNodes;
}
/**
* Insere um nó com expressão expression e valor-verdade truthValue em todas as folhas alcançáveis
* a partir deste nó, e retorna todos os nós adicionados por essa operação.
* A outra versão permite inserir dois nós.
*
* Exemplo: vector<Node*> insertedNodes = node.InsertFront("(~P)", false)
*/
vector<Node *> Node::insertFront(string expression1, bool truthValue1, string expression2, bool truthValue2)
{
vector<Node *> insertedNodes;
vector<Node *> reachableLeafs = getReachableLeafs();
for (Node *leaf : reachableLeafs)
{
Node *child1 = new Node(expression1, truthValue1);
Node *child2 = new Node(expression2, truthValue2);
insertNode(leaf, child1);
insertNode(child1, child2);
insertedNodes.push_back(child1);
insertedNodes.push_back(child2);
}
return insertedNodes;
}
void Node::insertNode(Node *parent, Node *frontChild)
{
parent->frontChild = frontChild;
frontChild->parent = parent;
}
void Node::insertNode(Node *parent, Node *leftChild, Node *rightChild)
{
parent->leftChild = leftChild;
parent->rightChild = rightChild;
leftChild->parent = rightChild->parent = parent;
}
vector<Node *> Node::getReachableLeafs()
{
vector<Node *> allNodes = traverse(new vector<Node *>);
vector<Node *> reachableLeafs;
for (Node *node : allNodes)
{
if (node->isLeaf())
reachableLeafs.push_back(node);
}
return reachableLeafs;
}
bool Node::isLeaf()
{
return leftChild == nullptr && rightChild == nullptr && frontChild == nullptr && !isContradictory;
}
/**
* Retorna todos os nós alcançáveis a partir deste nó em que se pode aplicar uma regra.
* Para retornar todos os nós da árvore em que se pode aplicar uma regra, o nó chamado deve ser a raiz.
*
* Exemplo: vector<Node*> appliableNodes = tableau.getAppliableNodes()
*/
vector<Node *> Node::getAppliableNodes()
{
vector<Node *> allNodes = traverse(new vector<Node *>);
vector<Node *> appliableNodes;
for (Node *node : allNodes)
{
if (node->isAppliable)
appliableNodes.push_back(node);
}
return appliableNodes;
}
vector<Node *> Node::traverse(vector<Node *> *nodes)
{
nodes->push_back(this);
if (frontChild != nullptr)
frontChild->traverse(nodes);
else if (leftChild != nullptr && rightChild != nullptr)
{
leftChild->traverse(nodes);
rightChild->traverse(nodes);
}
return *nodes;
}
/**
* Marca o nó como sido aplicado uma regra. Ele não aparecerá mais em chamadas de getAppliableNodes().
*
* Exemplo: node.markApplied()
*/
void Node::markApplied()
{
isAppliable = false;
}
/**
* Checa se a subárvore a partir deste nó está fechada.
* Para saber se o tableau inteiro está fechado, o nó chamado deve ser a raiz.
*
* Exemplo: bool isClosed = tableau.isClosed()
*/
bool Node::isClosed()
{
return getReachableLeafs().empty();
}
/**
* Checa se o nó contradiz algum outro do mesmo ramo.
*
* Exemplo: bool isContradictory = node.checkContradiction()
*/
bool Node::checkContradiction()
{
for (Node *it = this->parent; it != nullptr; it = it->parent)
{
if ((it->expression == this->expression && it->truthValue == !this->truthValue) || it->isContradictory)
return true;
}
return false;
}
/**
* Marca o nó como contraditório. Seu ramo será fechado e ignorado agora em diante.
*
* Exemplo: node.markContradiction()
*/
void Node::markContradiction()
{
isContradictory = true;
}
/**
* Imprime a árvore a partir do nó atual.
* Para imprimir toda a árvore, o nó chamado deve ser a raiz.
*
* Exemplo: tableau.printTree()
*/
void Node::printTree()
{
printTree(0);
}
void Node::printTree(int level)
{
for (int i = 0; i < level; i++)
cout << '\t';
cout << "v(" << this->expression << ") = " << this->truthValue << (isContradictory ? " X " : "") << endl;
if (frontChild != nullptr)
{
frontChild->printTree(level + 1);
}
else if (leftChild != nullptr && rightChild != nullptr)
{
leftChild->printTree(level + 1);
rightChild->printTree(level + 1);
}
}
/**
* Retorna a proposição do nó.
*
* Exemplo: string expression = node.getExpression()
*/
string Node::getExpression()
{
return expression;
}
/**
* Retorna o valor-verdade do nó.
*
* Exemplo: bool truthValue = node.getTruthValue()
*/
bool Node::getTruthValue()
{
return truthValue;
}