热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

C语言自学笔记

C语言自学笔记1.C语言编程预备知识2.基本的输入输出函数用法3.运算符4.流程控制什么是流程控制流程控制的分类顺序选择循环5.数组为什么需要数组数组的分类一维数组二维数组6.函数


C语言自学笔记

  • 1. C语言编程预备知识
  • 2. 基本的输入输出函数用法
  • 3. 运算符
  • 4. 流程控制
    • 什么是流程控制
    • 流程控制的分类
      • 顺序
      • 选择
      • 循环
  • 5. 数组
    • 为什么需要数组
    • 数组的分类
      • 一维数组
      • 二维数组
  • 6. 函数
  • 7. 变量
  • 8. 指针
  • 9. 结构体
  • 10. 枚举
  • 零散知识笔记




配合视频《郝斌C语言自学教程》


郝斌C语言自学教程程序源码

链接:https://pan.baidu.com/s/1fdUE77WRiqRQPzquDmFODg
提取码:obmv

1. C语言编程预备知识


  1. CPU,内存条,硬盘,显卡,主板,显示器,之间的关系
    例如要打开一部电影,鼠标双击电影,硬盘中的电影数据调入到内存条里面,CPU对内存条中的数据进行处理,如果是图像,就通过显卡在显示器上输出,如果是声音就通过声卡,发出声音;主板将这些模块连接在一起。
  2. Hello World 程序是如何运行起来的
  3. 什么是数据类型
    基本数据类型:
    整数:
      整型:int     4
      长整型:long int  8
      短整型:short int 2
    浮点数 [实数]
       单精度浮点数-- float–4
       双精度浮点数–double–8
    字符:
       char – 1
    复合数据类型:
        结构体
        枚举
        共用体
  4. 什么是变量
    变量的本质就是内存中一段存储空间
  5. 变量为什么必须的初始化
    所谓初始化就是赋值的意思
  6. 常量在C语言中是如何表示的
    在这里插入图片描述
  7. 什么是字节
    字节就是存储数据的单位,并且是硬件所能访问的最小单位
    1 字节 = 8位
    1 K = 1024字节
    1 M = 1024K
    1 G = 1024M

2. 基本的输入输出函数用法

printf() -----将变量的内容输出到显示器上

四种用法


  1. printf("字符串”);
  2. printf(" 输出控制符",输出参数) ;
  3. printf(“输出控制符1 输出控制符2。。。”,输出参数1,输出参数2, 。。。。);;
    输出控制符和输出参数的个数必须一 一对应
  4. printf (”输出控制符 非输出控制符",输出参数) ;
    在这里插入图片描述

为什么需要输出控制符


  1. 01 组成的代码可以表示数据也可以表示指令
  2. 如果01组成的代码表示的是数据的话,那么同样的01代码组合以不同的输出格式输出就会有不
    同的输出结果
    在这里插入图片描述

3. 运算符

在这里插入图片描述
在这里插入图片描述


4. 流程控制


什么是流程控制


  • 代码执行的顺序

流程控制的分类


顺序


选择

定义:某些代码可能执行,也可能不执行,有选择的执行某些代码

分类:

if 1. if 最简单的用法格式:if(表达式)语句功能:如果表达式为真,执行语句如果表达式为假,语句不执行2. if 的范围问题1.if (表达式)语句A; 语句B;解释: if默认只能控制语句A的执行或不执行if无法控制语句B的执行或不执行或者讲:语句B 一定会执行2.if (表达式){语句A;语句B;} 此时if可以控制语句A和语句B由此可见: if默认只能控制一个语句的执行或不执行,如果想控制多个语句的执行或不执行就必须把这些语句用 {} 括起来3. if...else...的用法4. if...else if...else... 的用法格式:if (1)A;else if (2)B;else if (3)C;else D;5. C语言对真假的处理非零是真.零就是假真用1表示假用零表示6. if 举例--求分数的等级7. if 的常见问题解析1> 空语句的问题if(3>2);等价于if(3>2); //这是一一个空语句2>if (表达式1)A;elseB;是正确的if (表达式1) ;A;elseB;是错误的3>if (表达式1)A;else if (表达式2)B;else if (表达式3)C;elseD;即便表达式12都成立,也只会执行A语句,因为如果 表达式1成立,就不会再执行表达式24>if (表达式1)A;else if (表达式2)B;else if (表达式3)C;这样写语法不会出错,但逻辑上有漏洞5>if (表达式1)A;else if (表达式2)B;else if (表达式3)C;else (表达式4) //7行D;这样写是不对的,正确的写法是:要么去掉7行的(表达式4),要么在7行的else后面加if6>if (表达式1)A;else if (表达式2)B;else if (表达式1)C;else(表达式4) ;D;这样写语法不会出错,但逻辑上是错误的else(表达式4) ;D;等价于else(表达式4) ;D;switch

循环

定义:某些代码会重复执行
分类:for1. 格式for(1;2;3)语句A;先执行 1 ,然后执行 22成立标志着此循环成立,开始执行循环,则执行 4,再执行 3, 3 执行完,标志着一次循环结束。是否需要执行下次循环,再次执行 2, 2 成立, 执行4,再执行33 结束,标志着第二次循环结束,(3 执行完标志着一次循环结束)以此类推,直到 2 不成立,则退出循环2. 执行的流程单个for循环的使用多个for循环的嵌套使用(1).for(1;2;3)for(4;5;6) 二A; 三B; 四整体是两个语句,一,二,三是一个语句,四是一个语句说明:先执行 1 ,再执行 22满足后执行:for(4;5;6)A;这个for循环结束之后,然后执行 3 ,一次大循环结束。 然后执行 2, 以此类推, 直到 2 不满足后,退出所有循环,执行 B

举例:

#include
int main()
{int i = 0;int j = 0;for(i = 0; i < 5; i++)for(j = 0; j < 5; j++)printf("第二个循环执行%d次\n",j); printf("**********************************************");return 0;}

输出结果:
在这里插入图片描述

(2).for(1;2;3)for(4;5;6){A;B;}整体是一个语句(3).for(7;8;9)for(1;2;3){A;B;for(4;5;6)C;}整体是一个语句3. 范围问题4. 举例:1 + 2 + 3 + ... + 100# include int main(void){int sum = 0;int i;for (i=1; i<101; ++i){sum = sum + i;}printf("sum = %d\n", sum);return 0;}1 + 1/2 + 1/3 + ... + 1/100# include int main(void){int i;float sum = 0;for (i=1; i<=100; ++i){sum = sum + 1.0/i; //是OK的 推荐使用//sum = sum + (float)(1/i); //这样写是不对的 //也可以这样写: sum = sum + 1 / (float)(i); //不推荐}printf("sum = %f\n", sum); //float必须用%f输出return 0;}

while

在这里插入图片描述

do...while

在这里插入图片描述
在这里插入图片描述

breakcontinuebreakbreak如果用于循环是用来终止循环break如果用于switch, 则是用于终止switchbreak不能直接用于if, 除非if属于循环内部的一个子句例子:for (i=0; i<3; ++i ){if(3>2)break; //break 虽然是if内部的语句,但break终止的确是外部的for循环printf("嘿嘿!\n"); //永远不会输出}在多层循环中,break 只能终止最里面包裹它的那个循环例子:for (i=0; i<3; ++i){for (j=1; j<4; ++j)break; //break 只能终止距离它最近的循环printf ("同志们好!\n");}在多层switch嵌套中,break只 能终止距离它最近的switch例子:int x=1, y=0, a=0, b=0;switch(x) //第一个switch{case 1:switch(y) // 第二个switch{case 0:a++;break; //终止的是第二个switchcase 1:b++;break;}b = 100;break; //终止的是第一个switchcase 2:a++;b++;break;}printf("%d %d\n" ,a, b); //26行最终输出结果是:1 100continue用于跳过本次循环余下的语句,转去判断是否需要执行下次循环例子:1. for(1;2;3).{A;B;continue; //如果执行该语句,则执行完该语句后,会执行语句3, C和D都会被跳过去, C// 和D不会被执行C;D;}2. while (表达式){A;B;continue; //如果执行该语句, 则执行完该语句后,//会执行表达式,C和D都会被跳过去, C和D不会被执行C;D;}

5. 数组


为什么需要数组


  • 为了解决大量同类型数据的存储和使用问题
  • 为了模拟现实世界

数组的分类


一维数组

怎样定义一维数组

为n个变量连续分配存储空间所有的变量数据类型必须相同所有变量所占的字节大小必须相等例子:int a[5];一维数组名不代表数组中所有的元素,一维数组名代表数组第-个元素的地址

有关一维数组的操作

初始化完全初始化int a[5] = {1,2,3,4,5};不完全初始化,未被初始化的元素自动为零int a[5] = {1,2, 3};不初始化,所有元素是垃圾值int a[5];清零int a[5] = {0};错误写法:int a[5];a[5] = {1,2,3,4,5}; //错误只有在定义数组的同时才可以整体赋值, 其他情况下整体赋值都是错误的int a[5] = {1,2,3,4, 5};a[5] = 100; //error 因为没有a[5]这个元素, 最大只有a[4]int a[5] = {1,2,3,4,5};int b[5];如果要把a数组中的值全部复制给b数组错误的写法:b = a;// error正确的写法for (i=0; i<5; ++i)b[i] = a[i];

二维数组

int a[3][4];
总共是12个元素,可以当做3行4列看待,这12个元素的名字依次是
a[0][0] a[0][1] a[0][2] a[0][3]
a[1][0] a[1][1] a[1][2] a[1][3]
a[2][0] a[2][1] a[2][2] a[2][3]
a[i][j]表示第i+1行第j+1列的元素
int a[m][n]; 该二维数组右下角位置的元素只能是a[m-1][n--1]初始化int a[3][4] = {1,2,3,4, 5, 6, 7,8, 9, 10,11,12}int a[3][4] = {{1,2,3,4},{5,6,7,8},{9,10,11,12}};操作输出二维数组内容# include int main(void){int a[3][4] = {{1, 2, 3, 4},{5, 6, 7, 8},{9, 10,11,12}};int i, j;//输出数组内容for (i=0; i<3; ++i){ for (j=0; j<4; ++j)printf("%d ", a[i][j]);printf("\n");}return 0;}

多维数组
是否存在多维数组
不存在
因为内存是线性一维的
n维数组可以当做每个元素是n-1维数组的一维数组
比如:
int a[3][4];
该数组是含有3个元素的一维数组
只不过每个元素都可以再分成4个小元素
int a[3][4][5];
该数组是含有3个元素的一维数组
只不过每个元素都是4行5列的二维数组


6. 函数


  • 为什么需要函数
    避免了重复性操作
    有利于程序的模块化
  • 什么叫函数

逻辑上: 能够完成特定功能的独立的代码块
物理上:能够接收数据[当然也可以不接受数据]能够对接受的数据进行处理

  • 如何定义函数
    函数的返回值函数的名字(函数的形参列表)
    {
      函数的执行体
    }
  • 函数定义的本质是详细描述函数之所以能够实现某个特定功能的具体方法
  • return 表达式的含义:

1>终止被调函数,向主调函数返回表达式的值2>如果表达式为空,则只终止函数,不向主调函数返回任何值3> break是用来终止循环和switch的,return是用来终止函数的
例子:void f(){return; //return只用来终止函数,不向主调函数返回任何值}int f()return 10; //第一: 终止函数, 第二: 向主调函数返回10}

  • 函数返回值的类型也称为函数的类型,因为如果函数名前的返回值类型和函数执行体中的return 表达式中表达式的类型不同的话,则最终函数返回值的类型以函数名前的返回值类型为准

例子:int f ()return 10. 5; //因为 函数的返回值类型是 int//所以最终 f 返回的是10而不是10.5

  • 函数的分类
    有参函数和无参函数
    有返回值函数和无返回值函数
    库函数和用户自定函数
    值传递函数和地址传递函数
    普通函数和主函数(main函数)
    一个程序必须有且只能有一一个主函数
    主函数可以调用普通函数普 通函数不能调用主函数
    普通函数可以相互调用
    主函数是程序的入口,也是程序的出口

  • 注意的问题
    函数调用和函数定义的顺序
    如果函数调用写在了函数定义的前面,则必须加函数前置声明

函数前置声明:


  1. 告诉编译器即将可能出现的若干个字母代表的是一个函数
  2. 告诉编译器即将可能出现的若干个字母所代表的函数的形参和返回值的具体情况
  3. 函数声明是一个语句,末尾必须加分号
  4. 对库函数的声明是通过# include <库函数所在的文件的名字. h>来实现的

形参和实参
个数相同
位置一 一对应,数据类型必须相互兼容


7. 变量

按作用域分:全局变量在所有函数外部定义的变量叫全局变量全局变量使用范围: 从定义位置开始到整个程序结束局部变量在一个函数内部定义的变量或者函数的形参都统称为局部变量void f(int i){int j=20;}i和j都属于局部变量局部变量使用范围: 只能在本函数内部使用注意的问题:全局变量和局部变量命名冲突的问题在一个函数内部如果定义的局部变量的名字和全局变量名一样时, 局部变量会屏蔽掉全局变量
按变量的存储方式静态变量自动变量寄存器变量

8. 指针

(见另一篇专门讲指针的文章)
链接: C语言指针详解.


9. 结构体


  • 为什么需要结构体
    为了表示一些复杂的事物,而普通的基本类型无法满足实际要求
  • 什么叫结构体
    把一些基本类型数据组合在一起形成的一个新的复合数据类型,这个叫做结构体
  • 如何定义结构体
    3种方式,推荐使用第一种:

第一种这只是定义了一一个新的数据类型,并没有定义变量
struct Student
{int age;float score;char sex;
};
第二种
struct Student2
{int age;float score;char sex;
}st2 ;
第三种
struct
{int age;float score;char sex ;
} st3;

  • 怎样使用结构体变量
    赋值和初始化
    定义的同时可以整体赋初值
    如果定义完之后,则只能单个的赋初值
  • 如何取出结构体变量中的每一个成员

1. 结构体变量名.成员名
2. 指针变量名->成员名(第二种方式更常用)指针变量名->成员名在计算机内部 会被转化成 (*指针变量名).成员名 的方式来执行所以说这两种方式是等价的
例子:struct Student{int age;float score;char sex;};int main(void){struct Student st = {80, 66.6, &#39;F&#39;}; // 初始化,定义的同时赋初值struct Student * pst = &st; // &st不能改成stpst->age = 88; // 第二种方式st.age = 10; // 第一种方式return 0;}
1.pst->age在计算机内部会被转化成(*pst).age
2.所以pst >age等价于(*pst).age 也等价于st.age
3.我们之所以知道pst->age 等价于st.age, 是因为pst->age 是被转化成了(*pst).age 来执行
4.pst->age的含义:pst 所指向的那个结构体变量中的age 这个成员

结构体变量和结构体指针变量作为函数参数传递的问题
推荐使用结构体指针变量作为函数参数来传递

结构体变量的运算
结构体变量不能相加,不能想减,也不能相互乘除
但结构体变量可以相互赋值

例子:
struct Student
{int age;char sex;char name[100];
}; //分号不能省
struct Student st1, st2;
st1+st2 st1*st2 st1/st2 都是错误的
st1 = st2 或者st2 = st1 都是正确的

10. 枚举

什么是枚举
把一个事物所有可能的取值一 一列举出来

# include
//只定义了一个数据类型,并没有定义变量, 该数据类型的名字是 enum WeekDay
enum WeekDay
{MonDay, TuesDay, WednesDay, ThursDay, FriDay, SaturDay, SunDay
};int main(void)
{//int day; //day定义成int类型不合适enum WeekDay day = SunDay;printf("%d\n", day);return 0;
}

怎样使用枚举

枚举的优缺点
代码更安全
书写麻烦


零散知识笔记

算法: 解题的方法和步骤
如何看懂一个程序,分三步:


  • 流程
  • 每个语句的功能
  • 试数

如何学习一些需要算法的程序【如何掌握一个程序】


  1. 尝试自己去编程解决它
    但要意识到大部分人都是自己无法解决的,这是不要气馁,也不要自卑,如果十五分钟还想不出来,就可以看答案
  2. 如果解决不了,就看答案
    关键是把答案看懂,这个要花很大的精力,也是学习的重点。
  3. 看懂之后尝试自己去修改程序,并且知道修改之后程序的输出结果的含义,
    不建议看懂程序之后就立即自己敲程序
  4. 照着答案去敲
  5. 调试错误
  6. 不看答案,自己独立把答案敲出来
  7. 如果程序实在无法彻底理解,就把它背会,不过无法彻底理解的程序非常少,

强制类型转化


  • 格式:(数据类型)(表达式)
  • 功能:把表达式的值强制转化为前面所执行的数据类型
  • 例子:

(int)(4.5 + 2.2) 最终值是 6
(float)(5) 最终值是 5.000000

浮点数的存错所带来的问题


  • float 和 double 都不能保证可以把所有的实数都准确的保存在计算机中
    例子
    在这里插入图片描述
    因为浮点数无法准确存储,所以就衍生出来两个编程问题:

有一个浮点型变量 x,如何判断 x 的值是否是零if (|x-0.000001| <= 0. 000001)是零else不是零循环中更新的变量不能定义成浮点型

一些琐碎的运算符知识

1. 自增【或者自减】分类:前自增: ++i ;后自增: i++ ;前自增和后自增的异同:相同:最终都使 i 的值加1不同前自增整体表达式的值是i加1之后的值后自增整体表达式的值是i加1之前的值# include int main(void){int i;int j;int k;int m;i = j = 3; //等价于 i = 3; j = 3;k = i++;m = ++j;printf("i = %d, j = %d, k = %d, m = %d\n", i, j, k, m);return 0;}/*在Vc++6.0中的输出结果是:----------------------i = 4, j = 4, k = 3, m = 4----------------------总结:前自增整体表达式的值是i加1之后的值后自增整体表达式的值是i加1之前的值*/

为什么会出现自增?代码更精练自增的速度更快

在这里插入图片描述


  1. 三目运算符
    在这里插入图片描述
  2. 逗号表达式
    格式:
    在这里插入图片描述

推荐阅读
  • 本文档旨在提供C语言的基础知识概述,涵盖常量、变量、数据类型、控制结构及函数定义等内容。特别强调了常量的不同类型及其在程序中的应用,以及如何正确声明和使用函数。 ... [详细]
  • 本文探讨了如何利用数组来构建二叉树,并介绍了通过队列实现的二叉树层次遍历方法。通过具体的C++代码示例,详细说明了构建及打印二叉树的过程。 ... [详细]
  • 本文详细探讨了如何处理包含多种分隔符的字符串分割问题,并提供了一个高效的C++实现方案。 ... [详细]
  • 深入解析mt_allocator内存分配器(二):多线程与单线程场景下的实现
    本文详细介绍了mt_allocator内存分配器在多线程和单线程环境下的实现机制。该分配器以2的幂次方字节为单位分配内存,支持灵活的配置和高效的性能。文章分为内存池特性描述、内存池实现、单线程内存池实现、内存池策略类实现及多线程内存池实现等部分,深入探讨了内存池的初始化、内存分配与回收的具体实现。 ... [详细]
  • 本文详细解析 Skynet 的启动流程,包括配置文件的读取、环境变量的设置、主要线程的启动(如 timer、socket、monitor 和 worker 线程),以及消息队列的实现机制。 ... [详细]
  • 深入解析轻量级数据库 SQL Server Express LocalDB
    本文详细介绍了 SQL Server Express LocalDB,这是一种轻量级的本地 T-SQL 数据库解决方案,特别适合开发环境使用。文章还探讨了 LocalDB 与其他轻量级数据库的对比,并提供了安装和连接 LocalDB 的步骤。 ... [详细]
  • 本文探讨了Java中有效停止线程的多种方法,包括使用标志位、中断机制及处理阻塞I/O操作等,旨在帮助开发者避免使用已废弃的危险方法,确保线程安全和程序稳定性。 ... [详细]
  • 2022年4月15日的算法练习题,包括最长公共子序列和线段树的应用。 ... [详细]
  • 本文详细介绍了Oracle RMAN中的增量备份机制,重点解析了差异增量和累积增量备份的概念及其在不同Oracle版本中的实现。通过对比两种备份方式的特点,帮助读者选择合适的备份策略。 ... [详细]
  • 离散数学与C语言在逻辑推理中的应用
    本文探讨了如何结合离散数学的知识和C语言编程技巧,解决具体的逻辑推理问题。通过实例分析,展示了从问题抽象到程序实现的完整过程。 ... [详细]
  • 本文介绍了一道来自LeetCode的编程题——拼写单词。题目要求从给定的词汇表中找出可以由指定字母表中的字母拼写出的单词,并计算这些单词的总长度。文章将展示如何通过使用数组替代哈希表来提高算法的执行效率。 ... [详细]
  • 本文介绍了进程的基本概念及其在操作系统中的重要性,探讨了进程与程序的区别,以及如何通过多进程实现并发和并行。文章还详细讲解了Python中的multiprocessing模块,包括Process类的使用方法、进程间的同步与异步调用、阻塞与非阻塞操作,并通过实例演示了进程池的应用。 ... [详细]
  • 深入解析C++ Atomic编程中的内存顺序
    在多线程环境中,为了防止多个线程同时修改同一数据导致的竞争条件,通常会使用内核级同步对象,如事件、互斥锁和信号量等。然而,这些方法往往伴随着高昂的上下文切换成本。本文将探讨如何利用C++11中的原子操作和内存顺序来优化多线程编程,减少不必要的开销。 ... [详细]
  • APP及其接口测试全面解析
    本文深入探讨了移动应用(APP)及其接口测试的关键点,包括安装与卸载、功能一致性、系统兼容性、权限管理等多个方面的测试策略,以及针对接口的功能、边界值、参数组合等专业测试方法。同时,介绍了几款常用的测试工具,帮助开发者提高测试效率和质量。 ... [详细]
  • JavaSE 基础语法详解
    本文详细介绍了JavaSE的基础语法,涵盖数据类型、变量与常量、流程控制语句及数组等内容,旨在为初学者提供全面的学习指南。 ... [详细]
author-avatar
mobiledu2502880273
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有