区别一:构成不一样
1、C程序的执行部分是由语句组成的。 程序的功能也是由执行语句实现的。
2、指令可以使编译器按不同的条件编译不同的程序部分,因而产生不同的目标代码文件。这对于程序的移植和调试是很有用的,尤其是针对于跨平台程序移植的时候。
区别二:实现功能不一样
1、C程序语言可以实现多种程序结构, 即顺序结构、分支(选择)结构、循环结构。
2、指令可以实现源代码的部分编译功能,可以根据表达式的值或者某个特定的宏来确定编译条件,以决定编译哪些代码,不编译哪些。
扩展资料C语言表达式语句
表达式语句由表达式加上分号“”组成。
其一般形式为:表达式执行表达式语句就是计算表达式的值和执行副作用。
例如: x=y+z;a=520;赋值语句;
y+z;加法运算语句,但计算结果不能保留,无实际意义;
i++; 自增1语句,i值增1。
i++; 是先运算i后再加1。
++i; 是先把i值增1后运算。
包括空语句,函数调用语句在内都属于表达式语句。
C语言条件编译指令
1、#if指令
该指令检测表达式值是否为真。如果表达式的值为真,则编译后面的代码直到出现 #else、#elif 或 #endif 为止,否则不编译。
2、#endif指令
该指令用于终止 #if 指令。
3、#else指令
该指令用于 #if 指令之后,当前面的 #if 指令的条件不为真时,就编译 #else 后面的代码。
4、#elif指令
该指令综合了 #else 和 #if 指令的作用。下面的示例代码演示了 #if、#else、#elif 与 #endif 的组合使用情况。
参考资料来源:百度百科—C语言基础语句
第一章:绪论?内核版本号格式:x.y.zz-www/x为主版本号,y为次版本号,zz为次次版本号,www为发行号/次版本号改变说明内核有重大变革,其偶数为稳定版本,奇数为尚在开发中的版本
第二章:基础?
文件种类:-:txt,二进制/d:目录/l:链接文件(link)/b:区块设备文件/c:字符设备文件/p:管道
目录结构:bin:可执行/boot:开机引导/dev:设备文件/etc:系统配置文件/lib:库文件/mnt:设备挂载点/var:系统日志/
命令:rmdir:删除空目录/find [path] [expression]/touch命令还可以修改指定文件的最近一次访问时间/tar -czvf usr.tar.gz path/tar –zxvf usr.tar.gz/tar –cjvf usr.tar.bz2 path/tar –jxvf usr.tar.bz2
gcc:预处理:-g/I在头文件搜索路径中添加目录,L在库文件搜索路径中
gdb:设置断点:b/查看断点信息:info
Makefile:make –f other_makefile/<:第一个依赖文件的名称/@:目标文件的完整名称/^:所有不重复的依赖文件/+:所有依赖文件(可能重复)
第三章:文件IO
read:read(fd, temp, size)/读fd中长度为size的值到temp/返回0表示file为NULL
write:write(fd, buf, buf_size)/写长度为buf_size的buf内容到fd中
lseek:lseek(fd, offset, SEEK_SET)/从文件开头向后增加offset个位移量
unlink:从文件系统中删除一个名字
open1:int open(const char * pathname, int flags, mode_t mode)/flags为读写方式/mode为权限设置/O_EXCL:测试文件是否存在/O_TRUNC:若存在同名文件则删除之并新建
open2:注意O_NONBLOCK
mmap.1:void *mmap(void *start, size_t length, int prot, int flags, int fd, off_t offsize)
mmap.2:mmap(start_addr, flength, PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0)
fcntl:上锁/int fcntl(int fd, int cmd, struct flock * lock)/对谁;做什么;设置所做内容
select:fd_max+1,回传读状况,回传写状况,回传异常,select等待的时间/NULL为永远等待/0为从不等待/凡需某状况则用之,反则(fd_set *)NULL之
FD_*那几个函数……
一般出错则返回-1
第四章:文件与目录
硬链接与符号链接?
chdir改变目录
0:in/1:out/2:err
第五章:内存管理
可执行文件存储时:代码区、数据区和未初始化区
栈:by编译器,向低址扩展,连续,效率高/堆:by程序员
/etc/syslog.conf,系统log记录文件/优先级为-20时最高
第六章:进程和信号
程序代码、数据、变量、文件描述符和环境/init的pid为1
execl族:int execl(const char * path, const char * arg, ....)/path即可执行文件的路径,一般为./最后一个参数以NULL结束
waitpid:waitpid(pid_t pid,int * status,int options)/option:一般用WNOHANG,没有已经结束的子进程则马上返回,不等待
kill:int kill(pid_t pid,int sig)/发送信号sig给pid
void (*signal(int signum, void(* handler)(int)))(int)/第一个参数被满足时,执行handler/第一个参数常用:SIG_IGN:忽略信号/SIG_DFL:恢复默认信号
第七章:线程
sem_init(sem_t *sem, int pshared, unsigned int value)/pshared为0/value即初始值
第八章:管道
1:write/0:read
第九章:信号量、共享内存和消息队列
临界资源: *** 作系统中只允许一个进程访问的资源/临界区:访问临界资源的那段代码
信号量:建立联系(semget),然后初始化,PV *** 作,最后destroy
共享内存没有提供同步机制
第十章:套接字
UDP:无连接协议,无主客端的区分/实时性
TCP:字节流/数据可靠性/网络可靠性
数据报:SOCK_STREAM/SOCK_DGRAM
其它
管道一章的both_pipe即父子进程间的全双工管道通讯
关系到信号和互斥的服务器-客户端程序
线程一章的class的multi_thread文件夹下的thread8.c
int main(void)
{
int data_processed
int file_pipes_1[2]
int file_pipes_2[2]
char buffer[BUFSIZ + 1]
const char some_data[] = "123"
const char ch2p[] = "this is the string from child to the parent!"
const char p2ch[] = "this is the string from parent to the child!"
pid_t fork_result
memset(buffer,'\0',sizeof(buffer))
if(pipe(file_pipes_1) == 0){
if(pipe(file_pipes_2) == 0){
fork_result = fork()
switch(fork_result){
case -1:
perror("fork error")
exit(EXIT_FAILURE)
case 0://child
close(file_pipes_1[1])
close(file_pipes_2[0])
printf("in the child!\n")
read(file_pipes_1[0],buffer, BUFSIZ)
printf("in the child, read_result is \"%s\"\n",buffer)
write(file_pipes_2[1],ch2p, sizeof(ch2p))
printf("in the child, write_result is \"%s\"\n",ch2p)
exit(EXIT_SUCCESS)
default://parent
close(file_pipes_1[0])
close(file_pipes_2[1])
printf("in the parent!\n")
write(file_pipes_1[1], p2ch, sizeof(p2ch))
printf("in the parent, write_result is \"%s\"\n",p2ch)
read(file_pipes_2[0],buffer, BUFSIZ)
printf("in the parent, read_result is \"%s\"\n",buffer)
exit(EXIT_SUCCESS)
}
}
}
}
#ifndef DBG
#define DBG
#endif
#undef DBG
#ifdef DBG
#define PRINTF(fmt, args...) printf("file->%s line->%d: " \
fmt, __FILE__, __LINE__, ##args)
#else
#define PRINTF(fmt, args...) do{}while(0)
#endif
int main(void)
{
PRINTF("%s\n", "hello!")
fprintf(stdout, "hello hust!\n")
return 0
}
#define N 5
#define MAX 5
int nput = 0
char buf[MAX][50]
char *buffer = "abcdefghijklmnopqrstuvwxyz0123456789"
char buf_r[100]
sem_t mutex,full,avail
void *productor(void *arg)
void *consumer(void *arg)
int i = 0
int main(int argc, char **argv)
{
int cnt = -1
int ret
int nput = 0
pthread_t id_produce[10]
pthread_t id_consume
ret = sem_init(&mutex, 0, 1)
ret = sem_init(&avail, 0, N)
ret = sem_init(&full, 0, 0)
for(cnt = 0cnt <6cnt ++ ){
//pthread_create(&id_produce[cnt], NULL, (void *)productor, &cnt)
pthread_create(&id_produce[cnt], NULL, (void *)productor, (void *)cnt)
}
pthread_create(&id_consume, NULL, (void *)consumer, NULL)
for(cnt = 0cnt <6cnt ++){
pthread_join(id_produce[cnt], NULL)
}
pthread_join(id_consume,NULL)
sem_destroy(&mutex)
sem_destroy(&avail)
sem_destroy(&full)
exit(EXIT_SUCCESS)
}
void *productor(void *arg)
{
while(1){
sem_wait(&avail)
sem_wait(&mutex)
if(nput >= MAX * 3){
sem_post(&avail)
//sem_post(&full)
sem_post(&mutex)
return NULL
}
sscanf(buffer + nput, "%s", buf[nput % MAX])
//printf("write[%d] \"%s\" to the buffer[%d]\n", (*(int*)arg), buf[nput % MAX],nput % MAX)
printf("write[%d] \"%s\" to the buffer[%d]\n", (int)arg, buf[nput % MAX],nput % MAX)
nput ++
printf("nput = %d\n", nput)
sem_post(&mutex)
sem_post(&full)
}
return NULL
}
void *consumer(void *arg)
{
int nolock = 0
int ret, nread, i
for(i = 0 i <MAX * 3i++)
{
sem_wait(&full)
sem_wait(&mutex)
memset(buf_r, 0, sizeof(buf_r))
strncpy(buf_r, buf[i % MAX], sizeof(buf[i % MAX]))
printf("read \"%s\" from the buffer[%d]\n\n",buf_r, i % MAX)
sem_post(&mutex)
sem_post(&avail)
//sleep(1)
}
return NULL
}
我们可以在C源程序中插入传给编译程序的各中指令,这些指令被称为预处理器指令,它们扩充了程序设计的环境。现把常用的预处理命令总结如下: \x0d\x0a1. 预处理程序\x0d\x0a按照ANSI标准的定义,预处理程序应该处理以下指令:\x0d\x0a#if #ifdef #ifndef #else #elif\x0d\x0a#endif\x0d\x0a#define\x0d\x0a#undef\x0d\x0a#line\x0d\x0a#error\x0d\x0a#pragma\x0d\x0a#include\x0d\x0a显然,上述所有的12个预处理指令都以符号#开始,,每条预处理指令必须独占一行。\x0d\x0a2. #define\x0d\x0a#define指令定义一个标识符和一个串(也就是字符集),在源程序中发现该标识符时,都用该串替换之。这种标识符称为宏名字,相应的替换称为宏代换。一般形式如下:\x0d\x0a#define macro-name char-sequence\x0d\x0a这种语句不用分号结尾。宏名字和串之间可以有多个空白符,但串开始后只能以新行终止。\x0d\x0a例如:我们使用LEFT代表1,用RIGHT代表0,我们使用两个#define指令:\x0d\x0a#define LEFT 1\x0d\x0a#define RIGHT 0\x0d\x0a每当在源程序中遇到LEFT或RIGHT时,编译程序都用1或0替换。\x0d\x0a定义一个宏名字之后,可以在其他宏定义中使用,例如:\x0d\x0a#define ONE 1\x0d\x0a#define TWO ONE+ONE\x0d\x0a#define THREE ONE+TWO\x0d\x0a宏代换就是用相关的串替代标识符。因此,如果希望定义一条标准错误信息时,可以如下定义:\x0d\x0a#define ERROR_MS “Standard error on input \n”\x0d\x0a如果一个串长于一行,可在行尾用反斜线”\”续行,如下:\x0d\x0a#define LONG_STRING “This is a very very long \\x0d\x0aString that is used as an example”\x0d\x0a3. #error\x0d\x0a#error指令强制编译程序停止编译,它主要用于程序调试。#error指令的一般形式是:\x0d\x0a#error error-message\x0d\x0a注意,宏串error-message不用双引号包围。遇到#error指令时,错误信息被显示,可能同时还显示编译程序作者预先定义的其他内容。\x0d\x0a4. #include\x0d\x0a程序中的#include指令要求编译程序读入另一个源文件。被读入文件的名字必须用双引号(“”)或一对尖括号()包围,例如:\x0d\x0a#include “stdio.h”\x0d\x0a#include \x0d\x0a都使C编译程序读入并编译头文件以用于I/O系统库函数。\x0d\x0a包含文件中可以包含其他#include指令,称为嵌套包含。允许的最大嵌套深度随编译器而变。\x0d\x0a文件名被双括号或尖括号包围决定了对指定文件的搜索方式。文件名被尖括号包围时,搜索按编译程序作者的定义进行,一般用于搜索某些专门放置包含文件的特殊目录。当文件名被双括号包围时,搜索按编译程序实时的规定进行,一般搜索当前目录。如未发现,再按尖括号包围时的办法重新搜索一次。\x0d\x0a通常,绝大多数程序员使用尖括号包围标准的头文件,双引号用于包围与当前程序相关的文件名。\x0d\x0a\x0d\x0a5. 条件编译指令\x0d\x0a若干编译指令允许程序员有选择的编译程序源代码的不同部分,这种过程称为条件编译。\x0d\x0a5.1#if、#else、#elif #endif\x0d\x0a条件编译指令中最常用的或许是#if,#else,#elif和#endif。这些指令允许程序员根据常数表达式的结果有条件的包围部分代码。\x0d\x0a#if的一般形式是:\x0d\x0a#if constant-expression\x0d\x0aStatement sequence\x0d\x0a#endif\x0d\x0a如#if后的常数表达式为真,则#if和#endif中间的代码被编译,否则忽略该代码段。#endif标记#if块的结束。\x0d\x0a#else指令的作用与C语言的else相似,#if指令失败时它可以作为备选指令。例如:\x0d\x0a#include \x0d\x0a#define MAX 100\x0d\x0aInt main(void)\x0d\x0a{\x0d\x0a#if MAX>99\x0d\x0aprintf(“Compiled for array greater than 99.\n”)\x0d\x0a#else\x0d\x0aprintf(“Complied for small array.\n”)\x0d\x0a#endif\x0d\x0areturn 0\x0d\x0a}\x0d\x0a注意,#else既是标记#if块的结束,也标记#else块的开始。因为每个#if只能写一个#endif匹配。\x0d\x0a#elif指令的意思是“否则,如果”,为多重编译选择建立一条if-else-if(如果-否则-如果链)。如果#if表达式为真,该代码块被编译,不测试其他#elif表达式。否则,序列中的下一块被测试,如果成功则编译之。一般形式如下:\x0d\x0a#if expression\x0d\x0aStatement sequence\x0d\x0a#elif expression1\x0d\x0aStatement sequence\x0d\x0a#elif expression2\x0d\x0aStatement sequence\x0d\x0a.\x0d\x0a.\x0d\x0a.\x0d\x0a#elif expression\x0d\x0aStatement sequence\x0d\x0a#endif\x0d\x0a5.2#ifdef和#ifndef\x0d\x0a条件编译的另一个方法是使用编译指令#ifdef和#ifndef,分别表示“如果已定义”和“如果未定义”。#ifdef的一般形式如下:\x0d\x0a#ifdef macro-name\x0d\x0aStatement sequence\x0d\x0a#endif\x0d\x0a如果macro-name原先已经被一个#define语句定义,则编译其中的代码块。\x0d\x0a#ifndef的一般形式是:\x0d\x0a#ifndef macro-name\x0d\x0aStatement sequence\x0d\x0a#endif\x0d\x0a如果macro-name当前未被#define语句定义,则编译其中的代码块。\x0d\x0a\x0d\x0a我认为,用这种,可以很方便的开启/关闭整个程序的某项特定功能。\x0d\x0a#ifdef和#ifndef都可以使用#else或#elif语句。\x0d\x0a#inlucde \x0d\x0a#define T 10\x0d\x0aInt main(void)\x0d\x0a{\x0d\x0a#ifdef t\x0d\x0aPrintf(“Hi T\n”)\x0d\x0a#else\x0d\x0aPrintf(“Hi anyone\n”)\x0d\x0a#endif\x0d\x0a#ifndef M\x0d\x0aPrintf(“M Not Defined\n”)\x0d\x0a#endif\x0d\x0aReturn 0\x0d\x0a}\x0d\x0a6. #undef\x0d\x0a#undef指令删除前面定义的宏名字。也就是说,它“不定义”宏。一般形式为:\x0d\x0a#undef macro-name\x0d\x0a7. 使用defined\x0d\x0a除#ifdef之外,还有另外一种确定是否定义宏名字的方法,即可以将#if指令与defined编译时 *** 作符一起使用。defined *** 作符的一般形式如下:\x0d\x0adefined macro-name\x0d\x0a如果macro-name是当前定义的,则表达式为真,否则为假。\x0d\x0a例如,确定宏MY是否定义,可以使用下列两种预处理命令之一:\x0d\x0a#if defined MY\x0d\x0a或\x0d\x0a#ifdef MY\x0d\x0a也可以在defined之前加上感叹号”!”来反转相应的条件。例如,只有在DEBUG未定义的情况下才编译。\x0d\x0a#if !defined DEBUG\x0d\x0aPrintf(“Final Version!\n”)\x0d\x0a#endif\x0d\x0a使用defined的一个原因是,它允许由#elif语句确定的宏名字存在。\x0d\x0a8. #line\x0d\x0a#line指令改变__LINE__和__FILE__的内容。__LINE__和__FILE__都是编译程序中预定义的标识符。标识符__LINE__的内容是当前被编译代码行的行号,__FILE__的内容是当前被编译源文件的文件名。#line的一般形式是:\x0d\x0a#line number “filename”\x0d\x0a其中,number是正整数并变成__LINE__的新值;可选的“filename”是合法文件标识符并变成__FILE__的新值。#line主要用于调试和特殊应用。\x0d\x0a\x0d\x0a9. #pragma\x0d\x0a#pragma是编译程序实现时定义的指令,它允许由此向编译程序传入各种指令。例如,一个编译程序可能具有支持跟踪程序执行的选项,此时可以用#pragma语句选择该功能。编译程序忽略其不支持的#pragma选项。#pragma提高C源程序对编译程序的可移植性。\x0d\x0a10. 预处理 *** 作符#和##\x0d\x0a有两个预处理 *** 作符:#和##,它们可以在#define中使用。\x0d\x0a *** 作符#通常称为字符串化的 *** 作符,它把其后的串变成用双引号包围的串。例如:\x0d\x0a#include \x0d\x0a#define mkstr(s) #s\x0d\x0aint main(void)\x0d\x0a{\x0d\x0aPrintf(mkstr(I like C))\x0d\x0aReturn 0\x0d\x0a}\x0d\x0a预处理程序把以下的语句:\x0d\x0aPrintf(mkstr(I like C))\x0d\x0a变成\x0d\x0aPrintf(“I like C”)\x0d\x0a *** 作符##把两个标记拼在一起,形成一个新标记。例如:\x0d\x0a#include \x0d\x0a#define concat(a,a) a##b\x0d\x0aint main(void)\x0d\x0a{\x0d\x0aInt xy = 10\x0d\x0aPrintf(“%d”,concat(x,y))\x0d\x0aReturn 0\x0d\x0a}\x0d\x0a预处理程序把以下语句:\x0d\x0aPrintf(“%d”,concat(x,y))\x0d\x0a变成\x0d\x0aPrintf(“%d”,xy)\x0d\x0a *** 作符#和##主要作用是允许预处理程序对付某些特殊情况,多数程序中并不需要。\x0d\x0a11. 预定义宏\x0d\x0aC规范了5个固有的预定义宏,它们是:\x0d\x0a__LINE__\x0d\x0a__FILE__\x0d\x0a__DATE__\x0d\x0a__TIME__\x0d\x0a__STDC__\x0d\x0a__LINE__和__FILE__包含正在编译的程序的行号和文件名。\x0d\x0a__DATE__和内容形如month/day/year(月/日/年)的串,代表源文件翻译成目标码的日期。\x0d\x0a__TIME__中的串代表源代码编译成目标码的时间,形如hour:minute:second(时:分:秒)\x0d\x0a如果__STDC__的内容是十进制常数1,则表示编译程序的实现符合标准C。欢迎分享,转载请注明来源:内存溢出
评论列表(0条)