说起比较运算,肯定第一时间想到了C语言中关于比较的相关运算符 “>、<、!=、>=、<=、==”,那么要比较两个字符串是否相等是不是直接用“==”比较就行了。下面就来看看这种方法行不行?
先看一个例子
void main( void )
{char s1[] = "abc";char s2[] = "abc"; __asm( "sim" ); //禁止中断SysClkInit();delay_init( 16 );LED_GPIO_Init();BEEP_GPIO_Init();Uart1_Init( 9600 );__asm( "rim" ); //开启中断while( 1 ){if( s1 == s2 ){printf( "equal" );}else{printf( "Not equal" );}}
}
复制
有两个字符串s1和s2,内容都是”abc”,在if语句中通过”==”运算符进行比较。
运行结果是s1和s2不相等,那为什么是不相等呢,通过右边的观察窗口可以看到,s1和s2是内部的两个存储空间,地址不一样,但是地址中的内容是一样的。那么用“ if( s1 == s2 )”这行语句去比较的时候,其实比较的是s1和s2的地址,不是比较地址中的内容,而s1和s2的地址是不相同的。
要比较内容的话,代码需要修改一下。
if(( s1[0] == s2[0] )&&( s1[1] == s2[1] )&&( s1[2] == s2[2] )){printf( "equal" );}else{printf( "Not equal" );}
复制
将代码修改为依次比较字符串中的内容
通过字符串的每一位依次比较,这样的比较方式,就可以判断出字符串相等。
将字符串数组改为指针看看
将s1和s2改为指针,可以看到通过”==”比较,结果是相等的。通过右边的观察窗口可以看出s1和s2的地址是相同的,也就是这两个字符串指向了同一个地址空间。说明此时比较的还是地址,而不是地址中的内容。我们需要比较的是内容。
将代码修改为获取指针的内容,发现比较结果也是相等的。那么能不能说明这种比较方法是可以的。将字符串内容修改一下在看看。
内容改变后,指针的地址也发生了改变,此时”s1==s2″语句就不成立了。
但是这时候”*s1==*s2″依然是相等的,这是为什么呢?通过右侧的观察窗口可以看到,此时*s1和*s2的内容是”a”,而不是整个字符串的内容。说明这行代码比较的是字符串的第一个字符,而不是整个字符串。那么要比较整个内容的话,还得每个字符挨个比较。
通过上面的分析可能会有一个疑问,为什么用”==”运算符比较单个字符的时候比较的就是内容,而比较字符串的时候,比较的却是地址?
这就要说下一字符串在C语言中比较特殊的一点,在C语言中处理一般的变量如整形、字符型、浮点型时,直接操作的是变量的值,比如 int a,b; a=b;在执行这行代码的时候,是将b的值拷贝一份然后复制给a。比较运算符“==”在使用的时候,也直接比较的是变量的值。而C语言在使用字符串的时候,是通过地址引用而不是值引用来操作的。
通过数组定义的字符串,系统分配的是两个不同的地址,虽然内容相同,但是实际上地址是不相同的,通过相等运算符”==”比较时,比较的是s1和s2本身的值,s1和s2本身的值是一个地址,这两个地址不同,所以比较结果就不相等。
用指针定义的两个字符串,系统分配的是同一个地址空间,说明虽然他们的名字不一样,但是实际上是共用了同一个地址。通过相等运算符”==”比较时,比较的是s3和s4本身的值,s3和s4本身的值是一个地址,这两个地址相同,所以比较结果相等。
通过上面的分析,发现在处理字符串的时候不能直接用”==”运算符去比较,那么要如何去比较呢?
可以直接使用 C 库函数 int strcmp(const char *str1, const char *str2) 把 str1 所指向的字符串和 str2 所指向的字符串进行比较。
该函数返回值如下:
- 如果返回值小于 0,则表示 str1 小于 str2。
- 如果返回值大于 0,则表示 str1 大于 str2。
- 如果返回值等于 0,则表示 str1 等于 str2。
strcmp函数实际上是对字符的ASCII码进行比较,实现原理如下:首先比较两个串的第一个字符,若不相等,则停止比较并得出两个ASCII码大小比较的结果;如果相等就接着 比较第二个字符然后第三个字符等等。
这个函数其实就是相当于把上面例程中对字符串每个字符独立比较方法的一个封装,内部函数实现方式类似于下面这样。
由于字符串在C语言中的处理比较特殊,所以C语言提供了一个专门操作字符串的库。关于字符串的操作在”string.h”这个头文件中可以看到。
有字符串增加,字符串删除,字符串拷贝,字符串比较,字符串查找等等各种功能的函数封装,这样在处理字符串的时候就可以直接调用库函数来实现。
比如在处理串口命令的时候,如果串口命令为字符串,就可以直接用字符串比较命令去实现。
if ( strcmp( uartRecStr, "led1 on" ) == 0 ){led1_on();}else if ( strcmp( uartRecStr, "led1 off" ) == 0 ){led1_off();}else if ( strcmp( uartRecStr, "led2 on" ) == 0 ){led1_off();}else if ( strcmp( uartRecStr, "led2 off" ) == 0 ){led1_off();}
复制
将接收到字符串和指定的字符串比较,如果字符串相等,就调用相关的函数,这样通过if语句就可以简单的处理串口的指令。
不过通过if语句比较的话,程序的分支太多,程序看起来比较臃肿,能不能用switch语句去实现呢?但是switch语句中不支持字符串的比较直接比较,那要怎么实现?
char *cmdStr[] = {"led1 on", "led1 off", "led2 on", "led2 off"}; //字符串命令数组
len = sizeof( cmdStr ) / sizeof( cmdStr[0] ); //计算字符串个数for( i = 0; i < len; i++ ){if( strcmp( uartRecStr, cmdStr[i] ) == 0 ){switch( i )//分支语句不支持字符串,通过字符串数组的下标确定是哪个字符串。{case 0:led1_on();break;case 1:led1_off();break;case 2:led2_on();break;case 3:led2_off();break;default:break;}}}
复制
可以将字符串存储在数组中,通过数组的下标去调用不同的字符串,然后使用字符串比较函数去判断字符串,这样switch语句的分支判断就可以转换为字符串的下标。
通过这种方法就可以直接在串口中处理简单的通信命令了。