离散数学——用c/c++求命题公式的主范式

本文借鉴了这篇文章https://blog.csdn.net/DaDaMr_X/article/details/51265230
离散数学又被称为计算机数学,本篇文章就介绍了如何用C++实现求离散数学中的主范式
要求主范式,具体可分为以下几步操作:
首先,由于某些特殊符号不好在计算机中打出,因此需要用相应的符号代替,并定义各个运算符的优先级,如下

替换方式
现符号优先级
5
合取&4
析取|3
条件-2
双条件+1

接下来需要按运算符优先级对公式进行转换,然后用位运算的方式写出五种运算符的运算方法
假设有两个命题变元a,b(计算时a,b的值为0或1),下面列出位运算方法(可以举例子试一下):
非运算:非a:  return (a+1)&1;(例如,a为1,a+1=2, 2在二进制中的表示为10,而位运算是取二进制形式的最后一位数进行运算,则为0&1,
其结果显然为0,即返回值为0,达到了非a的运算效果,以下运算方式类似)
合取:a&b: return a*b;
析取:if(a+b) return 1;else return 0;
条件:if(a==1 && b==0) return 0;else return 1;
双条件:return !((a+b)&1);

然后需要写一个函数来对所有存在的命题变元进行赋值,每个命题变元都有0和1两种情况,这一步可以采用递归来写(具体见代码),每完成一组赋值后都要都要以该组赋值情况确定下标(例如,有a,b,c,三个命题变元,假设某组赋值为101,则其下标为1*(2^0)+1*(2^2)=5,则最后输出范式时下标就为m5或M5,至于是主析取范式还是主合取范式则需看该组赋值的运算结果)

具体内容见代码:

#include<bits/stdc++.h>
using namespace std;

/*注意: 
    非:!
	合取:&
	析取:|
	条件:-
	双条件:+ 
	命题字母小写 
*/

const int N=1e4;
char s[N];//存放初始字符串 
bool table[30];//标记命题变元是否存在 
int explain[30];//存放每个命题变元的赋值(0或1) 
int value[1000010];//存放每组赋值的最后运算结果 
int sum=0;//下标 
 
int youxian(char c){
	switch(c){//定义运算符的优先顺序 
		case '#':return -1;
		case '!':return 5;
		case '&':return 4;
		case '|':return 3;
		case '-':return 2;
		case '+':return 1;
		case '(':return 0;
		default:return 0;
	}
}

void get_chars(){//将输入的字符串转换为逆波兰式 
	char post[N]={'\0'};int po=-1;//post存放转换后的命题形式,初始化为\0使strlen求长度时准确 
	char stack[N]={'#'};int st=0;//stack暂时存放运算符 
	int len=strlen(s);
	for(int i=0;i<len;i++){//当该字符为命题字母时 ,直接存入post 
		if(s[i]>='a' && s[i]<='z'){
			post[++po]=s[i];
			continue;
		}
		if(s[i]=='!'||s[i]=='&'||s[i]=='|'||s[i]=='-'||s[i]=='+'){
			/*下面一行包含两种情况,一是当前运算符优先级比上一个的小,则先把上一个运算符存入post,再把当前
			运算符暂时存入stack,以便下次比较,二是两个运算符优先级相同,则按从左到右的顺序运算,操作和第一
			种情况相同,还有一种运算符优先级大于上一个的情况,在60行得到了处理*/ 
			while(youxian(s[i])<=youxian(stack[st])) post[++po]=stack[st--];
			stack[++st]=s[i];
			continue;
		}
		if(s[i]=='('){
			stack[++st]=s[i];
			continue;
		}
		if(s[i]==')'){
			while(stack[st]!='(') post[++po]=stack[st--];//把括号之间的运算符加到post里 
			st--;//完成了一个括号内所有字符转换后,将st重新返回0,为后面的转换做准备 
			continue;
		}
	}
	while(st) post[++po]=stack[st--];/*该操作主要处理运算符优先级大于上一个的情况,把之前存在stack里
	的运算符一一存入post,至此,已完成全部字符串转换为逆波兰式的操作*/ 
	strcpy(s,post);//由于post不是全局变量,所以需要复制给s
	int l=strlen(s);
}
	
void settable(){//统计命题变元的个数 
	memset(table,0,sizeof(table));//将table数组初始化为0 
	int len=strlen(s);
	for(int i=0;i<len;i++){
		if(s[i]>='a'&&s[i]<='z') table[s[i]-'a']=true;//若含有某个变元,则将对应的table值变为1 
	}
	for(int i=0;i<26;i++){
		if(table[i]) sum++;//求命题变元的个数 
	}
	sum=pow(2,sum);//一共2^ans个赋值情况 
}

int boti(){//计算下标,如010表示2 
	int sum=0,wei=1;
	for(int i=25;i>=0;i--){//从后往前依次成二倍相加,把二进制数转换为十进制数 
		if(table[i]){
			if(explain[i]){ 
			    sum+=wei;/*expaain值为0时不需要加,只需加上1对应的值即可
			    如10010为1*2+1*2^4=20 */ 
		    }   wei*=2;
		}
	}
	return sum;//sum为最大情况个数 
} 

int cal(int a,int b,char c){//采用位运算对四种运算符进行操作 
	switch(c){
		case'&':return a*b;//合取 
		case'|':if(a+b) return 1;else return 0;//析取 
		case'-':if(a==1 && b==0) return 0;else return 1;//条件,a为真,b为假时为0,其他都为1; 
		case'+':return !((a+b)&1);//按位运算,比如2的二进制为10,则选0&1为结果 
	}
} 

int work(){//按照逆波兰式两两结合计算出最后结果 
	int stack[N],st=-1;
	int len=strlen(s);
	for(int i=0;i<len;i++){
		if(s[i]>='a' && s[i]<='z'){
			stack[++st]=explain[s[i]-'a'];//存入命题的一次赋值(为0或1) 
			continue;
		}
		if(s[i]=='!'){//是!时对上一个字符取非 
			stack[st]=(stack[st]+1)&1;//位运算:+1后再跟1进行与运算可实现非的效果 
			continue;
	    }
	    int ans=cal(stack[st-1],stack[st],s[i]);//两两结合运算 
	    stack[--st]=ans;//每次将st归零 ,下次将stack[0]与stack[1]运算 
	}
	return stack[0];//返回的是最终运算后的结果 
} 

void assign(){//计算一组赋值后的结果,如00010,10100对应的结果 
	int x=boti();
	int ans=work();
	value[x]=ans;
}
void generate(char c){//利用真值表列出所有的取值情况 
	while(c<='z'&&table[c-'a']==false) c++;//因为命题字母可以随意选择,所以要把26个遍历一遍 
	if(c>'z'){//超出z后再计算该组赋值的结果 
		assign();
		return ;
    }
    explain[c-'a']=0;//类似树状图,遍历所有命题变元的0,1取值 
    generate(c+1);
    explain[c-'a']=1;
    generate(c+1); 
}

void output1(){//输出析取范式 
	int i=0;
	while(i<sum && !value[i]) i++;/*此处sum可理解为最大情况总数, 
	在generate函数中每组数据以sum为下标都有对应的value[sum]值 
	*/ 
	if(i >= sum){
		printf("无主析取范式\n");
		return ;
	}
	printf("主析取范式为:m%d",i);
	for(i++;i<sum;i++){
		if(value[i]) printf(" V m%d",i);
	}
	printf("\n");
}

void output2(){//输出合取范式 
	int i=0;
	while(i<sum && value[i]) i++;
	if(i >= sum){
		printf("无主合取范式\n");
		return ;
	}
	printf("主合取范式为:M%d",i);
	for(i++;i<sum;i++){
		if(!value[i]) printf(" ∧ M%d",i);
	}
	printf("\n");
}

int main(){
	cin>>s;
	get_chars();
	settable();
	memset(value,0,sizeof(value));
	memset(explain,0,sizeof(explain));
	generate('a');//从a开始遍历26个字母 
	output1();
	output2();
	return 0;
}

测试样例如下:

  • 9
    点赞
  • 30
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: 本章要针对初学者,简要介绍了c/c++的基本语法和常见的编译错误。首先,介绍了程序的基本结构和常用的数据类型。c语言的基本语句包括赋值语句、条件语句和循环语句等,这些语句都可以组成程序的逻辑结构。接下来,介绍了函数的定义和调用,以及参数传递的方式。对于c++,还介绍了一些面向对象的概念,例如类、对象、成员函数等。同时,也提到了头文件和命名空间的使用方法。 在编程过程中,常常会出现各种编译错误,例如语法错误、类型不匹配、语义错误等,需要学会如何查看和解决这些错误。此外,还介绍了调试工具和各种常用的运算符和表达式,这些都是初学者必须掌握的基础知识。 总的来说,本章是关于c/c++快速入门的一篇简介性文章。虽然只是涉及到了基础的语法和知识点,但对于初学者而言是一个很好的起点。在学习过程中需要不断实践,积累经验,并不断深入了解更高级的编程技术和工具。 ### 回答2: C/C++是一种广泛使用的编程语言,它具有高效、灵活、可移植等特点,在各个领域得到广泛应用。C语言C++语言的基础,在学习C++之前,需要先掌握C语言的基础知识。 本文介绍的是C/C++教程中的第二章——快速入门C/C++。在这一章节中,我们将介绍C语言的基本语法、变量、运算符和流程控制语句等基础知识。以下是C语言的一些基本知识点。 C语言的基本语法: C语言程序由多个函数组成,其中一个函数必须命名为main(),程序从该函数开始执行。C程序中的语句以分号结束,注释使用“//”表示单行注释,“/* */”表示多行注释。 变量和数据类型: C语言中变量的定义格式为“数据类型 变量名”;数据类型包括基本类型和用户自定义类型。C语言中的基本类型有int类型、char类型、float类型和double类型等。其中,int类型表示整型,char类型表示字符型,float类型和double类型表示浮点型。 运算符: C语言中的运算符包括算术运算符、关系运算符、逻辑运算符等。例如,“+”表示加法运算符,“>=”表示大于等于运算符,“&&”表示逻辑与运算符。 流程控制语句: C语言中的流程控制语句包括if语句、switch语句、while语句、do-while语句和for语句等。这些语句可以根据条件执行相应的语句块。 总之,本章节的快速入门C/C++,具有基本语法、变量、运算符和流程控制语句等基础知识。初学者可以通过这些基础知识,轻松入门C/C++,为后续学习打下基础。同时,要注意编写代码的规范和逻辑性,才能更好的理解和使用C/C++语言。 ### 回答3: C语言是一门广泛使用的编程语言,具有高效、灵活、稳定等特点,被广泛应用于嵌入式系统、操作系统、驱动程序、多媒体应用等领域。学习C语言是程序员的必备技能之一。 第二章的快速入门C/C++教程,要介绍了C/C++语言的基础知识,重点是程序的结构和输入输出。其中程序结构包括函数、语句、变量、表达式等,而输入输出则包括scanf、printf、getchar和putchar等函数。 # 程序结构 程序结构是指程序的基本构成单元,包括函数、语句、变量、表达式等。C语言中的程序结构要包含以下几个方面。 ## 函数 C语言中,函数是程序的基本组成单元。一个C程序可以由一个或多个函数组成,每个函数可以完成一个任务。函数的格式如下: ```c 返回类型 函数名(参数1, 参数2, ...){ // 函数体 return 返回值; } ``` 其中,返回类型指函数执行后的返回值类型;函数名是由程序员定义的,用于调用函数时识别函数;参数列表是函数的输入参数,可以有多个参数,每个参数由类型和变量名组成;函数体是函数要执行的代码块;return语句可以返回函数的执行结果。 ## 语句 语句是完成特定功能的一组指令。C语言中的语句包括赋值语句、条件语句、循环语句等。C语言中通常使用花括号来表示语句块。例如,下面是一个if语句的例子。 ```c if(条件){ // if语句块 }else{ // else语句块 } ``` 如果条件为真,则执行if语句块中的代码;否则执行else语句块中的代码。 ## 变量 变量是用于存储数据的一种容器。在C语言中,一个变量包括变量名、类型和值。变量名由程序员定义,用于识别变量;类型指变量的数据类型,如整型、字符型、实型等;值是存储在变量中的数据。变量的定义格式如下。 ```c 数据类型 变量名 = 值; ``` 例如,下面是一个整型变量的定义。 ```c int num = 10; ``` ## 表达式 表达式是由变量、运算符和常量组成的一个具有返回值的语句。C语言中的运算符分为算术运算符、关系运算符、逻辑运算符等,例如加号、减号、乘号、除号等。下面是一个简单的表达式。 ```c a = 5 + 6 * 3 / 2 - 1; ``` 这个表达式将计算5加6乘3除以2减1的值,并将结果赋给a变量。 # 输入输出 输入输出是程序中非常重要的部分,可以让程序与用户进行交互。C语言中有多种输入输出函数,其中一些最常用的是scanf、printf、getchar和putchar函数。 ## scanf函数 scanf函数用于从标准输入读取格式化数据,并将读取的数据存储到变量中。它的格式如下。 ```c scanf("格式控制字符串", 变量列表); ``` 其中,格式控制字符串指示scanf函数需要读取的数据类型和格式,变量列表指向要读取的变量。下面是一个scanf函数的例子。 ```c int num; scanf("%d", &num); ``` 这个代码段将从标准输入读取一个整数,并将其存储到num变量中。 ## printf函数 printf函数用于将格式化数据输出到标准输出。它的格式如下。 ```c printf("格式控制字符串", 参数列表); ``` 其中,格式控制字符串指示printf函数需要输出的数据类型和格式,参数列表包含要输出的变量和常量。下面是一个printf函数的例子。 ```c int num = 5; printf("num的值是%d\n", num); ``` 这个代码段将输出“num的值是5”。 ## getchar和putchar函数 getchar函数用于从标准输入读取一个字符,putchar函数用于将一个字符输出到标准输出。它们的用法非常简单,例如下面的代码将读取一个字符并将其转换成大写字母后输出。 ```c char c = getchar(); putchar(toupper(c)); ``` 以上就是第二章中的快速入门C/C++教程的要内容,包括程序结构和输入输出方面的基础知识。熟练掌握这些内容,对于学习C语言来说是非常重要的。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值