APUE 学习笔记(五) 进程环境

1.main函数

C程序总是从main函数开始执行,当内核执行C程序时,在调用main函数之前先调用exec函数从内核获取命令行参数和环境变量值

2.进程终止

正常终止:

(1)在main函数内执行return语句

(2)调用exit

(3)最后一个线程从其启动例程返回

(4)最后一个线程调用pthread_exit

异常中止:

(1)调用abort

(2)接收到一个信号并终止(内存越界或除0)

(3)最后一个线程对取消请求作出响应

内核使程序执行的唯一方法就是调用exec函数,进程自愿终止的唯一方法就是调用exit

3.C程序的存储空间布局

a.out中还有其他类型的段,比如 符号表段、调试信息段、动态共享库段

size命令查看 正文段、数据段和bss(未初始化数据段)的长度。 如  size  /usr/bin/cc

4. setjmp longjmp


  #include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <setjmp.h>

static jmp_buf jmpbuf;
static int globval = 1;

static void func2(void)
{
longjmp(jmpbuf, 1);
}

static void func1(int i, int j, int k, int l)
{
fprintf(stdout, "in func1:\n");
fprintf(stdout, "globval = %d,"
"autoval = %d,"
"regival = %d,"
"volaval = %d,"
"statval = %d\n", globval, i, j, k , l);
func2();
}
int main(int argc, char* argv[])
{
int autoval = 2;
register int regival = 3;
volatile int volaval = 4;
static int statval = 5;
if (setjmp(jmpbuf) != 0) {
fprintf(stdout, "after longjmp:\n");
fprintf(stdout, "globval = %d,"
"autoval = %d,"
"regival = %d,"
"volaval = %d,"
"statval = %d\n",
globval, autoval, regival, volaval, statval);
exit(0);
}

globval = 95;
autoval = 96;
regival = 97;
volaval = 98;
statval = 99;

func1(autoval, regival, volaval, statval);
exit(0);
}

编译时不进行优化,结果如下:

in func1:
globval = 95,autoval = 96,regival = 97,volaval = 98,statval = 99
after longjmp:
globval = 95,autoval = 96,regival = 97,volaval = 98,statval = 99

编译时进行全部优化,结果如下:

in func1:
globval = 95,autoval = 96,regival = 97,volaval = 98,statval = 99
after longjmp:
globval = 95,autoval = 2,regival = 3,volaval = 98,statval = 99

我们可以看到:全局、静态、易失变量不受优化的影响,在调用longjmp之后,他们的值是最近所呈现的值

未优化时,所有5个变量都存放在存储器中,而进行优化之后,autoval 和 regival 都存放在寄存器中,而 volatile变量仍存放在
存储器中

对于 volatile 修饰符用法,这里有一篇很好的文章:http://hedengcheng.com/?p=725

时间: 2024-10-03 22:29:03

APUE 学习笔记(五) 进程环境的相关文章

Linux System Programming 学习笔记(五) 进程管理

1. 进程是unix系统中两个最重要的基础抽象之一(另一个是文件) A process is a running program A thread is the unit of activity inside of a process the virtualization of memory is associated with the process, the threads all share the same memory address space 2. pid The idle pro

APUE 学习笔记(六) 进程控制

1. fork 创建新进程 fork创建的新进程称为子进程,fork函数调用一次,返回两次. 两次返回的唯一区别就是子进程的返回值是0,而父进程的返回值是新子进程的进程ID 在fork之后是父进程先执行还是子进程先执行是不确定的,这取决于内核的调度算法 fork的一个特性就是父进程的所有打开文件描述符都被复制到子进程中,父子进程的每个相同的打开描述符共享一个文件表项 在fork之后处理文件描述符有两种常见情况: (1)父进程等待子进程完成.在这种情况下,父进程无需对其描述符做任何处理,因为子进程

APUE学习笔记:第七章 进程环境

7.1 引言 本章将学习:当执行程序时,其main函数是如何被调用的:命令行参数是如何传送给执行程序的:典型的存储器布局是什么样式:如何分配另外的存储空间:进程如何使用环境变量:各种不同的进程终止方式等:另外还将说明longjmp和setjmp函数以及它们与栈的交互作用:还将介绍研究进程的资源限制 7.2 main函数 C程序总是从main函数开始执行.当内核执行C程序时,在调用main前先调用一个特殊的启动例程.可执行程序文件将此启动例程指定为程序的起始地址——这是由连接编辑器设置的,而连接编

APUE学习笔记:第五章 标准I/O库

5.1 引言 标准I/O库处理很多细节,例如缓冲区分配,以优化长度执行I/O等.这些处理不必担心如何使用正确的块长度.这使得它便于用户使用,但是如果不较深入地了解I/O库函数的操作,也会带来一些问题 5.2 流和FILE对象 对于ASCII字符集,一个字符用一个字节表示.对于国际字符集,一个字符可用多个字节表示.标准I/O文件流可用于单字节或多字节字符集. 流的定向决定了所读.写的字符是单字节还是多字节的.当一个流最初被创建时,它并没有定向.如若在未定向的流上使用一个多字节I/O函数,则将该流的

APUE学习笔记:第九章 进程关系

9.1 引言 本章将更详尽地说明进程组以及POSIX.1引入的会话的概念.还将介绍登陆shell(登录时所调用的)和所有从登陆shell启动的进程之间的关系. 9.1 终端登陆 系统管理员创建通常名为/etc/ttys的文件,其中每个终端设备都有一行,每一行说明设备名传递给getty程序的参数.当系统自举时,内核创建进程ID为1的进程,依旧是init进程.init进程使系统进入多用户状态.init进程读文件/etc/ttys,对每一个允许登陆的终端设备,init调用一次fork,所生成的子进程则

APUE学习笔记:第八章 进程控制

8.1 引言 本章介绍UNIX的进程控制,包括创建新进程.执行程序和进程终止.还将说明进程属性的各种ID-----实际.有效和保存的用户和组ID,以及他们如何受到进程控制原语的影响.本章还包括了解释器文件和system函数.本章最后讲述大多数UNIX系统所提供的进程会计机制.这种机制使我们能够从另一个角度了解进程的控制功能. 8.2 进程标识符 每个进程都有一个非负整型表示的惟一进程ID.因为进程标识符是惟一的,常将其用作其他标识符的一部分以保证其惟一性.虽然是惟一的,但是进程ID可以重用.(大

APUE学习笔记:第一章 UNUX基础知识

1.2 UNIX体系结构 从严格意义上,可将操作系统定义为一种软件(内核),它控制计算机硬件资源,提供程序运行环境.内核的接口被称为系统调用.公用函数库构建在系统调用接口之上,应用软件即可使用公用函数库,也可使用系统调用.shell是一种特殊的应用程序,它为运行其他应用程序提供了一个接口 从广义上,操作系统包括了内核和一些其他软件,这些软件使得计算机能够发挥作用,并给予计算机以独有的特性(软件包括系统实用程序,应用软件,shell以及公用函数库等) 1.3  shell shell是一个命令行解

APUE学习笔记:第二章 UNIX标准化及实现

2.2UNIX标准化 2.2.1 ISO C 国际标准化组织(International Organization for Standardization,ISO) 国际电子技术委员会(International Electrotechnical Commission,IEC) ISO C标准的意图是提供C程序的可移植性,使其能适合于大量不同的操作系统,而不只是UNIX系统.此标准不仅定义了C程序设计语言的语法和语义,还定义了其标准库.因为所有现今的UNIX系统都提供C标准中定义的库例程,所以该

APUE 学习笔记(九) 高级I/O

1. 非阻塞I/O 低速系统调用时可能会使进程永远阻塞的一类系统调用,包括以下调用: (1)某些文件类型你(网络socket套接字.终端设备.管道)暂无可使用数据,则读操作可能会使调用者永远阻塞 (2)如果数据不能立即被(1)中文件类型接受,则写操作会使调用者永远阻塞 (3)某些进程间通信函数 非阻塞I/O使我们可以调用open.read.write这样的I/O操作,并使这些操作不会永远阻塞,如果这种操作不能完成,则调用立即出错返回 对于一个给定的文件有两种方法对其指定非阻塞I/O: (1)调用