逗号表达式
逗号表达式:就是用逗号隔开的多个表达式,从左向右依次执行
exp1,exp2,wxp3,......wxpn
注意:整个表达式的结果是最后一个表达式的结果。
void test()
{
int a = 1;
int b = 2;
int c = (a > b, a = b + 10, b = a + 1);
//可知最后一个表达式的结果,就是整个表达式的结果。
//那么从左到右依次计算就会得到c = 13;
}
下表引用、函数调用和结构成员
1. [ ]:下表引用符操作符
操作数 : 一个数组名 + 以后索引值
int arr[10];
arr[9] = 10;
[ ]:有两个操作数一个是arr,另外一个就是9.
注意:下表引用常用与数组的遍历修改
2.( ):函数调用操作符
操作数 : 一个函数名 + [若干传递给函数的参数]
#include<stdio.h>
void test()
{
printf("hehe\n");
}
void test2(const char* str)
{
printf("%s\n",str);
}
int main()
{
test();
test2("hello worl");
return 0;
}
3.访问一个结构体成员
. :结构体.成员名
-> :结构体指针->成员名
#include<stdio.h>
struct stu{
char name[20];
int age;
};
void test(struct stu stu)
{
stu.age = 18;
}
void test2(struct stu * pstu)
{
pstu->age = 18;
}
itn main()
{
struct stu stu;
struct stu* pstu = &stu;
stu.age = 10;
test(stu);
pstu->age = 20;
test2(pstu);
return 0;
}
表达式求值
众所周知表达式的求值,一部分有操作符的优先级和结合性决定,但是还有一部分在求职的过程中需要进行类型转换才能够进行计算。
隐式类型转换:
关于隐式类型转换,是指发生在没有明确说明的情况下(C语言风格的强制类型转换就是属于我们程序员有明确说明的),编译器自动帮我们执行的类型转换。通常同类型的数据进行运算、比较和赋值的时候我们是不需要担心的,这里我只是说明不同类型的数据进行运算、比较和赋值时,且我们程序员没有指定类型转换时,编译器是如何帮我们进行处理类型之间的转换的,只有知道这个过程,才能让我们知道程序设计时应该注意和避免的地方。
(1)比int类型小的隐式类型转换:整型提升(Integer Promotion)。
C语言的整形算术运算总是以缺省整形的精度来进行的,为了获得这个精度那么,表达式中的字符和短整型操作数在使用之前被转换为普通整型,这种转换称为整型提升
(2) 为什么要整型提升(Integer Promotion)?
我们知道表达式的整形运算要在CPU的相应的运算器件中完成,而这个期间就是逻辑运算单元(ALU)我们也叫做整形运算器,ALU的操作数长度就是int的字节长度一也就是32个比特位,同时也就是cpu通用寄存器的长度。那么此时如果我们要就是计算两个char类型的相加,而CPU中执的标椎长度是32位,但是char只有8位,如果要进行计算就要先报char类型变成32位然后通过寄存器送入ALU进行计算。因此,表达式中比int类型小的类型就要先转换成int或unsigned int类型,才能计算。而这个转换过程也就是整形提升,这也是整形提升存在的意义;;;;
(3)如何进行整形提升
整形提升是按照变量的数据类型的符号位来提升的
//负数的提升
char c1 = -1;变量c1的二进制位(补码)中只有8个
比特位:11111111
因为char为有符号的char,所以整形提升时,高位补
符号位,
提升之后:11111111111111111111111111111111
//正数的整形提升
char c2 = 1;变量c1的二进制位(补码)中只有8个
比特位:00000001
因为char为有符号的char,所以整形提升时,高位补
符号位,
提升之后:00000000000000000000000000000001
//无符号整形提升,,/*高位补0*/就OK了
//栗子
void test()
{
char c = 1;
printf("%u\n", sizeof(c));//输出1
printf("%u\n", sizeof(+c));//输出4
printf("%u\n", sizeof(!c));//输出1
}
//只要c参加表达式运算就会发生整形提升,所以+c就会发送提升,所以输出4,而其他没有发生表达式运算
(4)赋值时的隐式类型转换
上面说到都是在计算式的隐式类型转换,但是还要特别说明的就是在进行赋值操作时,赋值运算符右边的数据类型必须转换成赋值号左边的类型,若右边的数据类型的长度大于左边,则要进行截断或舍入操作。
char ch;
int i,result;
float f;
double d;
result = ch/i + (f*d-i);
(1)首先计算 ch/i,ch → int型,ch/i → int型。
(2)接着计算 f*d-i,由于最长型为double型,故f→double型,i→double型,f*d-i→double型。
(3)(ch/i) 和(f*d-i)进行加运算,由于f*d-i为double型,故ch/i→double型,ch/i+(f*d-i)→double型。
(4)由于result为int型,故ch/i+(f*d-i)→double→int,即进行截断与舍入,最后取值为整型