Linux第二课:LinuxC高级 学习记录day04

6、shell中的语句

6.3、结构性语句

6.3.1、if

if...then...fi

1、结构

1)基本结构

if 表达式

then

命令表

fi
if [ 表达式 ] // 【】两侧有空格

then

命令表

fi

2)分层结构

if 表达式

then

命令表1

else

命令表2

fi

3)嵌套结构

if 表达式

then

if

then

命令表1

else

命令表2

fi

else

if

then

命令表3

else

命令表4

fi

fi

4)elif

if 表达式1

then

命令表1

elif 表达式

then

命令表2

......

else

命令表n

fi

6.3.2、case语句

结构

case 变量 in // 取值后面必须为关键字 in,取值可以为变量或者常量

模式1) // 每个模式必须以右括号结束

命令表1 // 一旦模式匹配,其间所有命令开始执行,直到 ;; 结束

;; // 执行完匹配模式相应命令后,不hi再继续匹配其他格式

模式2) // 取值检测匹配的每一个模式

命令表2

;;

...

*) // 如果没有模式匹配成功,使用 * 捕获该变量

命令表 n

;;

esac


6.3.3、for循环

结构:

for 变量名 in 单词表 // 变量依次取单词表中的各单词

do // 每取一次单词,就执行循环体中的命令

命令表 // 可以是一条,也可以是换行符分开的多条

done // 循环次数由单词表中的单词数确定

书写格式:

1)变量 I 从单词表中取值

2)变量I 从命令行取值,省略 in 和单词表

3)变量i在单词表内通过通配符取值


4)书写格式类似C语言 // 两个空格里的表达式两端用空格与括号隔开

5)输出当前目录下的所有文件名字

6.3.4、while循环

结构

while 命令表达式 // 首先测试其后的命令或表达式的值

do // 为真进入循环

命令表

done // 为假退出循环

6.3.5、循环控制语句

break:

break // 结束本层循环

break n // 结束 n 层循环,内层往外数 n 层,都结束

continue:

continue // 跳过本层本次循环

continue n // 跳过 n 层本次循环,内层往外数 n 层,这 n 层本次循环都跳过

7、分文件

整个工程包含多个.c和多个.h文件

7.1、分文件包括

1、main.c

main() // 这些文件统一由 main 函数调用

2、xxx.c

多个.c // 不同功能的函数接口

3、.h 头文件包含:

1、包含其他的头文件

2、函数声明

3、构造数据类型,可以定义类型之后定义变量,但是不能同时定义

4、宏定义

5、重定义的名字

6、全局变量

7.2、分文件使用

include 引用时 " " 和 < > 的区别

<stdio.h> // 系统定义的头文件:去系统目录下查找头文件

"add.h" // 自定义头文件:先从当下目录下查找,如果没有,再去系统目录下查找头文件

7.3、条件编译

编译器根据条件的真假,决定是否编译相关代码

1、根据宏是否定义

格式

#ifdef 宏名

代码块1 // 宏定义,代码块1被编译

#else

代码块2 // 宏没有被定义,代码块2被编译

#endif

cs 复制代码
#include <stdio.h>
#define EDF

int main(int argc, char const *argv[])
{
    #ifdef EDF
        printf("yes\n");
    #else
        printf("NO\n");
    #endif
    return 0;
}
2、根据宏的值是否为真

格式:

#if 宏名

代码块1 // 宏的值为真,非0,代码块1被编译

#else

代码块2 // 宏的值为假,0,代码块2被编译

#endif

cs 复制代码
#include <stdio.h>
#define EDF 1

int main(int argc, char const *argv[])
{
    #if EDF
        printf("yes\n");
    #else
        printf("NO\n");
    #endif
    return 0;
}
3、根据宏是否定义

防止头文件重复包含

结构:

#ifndef 宏名 // 第一次展开,询问是否有宏,没有编译下列代码块,

// 第二次展开,询问是否有宏,第一次展开的时候定义了宏,

所以这一次有有宏,下面的代码块不进行编译

#define 宏名 // 定义一个同名宏,然后继续执行后面的代码块,

......

#endif

8、make工具

8.0、gcc 编译步骤

预处理:

处理以#开头的内容,展开头文件,替换宏定义,删除注释,不进行语法检查

gcc -E xxx.c -o xxx.i

编译:

检查语法错误,有错报错,没错生成汇编文件

gcc -S xxx.i -o xxx.s

汇编:

生成不可执行的二进制文件

gcc -c xxx.s -o xxx.o

链接:

链接库文件,生成可执行的二进制文件

gcc xxx.o -o xxx

8.1、make工具的内容

1、make工具又叫工程管理器,作用是在文件过多时管理文件

// 能根据文件时间戳,发现更新过的文件而减少编译工作量的同时,它通过读入文件的内容,来执行大量的编译工作

2、Makefile 是 make 读入的唯一配置文件

// Makefile 工程文本文件

8.2、Makefile 书写格式

8.2.1、基础书写方式

目标(可执行文件): 依赖文件(二进制文件)

<tab>命令 // gcc xxx.o -o xxx

目标(二进制文件): 依赖文件(.c文件)

<tab>命令 // gcc -c xxx.c -o xxx.o

.PHONY: clean // 目的不是创建clean文件,而是执行下列命令

目标(clean): // 删除 .o 文件和可执行文件

<tab>伪命令 // rm -rf *.o main

示例:写一个makefile文件

main: main.o fun.o
gcc main.o fun.o -o main

main.o: main.c

gcc -c main.c -o main.o

fun.o: fun.c

gcc -c fun.c -o fun.o

.PHONY: clean

clean:

rm -rf *.o main

执行:
make // 执行前面的编译过程

make clean // 执行删除命令

8.2.2引入自定义变量和预定义变量

通过定义变量来重新书写makefile文件

自定义变量

一般用大写表示变量名,取变量值用$(变量名)

= // 递归赋值(以最后一次赋值为准),找到最后一次赋值之后,前面所有该变量全都变成最后一次赋值的值

:= // 立即赋值(当前是什么就立即是什么值),以当前值为准,直接确定

+= // 追加赋值(追加新的值),以字符串的形式追加,追加前自动增加分隔符

?= // 条件赋值(判断之前是否定义,如果定义,不重新赋值,否则赋值),前面有值了,就保持,前面没有值就赋值

预定义变量

系统预先定义好的一些变量,可能有默认值,可能没有

RM // 文件删除程序的名称,默认值为rm -f

CC // C编译器的名称,默认值是cc

CPP // C预编译器的名称,默认值是$(CC) -E

CFLAGS // C编译器的选项,无默认值

// -g 调试 -Wall 加警告

OBJS // 生成的二进制文件或者目标文件,自己定义的

makefile文件书写方式2

EXE=main # 保存可执行文件

OBJS=main.o fun.o

CC=gcc

CFLAGS=-c -g -Wall # -g 调试 -Wall:警告

(EXE): (OBJS)

(CC) (CFLAGS) -o $(EXE)

main.o: main.c

(CC) (CFLAGS) main.c -o main.o

fun.o: fun.c

(CC) (CFLAGS) fun.c -o fun.o

.PHONY: clean

clean:

(RM) (OBJS) $(EXE)

8.2.3、引入自动变量和通配符修改makefile

自动变量

$< // 第一个依赖文件

$@ // 目标文件的完整名称

$^ // 所有不重复的依赖文件,以空格分开

通配符

% // 字符串的匹配模式,在Makefile中作用是,获取一个.o 文件,然后匹配一个同名.c

// %.o: %.c

makefile文件书写方式3

EXE=main # 保存可执行文件
OBJS=main.o fun.o

CC=gcc

CFLAGS=-c -g -Wall # -g调试 -Wall:警告

(EXE): (OBJS)

(CC) ^ -o $@

%.o: %.c

(CC) (CFLAGS) \< -o @

.PHONY: clean

clean:

(RM) (OBJS) $(EXE)

8.2.4、引入函数

1、要求:

make 中提供了内置函数 // 内置函数是帮助程序员查找文件信息的,所以要求在查找目录下,只有程序需要的 .c 文件,没有其他程序的 .c 文件

2、wildcard

功能:

根据给的条件,获取指定的文件名(找文件)

命令:

$(wildcard 指定字符串的格式)

$(wildcard *.c) // 找到当前路径下,所有的 .c 文件的文件名

3、patsubst

功能:

模式匹配替换字符串

命令:

$(patsubst 原格式,目标格式,要替换的字符串)

$(patsubst %.c,%.o,main.c fun.c) // 获取到main.c fun.c 字符串,根据模式匹配,得到main.o fun.o 字符串每个参数之间以','作为分隔,要替换的字符串之间以空格作为分隔

makefile文件书写方式4

EXE=main # 保存可执行文件
FILES=$(wildcard *.c)

OBJS=(patsubst %.c,%.o,(FILES))

CC=gcc

CFLAGS=-c -g -Wall # -g调试 -Wall:警告

(EXE): (OBJS)

(CC) ^ -o $@

%.o: %.c

(CC) (CFLAGS) \< -o @

.PHONY: clean

clean:

(RM) (OBJS) $(EXE)

相关推荐
满怀101530 分钟前
Python扩展知识详解:lambda函数
开发语言·python
佚名涙1 小时前
go中锁的入门到进阶使用
开发语言·后端·golang
猫猫的小茶馆1 小时前
【PCB工艺】软件是如何控制硬件的发展过程
开发语言·stm32·单片机·嵌入式硬件·mcu·51单片机·pcb工艺
勘察加熊人2 小时前
wpf+c#路径迷宫鼠标绘制
开发语言·c#·wpf
小黄人软件3 小时前
C# ini文件全自动界面配置:打开界面时读ini配置到界面各控件,界面上的控件根据ini文件内容自动生成,点保存时把界面各控件的值写到ini里。
开发语言·c#
多多*5 小时前
Java设计模式 简单工厂模式 工厂方法模式 抽象工厂模式 模版工厂模式 模式对比
java·linux·运维·服务器·stm32·单片机·嵌入式硬件
Android洋芋6 小时前
C语言深度解析:从零到系统级开发的完整指南
c语言·开发语言·stm32·条件语句·循环语句·结构体与联合体·指针基础
bjxiaxueliang6 小时前
一文详解QT环境搭建:Windows使用CLion配置QT开发环境
开发语言·windows·qt
白夜易寒6 小时前
Docker学习之私有仓库(day10)
学习·docker·容器
Run_Teenage7 小时前
C语言 【初始指针】【指针一】
c语言·开发语言