提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档
前言
提示:这里可以添加本文要记录的大概内容:
考察 命题逻辑归结推理 要求 熟悉C++迭代器使用,有一定离散数学基础 复杂式子无法实现
提示:以下是本篇文章正文内容,下面案例可供参考
一、流程
二、步骤
0.声明
typedef string Formula;
typedef vector<string> SubsentenceSet;
typedef string::iterator FormulaIter;
typedef string::reverse_iterator FormulaRevIter;
const char EQ = '#';
const char UQ = '@';
const char IMPLICATION = '>';
const char NEGATION = '~';
const char CONJUNCTION = '&';
const char DISJUNCTION = '|';
const char CONSTANT_ALPHA[] = {'a', 'b', 'c', 'd', 'e', 'i', 'j', 'k'};
const char FUNC_ALPHA[] = {'f', 'g', 'h', 'l', 'm', 'n', 'o'};
1.消去蕴含词和等值词
Formula &RemoveImplication(Formula &f)
{
FormulaIter iter;
while ((iter = find(f.begin(), f.end(), IMPLICATION)) != f.end())
{
*iter = DISJUNCTION;
FormulaRevIter revIter(iter);
revIter = GetBeginOfFormula(revIter, f.rend());
iter = revIter.base();
f.insert(iter, NEGATION);
}
return f;
}
2.使否定词仅作用于原子公式
Formula &MoveNegation(Formula &f)
{
FormulaIter iter = find(f.begin(), f.end(), NEGATION);
while (iter != f.end())
{
if (*(iter + 1) == '(')
{
if (*(iter + 2) == EQ || *(iter + 2) == UQ)
{
*(iter + 2) == EQ ? *(iter + 2) = UQ : *(iter + 2) = EQ;
string leftDonePart(f.begin(), iter + 5);
leftDonePart.erase(find(leftDonePart.begin(), leftDonePart.end(), NEGATION));
string rightPart(iter + 5, f.end());
rightPart.insert(rightPart.begin(), NEGATION);
MoveNegation(rightPart);
string(leftDonePart + rightPart).swap(f);
return f;
}
else
{
iter = f.insert(iter + 2, NEGATION);
while (1)
{
iter = FindFormula(iter, f.end());
FormulaIter iter2 = FindPairChar(
iter, f.end(), '(', ')');
++iter2;
if (IsConnector(*iter2))
{
*iter2 == DISJUNCTION ? *iter2 = CONJUNCTION
: *iter2 = DISJUNCTION;
iter = f.insert(iter2 + 1, NEGATION);
}
else
break;
}
f.erase(find(f.begin(), f.end(),
NEGATION));
return MoveNegation(f);
}
}
else if (*(iter + 1) == NEGATION)
{
f.erase(iter, iter + 2);
return MoveNegation(f);
}
else
{
string leftDonePart(f.begin(), iter + 1);
string rightPart(iter + 1, f.end());
MoveNegation(rightPart);
string(leftDonePart + rightPart).swap(f);
return f;
}
}
return f;
}
3.使量词间不含同名指导变元
Formula &StandardizeValues(Formula &f)
{
set<char> checkedAlpha;
FormulaIter iter = FindQuantifier(f.begin(), f.end());
while (iter != f.end())
{
char varName = *++iter;
if (checkedAlpha.find(varName) == checkedAlpha.end())
{
checkedAlpha.insert(varName);
}
else
{
char newName = FindNewLowerAlpha(checkedAlpha);
checkedAlpha.insert(newName);
FormulaIter rightBorder = FindPairChar(
iter + 2, f.end(), '(', ')');
*iter = newName;
replace(iter, rightBorder, varName, newName);
iter = rightBorder;
}
iter = FindQuantifier(iter, f.end());
}
return f;
}
4.化为前束范式
Formula &TransformToPNF(Formula &f)
{
FormulaIter iter = FindQuantifier(f.begin(), f.end());
if (iter == f.end())
return f;
else if (iter - 1 == f.begin())
{
iter += 3;
string leftPart(f.begin(), iter);
string rightPart(iter, f.end());
TransformToPNF(rightPart);
(leftPart + rightPart).swap(f);
}
else
{
string quantf(iter - 1, iter + 3);
f.erase(iter - 1, iter + 3);
f.insert(f.begin(), quantf.begin(), quantf.end());
return TransformToPNF(f);
}
return f;
}
5.消去存在量词
Formula &RemoveEQ(Formula &f)
{
set<char> checkedAlpha;
FormulaIter eqIter = find(f.begin(), f.end(), EQ);
if (eqIter == f.end())
return f;
FormulaRevIter left = leftleft(FormulaRevIter(eqIter), f.rend());
FormulaRevIter uqIter = findRev(left, f.rend(), UQ);
if (uqIter == f.rend())
{
char varName = *(eqIter + 1);
char newName = GetNewConstantAlha(f);
auto rightBound = FindPairChar(eqIter + 3, f.end(), '(', ')');
replace(eqIter + 3, rightBound, varName, newName);
f.erase(eqIter - 1, eqIter + 3);
}
else
{
copy_if(f.begin(), f.end(), checkedAlpha);
const char oldName = *(eqIter + 1);
const char funcName = FindFuncAlpha(checkedAlpha);
string funcFormula;
funcFormula = funcFormula + funcName + '(' + *(uqIter - 1) + ')';
f.erase(eqIter - 1, eqIter + 3);
ReplaceAlphaWithString(f, oldName, funcFormula);
}
return RemoveEQ(f);
}
6.消去全称量词
Formula &RemoveUQ(Formula &f)
{
FormulaIter uqIter = find(f.begin(), f.end(), UQ);
while (uqIter != f.end())
{
uqIter = f.erase(uqIter - 1, uqIter + 3);
uqIter = find(uqIter, f.end(), UQ);
}
return f;
}
7.化公式为合(&)取范式
Formula &TransformToConjunction(Formula &f)
{
FormulaIter dis = find(f.begin(), f.end(), DISJUNCTION);
FormulaRevIter left = leftfind((FormulaRevIter)dis, f.rend());
FormulaIter leftt = left.base() - 1;
FormulaIter right = rightfind(dis + 1, f.end());
FormulaRevIter leftcon = findRev((FormulaRevIter)dis, left, CONJUNCTION);
FormulaIter lefttcon = leftcon.base() - 1;
FormulaIter rightcon = find(dis, right, CONJUNCTION);
if (leftt != lefttcon && rightcon == right)
{
cout << string(dis + 1, right) << endl;
cout << string(leftt + 1, lefttcon) << endl;
cout << string(lefttcon + 1, dis) << endl;
string temp = "(" + string(dis + 1, right) + "|" + string(leftt + 1, lefttcon) + ")&(" + string(dis + 1, right) + "|" + string(lefttcon + 1, dis - 1) + ")";
f.replace(leftt, right, temp);
}
else if (leftt == lefttcon && right != rightcon)
{
string temp = "(" + string(leftt + 1, dis) + "|" + string(dis + 1, rightcon) + ")&(" + string(leftt + 1, dis) + "|" + string(rightcon + 1, right - 1) + ")";
f.replace(leftt, right, temp);
}
else if (leftt != lefttcon && right != rightcon)
{
string str1 = string(leftt + 1, lefttcon);
string str2 = string(lefttcon + 1, dis);
string str3 = string(dis + 1, rightcon);
string str4 = string(rightcon + 1, right);
cout << str1 << endl;
cout << str2 << endl;
cout << str3 << endl;
cout << str4 << endl;
string temp;
if (str1 == str3)
{
temp = str1 + "&(" + str2 + "|" + str4 + ")";
}
else if (str1 == str4)
{
temp = str1 + "&(" + str2 + "|" + str3 + ")";
}
else if (str2 == str3)
{
temp = str2 + "&(" + str1 + "|" + str4 + ")";
}
else if (str2 == str4)
{
temp = str2 + "&(" + str1 + "|" + str3 + ")";
}
f.replace(leftt, right + 1, temp);
}
return f;
}
8.消去合取词,得到子句集
这里还需要对子句做标准化,即不同的子句用不同的变元,和前面的标准化大同小异。
void ExtractSubsentence(SubsentenceSet &subset, Formula &f)
{
FormulaIter leftIter = f.begin();
FormulaIter middleIter = find(f.begin(), f.end(), CONJUNCTION);
while (middleIter != f.end())
{
if (*leftIter == '(' && *(middleIter - 1) == ')')
subset.push_back(string(leftIter + 1, middleIter - 1));
else
subset.push_back(string(leftIter, middleIter));
leftIter = middleIter + 1;
middleIter = find(middleIter + 1, f.end(), CONJUNCTION);
}
if (*leftIter == '(' && *(middleIter - 1) == ')')
subset.push_back(string(leftIter + 1, middleIter - 1));
else
subset.push_back(string(leftIter, middleIter));
}
9.合一算法
合一代码,直接调用syncretism函数。
bool syncretism(const string tf1, const string tf2, vector<transform> &mgu)
{
string f1 = tf1, f2 = tf2;
while (!same(f1, f2))
{
transform t = dif(f1, f2);
int flag = legal(t);
if (flag == 0)
return false;
else
{
mgu.push_back(t);
f1 = change(f1, mgu.back());
f2 = change(f2, mgu.back());
cout << "after change:" << endl;
cout << "f1:" << f1 << endl;
cout << "f2:" << f2 << endl;
if (same(f1, f2))
return true;
}
}
return false;
}
bool same(const string f1, const string f2)
{
if (f1.length() == f2.length())
{
int i = 0;
while (i < f1.length() && f1.at(i) == f2.at(i))
i++;
if (i == f1.length())
return true;
else
{
return false;
}
}
else
return false;
}
transform dif(const string f1, const string f2)
{
int i = 0;
transform t;
while (f1.at(i) == f2.at(i))
i++;
int j1 = i;
while (j1 < f1.length() - 1 && f1.at(j1) != ',')
j1++;
if (j1 - i == 0)
return t;
t.t_f1 = f1.substr(i, j1 - i);
int j2 = i;
while (j2 < f2.length() - 1 && f2.at(j2) != ',')
j2++;
if (j2 - i == 0)
return t;
t.t_f2 = f2.substr(i, j2 - i);
while (t.t_f1[j1 - i - 1] == t.t_f2[j2 - i - 1])
{
t.t_f1.erase(j1 - 1 - i);
t.t_f2.erase(j2 - i - 1);
j1--;
j2--;
}
return t;
}
int legal(transform &t)
{
if (t.t_f1.length() == 0 || t.t_f2.length() == 0)
return 0;
if (var(t.t_f2))
{
if (var(t.t_f1) && (varData(t.t_f1) == varData(t.t_f2)))
return 0;
else
return 2;
}
if (!var(t.t_f1))
return 0;
string temp;
temp = t.t_f1;
t.t_f1 = t.t_f2;
t.t_f2 = temp;
return 1;
}
string varData(string s)
{
if (s.length() == 1 || s.length() == 0)
return s;
if (s.length() > 1)
{
int i = 0;
while (i < s.length() && s.at(i) != '(')
i++;
int j = i;
while (j < s.length() && s.at(j) != ')')
j++;
string ss = s.substr(i + 1, j - i - 1);
return varData(ss);
}
}
bool var(const string s)
{
if (s.length() == 0)
return false;
if (s.length() == 1 && s[0] >= 'A' && s[0] <= 'Z')
return false;
if (s.length() > 1)
{
int i = 0;
while (i < s.length() && s.at(i) != '(')
i++;
int j = i;
while (j < s.length() && s.at(j) != ')')
j++;
string ss = s.substr(i + 1, j - i - 1);
return var(ss);
}
else
{
return true;
}
}
string change(string f, transform q)
{
int i = f.find(q.t_f2);
while (i < f.length())
{
i = f.find(q.t_f2);
if (i < f.length())
f = f.replace(i, q.t_f2.length(), q.t_f1);
}
return f;
}
10.归结原理
bool resolution(SubsentenceSet &subset)
{
for (vector<string>::reverse_iterator rbegin = subset.rbegin(); rbegin != subset.rend();)
{
bool f = false;
bool rev = false;
string s = *rbegin;
string ss;
FormulaIter left = s.begin();
FormulaIter middle = find(left, s.end(), DISJUNCTION);
string goal = string(left, middle);
string _goal = pdrev(goal, rev);
vector<string>::reverse_iterator iter;
while (middle != s.end())
{
iter = findreviter(subset, _goal, rev);
if (iter == subset.rend())
{
left = middle + 1;
middle = find(left, s.end(), DISJUNCTION);
goal = string(left, middle);
_goal = pdrev(goal, rev);
}
else
{
f = true;
break;
}
}
if (!f)
{
iter = findreviter(subset, _goal, rev);
if (iter == subset.rend())
{
rbegin++;
continue;
}
}
ss = (*iter);
char cc = goal[0];
int len = goal.length();
if (rev)
{
cc = goal[1];
len -= 1;
}
dosyncretism(s.substr(s.find(cc), len), ss.substr(ss.find(cc), len), *iter);
(*rbegin).erase((*rbegin).find(goal), goal.length());
string c = "";
if (rev)
{
c += _goal[0];
}
else
{
c += _goal[0] + _goal[1];
}
(*iter).erase((*iter).find(c), _goal.length());
string newstr = (*rbegin) + "|" + (*iter);
for (int i = 0; i < newstr.length(); i++)
{
if ((i == 0 || i == newstr.length() - 1) && newstr[i] == DISJUNCTION)
{
newstr.erase(i, 1);
}
else if (newstr[i] == DISJUNCTION && newstr[i + 1] == DISJUNCTION)
{
newstr.erase(i + 1, 1);
}
}
cout << endl;
subset.erase(subset.rbegin().base());
subset.erase(iter.base() - 1);
subset.push_back(newstr);
rbegin = subset.rbegin();
count(subset);
if (newstr == "")
{
return true;
}
}
return false;
}
运行结果
1.
Input:
(@x)(D(x)>(B(x)&F(x)))&(@x)(F(x)>N(x))&(@x)(G(x)>D(x))&~(@x)(G(x)>N(x))
Elimination of implied symbols:
(@x)(~D(x)|(B(x)&F(x)))&(@x)(~F(x)|N(x))&(@x)(~G(x)|D(x))&~(@x)(~G(x)|N(x))
Move the negation sign to the front of each predicate:
(@x)(~D(x)|(B(x)&F(x)))&(@x)(~F(x)|N(x))&(@x)(~G(x)|D(x))&(#x)(G(x)&~N(x))
Standardization of variables:
(@x)(~D(x)|(B(x)&F(x)))&(@y)(~F(y)|N(y))&(@z)(~G(z)|D(z))&(#q)(G(q)&~N(q))
Eliminate existential quantifiers:
(@x)(~D(x)|(B(x)&F(x)))&(@y)(~F(y)|N(y))&(@z)(~G(z)|D(z))&(G(a)&~N(a))
Convert to a front bundle:
(@x)(@y)(@z)(~D(x)|(B(x)&F(x)))&(~F(y)|N(y))&(~G(z)|D(z))&(G(a)&~N(a))
Omit universal quantifiers:
(~D(x)|(B(x)&F(x)))&(~F(y)|N(y))&(~G(z)|D(z))&(G(a)&~N(a))
Standardization of Conjunction:
(~D(x)|B(x))&(~D(x)|F(x))&(~F(y)|N(y))&(~G(z)|D(z))&G(a)&~N(a)
Eliminate the conjunction:
~D(x)|B(x)
~D(x)|F(x)
~F(y)|N(y)
~G(z)|D(z)
G(a)
~N(a)
after change:
f1:N(a)
f2:N(a)
mgu={ a/y }
~D(x)|B(x)
~D(x)|F(x)
~G(z)|D(z)
G(a)
~F(a)
after change:
f1:F(a)
f2:F(a)
mgu={ a/x }
~D(x)|B(x)
~G(z)|D(z)
G(a)
~D(a)
after change:
f1:D(a)
f2:D(a)
mgu={ a/z }
~D(x)|B(x)
G(a)
~G(a)
cannot be syncretized
~D(x)|B(x)
Success
2.
Input:
~((#x)(P(x)&(@y)D(y)))
Elimination of implied symbols:
~((#x)(P(x)&(@y)D(y)))
Move the negation sign to the front of each predicate:
((@x)(~P(x)|(#y)~D(y)))
Standardization of variables:
((@x)(~P(x)|(#y)~D(y)))
Eliminate existential quantifiers:
((@x)(~P(x)|~D(f(x))))
Convert to a front bundle:
(@x)((~P(x)|~D(f(x))))
Omit universal quantifiers:
((~P(x)|~D(f(x))))
Standardization of Skolem:
((~P(x)|~D(f(x))))
Eliminate the conjunction:
(~P(x)|~D(f(x)))
Failure
代码简陋,还请包涵
|