Linux C语言编程基本原理与实践(2018-06-16 19:12:15)linux
高效的学习带着目的性: 是什么 -> 干什么 -> 怎么用nginx
nginx:c apache:c++
linux嵌入式c++
Ubuntu:apache
PS: 尽可能在Linux环境下开发C语言程序编程
sudo apt install 【软件名】
sudo apt update
Ctrl+Alt+T
:打开终端cd ~
:进入当前用户的根目录pwd
:查看当前所在路径ls
:当前章目录包含哪些文件ls -l
:显示当前文件的类型,权限,建立时间,名字ls -al
或ll
:显示隐藏文件若是前面是d
就是文件夹,-
就是普通类型的文件
touch **
:建立字符型文件rm **
:删除mkdir **
:建立目录(文件夹)vi **
:打开(进入)文件vi 一个不存在的文件,进入后没法输入内容,因当前在命令模式下;按字符i,可进入INSERT插入模式,就可输入内容,按Esc返回命令模式;
在命令模式下:ubuntu
:w
:保存该文件:q
:退出i
:当前光标前面插入字符Shift+i
:跳到本行行首插入字符a
:当前光标后面插入字符Shift+a
:跳到本行末尾插入字符o
:在当前下一行插入字符Shift+o
:在当前上一行插入字符x
:删除当前光标所处字符d+d
:删除整行Linux下最好用的文本编辑器: emacs, vim小程序
cc -v(gcc -v)
: 查看编译器版本clear
:清洁屏幕linux下通常不用void main
,最新c语言标准,int main
vim
#include <stdio.h> int main() { printf("hello,world!\n"); return 0; }
cc a.c
默认会为咱们编译并生成可执行文件a.out(可读可写可执行)数组
./
表示当前路径下,./a.out
执行当前路径下的a.out文件r
表示可读 w
表示可写 x
表示可执行三组重复的顺序为"建立者","用户组","任意其余用户"并发
c语言是一个结构化的程序语言,是支持多函数的。程序可由若干个函数组成。
vim hello.c
最原始版本的实现(hello.c):
#include <stdio.h> int max(int a, int b) { if(a>b){ return a; }else{ return b; } } int main() { int a1 = 33; int a2 = 21; int maxNum = max(a1,a2); printf("the max value is %d\n",maxNum); return 0; }
附加知识: vim分屏显示
:sp 文件名
//建立(打开)新文件ctrl+w+上箭头
ctrl+w+下箭头
:set nu
//这两个不用点冒号
若是就是上图代码直接编译会报错,这是一个未声明的函数。有两种分离方案:
int max(int a,int b);
,在hello.c中声明该方法,而后编译的时候须要加上max.c#include "max.c"
而后编译的时候就不须要加上max.c一块儿编译版本1:
0-hello.c:
#include <stdio.h> int max(int a,int b); int main() { int a1 = 33; int a2 = 21; int maxNum = max(a1,a2); printf("the max value is %d\n",maxNum); return 0; }
0-max.c:
int max (int a, int b) { if(a>b){ return a; }else{ return b; } }
编译命令:
gcc 0-hello.c 0-max.c -o 0-hello.out
若是不加上0-max.c一块儿编译,会出现错误
gcc 0-hello.c -o 0-hello.out /tmp/cc8GuaAH.o:在函数‘main’中: 0-hello.c:(.text+0x21):对‘max’未定义的引用 collect2: error: ld returned 1 exit status
版本2
1-hello.c:
#include <stdio.h> #include "0-max.c" int main() { int a1 = 33; int a2 = 21; int maxNum = max(a1,a2); printf("the max value is %d\n",maxNum); return 0; }
0-max.c与原来的一致
编译命令:
gcc 1-hello.c -o 1-hello.out
若是此时多加了0-max.c一块儿编译
gcc 1-hello.c 0-max.c -o 1-hello.out /tmp/ccjcCmVa.o:在函数‘max’中: 0-max.c:(.text+0x0): `max'被屡次定义 /tmp/cclIxMtD.o:1-hello.c:(.text+0x0):第一次在此定义 collect2: error: ld returned 1 exit status
终端下:
gcc 文件名.c -o 命名.out
#include <>
表示在预装的库里查找#include "max.c"
表示在当前目录内查找文件
include "max.c"
至关于把整个函数复制进来了。效果等同于写进来
wqa
是将多个文件一块儿保存把函数的声明和定义分离开来
代码没有main函数不能执行,main是入口。
mtianyan@ubuntu:~/Desktop/zjuPlan/CSF878/CCode/linux_c/2-lesson/part1$ ls 0-max.c 1-hello.c
加快编译速度
gcc -c 0-max.c -o 0-max.o
将max.c变成max.o以后,咱们须要把hello.c中的include注释掉并添上方法声明
#include <stdio.h> //#include "0-max.c" int max(int a,int b);
可读可写不可执行,max.o至关于计算器对于源代码进行了翻译,变成计算机可识别的机器码
gcc 0-max.o 1-hello.c -o 1-hello.out
新建一个min.c
int min (int a, int b) { if(a<b){ return a; }else{ return b; } }
hello.c中进行minNum的调用
#include <stdio.h> //#include "0-max.c" int max(int a,int b); int min(int a,int b); int main() { int a1 = 33; int a2 = 21; int maxNum = max(a1,a2); int minNum = min(a1,a2); printf("the max value is %d\n",maxNum); printf("the min value is %d\n",minNum); return 0; }
编译命令:
gcc -c min.c -o min.o
gcc 0-max.o min.o 1-hello.c -o 2-hello.out
加快编译速度。不会再修改的函数,公共框架和公共类编译生成静态库。
gcc的编译流程分为4步:
预处理(Pre-Processing) -> 编译(Compling) -> 汇编(Assembling) -> 链接(Linking)
预处理: 处理#include
#define
#ifdef
等宏命令
编译: 把预处理完的文件编译为汇编程序.s
汇编: 把汇编程序.s
编译为.o
二进制文件
gcc -c min.c -o min.o
//把文件min.c预编译成文件min.ocp 文件名a 文件b
//把文件a拷贝成新的文件byy
复制一行 使用 行号n+yy
复制n行p
对复制的行进行粘贴cat
命令能够查看那么问题又来了,咱们如今的max函数和min函数是本身编写的,即便不加声明,咱们也知道须要哪些参数,参数是什么类型,返回值是什么类型。若是这个函数不是咱们编写的,别人又编写成了max.o min.o 咱们看不到源代码,又不知道函数的传入参数与返回。.h
文件的好处就来了
咱们建立一个文件夹part2
max.h 代码:
int maxNum(int a, int b);
min.h 代码:
int minNum(int a ,int b);
hello.c代码:
#include <stdio.h> #include "max.h" #include "min.h" int main() { int a1 = 33; int a2 = 21; int maxNum = max(a1,a2); int minNum = min(a1,a2); printf("the max value is %d\n",maxNum); printf("the min value is %d\n",minNum); }
gcc max.o min.o hello.c -o hello.out
warning: implicit declaration of function ‘max’; did you mean ‘main’?
出现警告,但不影响正常的编译运行。
make -v
sudo apt-get install make
make工具的内部也是使用的gcc
咱们本身开发仍是安装软件都要使用到make
和make install
这两个命令
由于当咱们的源文件不少不少的时候
gcc max.c min.c hello.c -o hello.out
命令就会很长很长。
编写一个Makefile能够同时编译多个文件,告诉依赖关系。
hello.out:max.o min.o hello.c gcc max.o min.o hello.c -o hello.out max.o:max.c gcc -c max.c min.o:min.c gcc -c min.c # 添加注释
make
编写 Makefile 缩进使用 tab 键(八个空格,不然出错)
Makefile:3: *** 遗漏分隔符 (null)。 中止。
mv MakeFile makefile
从上往下找,从下往上编译出来。 已经生成出来的文件不会再从新生成。
make: “hello.out”已经是最新。 (up to date)
main.c:
#include <stdio.h> int main(int argc,char* argv[]) //main函数完整形式 { printf("hello,world\n"); return 101; }
gcc main.c -o main.out && ./main.out
gcc main.c -o main.out && ./ main.out
能够依次执行两条命令return 0
用来验证程序运行是否成功。echo $?
用来查看返回值./main.out && ls
打印出helloworld的同时,列出当前目录。由于main.out的return值为0,断定为成功执行。
./main2.out && ls
打印出helloworld,未列出当前目录。由于main.out的return值为101(非0),断定为不成功执行。
int main(int argc, char* argv[])//main函数完整形式
argc统计输入参数的个数。只输入文件名时个数为1
argv[]
存放每一个参数的内容
如:
argc(argument count),存放着传入main函数的参数个数是几个。argv(argument vector),存放具体传入的参数
main3.c:
#include <stdio.h> int main(int argc,char* argv[]) { printf("argc is %d\n",argc); //int i; for(int i =0;i<argc;i++){ printf("argv[%d]is %s\n",i,argv[i]); } return 101; }
$ ./main3.out -a -l argc is 3 argv[0]is ./main3.out argv[1]is -a argv[2]is -l
linux把全部东西看成文件处理
stdin
键盘stdout
显示器stderr
fprintf fscanf
:将输入输出放入...printf("")
是对fprintf(stdout,"")
函数的封装.scanf("")
是对fscanf(stdin,"")
函数的封装#include <stdio.h> int main() { // printf("hello world!\n"); fprintf(stdout,"hello world \n"); int a; //scanf("%d",&a); fscanf(stdin,"%d",&a); if(a<0){ fprintf(stderr,"the value must >0\n"); # 返回值不等于0代表函数出错 return 1; } //printf("input value is: %d\n",a); return 0; }
Linux几乎能够用于任何领域,这里咱们不得不提出linux的通道。管道
起到了很重要的做用,不一样应用程序之间要配合使用,就须要用到管道。
Demo:main.c
先理解输入流,输出流和错误流的 重定向机制,对于管道的理解会比较容易些。
#include <stdio.h> int main() { int i,j; printf("input the int value i:\n"); \\printf其实对fprintf的封装,是从标准输出流(即stdout)来输出这个过程 scanf("%d", &i); //默认输入流是键盘 printf("input the int value j:\n"); scanf("%d", &j); printf("i+j=%d\n", i+j); }
cc main.c
, 获得a.out
,运行a.out,咱们分别输入3和5输入到终端../a.out 1>> a.txt
,其中>>
符号(不写参数就是输出流),以前默认输出流是终端,如今咱们则改成输出到a.txt中,咱们执行命令后,分别输入3回车后再输入5。再使用命令cat a.txt,咱们能够看到咱们已经输出到文件里的内容。咱们标准输出流是1>>
,输入流是0>>
./a.out >> a.txt
,咱们再次输入参数,完成后咱们再次使用cat
来查看a.txt文件里的内容,发现以前的内容还在,新的输出内容追加到了后面。ls /etc >> etc.txt
,咱们将ls目录下的内容输入到了etc.txt文件中>>
改成单箭头>
,则文件中先前的内容就会被覆盖掉。18 9
./a.out < input.txt
,不存在追加模式,因此咱们用单箭头<
,咱们能够将要输入的内容所有在input.txt中准备好,命令执行后,咱们便在终端上能够看到结果。#include <stdio.h> int main() { int i,j; printf("input the int value i:\n"); scanf("%d", &i); //默认输入流是键盘 printf("input the int value j:\n"); scanf("%d", &j); if(0!=j){ printf("%d/%d=%d\n",i,j,i/j); }else{ fprintf(stderr,"j != 0\n"); return 1; } return 0; }
注意代码规范小技巧,0和j的比较,0放前面若是漏了等号会报错
echo $?
查看返回值
./main.out 1>true.txt 2>false.txt < input.txt
而当值错误,产生除零错误时。
上图是当j=0时的运行状况。
重定向到其余地方。
管道:
把前面的输出流做为后面工具的输入流,用一个|
表示.
grep
:查看指定文本,搜出包含字符的文本ls /etc/ | grep ab
ls /etc/
的输出做为grep
的输入
ls /etc/ | grep ab
在/etc/文件下查找含有ab字符的文件名ps -e
:查看系统运行的进程ps -e | grep ssh
查看包含ssh的进程
avg.c :
#include <stdio.h> int main() { int s,n; scanf("%d,%d",&s,&n); float v = s/n; printf("v =%f\n",v); return 0; }
编译命令:
cc avg.c -o avg.out
input.c:
#include <stdio.h> int main() { int flag =1; int i; int s = 0; int count=0; while(flag){ scanf("%d",&i); if(0==i) break; #等于零跳出 count++; s+=i; } printf("%d,%d\n",s,count); return 0; }
cc input.c -o input.out
注意上面的代码中应该将s初始化为0.不然会产生问题。
使用时:
./input.out | ./avg.out
实现将input的数据之间求平均值。