这篇文章给大家分享的是有关C语言入门基础之操作符的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。操作符首先第一部分操作符分类如上,具体不再用文字阐述。算术操作符首先算术操作符,有除号值得一讲,若想得浮点数
这篇文章给大家分享的是有关C语言入门基础之操作符的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。
首先第一部分操作符
分类如上,具体不再用文字阐述。
首先算术操作符,有除号值得一讲,若想得浮点数,两端操作数至少有一个为浮点数,否则就算变量用float定义也不行。
int main(){//除号任意两端有浮点数则,进行浮点数除法//全为整数则进行整数除法float a = 5 / 2;printf("%f\n", a);//2.0000float b = 5.0 / 2;printf("%f\n", b);//2.5000float c = 5 / 2.0;printf("%f\n", c);//2.5000float d = 5.0 / 2.0;printf("%f\n", d);//2.5000 return 0;}
接下来是移位操作符,分为左移和右移,右移较复杂先不讲,先讲左移。按二进制位向左移动一位,即把32个bit的二进制序列写出来,整体向左移动一位,左侧移出去就删去,右侧补零。
当然如果你写的多的话就可以看出来左移一位即十进制数字乘以二。
具体看代码
int main(){//00000000 00000000 00000000 00001100 - 12//00000000 00000000 00000000 00011000 - 24int a = 12;int b = a << 1;printf("%d\n", b);//00000000 00000000 00000000 00000110 - 6//00000000 00000000 00000000 00001100 - 12int c = 6;int d = c << 1;printf("%d\n", d);//00000000 00000000 00000000 00000010 - 2//00000000 00000000 00000000 00000100 - 4int e = 2;int f = e << 1;printf("%d\n", f);//由2^0左移一位为2^1;//由2^1左移一位为2^2;//由2^2+2^3左移一位为2^3+2^4;//二进制左移一位即十进制乘以二return 0;}
接下来是位操作符,有按位与,按位异或和按位或,按位与和按位或是反义的。
具体方法是将两个操作数的二进制位写出来,相应位对比。
按位与:有0则0,全1则1。
按位异或:相同为0,相异为1。(异或嘛~)
按位或:有1则1,全0才0。
具体看代码
int main(){int a = 3;int b = 5;//00000000000000000000000000000011 - a//00000000000000000000000000000101 - b//00000000000000000000000000000001 - a & b//对应的二进制位有0就为0,全是1则为1int c = a & b;printf("%d\n", c);//1//00000000000000000000000000000011 - a//00000000000000000000000000000101 - b//00000000000000000000000000000110 - a ^ b//对应的二进制位相同为0,相异为1int d = a ^ b;printf("%d\n", d);//6//00000000000000000000000000000011 - a//00000000000000000000000000000101 - b //00000000000000000000000000000111 - a | b//对应二进制位有1就是1,全为0才得0int e = a | b;printf("%d\n", e);//7return 0;}
赋值操作符没什么好讲的,也就把a=a+1简写成了a+=1这样差不多的操作符。
接下来是单目操作符
首当其冲是逻辑反操作!,这就涉及到了真假的问题。C语言中规定非0就是真,只有0是假。所以!上一个任意不为零的数都是0,但!0呢,规定!0就为1。
sizeof是个操作符,这也是很多人会忽略的一点。
按位取反~,经过上面移位和位操作符的讲解,应该不难得出按位取反的含义,就是把二进制位列出来然后1变0,0变1。
当然不止上面这么简单,正数是这样,那负数呢?
首先负整数时有符号的整数,二进制位最高位如果是0,则该数为正数,如果时1,则为负数。
其实计算机在内存中存储整数的时侯呀,存储的是二进制,这大家都知道。然而一个整数的二进制的表示形式有三种分别是 原码,反码,补码
如果是正数,那么原码反码补码相同,那如果是负数呢,它的原反补是需要计算的。
其次我们应该知道原反补是如何计算的。反码,原码符号位不变,其他位按位取反。补码,反码+1。
但是最重要也是最容易让初学者混淆的一点是,计算机存储整数时,往内存里存的是补码,而不是大多数人想象的原码,这就需要我们反过来计算原反补了。
可以以0为例,0的二进制位全为0,所以可以看成是个正数,所以其原反补相同。
如果我们想知道~0(对0按位取反)是个什么结果的话,
先对0的补码按位取反的~0的补码,
再反过来计算,补码-1得反码,
然后再符号位不变,其他按位取反得其原码,
这样就得到了~0的原码,就可计算其十进制数了
由此可得的重要结论,对一个数按位取反,反的是二进制位的补码!
int main(){//整数在内存中存储的时候,存储的是二进制//一个整数的二进制表示有3种形式:原码,反码,补码//正整数:原码,反码,补码相同;//负整数:原码,反码,补码需计算;//有符号的整数,最高位是0,表示为正,// 最高位是1,表示为负;//eg: // int a = 1;//00000000000000000000000000000001 - 原码//00000000000000000000000000000001 - 反码//00000000000000000000000000000001 - 补码// int a = -1;//10000000000000000000000000000001 - 原码//11111111111111111111111111111110 - 反码 - 符号位不变,其他位按位取反//11111111111111111111111111111111 - 补码 - 反码+1//内存存储整数时,存储的是二进制的补码//计算时也是从补码开始计算//~按(二进制)位取反int b = 0;printf("%d\n", ~b);//-1//00000000000000000000000000000000 - 0的补码//11111111111111111111111111111111 - ~0的补码//11111111111111111111111111111110 - ~0的反码//10000000000000000000000000000001 - ~0的原码//故由0的补码得~0的补码,补码-1得反码,再(符号位不变)按位取反得原码;//最后原码代表的就是结果的二进制序列;return 0;}
操作符++,--,值得一提。很多学校喜欢考各种各样的奇葩++--题,不同的编译器得到的结果可能不同,所以那就是道错题。
在真正编程时,使用++,--就老老实实使用。别搞别人看不懂的那一套,没意思的,实力不是靠那个体现出来的。就分为前置和后置,也就是先++,在使用,还是先使用,再++的区别。
int main(){int a = 2;//a = a + 1;//a += 1;//printf("%d\n", a);//前置++;后置++;int c = ++a;//前置++;先++,后使用printf("++a=%d\n", c);//3printf(" a=%d\n", a);//3a = 2;int d = a++;//后置++;先使用,后++printf("a++=%d\n", d);//2printf(" a=%d\n", a);//3a = 2;int e = --a;//前置--;先--,后使用printf("--a=%d\n", e);//1printf(" a=%d\n", a);//1a = 2;int f = a--;//后置--;先使用,后--printf("a--=%d\n", f);//2printf(" a=%d\n", a);//1return 0;}
取地址操作符和解引用操作符是一对,放在指针部分再讲。
关系操作符没什么好讲的,和数学上一个含义。
逻辑操作符嘛,就如同数学里的逻辑真假一样,逻辑与和逻辑或,分别是并且和或者的关系,他们两边分别是两个条件,如果两个都为真,逻辑与表达式就为真。如果两个有一个真的,逻辑或表达式就是真。
int main(){//逻辑与 - &&(并且)//逻辑或 - ||(或者)int a = 1;int b = 4;if ((a = 1) && (b = 4)){printf("&&\n");}if ((a = 3) || (b = 4)){printf("||\n");}return 0;}
条件操作符,x?y1:2,?的两边分别是两个条件,前面的成立则整个表达式的值为1,反正则2。
逗号表达式,( , , ,) ,如这样的一个例子,每一个表达式都是一个算式,从左向右依次计算,最后一个表达式的结果作为整个逗号表达式的值。
//条件操作符(三目操作符)int main(){int a = 10;int b = 0;//if (a == 5)//{//b = -6;//}//else//{//b = 6;//}b = (a = 5) ? (6) : (-6);printf("%d\n", b);return 0;}//逗号表达式//( , , ... , );//表达式从左向右计算,整个表达式的结果为最后一个表达式的值int main(){int a = 0;int b = 3;int c = -1;int d = (a = b - 5, b = a + c, c = a + b, c -= 5);printf("%d\n", d);return 0;}
OK,关于操作符的内容就先介绍到这儿。
下面是关键字的内容。
上图为常见关键字的思维导图,接下来请随我一同探讨。
这些是C语言里常见的各种关键词,下面将对其部分稍作讨论。
首先是typedef,翻译来就是类型重命名。顾名思义,就是将定义变量的类型的名字如int,char等,重新取个名字代替。当然一般用于非常长的类型名如unsigned int这样,或者是结构体类型。 如此之后就可以把unsigned int 改写成 unint了。
//typedef 变量类型重命名typedef unsigned int unint;
然后是extern,翻译过来就是外部的意思,故用于声明引用外部(其他.c文件)的文件或者是函数等,但由于函数自带外部链接属性,所以一般不用于声明外部函数。用法如extern int g_val;(g_val是外部的变量)
紧接着是static,它分别有修饰局部变量,修饰全局变量和修饰函数三种不同的用法,但个人认为修饰全局变量和函数的意义相同。
修饰局部变量时,使其出作用域不会被销毁,准确的来说就是延长了他的生命周期,但不影响作用域。
修饰全局变量和函数时,会使其外部链接属性失效,也是就使其不可以再其他源文件中被使用。
void test(){//修饰局部变量//改变其生命周期,不影响作用域static int a = 1;a++;printf("%d ", a);}int main() {int i = 0;while (i < 10){test();i++;}//static修饰全局变量printf("%d\n", g_val);int a = 10;int b = 20;//static修饰的函数int c = Add(a, b);printf("%d\n", c);return 0;}
//static修饰全局变量//使其不可跨文件使用(外部链接属性失效)static int g_val = 2021;//static修饰的函数//函数被static修饰,使其外部链接属性变内部链接属性static int Add(int x, int y){return x + y;}
其它如,auto,Goto,reGISter,uNIOn,稍微了解一下。
定义常量时,也是非常简单,例如:#define N 10; 就定义了一个不可被修改的常量其值为10。
#define MAX(x,y) (x>y?x:y)
类似于函数,但又有别于函数,MAX(x,y)是宏,(x>y?x:y) 是宏体。MAX(x,y),像是函数名和传参放在一块,()就像是函数内容。
//定义常量#define NUM 100//定义宏#define MAX(X,Y) (X>Y?X:Y)int main(){printf("%d\n", NUM);int a = 0;int b = 10;int c = MAX(a, b);//实际操作,替换宏体//int c = (a > b ? a : b);printf("%d\n", c);return 0;}
指针一直是我之前自学的时候最害怕的内容,但这次初识C语言让我消除了对指针的恐惧,一步步的了解指针。
指针嘛,指向变量的内存地址,故讲指针之前必须把内存搞清楚。
为了可以有效使用内存,我们把内存划分了一个个小的内存单元,每个内存单元的大小为1byte。
我们需对内存进行编号,当然需要二进制位序列表示(默认我们是32位机器)。
每个二进制序列有32个bit,从数学全排列角度看,一共有2的32次方种排列可能(32个全0到32个全1)。
所以若想对其进行编号,不如一人一个码(一个内存单元用一个二进制序列表示)。并且我们把这些编号成为地址。
当然,二进制也可以转化为十进制或者十六进制,所以我们再调试时调用内存会看到自动显示为十进制数字。
我们了解了内存,现在我们看看如何取出内存的地址
int main(){ int num = 1;//先创建一个变量 #//然后取出它的地址 printf("%p",&num);//最后以%p的形式打印地址return 0;}
这样我们就得到了num的地址,以十六进制数字展示。
我们讲清楚了内存,现在再来看看指针。
我们既然已经得到了地址,那我们如何去储存这些东西呢,这是程序员们就想到了一种东西叫指针变量,它用于存放地址。
关于该(指针)变量如何定义看下列代码。
#include <stdio.h>int main(){ int num = 10; int *p = # *p = 20;return 0;}
上述代码中我们可以看到,指针变量的类型时 int * 。而有了指针变量后,在其前面加上*有个可以改变原变量的值。当然之所以是int*而不是char*,是因为原变量是int型的。
这里我们介绍一下,两个操作符分别是&取地址操作符和*解引用操作符。
&+变量名 可以取出变量的地址。
*+指针变量名 就可以把它当作原变量使用,通过这样就可以进行改变原变量的这样一系列的操作
可以说 pa = &a , *pa = a。
那么我们既然知道了有种变量叫指针变量,那么他们的类型大小是多少呢?
答案是每种指针变量类型大小都为4个字节(32位机器),因为指针变量存放地址,地址为二进制序列,32个bit,正好占4个byte。当然64位机器就是8个字节。
结构体的出现使得C语言具有了描述复杂类型的能力。
C语言的类型int,char,float等可以描述很多东西,但是这毕竟太单一,使用结构体可以描述更复杂的对象。
比如最经典的例子,如学生,书籍等。
描述学生的信息有名字,性别,年龄,学号等,下面且与我一同欣赏如何定义学生结构体。(记得大括号后面有个分号,vs2019自动带上)
struct stu{char name[20];//姓名int age;//年龄char sex[5];//性别char id[20];//学号};
或者是针对书籍的描述,有书名,价格,作者名等
struct book{char name[20];int price;char author[20];};
注意,struct stu 这一整个相当于 int float double 。
这样我们就完成了结构体变量类型的定义。
下面我们定义一个个的学生(结构体)变量。
//创建结构体变量struct stu s1 = {"芜湖大司马",40,"男","2020313222"};struct stu s2 = {"lisa",22,"女","2020313232"};struct book b1 = {"C语言详解",55,"谭浩强"};
创建好了我们如何去使用呢?最简单的输出方式,使用操作符 .
形式上是 结构体变量.成员名。
//输出1printf("name: %s,age: %d,sex: %s,id: %s\n", s1.name, s1.age, s1.sex, s1.id);printf("书名: %s,价格: %d,作者: %s\n", b1.name, b1.price, b1.author);
既然这样可以的话,我们还可以定义指针变量代替变量名,用(*pb)代替b1。
struct book * pb = &b1; //先定义一下指针变量printf("%s %d %s\n", (*pb).name, (*pb).price, (*pb).author); //指针变量解引用,就可以当作原变量使用
当然有更方便的操作符 ->,这样我们可以直接使用指针啦,如结构体指针->成员名。
struct book * pb = &b1;//别忘了定义指针printf("%s %d %s\n", pb->name, pb->price, pb->author); //指针变量名直接加 —> 再加成员名
感谢各位的阅读!关于“C语言入门基础之操作符的示例分析”这篇文章就分享到这里了,希望以上内容可以对大家有一定的帮助,让大家可以学到更多知识,如果觉得文章不错,可以把它分享出去让更多的人看到吧!
--结束END--
本文标题: C语言入门基础之操作符的示例分析
本文链接: https://lsjlt.com/news/299589.html(转载时请注明来源链接)
有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341
2024-03-01
2024-03-01
2024-02-29
2024-02-29
2024-02-29
2024-02-29
2024-02-29
2024-02-29
2024-02-29
2024-02-29
回答
回答
回答
回答
回答
回答
回答
回答
回答
回答
0