【C语言进阶剖析】18.C语言中的三目运算符和逗号表达式

724 阅读3分钟

本文已参与「新人创作礼」活动,一起开启掘金创作之路

文章目录

一、三目运算符

二、逗号表达式

三、小结


一、三目运算符

  • 三目运算符( a ? b : c)可以作为逻辑运算的载体
  • 规则:当 a 的值为真时,返回 b 的值;否则返回 c 的值

        下面看一段代码:

#include <stdio.h>

int main()
{
    int a = 1;
    int b = 2;
    int c = 0;
    
    c = a < b ? a : b;
    
    (a < b ? a : b) = 3;
    
    printf("%d\n", a);
    printf("%d\n", b);
    printf("%d\n", c);
    
    return 0;
}

        编译时会发现报错: 

​编辑

         这是因为三目运算符返回的是值,而不是变量,所以报错。

        看看下面做法,将地址作为三目运算符的返回,就可以:

#include <stdio.h>

int main()
{
    int a = 1;
    int b = 2;
    int c = 0;
    
    c = a < b ? a : b;
    
    *(a < b ? &a : &b) = 3;
    
    printf("%d\n", a);
    printf("%d\n", b);
    printf("%d\n", c);
    
    return 0;
}

        下面为输出结果: 

​编辑

  • 三目运算符( a ? b : c)的返回类型

    • 通过隐式类型转换规则返回 b 和 c 中的较高类型
    • 当 b 和 c 不能隐式转换到同一类型时将编译出错

        下面看一段代码,看看三目运算符的返回类型:

#include <stdio.h>

int main()
{   
    char c = 0;
    short s = 0;
    int i = 0;
    double d = 0;
    char* p = "str";
        
    printf( "%d\n", sizeof(c ? c : s) );
    printf( "%d\n", sizeof(i ? i : d) );
    //printf( "%d\n", sizeof(d ? d : p) );

    return 0;
}

        下面为输出结果:

​编辑

        char 和 short 在一起,返回 int 类型,占 4 个字节。 

        关于 char 和 short 返回 int 类型,我的理解如下:

         如果一个运算符两边的运算数类型不同,先要将其转换为相同的类型,即较低类型转换为较高类型,然后再参加运算,转换规则如下图所示。
double ←── float 高

long

unsigned

int ←── char,short 低

二、逗号表达式

  • 逗号表达式是C语言中的“粘贴剂”
  • 逗号表达式用于将多个子表达式连接为一个表达式
  • 逗号表达式的值为最后一个子表达式的值
  • 逗号表达式中的前 N-1 个子表达式可以没有返回值
  • 逗号表达式按照从左向右的顺序计算每个子表达式的值

        如下:

​编辑

         下面看一个逗号表达式的示例:

#include <stdio.h>

void hello()
{
    printf("Hello!\n");
}

int main()
{   
    int a[3][3] = {
        (0, 1, 2),
        (3, 4, 5),
        (6, 7, 8)
    };
    
    int i = 0;
    int j = 0;
    
    while( i < 5 )
        printf("i = %d\n", i),
        
    hello(),
    
    i++;
        
    for(i=0; i<3; i++)
    {
        for(j=0; j<3; j++)
        {
            printf("a[%d][%d] = %d\n", i, j, a[i][j]);
        }
    }

    return 0;
}

        下面为输出结果:

​编辑

         为什么打印出来的 a 数组和我们预想的不一样呢,这是因为 

        int a[3][3] = {
(0, 1, 2),
(3, 4, 5),
(6, 7, 8)
};

        里面的逗号构成了逗号表达式,逗号表达式的值为最后一个子表达式的值,相当于

        int a[3][3] = {
2,
5,
8
};

        而且,二维数组的初始化也不是那样,而是

        int a[3][3] = {
{0, 1, 2},
{3, 4, 5},
{6, 7, 8}
};

        这点一定要注意,不要弄错!!!

        下面再来看一个一行代码实现 strlen :

#include <stdio.h>
#include <assert.h>

int strlen(const char* s)
{  
    return assert(s), (*s ? strlen(s + 1) + 1 : 0);
}

int main()
{   
    printf("len = %d\n", strlen("Autumn"));
    printf("len = %d\n", strlen(NULL));
    
    return 0;
}

         下面为输出结果:

​编辑

三、小结

  • 三目运算符返回变量的值,而不是变量本身
  • 三目运算符通过隐式类型转换规则确认返回值类型
  • 逗号表达式按照从左向右的顺序计算每个子表达式的值
  • 逗号表达式的值为最后一个子表达式的值