poj3295 Tautology —— 构造法

题目链接:http://poj.org/problem?id=3295


题意:

输入由p、q、r、s、t、K、A、N、C、E共10个字母组成的逻辑表达式,

其中p、q、r、s、t的值为1(true)或0(false),即逻辑变量;

K、A、N、C、E为逻辑运算符,

K --> and:  x && y

A --> or:  x || y

N --> not :  !x

C --> implies :  (!x)||y

E --> equals :  x==y

问这个逻辑表达式是否为永真式。

PS:输入格式保证是合法的。


题解:一步一步地解决。1.首先需要记录变量,且操作时需要记录变量的值。所以就用一个ch[]记录变量,val[]记录变量的值(以val[ch[]-97]的形式记录,免去查找),并且记录变量的个数n。2.然后,变量需要“变”,然后就想到用dfs(),所以在每种情况中,变量都有确定的值。3.最后就是要处理这个逻辑表达式,一开始想直接在一个数组中处理,发现很难,要是逻辑表达式中海油逻辑表达式……,那该怎么处理。后来想到逆波兰表达式(没学,但却给了点启示),好像是用两个栈,对四则运算符和数字分开,什么遇到运算符就对数字进行操作,然后更新的,大概是这样。然后就想到:设一个主栈,初始化为输入的逻辑表达式,并设一个副栈,储存操作数。对主栈进行处理,遇到变量压入副栈,遇到操作码,提取副栈的操作数,进行操作并更新。一直到处理完主栈,最后副栈只剩下一个值。这个值就是:在变量为某些确定值时,这个逻辑表达式的真或假。



代码如下:


#include<stdio.h>//poj3295
#include<string.h>
//a为原始输入的字符串, ch用于保存变量,val用于保存变量的值
char a[105],ch[105],sm[105];
int val[30],ss[105];

int sum()
{   //sm为主栈,ss为副栈,op为操作数,rm和rs为对应的栈顶指针
    int op1,op2,rm,rs = -1;
    strcpy(sm,a);//初始化主栈
    rm = strlen(sm);
    while(--rm>=0)
    {   /*对主栈进行出栈操作,如果遇到变量,则压入副栈
          如果遇到操作码,则对副栈的数进行操作更新*/
        if(sm[rm]>='a' && sm[rm]<='z')
        {
            ss[++rs] = val[sm[rm]-97];
        }

        else
        {
            if(sm[rm]=='K')
            {
                op1 = ss[rs--], op2 = ss[rs];
                ss[rs] = (op1 && op2);
            }

            else if(sm[rm]=='A')
            {
                op1 = ss[rs--], op2 = ss[rs];
                ss[rs] = (op1 || op2);
            }

            else if(sm[rm]=='N')
            {
                ss[rs] = !ss[rs];
            }

            else if(sm[rm]=='C')
            {
                op1 = ss[rs--], op2 = ss[rs];
                ss[rs] = ((!op1)||op2);
            }

            else if(sm[rm]=='E')
            {
                op1 = ss[rs--], op2 = ss[rs];
                ss[rs] = (op1 == op2);
            }
        }
    }
    //返回最后的操作数,0或1
    return ss[0];
}

int dfs(int i,int n)
{   //用dfs对变量进行变化
    if(i==n)
    {
        if(sum())return 1;
        else return 0;
    }
    //如果出现假的情况,那肯定不是tautology, 那么余下的就不必判断,直接退出
    //所以将dfs()放到判断中去
    val[ch[i]-97] = 0;
    if(!dfs(i+1,n))return 0;
    val[ch[i]-97] = 1;
    if(!dfs(i+1,n))return 0;
    return 1;
}

int main()
{
    while(scanf("%s",a) && a[0]!='0')
    {
        int n = 0;
        for(int i = 0; a[i]!=0; i++)
        {   //记录变量
            if(a[i]>='a' && a[i]<='z')
            {
                int j;
                for(j = 0; j<n; j++)
                    if(a[i]==ch[j]) break;
                if(j==n)
                    ch[n++] = a[i];
            }
        }

        if(dfs(0,n))
            puts("tautology");
        else
            puts("not");
    }
    return 0;
}



看了一下网上的解题报告,可以对自己的代码进行优化:1.由于变量只有5个,所以可设5重循环来代替递归dfs(但是dfs根据变量的个数决定递归层次的,而循环则有点盲目)

2.由于变量最多只有5个,所以可直接射变量为p,q,r,s,t(但自己的方法可以是随便的变量)。3.处理逻辑表达式时,只需一个栈就够了,这个栈为上述代码的副栈。


代码如下(转载):


#include<string.h>
#include<stdio.h>
const int maxn=120;
int sta[maxn]; //数组模拟堆栈
char str[maxn];
int p,q,r,s,t;
void doit()
{
    int top=0;
    int len=strlen(str);
    for(int i=len-1;i>=0;i--)
    {
        if(str[i]=='p') sta[top++]=p;
        else if(str[i]=='q') sta[top++]=q;
        else if(str[i]=='r') sta[top++]=r;
        else if(str[i]=='s') sta[top++]=s;
        else if(str[i]=='t') sta[top++]=t;
        else if(str[i]=='K')
        {
            int t1=sta[--top];
            int t2=sta[--top];
            sta[top++]=(t1&&t2);
        }
        else if(str[i]=='A')
        {
            int t1=sta[--top];
            int t2=sta[--top];
            sta[top++]=(t1||t2);
        }
        else if(str[i]=='N')
        {
            int t1=sta[--top];
            sta[top++]=(!t1);
        }
        else if(str[i]=='C')
        {
            int t1=sta[--top];
            int t2=sta[--top];
            if(t1==1&&t2==0) sta[top++]=0;
            else sta[top++]=1;
        }
        else if(str[i]=='E')
        {
            int t1=sta[--top];
            int t2=sta[--top];
            if((t1==1&&t2==1)||(t1==0&&t2==0)) sta[top++]=1;
            else sta[top++]=0;
        }
    }
}
 
bool solve()
{   //5重循环,枚举2^5  32种可能 如果都满足 return 1 
    for(p=0;p<2;p++)
        for(q=0;q<2;q++)
            for(r=0;r<2;r++)
                for(s=0;s<2;s++)
                    for(t=0;t<2;t++)
                    {
                        doit();
                        if(sta[0]==0)return 0;
                    }
     return 1;
}
 
int main()
{
    while(scanf(%s,&str))
    {
        if(strcmp(str,0)==0)break;
        if(solve()) printf(tautology
);
        else printf(not
);
    }
    return 0;
}


有空再改进一下自己的代码,但仍喜欢用dfs。

等等,用dfs还怎么知道是哪个变量,看来还是用循环好,毕竟对题目有针对性。


posted on 2017-02-13 15:04  h_z_cong  阅读(295)  评论(0编辑  收藏  举报

导航