fork函数的2个返回值说明

fork函数的2个返回值说明
fork函数的2个返回值说明

fork简介:

fork英文原意是“分岔,分支”的意思,而在操作系统中,乃是著名的Unix(或类Unix,如Linux,Minix)中用于创建子进程的系统调用。

【NOTE1】

fork () 的作用是什么?换句话说,你用fork () 的目的是什么?

――是为了产生一个新的进程,地球人都知道:)

产生一个什么样的进程?

――和你本来调用fork () 的那个进程基本一样的进程,其实就是你原来进程的副本;

真的完全一样吗?

――当然不能完全一样,你要两个除了pid 之外其它一模一样的进程干什么,就算memory 再多也不用这么摆谱吧?

哪里不一样?

――当然最重要的是fork () 之后执行的代码不一样,you know, i know :)

怎么实现呢?

――如果是Windows,它会让你在fork () 里面提供一大堆东西,指明这个那个什么的…… 我用的是unix 啊

――所以很简单,unix 会让两个进程(不错,原来是一个,unix 替你复制了一个,现在有两个)

在fork () 之后产生不同:返回值不同。其中一个进程(使用新的pid)里面的fork () 返回零,

这个进程就是“子进程”;而另一个进程(使用原来的pid)中的fork () 返回前面那个子进程的

pid,他自己被称为“父进程”

然后呢?

――写代码的人又不笨,当然就根据返回值是否非零来判断了,现在我是在子进程里面呢,还是在

父进程里面?在子进程里面就执行子进程该执行的代码,在父进程里面就执行父进程的代码……

有铁杆windows fans 借此说明,windows 好啊,子进程用子进程的代码,父进程用父进程的,

你unix 笨了吧,子进程包含父进程、子进程的代码,父进程包含父进程子进程的代码,岂不是多占用内存了吗?

――据我所知,unix 代码段都是可重入代码,也就是说,进程复制,并不复制代码段,若

干个进程

共享同一代码段,增加的只是全局共享数据和对文件描述符的引用等,另外就是堆栈。你一个代码

长达10M 的进程,fork () 出三四个子进程,只是增加一点内存占用(如果你没有使用很多全局变量

的话),而不是占用40M 以上的内存。

【NOTE2】

程序从fork 开始分支(称分支不准确),一路是主进程pid > 0 (pid 是子进程ID) 一路是子进程pid == 0 自此分成两个任务

其实fork的时候已经两个分支了,数据段被复制了一份,因此pid有两份

执行pid=fork()时,返回值赋给pid在两个进程中运行,

fork会返回给父进程的那个>0的值,告诉调用者新建进程的pid

子进程的fork返回值是0

更不用说if...else的比较也是在两个进程中都做的了

【NOTE3】

fork的精辟剖析

程序如下:

#include ;

#include ;

main ()

{ pid_t pid;

pid=fork();

if (pid < 0) printf("error in fork!");

else if (pid == 0)

printf("i am the child process, my process id is %dn",getpid());

else

printf("i am the parent process, my process id is %dn",getpid());

}

结果是

[root@localhost c]# ./a.out

i am the child process, my process id is 4286

i am the parent process, my process id is 4285

一:

要搞清楚fork的执行过程,就必须先讲清楚操作系统中的“进程(process)”概念。一个进程,主要包含三个元素:

o. 一个可以执行的程序;

o. 和该进程相关联的全部数据(包括变量,内存空间,缓冲区等等);

o. 程序的执行上下文(execution context)。

不妨简单理解为,一个进程表示的,就是一个可执行程序的一次执行过程中的一个状态。操作系统对进程的管理,典型的情况,是通过进程表完成的。进程表中的每一个表项,记录的是当前操作系统中一个进程的情况。对于单CPU的情况而言,每一特定时刻只有一个进程占用CPU,但是系统中可能同时存在多个活动的(等待执行或继续执行的)进程。一个称为“程序计数器(program counter, pc)”的寄存器,指出当前占用CPU的进程要执行的下一条指令的位置。当分给某个进程的CPU时间已经用完,操作系统将该进程相关的寄存器的值,保存到该进程在进程表中对应的表项里面;把将要接替这个进程占用CPU的那个进程的上下文,从进程表中读出,并更新相应的寄存器(这个过程称为“上下文交换(process context switch)”,实际的上下文交换需要涉及到更多的数据,那和fork无关,不再多说,主要要记住程序寄存器pc指出程序当前已经执行到哪里,是进程上下文的重要内容,换出CPU的进程要保存这个寄存器的值,换入CPU的进程,也要根据进程表中保存的本进程执行上下文信息,更新这个寄存器)。

好了,有这些概念打底,可以说fork了。当你的程序执行到下面的语句:

pid=fork();

操作系统创建一个新的进程(子进程),并且在进程表中相应为它建立一个新的表项。新进程和原有进程的可执行程序是同一个程序;上下文和数据,绝大部分就是原进程(父进程)的拷贝,但它们是两个相互独立的进程!此时程序寄存器pc,在父、子进程的上下文中都

声称,这个进程目前执行到fork调用即将返回(此时子进程不占有CPU,子进程的pc不是真正保存在寄存器中,而是作为进程上下文保存在进程表中的对应表项内)。问题是怎么返回,在父子进程中就分道扬镳。

父进程继续执行,操作系统对fork的实现,使这个调用在父进程中返回刚刚创建的子进程的pid(一个正整数),所以下面的if语句中pid<0, pid==0的两个分支都不会执行。所以输出i am the parent process...

子进程在之后的某个时候得到调度,它的上下文被换入,占据CPU,操作系统对fork的实现,使得子进程中fork调用返回0。所以在这个进程(注意这不是父进程了哦,虽然是同一个程序,但是这是同一个程序的另外一次执行,在操作系统中这次执行是由另外一个进程表示的,从执行的角度说和父进程相互独立)中pid=0。这个进程继续执行的过程中,if语句中pid<0不满足,但是pid= =0是true。所以输出i am the child process...

为什么看上去程序中互斥的两个分支都被执行了?在一个程序的一次执行中,这当然是不可能的;但是你看到的两行输出是来自两个进程,这两个进程来自同一个程序的两次执行。

fork之后,操作系统会复制一个与父进程完全相同的子进程,虽说是父子关系,但是在操作系统看来,他们更像兄弟关系,这2个进程共享代码空间,但是数据空间是互相独立的,子进程数据空间中的内容是父进程的完整拷贝,指令指针也完全相同,但只有一点不同,如果fork成功,子进程中fork的返回值是0,父进程中fork的返回值是子进程的进程号,如果fork不成功,父进程会返回错误。

可以这样想象,2个进程一直同时运行,而且步调一致,在fork之后,他们分别作不同的工作,也就是分岔了。这也是fork为什么叫fork的原因。

在程序段里用了fork()之后程序出了分岔,派生出了两个进程。具体哪个先运行就看该系统的调度算法了。

如果需要父子进程协同,可以通过原语的办法解决。

二:

进程的创建:

创建一个进程的系统调用很简单.我们只要调用fork函数就可以了.

#include

pid_t fork();

当一个进程调用了fork以后,系统会创建一个子进程.这个子进程和父进程不同的地方只有他的进程ID和父进程ID,其他的都是一样.就象父进程克隆(clone)自己一样.当然创建两个一模一样的进程是没有意义的.为了区分父进程和子进程,我们必须跟踪fork的返回值. 当fork 掉用失败的时候(内存不足或者是用户的最大进程数已到)fork返回-1,否则fork的返回值有重要的作用.对于父进程fork返回子进程的ID,而对于fork子进程返回0.我们就是根据这个返回值来区分父子进程的. 父进程为什么要创建子进程呢?前面我们已经说过了Linux是一个多用户操作系统,在同一时间会有许多的用户在争夺系统的资源.有时进程为了早一点完成

任务就创建子进程来争夺资源. 一旦子进程被创建,父子进程一起从fork处继续执行,相互竞争系统的资源.有时候我们希望子进程继续执行,而父进程阻塞,直到子进程完成任务.这个时候我们可以调用wait或者waitpid系统调用.

总结一下有三:

1,派生子进程的进程,即父进程,其pid不变;

2,对子进程来说,fork返回给它0,但它的pid绝对不会是0;之所以fork返回0给它,是因为它随时可以调用getpid()来获取自己的pid;

3,fork之后父子进程除非采用了同步手段,否则不能确定谁先运行,也不能确定谁先结束。认为子进程结束后父进程才从fork返回的,这是不对的,fork不是这样的,vfork才这样。

【NOTE4】

首先必须有一点要清楚,函数的返回值是储存在寄存器eax中的。

其次,当fork返回时,新进程会返回0是因为在初始化任务结构时,将eax设置为0;

在fork中,把子进程加入到可运行的队列中,由进程调度程序在适当的时机调度运行。也就是从此时开始,当前进程分裂为两个并发的进程。

无论哪个进程被调度运行,都将继续执行fork函数的剩余代码,执行结束后返回各自的值。

【NOTE5】

对于fork来说,父子进程共享同一段代码空间,所以给人的感觉好像是有两次返回,其实对于调用fork的父进程来说,如果fork出来的子进程没有得到调度,那么父进程从fork系

统调用返回,同时分析sys_fork知道,fork返回的是子进程的id。再看fork出来的子进程,由copy_process函数可以看出,子进程的返回地址为ret_from_fork(和父进程在同一个代码点上返回),返回值直接置为0。所以当子进程得到调度的时候,也从fork返回,返回值为0。

关键注意两点:1.fork返回后,父进程或子进程的执行位置。(首先会将当前进程eax的值做为返回值)2.两次返回的pid存放的位置。(eax中)

进程调用copy_process得到lastpid的值(放入eax中,fork正常返回后,父进程中返回的就是lastpid)

子进程任务状态段tss的eax被设置成0,

fork.c 中

p->tss.eax=0;(如果子进程要执行就需要进程切换,当发生切换时,子进程tss中的eax 值就调入eax寄存器,子进程执行时首先会将eax的内容做为返回值)

当子进程开始执行时,copy_process返回eax的值。

fork()后,就是两个任务同时进行,父进程用他的tss,子进程用自己的tss,在切换时,各用各的eax中的值.

所以,“一次调用两次返回”是2个不同的进程!

看这一句:pid=fork()

当执行这一句时,当前进程进入fork()运行,此时,fork()内会用一段嵌入式汇编进行系统调用:int 0x80(具体代码可参见内核版本0.11的unistd.h文件的133行_syscall0函数)。这时进入内核根据此前写入eax的系统调用功能号便会运行sys_fork系统调用。接着,sys_fork中首先会调用C函数find_empty_process产生一个新的进程,然后会调用C函数copy_process将父进程的内容复制给子进程,但是子进程tss中的eax值赋值为0(这也是为什么子进程中返回0的原因),当赋值完成后,copy_process会返回新进程(该子进程)的pid,这个值会被保存到eax中。这时子进程就产生了,此时子进程与父进程拥有相同的代码空间,程序指针寄存器eip指向相同的下一条指令地址,当fork正常返回调用其的父进程后,因为eax中的值是新创建的子进程号,所以,fork()返回子进程号,执行else (pid>0);当产生进程切换运行子进程时,首先会恢复子进程的运行环境即装入子进程的tss任务状态段,其中的eax 值(copy_process中置为0)也会被装入eax寄存器,所以,当子进程运行时,fork返回的是0执行if(pid==0)。

【NOTE5】

理解它关键在于理解堆栈的切换和压栈,弹栈!

关于子进程的返回:

子进程复制了父进程的栈内容,从高到低

SS

ESP

EFLAGS

CS

EIP -----此是int 0x80 的下一条指令,也是子进程开始执行的地方!!!!

DS

ES

FS

EDX

ECX

EBX

GS

ESI

EDI

EBP

EAX(0)

由于EAX = 0,所以子进程返回0 给fork.

注:新进程的用户栈设为其父进程的用户栈(最后弹出的SS,ESP)。如果父子进程以

copy_on_write方式共用用户堆栈

(Linux之下就是这样的),而且在此之前父进程修改了该堆栈(如果父进程先返回,这几乎是肯定的),那么,系统已经为父进程创建了该用户栈的副本,父进程原来的用户栈留给了子进程。那么新进程的系统栈已经清空,新进程回到了用户态,返回到了函数fork。

【NOTE6】

关于fork的讨论与评价:

fork好不好?相比其他操作系统如Windows,Windows会有诸如CreateProcess这样的函数来创建一个与生俱来两手空空的独立的新进程。然后还有一大堆参数,指手画脚的告诉你这个那个是什么。。。

烦!!!

K.I.S.S. (Keep it simple,stupid.)是Unix的至高原则。

fork 起源于Unix 操作系统。那是贝尔实验室的K&R (这两人是Unix和C语言之父) 的一项天才发明!!! Linux由于与生俱来就与Unix血浓于水,所以继承了它的这个天才发明。

这种方法效率是很高的。因为复制的代价是很低的。在计算机网络的实现中,以及在

client/server 系统中的server 一方的实现中,fork 常常是最自然,最有效,最适宜的手段。很多人甚至怀疑,到底是先有fork 还是先有client/server,因为fork 似乎就是专门为此而设计的!更重要的好处是,这样有利于父子进程间通过pipe 建立起一种简单有效的进程间通信管道,并且产生了操作系统的用户界面即shell 的管道机制。这一点,对于Unix 的发展和应用推广,对于Uinx 程序设计环境的形成,对于Unix 程序设计风格的形成,都有非常深远的影响。可以说这是一项天才的发明,它在很大程度上改变了操作系统的发展方向。

19.1.1《变量与函数》反思

19.1.1《变量与函数》教学反思 本节课是八年级学生初步接触函数的入门课,必须让学生准确认识变量与常量的特征,初步感受现实世界各种变量之间相互联系的复杂性,同时感受到数学研究方法的化繁为简,知道在初中阶段主要研究两个变量之间的特殊对应关系。 函数定义的关键词是:“两个变量”、“唯一确定”、“与其对应”;函数的要点是:1 有两个变量,2 一个变量的值随另一个变量的值的变化而变化,3 一个变量的值确定另一个变量总有唯一确定的值与其对应;函数的实质是:两个变量之间的对应关系;学习函数的意义是:用运动变化的观念观察事物。与学习进行仔细的研究,有助于函数意义的理解,但是,不可能在一课的学时内真正理解函数的意义,继续布置作业:每个同学列举出几个反映函数关系的实例,培育学生用函数的观念看待现实世界,最后,我还说明了,函数的学习,是我们数学认识的第二个飞跃,代数式的学习,是数学认识的第一次飞跃:由具体的数、孤立的数到一般的具有普遍意义的数,函数的学习,是由静止的不变的数到运动变化的数。 在函数概念的教学中,应突出“变化”的思想和“对应”的思想。从概念的起源来看,函数是随着数学研究事物的运动、变化而出现的,他刻画了客观世界事物间的动态变化和相互依存的关系,这种关系反映了运动变化过程中的两个变量之间的制约关系。因此,变化是函数概念产生的源头,是制约概念学习的关节点,同时也是概念教学的一个重要突破口。教师可以通过大量的典型实例,让学生反复观察、反复比较、反复分析每个具体问题的量与量之间的变化关系,把静止的表达式看动态的变化过程,让他们从原来的常量、代数式、方程式和算式的静态的关系中,逐步过渡到变量、函数这些表示量与量之间的动态的关系上,使学生的认识实现 为了快速明了的引出课题,课前让学生收集一些变化的实例,从学生的生活入手,开门见山,来指明本节课的学习内容。本课的引例较为丰富,但有些内容学生解决较为困难,于是我采取了三种不同的提问方式:1.教师问,学生答; 2.学生自主回答; 3.学生合作交流回答。为了较好的突出重点突破难点,在处理教学活动过程中,让学生思考每个变化活动中反映的是哪个量随哪个量的变化而变化,并提出一个量确定时另一个量是否唯一确定的问题,在得出变量和常量概念的同时渗透函数的概念.为了更好的让学生理解变量和常量的意义,由“问题中分别涉及哪些量?哪些量是变化的,哪些量是始终不变的?”一系列问题,在借助生活实例回答的过程中,归纳总结出变量与常量的概念,并能指出具体问题中的变量与常量。函数的概念是把学生由常量数学的学习引入变量数学的学习的过程,学生初步接触函数的概念,难以理解定义中“唯一确定”的准确含义,我设置了以下二个问题:1.在前面研究的每个问题中,都出现了几个变量?它们之间是相互影响,相互制约的。2.在二个变量中,一个量在变化的过程中每取一个值,另一个量有多少个值与它对应?来理解具体实例中二个变量的特殊对应关系,初步理解函数的概念。为了进一步让学生理解“唯一对应”关系,借助函数图像,使学生直观的感受二个变量之间特殊对应关系-----唯一对应。通过这种从实际问题出发的探究方式,使学生体验从具体到抽象的认识过程,及时给出函数的定义。再从抽象转化到实际应用中去,加深学生对函数概念的理解。为了加强学生辨析函数的能力,我准备了一道思考题,Y2=X中对于X的每一个值Y都

fork函数和子进程

Fork函数 函数pid_t fork(void) 正确返回:在父进程中返回子进程的进程号,在子进程中返回0 错误返回:-1 子进程是父进程的一个拷贝。即,子进程从父进程得到了数据段和堆栈段的拷贝,这些需要分配新的内存;而对于只读的代码段,通常使用共享内存的方式访问。fork返回后,子进程和父进程都从调用fork函数的下一条语句开始执行。父进程与子进程的不同之处在于:fork的返回值不同——父进程中的返回值为子进程的进程号,而子进程为0。 以下是fork的两个示例程序: //fork.c #include #include void main () { int pid; //printf("Process [%d] begin",getpid()); //print twice printf("Process [%d] begin\n",getpid()); //print once //由于fork时pc等值的拷贝,子进程只会从fork处开始执行 pid = fork(); if (pid < 0) printf("error in fork!"); else if (pid == 0) printf("I'm child process, my pid is %d\n", getpid()); else printf("I'm parent process, my pid is %d\n", getpid()); printf("Process [%d] end\n",getpid()); return; } 输出结果: 使用printf("Process [%d] begin\n",getpid())时 Process [11155] begin I'm parent process, my pid is 11155

!函数返回值

函数返回值 int Count() { int i,j; i=100; j=200; return i+j; } 测试函数: void Test() { int k=Count(); printf("\n k[%d]\n"); } C/C++的函数返回值一般是放在寄存器eax里的,而不是在栈里。 你的这一句int k = Count()的汇编语句就是这样: mov [esp - 4], eax //eax里是300,esp - 4是局部变量k的位置 你可以在vc里做个实验: int add(int a, int b) { __asm { mov eax,a // 把参数1存入eax add eax,b // eax += 参数2, 结果在eax里 } } int main() { printf("%d\n", add(3, 4)); return 0; } 楼主需要了解下寄存器这一概念,我就不把C/C++函数的汇编代码给发出来了。 还有在汇编层面来看,函数的返回值根本就没有定论,函数可以通过多种方式返回。保存返回值在eax里只是C/C++的一个约定而已。

返回值可以放在栈里,但你在C的语言层面上可能做不到,其实随着函数的结束,mov esp, ebp这条指令过后,函数内部的局部变量就报废了。如果你之后没改变过栈的内容,你可以用栈来存返回值,但比起用寄存器来存储,存储和读取要慢的多。 自己突发奇想在vc下试了下用栈“返回”值,写了段代码: #include void __declspec(naked) __stdcall return_a_value() { int local; local = 1990; // 栈空间 __asm ret } int main() { int local = 1; return_a_value(); // 用栈返回值 printf("%d\n", local); return 0; } 汇编看c之一,简单函数调用 简单的函数调用,通过简单的函数调用反汇编可以清楚了解如下 1.栈到底是什么,如何操纵栈的? 2.参数和临时变量是以什么形式在哪存放? 3.如何传递返回值? 举例: #include

操作系统实验2课前说明-fork函数

关于fork函数的多进程编程研究 首先我们来看一下多进程的使用,我们简单的使用fork函数来实现。第一步:我们man fork一下,发现它所依赖的头文件是:sys/types.h 和unistd.h好吧,于是我们开始来使用它。 代码一: #include #include #include main() {pid_t pid; //pid_t 类型实际上就是int型 pid = fork(); if(pid < 0) printf("erro \n"); else if(pid == 0){ printf("child \n"); } else{ printf("parnts \n"); }} 这个就是最简单的fork使用了,编译一下,输出: [xdyang@SEP4020 learning]$ ./fork child

parnts 好了,大家就要开始问了,为什么会这样?同样一个if的判断语句,为什么会要进去两次?其实很简单,这个我们今天这段代码所要实现的功能:多进程。当一个进程调用fork函数后,就会创建一个子进程,子进程会拷贝(见附录二)父进程的代码段,但是拥有自己的数据段。也就是说,其实我们调用了fork函数后,相当于把这个函数代码复制了一遍,也就是产生了类似下面这样的代码: #include #include #include main(){ //进程一 pid_t pid; pid = fork(); if(pid < 0) printf("erro \n"); else if(pid == 0){ printf("child \n"); } else{ printf("parnts \n"); } //进程二 if(pid < 0) printf("erro \n"); else if(pid == 0){ printf("child \n"); } else{ printf("parnts \n"); } }

最新初中函数知识点总结与练习大全资料

考点一、平面直角坐标系 1、平面直角坐标系 在平面内画两条互相垂直且有公共原点的数轴,就组成了平面直角坐标系。 其中,水平的数轴叫做x 轴或横轴,取向右为正方向;铅直的数轴叫做y 轴或纵轴,取向上为正方向;两轴的交点O (即公共的原点)叫做直角坐标系的原点;建立了直角坐标系的平面,叫做坐标平面。 为了便于描述坐标平面内点的位置,把坐标平面被x 轴和y 轴分割而成的四个部分,分别叫做第一象限、第二象限、第三象限、第四象限。 注意:x 轴和y 轴上的点,不属于任何象限。 2、点的坐标的概念 点的坐标用(a ,b )表示,其顺序是横坐标在前,纵坐标在后,中间有“,”分开,横、纵坐标的位置不能颠倒。平面内点的坐标是有序实数对,当b a ≠时, (a ,b )和(b ,a )是两个不同点的坐标。 考点二、不同位置的点的坐标的特征 1、各象限内点的坐标的特征 点P(x,y)在第一象限0,0>>? y x 点P(x,y)在第二象限0,0>?y x 2、坐标轴上的点的特征 点P(x,y)在x 轴上0=? y ,x 为任意实数 点P(x,y)在y 轴上,y 0=?x 为任意实数 点P(x,y)既在x 轴上,又在y ?轴上x ,y 同时为零,即点P 坐标为(0,0) 3、两条坐标轴夹角平分线上点的坐标的特征 点P(x,y)在第一、三象限夹角平分线上?x 与y 相等 点P(x,y)在第二、四象限夹角平分线上?x 与y 互为相反数 4、和坐标轴平行的直线上点的坐标的特征 位于平行于x 轴的直线上的各点的纵坐标相同。 位于平行于y 轴的直线上的各点的横坐标相同。 5、关于x 轴、y 轴或远点对称的点的坐标的特征 点P 与点p ’关于x 轴对称?横坐标相等,纵坐标互为相反数 点P 与点p ’关于y 轴对称?纵坐标相等,横坐标互为相反数 点P 与点p ’关于原点对称?横、纵坐标均互为相反数 6、点到坐标轴及原点的距离 点P(x,y)到坐标轴及原点的距离: (1)点P(x,y)到x 轴的距离等于y (2)点P(x,y)到y 轴的距离等于 x (3)点P(x,y)到原点的距离等于2 2y x + 考点三、函数及其相关概念 1、变量与常量 在某一变化过程中,可以取不同数值的量叫做变量,数值保持不变的量叫做常量。 一般地,在某一变化过程中有两个变量x 与y ,如果对于x 的每一个值,y 都有唯一确定的值与它对应,那么就说x 是自变量,y 是x 的函数。 2、函数解析式 用来表示函数关系的数学式子叫做函数解析式或函数关系式。 使函数有意义的自变量的取值的全体,叫做自变量的取值范围。 3、函数的三种表示法及其优缺点 (1)解析法 :两个变量间的函数关系,有时可以用一个含有这两个变量及数字运算符号的等式表示,这种表示法叫做解析法。 (2)列表法:把自变量x 的一系列值和函数y 的对应值列成一个表来表示函数关系,这种表示法叫做列表法。 (3)图像法:用图像表示函数关系的方法叫做图像法。 4、由函数解析式画其图像的一般步骤:(1)列表:列表给出自变量与函数的一些对应值 (2)描点:以表中每对对应值为坐标,在坐标平面内描出相应的点 (3)连线:按照自变量由小到大的顺序,把所描各点用平滑的曲线连接起来。 考点四、正比例函数和一次函数 1、正比例函数和一次函数的概念 一般地,如果 b kx y +=(k ,b 是常数,k ≠0),那么y 叫做x 的一次函数。

C#多线程函数如何传参数和返回值

C#多线程函数如何传参数和返回值 提起多线程,不得不提起委托(delegates)这个概念. 我理解的委托就是具有同样参数和返回值的函数的集合. 比如 public delegate void MyDelegate(int arg); 就是这种形式的函数 void Myfuntion(int i); 的集合. 如何将一个函数加入委托的集合? MyDelegate dele = new MyDelegate(Myfuntion1); 再增加一个 dele += new MyDelegate(Myfuntion2); ... 委托函数 dele 就是具有整数参数和空返回值的函数 Myfuntion1,2的集合. 调用这个委托函数 dele(1); 就是逐个调用 Myfuntion1,2,... 一般线程函数的声明和启动 Thread t = new Thread(new ThreadStart(MyFunction)); t.Start(); 正是调用了没有参数和返回值的委托函数 ThreadStart 其中的参数MyFunction 是这个委托函数中的一员. 很明显这样无法传参数和返回值,那我们该怎么办? 答案就在委托的BeginInvoke() 方法上, BeginInvoke() 也是(异步)启动一个新线程. 例如 MyDelegate dele = new MyDelegate (MyFunction); dele.BeginInvoke(10,"abcd"); void MyFunction(int count, string str); 可以实现参数的传递. 如何收集线程函数的返回值? 与BeginInvoke 对应有个 EndInvoke 方法,而且运行完毕返回 IAsyncResult 类型的返回值.这样我们可以这样收集线程函数的返回值 MyDelegate dele = new MyDelegate (MyFunction); IAsyncResult ref = dele.BeginInvoke(10,"abcd"); ...

(完整版)初中函数知识点总结非常全

知识点一、平面直角坐标系 1、平面直角坐标系 在平面内画两条互相垂直且有公共原点的数轴,就组成了平面直角坐标系。 其中,水平的数轴叫做x 轴或横轴,取向右为正方向;铅直的数轴叫做y 轴或纵轴,取向上为正方向;两轴的交点O (即公共的原点)叫做直角坐标系的原点;建立了直角坐标系的平面,叫做坐标平面。 为了便于描述坐标平面内点的位置,把坐标平面被x 轴和y 轴分割而成的四个部分,分别叫做第一象限、第二象限、第三象限、第四象限。 注意:x 轴和y 轴上的点,不属于任何象限。 2、点的坐标的概念 点的坐标用(a ,b )表示,其顺序是横坐标在前,纵坐标在后,中间有“,”分开,横、纵坐标的位置不能颠倒。平面内点的坐标是有序实数对,当b a ≠时,(a ,b )和(b ,a )是两个不同点的坐标。 知识点二、不同位置的点的坐标的特征 1、各象限内点的坐标的特征 点P(x,y)在第一象限0,0>>?y x 点P(x,y)在第二象限0,0>?y x 2、坐标轴上的点的特征 点P(x,y)在x 轴上0=?y ,x 为任意实数 点P(x,y)在y 轴上0=?x ,y 为任意实数 点P(x,y)既在x 轴上,又在y 轴上?x ,y 同时为零,即点P 坐标为(0,0) 3、两条坐标轴夹角平分线上点的坐标的特征 点P(x,y)在第一、三象限夹角平分线上?x 与y 相等 点P(x,y)在第二、四象限夹角平分线上?x 与y 互为相反数 4、和坐标轴平行的直线上点的坐标的特征 位于平行于x 轴的直线上的各点的纵坐标相同。 位于平行于y 轴的直线上的各点的横坐标相同。 5、关于x 轴、y 轴或远点对称的点的坐标的特征 点P 与点p ’关于x 轴对称?横坐标相等,纵坐标互为相反数 点P 与点p ’关于y 轴对称?纵坐标相等,横坐标互为相反数 点P 与点p ’关于原点对称?横、纵坐标均互为相反数 6、点到坐标轴及原点的距离 点P(x,y)到坐标轴及原点的距离: (1)点P(x,y)到x 轴的距离等于y (2)点P(x,y)到y 轴的距离等于x (3)点P(x,y)到原点的距离等于22y x + 知识点三、函数及其相关概念 1、变量与常量 在某一变化过程中,可以取不同数值的量叫做变量,数值保持不变的量叫做常量。 一般地,在某一变化过程中有两个变量x 与y ,如果对于x 的每一个值,y 都有唯一确定的值与它对应,那么就说x 是自变量,y 是x 的函数。 2、函数解析式 用来表示函数关系的数学式子叫做函数解析式或函数关系式。 使函数有意义的自变量的取值的全体,叫做自变量的取值范围。 3、函数的三种表示法及其优缺点 (1)解析法 两个变量间的函数关系,有时可以用一个含有这两个变量及数字运算符号的等式表示,这种表示法叫做解析法。 (2)列表法 把自变量x 的一系列值和函数y 的对应值列成一个表来表示函数关系,这种表示法叫做列表法。 (3)图像法 用图像表示函数关系的方法叫做图像法。 4、由函数解析式画其图像的一般步骤 (1)列表:列表给出自变量与函数的一些对应值 (2)描点:以表中每对对应值为坐标,在坐标平面内描出相应的点 (3)连线:按照自变量由小到大的顺序,把所描各点用平滑的曲线连接起来。 知识点四、正比例函数和一次函数 1、正比例函数和一次函数的概念

fork函数实验总结

针对fork函数难以理解,根据网上的解释,参考他人代码,做了如下实验,并附以实验分析 2 #include 3 #include 4 #include 5 #include 6 #include 7 8 int main () 9 { 10 pid_t pc,pr; 11 pc=fork(); (gdb) 12 13 if (pc<0) 14 { 15 printf("error fork.\n"); 16 17 } 18 else if (pc==0) 19 { 20 printf("this is pc=%d\n",getpid()); 21 sleep(5); (gdb) 22 printf("5 s over\n"); 23 //exit(0); 24 } 25 pr=fork(); 26 if (pr==0) 27 { 28 printf("this is pr =%d\n",getpid()); 29 } 30 31 else if (pr>0&&pc>0) (gdb) 32 printf("this is main =%d",getpid()); 33 34 35 36 37 38 } (gdb) b 12

Breakpoint 1 at 0x804849d: file /home/lsp/fork3.c, line 12. (gdb) b 19 Breakpoint 2 at 0x80484b7: file /home/lsp/fork3.c, line 19. (gdb) b 24 Breakpoint 3 at 0x80484e4: file /home/lsp/fork3.c, line 24. (gdb) b 26 Breakpoint 4 at 0x80484ec: file /home/lsp/fork3.c, line 26. (gdb) run Starting program: /home/lsp/fork3 Detaching after fork from child process 13200. ---说明pc=fork()函数已经建立子进程 this is pc=13200 Breakpoint 1, main () at /home/lsp/fork3.c:13 13 if (pc<0) (gdb) 5 s over this is pr =13201 --说明pc=fork()进程13200启动了新的子进程pr 其pid=13201 next Breakpoint 3, main () at /home/lsp/fork3.c:25 25 pr=fork(); --父进程停在pr=fork()处, (gdb) next Detaching after fork from child process 13254. this is pr =13254 --此处pr的pid=13254 与上一个pr=13201不同,这说明此处的pr是由main创建的 Breakpoint 4, main () at /home/lsp/fork3.c:26 26 if (pr==0) (gdb) next 31 else if (pr>0&&pc>0) (gdb) next 32 printf("this is main =%d",getpid()); (gdb) next 38 } (gdb) next 0x00a6d5d6 in __libc_start_main () from /lib/libc.so.6 (gdb) next Single stepping until exit from function __libc_start_main, which has no line number information. this is main =13199 ---main函数退出,器pid=13199 Program exited with code 023. (gdb)

函数的参数

如果把函数比喻成一台机器,那么参数就是原材料,返回值就是最终产品;函数的作用就是根据不同的参数产生不同的返回值。 函数的参数 在函数定义中出现的参数可以看做是一个占位符,它没有数据,只能等到函数被调用时接收传递进来的数据,所以称为形式参数,简称形参。 函数被调用时给出的参数包含了实实在在的数据,会被函数内部的代码使用,所以称为实际参数,简称实参。 形参和实参的功能是作数据传送,发生函数调用时,实参的值会传送给形参。 形参和实参有以下几个特点: 1) 形参变量只有在函数被调用时才会分配内存,调用结束后,立刻释放内存,所以形参变量只有在函数内部有效,不能在函数外部使用。 2) 实参可以是常量、变量、表达式、函数等,无论实参是何种类型的数据,在进行函数调用时,它们都必须有确定的值,以便把这些值传送给形参,所以应该提前用赋值、输入等办法使实参获得确定值。 3) 实参和形参在数量上、类型上、顺序上必须严格一致,否则会发生“类型不匹配”的错误。

函数调用中发生的数据传送是单向的,只能把实参的值传送给形参,而不能把形参的值反向地传送给实参。因此在函数调用过程中,形参的值发生改变,而实参中的值不会变化。 【示例】计算1+2+3+...+(n-1)+n 的值。 1.#include 2.int sum(int n){ // 有参函数 3.int i; 4.for(i=n-1; i>=1; i--){ 5. n+=i; 6.} 7.printf("The inner n = %d\n",n); 8.return n; 9.} 10.int main(){ // 无参函数 11.int m, total; 12.printf("Input a number: "); 13.scanf("%d",&m); 14. total =sum(m); 15.printf("The outer m = %d \n", m); 16.printf("1+2+3+...+%d+%d = %d\n", m-1, m, total); 17.return0; 18.} 运行结果: Input a number: 100↙ The inner n = 5050 The outer m = 100

函数参数返回值总结

函数的参数、返回值总结 (一)参数 ◆函数分: 有参函数:函数名(实参列表) 无参函数:函数名() ◆有参函数调用语句中的实参应与被调函数中的形参在个数、类型、顺序上一致。 ◆参数传递时,实参向形参一一对应进行单向的值传递。值:可是数值(变量或数 组元素)或数值的地址值(指针或数组名)。 (二)返回值 函数的返回值即为函数调用后的结果,可有如下返回结果的方法: (1)通过return语句返回一个值; (2)利用地址做参数返回一个或多个值; (3)利用全局变量返回一个或多个值。 (三)例 1、170页实验内容(1):打印由正三角和倒三角组成的图形。 有一个参数,无返回值。实参向形参传递一个数值。 #include /* 有一个参数,无返回值的函数,打印正三角 */ void f1(int n) /* 形参只能是变量,用来接收实参传来的数值 */ { int i,j,k; for(k=1;k<=n;k++) {for(i=1;i<=10-k;i++) printf(" "); for(j=1;j<=k;j++) printf(" *"); printf("\n");} } /* 有一个参数,无返回值的函数,打印倒三角*/ void f2(int n) {int i,j,k; for(k=n;k>=1;k--) {for(i=1;i<=10-k;i++) printf(" "); for(j=1;j<=k;j++) printf(" *"); /*双引号内应为“空格加半角星号”*/ printf("\n");} } main() { int n; scanf("%d",&n);

一次函数知识点总结与常见题型

三乐教育名师点拔中心 学生姓名: 家长签名 基本概念 1、变量:在一个变化过程中可以取不同数值的量。 常量:在一个变化过程中只能取同一数值的量。 例题:在匀速运动公式vt s =中,v 表示速度,t 表示时间,s 表示在时间t 内所走的路程,则变量是________,常量是_______。在圆的周长公式C =2πr 中,变量是________,常量是_________. 2、函数:一般的,在一个变化过程中,如果有两个变量x 和y ,并且对于x 的每一个确定的值,y 都有唯一确定的值与其 对应,那么我们就把x 称为自变量,把y 称为因变量,y 是x 的函数。 *判断Y 是否为X 的函数,只要看X 取值确定的时候,Y 是否有唯一确定的值与之对应 例题:下列函数(1)y =πx (2)y =2x -1 (3)y =1 x (4)y =21-3x (5)y =x 2-1中,是一次函数的有( ) (A )4个 (B )3个 (C )2个 (D )1个 3、定义域:一般的,一个函数的自变量允许取值的范围,叫做这个函数的定义域。 4、确定函数定义域的方法: (1)关系式为整式时,函数定义域为全体实数;(2)关系式含有分式时,分式的分母不等于零; (3)关系式含有二次根式时,被开放方数大于等于零;(4)关系式中含有指数为零的式子时,底数不等于零; (5)实际问题中,函数定义域还要和实际情况相符合,使之有意义。 例题:下列函数中,自变量x 的取值范围是x ≥2的是( ) A .y B .y C .y D .y 函数y = x 的取值范围是___________. 已知函数22 1 +-=x y ,当11≤<-x 时,y 的取值范围是 ( ) A .2325≤<-y B .2523<0时,直线y =kx 经过三、一象限,从左向右上升,即随x 的增大y 也增大;当k <0时,?直线y =kx 经过二、四象限,从左向右下降,即随x 增大y 反而减小. (1) 解析式:y =kx (k 是常数,k ≠0) (2) 必过点:(0,0)、(1,k ) (3) 走向:k >0时,图像经过一、三象限;k <0时,?图像经过二、四象限 (4) 增减性:k >0,y 随x 的增大而增大;k <0,y 随x 增大而减小 (5) 倾斜度:|k |越大,越接近y 轴;|k |越小,越接近x 轴 例题:(1).正比例函数(35)y m x =+,当m 时,y 随x 的增大而增大. (2)若23y x b =+-是正比例函数,则b 的值是 ( ) A .0 B . 23 C .23- D .32 - .(3)函数y =(k -1)x ,y 随x 增大而减小,则k 的范围是 ( ) A .0k C .1≤k D .1

Linux的fork、exec、wait函数的分析

Linux 的fork 、exec 、wait 函数分析 I 数学与计算机学院 课程设计说明书 课 程 名 称: 操作系统原理-课程设计 课 程 代 码: 8404061 题 目: Linux 的fork 、exec 、wait 函数的分析 年级/专业/班: 学 生 姓 名: 学 号: 3 开 始 时 间: 2010 年 12 月 12 日 完 成 时 间: 2011 年 01 月 09 日 课程设计成绩: 指导教师签名: 年 月 日

Linux 的fork 、exec 、wait 函数分析 II 目 录 1 引 言 ................................................................. 1 1.1 问题的提出 ...................................................................................................................... 1 1.2国内外研究的现状 ........................................................................................................... 1 1.3任务与分析 ....................................................................................................................... 1 2代码分析结果 ............................................................ 2 2.1 数据结构 ......................................................................................................................... 2 2.1.1 struct task_struct ............................................................................................. 2 2.1.2 task ......................................................................................................................... 3 2.1.3 tarray_freelist ................................................................................................... 3 2.1.4 struct--linux_binprm ......................................................................................... 3 2.1.5进程状态 .................................................................................................................. 4 2.2常量和出错信息的意义 .................................................................................................. 4 2.3调用关系图 ...................................................................................................................... 4 2.4各模块/函数的功能及详细框图 .................................................................................... 5 2.4.1 do_fork 模块 .......................................................................................................... 5 2.4.2 get_pid 模块 .......................................................................................................... 8 2.4.3 do_execve 模块 ........................................................................................................ 10 3.4.4 do_exit 模块 ........................................................................................................ 14 3.4.5 sys_wait4模块 .................................................................................................. 18 3 总结与体会 ............................................................ 20 4 参考文献 .. (20)

变量与函数 知识讲解

变量与函数 【学习目标】 1.知道现实生活中存在变量和常量,变量在变化的过程中有其固有的范围(即变量的取值范围); 2.能初步理解函数的概念;能初步掌握确定常见简单函数的自变量取值范围的基本方法;给出自变量的一个值,会求出相应的函数值. 3. 理解函数图象上的点的坐标与其解析式之间的关系,会判断一个点是否在函数的图象上,明确交点坐标反映到函数上的含义. 4. 初步理解函数的图象的概念,掌握用“描点法”画一个函数的图象的一般步骤,对已知图象能读图、识图,从图象解释函数变化的关系. 【要点梳理】 要点一、变量、常量的概念 在一个变化过程中,我们称数值发生变化的量为变量.数值保持不变的量叫做常量. 要点诠释:一般地,常量是不发生变化的量,变量是发生变化的量,这些都是针对某个变化过程而言的.例如,60s t =,速度60千米/时是常量,时间t 和里程s 为变量. 要点二、函数的定义 一般地,在一个变化过程中. 如果有两个变量x 与y ,并且对于x 的每一个确定的值,y 都有唯一确定的值与其对应,那么我们就说 x 是自变量,y 是x 的函数. 要点诠释:对于函数的定义,应从以下几个方面去理解: (1)函数的实质,揭示了两个变量之间的对应关系; (2)对于自变量x 的取值,必须要使代数式有实际意义; (3)判断两个变量之间是否有函数关系,要看对于x 允许取的每一个值,y 是否 都有唯一确定的值与它相对应. (4)两个函数是同一函数至少具备两个条件: ①函数关系式相同(或变形后相同); ②自变量x 的取值范围相同. 否则,就不是相同的函数.而其中函数关系式相同与否比较容易注意到,自变 量x 的取值范围有时容易忽视,这点应注意. 要点三、函数的定义域与函数值 函数的自变量允许取值的范围,叫做这个函数的定义域. 要点诠释:考虑自变量的取值必须使解析式有意义。 (1)当解析式是整式时,自变量的取值范围是全体实数; (2)当解析式是分式时,自变量的取值范围是使分母不为零的实数; (3)当解析式是二次根式时,自变量的取值范围是使被开方数不小于零的实数; (4)当解析式中含有零指数幂或负整数指数幂时,自变量的取值应使相应的底数 不为零; (5)当解析式表示实际问题时,自变量的取值必须使实际问题有意义. y 是x 的函数,如果当x =a 时y =b ,那么b 叫做当自变量为a 时的函数值.在函数用记号()y f x =表示时,()f a 表示当x a =时的函数值. 要点诠释: 对于每个确定的自变量值,函数值是唯一的,但反过来,可以不唯一,即一个函数值对

进程创建之fork系统调用

4 进程创建 (1) 4.1 实验内容及要求 (1) 4.2 实验目的 (1) 4.3 实验环境 (1) 4.4 实验思路 (1) 4.5 实验代码 (2) 4.6 运行结果 (3) 4.7 实验心得 (3) 4 进程创建 4.1 实验内容及要求 利用fork()系统调用创建进程。要求如下: 编制一段程序,使用系统调用fork( )创建两个子进程,这样在此程序运行时,在系统中就有一个父进程和两个子进程在活动。每一个进程在屏幕上显示一个字符,其中父进程显示字符A,子进程分别显示字符 B和字符C。试观察、记录并分析屏幕上进程调度的情况。 4.2 实验目的 了解进程的创建过程,进一步理解进程的概念,明确进程和程序的区别。 4.3 实验环境 Ubuntu 18.04.1 LTS 64位,编译器gcc 7.3.0 (Ubuntu 7.3.0-16ubuntu3) 4.4 实验思路 (1)可用fork()系统调用来创建一个新进程。 系统调用格式:pid=fork() fork()返回值意义如下: =0:若返回值为0,表示当前进程是子进程。 >0:若返回值大于0,表示当前进程是父进程,返回值为子进程的pid值。

<0:若返回值小于0,表示进程创建失败。 如果fork()调用成功,它向父进程返回子进程的pid,并向子进程返回0,即fork()被调用了一次,但返回了两次。此时OS在内存中建立一个新进程,所建的新进程是调用fork()父进程的副本,称为子进程。子进程继承了父进程的许多特性,并具有与父进程完全相同的用户级上下文。父进程与子进程并发执行。 (2)编译和执行的方法: 编译:在shell提示符下输入gcc 源文件名 -o 可执行文件名 运行:在shell提示符下输入 ./可执行文件名 4.5 实验代码 #include #include #include int main(void) { pid_t p1=fork();pid_t p2=fork(); //迭代调用fork(),创建三个新进程printf("this is parent, pid = %d\n", getpid()); //父进程 if(p1<0||p2<0){ //fork()失败 printf("fork failed with p1=%d, p2=%d\n", p1, p2); exit(1); }if(p1==0&&p2==0){ //子进程B创建的子进程D,即孙进程 printf("D in grandson Damson, pid = %d\n", getpid()); }if(p1==0&&p2>0){ //父进程创建的子进程B printf("B in child Blueberry, pid = %d\n", getpid()); }if(p1>0&&p2==0){ //父进程创建的子进程C printf("C in child Carambola, pid = %d\n", getpid()); }if(p1>0&&p2>0){ //父进程 printf("A in parent Apple, pid = %d\n", getpid()); }return 0; }

相关文档
最新文档