实验十七 Linux下进程线程的创建

合集下载

kernel创建线程的方式

kernel创建线程的方式

在Linux内核中,可以使用`kthread_create`函数来创建新的线程。

这个函数需要传入一个`kthread_create_zombie`结构体,其中包含线程的属性、入口函数、参数等信息。

下面是`kthread_create_zombie`结构体的定义:```cstruct kthread_create_zombie {struct task_struct __task;void (*threadfn)(void *data);void *data;char name[16];};```其中,`task_struct`是Linux内核中表示进程或线程的结构体,`threadfn`是线程的入口函数,`data`是传递给入口函数的参数,`name`是线程的名称。

使用`kthread_create`函数创建线程的示例代码如下:```c#include <linux/kernel.h>#include <linux/module.h>#include <linux/kthread.h>static int my_thread(void *data){printk(KERN_INFO "My thread starts running.\n"); /* Do some work here... */printk(KERN_INFO "My thread is done.\n");return 0;}static int __init my_module_init(void){struct kthread_create_zombie thread = {.threadfn = my_thread,.data = NULL,.name = "my-thread",};struct task_struct *task;task = kthread_create(&thread, &task, NULL);if (IS_ERR(task)) {printk(KERN_ERR "Failed to create thread.\n");return PTR_ERR(task);}wake_up_process(task);return 0;}static void __exit my_module_exit(void){printk(KERN_INFO "Module exit.\n");}module_init(my_module_init);module_exit(my_module_exit);MODULE_LICENSE("GPL");```在上面的示例中,我们定义了一个名为`my_thread`的线程入口函数,并在模块初始化函数中创建了一个新的线程。

操作系统实验报告(进程的创建)

操作系统实验报告(进程的创建)

wait(0);printf("parent process doesn't change the glob and loc:\n");printf("glob=%d,loc=%d\n",glob,loc);exit(0);}运行结果:2、理解vofork()调用:程序代码:#include<stdio.h>#include<sys/types.h>#include<unistd.h>int glob=3;int main(void){pid_t pid;int loc=3;if((pid=vfork())<0){printf("vfork() error\n");exit(0);}else if(pid==0){glob++;loc--;printf("child process changes the glob and loc\n");exit(0);}elseprintf ("parent process doesn't change the glob and loc\n");printf("glob=%d,val=%d\n",glob,loc);}运行结果:3、给进程指定一个新的运行程序的函数exec().程序代码:printe1.c代码:#include<stdio.h>int main(int argc,char * argv[]){int n;char * * ptr;extern char * * environ;for(n=0;n<argc;n++)printf("argv[%d]:%s\n",n,argv[n]);for(ptr=environ; * ptr!=0;ptr++)printf("%s\n",* ptr);exit(0);}file4.c代码如下:#include<stdio.h>#include<sys/types.h>#include<unistd.h>#include<sys/wait.h>char * env_list[]={"USER=root","PATH=/root/",NULL};int main(){pid_t pid;if((pid=fork())<0){printf("fork error!\n");exit(0);}else if(pid==0){if(execle("/root/print1","print1","arg1","arg2",(char *)0,env_list)<0) printf("execle error!\n");exit(0);}if((waitpid(pid,NULL,0))<0)printf("WAIT ERROR!\n");exit(0);if((pid=fork())<0){printf("fork error!\n");exit(0);}else if(pid==0){if(execlp("print1","print1","arg1",(char *)0)<0)printf("execle error!\n");exit(0);}exit(0);}运行结果:4、进程终止函数exit()。

linux下的CC++多进程多线程编程实例详解

linux下的CC++多进程多线程编程实例详解

linux下的CC++多进程多线程编程实例详解linux下的C\C++多进程多线程编程实例详解1、多进程编程#include <stdlib.h>#include <sys/types.h>#include <unistd.h>int main(){pid_t child_pid;/* 创建⼀个⼦进程 */child_pid = fork();if(child_pid == 0){printf("child pid\n");exit(0);}else{printf("father pid\n");sleep(60);}return 0;}2、多线程编程#include <stdio.h>#include <pthread.h>struct char_print_params{char character;int count;};void *char_print(void *parameters){struct char_print_params *p = (struct char_print_params *)parameters;int i;for(i = 0; i < p->count; i++){fputc(p->character,stderr);}return NULL;}int main(){pthread_t thread1_id;pthread_t thread2_id;struct char_print_params thread1_args;struct char_print_params thread2_args;thread1_args.character = 'x';thread1_args.count = 3000;pthread_create(&thread1_id, NULL, &char_print, &thread1_args);thread2_args.character = 'o';thread2_args.count = 2000;pthread_create(&thread2_id, NULL, &char_print, &thread2_args);pthread_join(thread1_id, NULL);pthread_join(thread2_id, NULL);return 0;}3、线程同步与互斥1)、互斥pthread_mutex_t mutex;pthread_mutex_init(&mutex, NULL);/*也可以⽤下⾯的⽅式初始化*/pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER; pthread_mutex_lock(&mutex);/* 互斥 */thread_flag = value;pthread_mutex_unlock(&mutex);2)、条件变量int thread_flag = 0;pthread_mutex_t mutex;pthread_cond_t thread_flag_cv;\void init_flag(){pthread_mutex_init(&mutex, NULL);pthread_cond_init(&thread_flag_cv, NULL);thread_flag = 0;}void *thread_function(void *thread_flag){while(1){pthread_mutex_lock(&mutex);while(thread_flag != 0 ){pthread_cond_wait(&thread_flag_cv, &mutex);}pthread_mutex_unlock(&mutex);do_work();}return NULL;}void set_thread_flag(int flag_value){pthread_mutex_lock(&mutex);thread_flag = flag_value;pthread_cond_signal(&thread_flag_cv);pthread_mutex_unlock(&mutex);}感谢阅读,希望能帮助到⼤家,谢谢⼤家对本站的⽀持!。

Linux系统编程之进程控制(进程创建、终止、等待及替换)

Linux系统编程之进程控制(进程创建、终止、等待及替换)

Linux系统编程之进程控制(进程创建、终⽌、等待及替换)进程创建在上⼀节讲解进程概念时,我们提到fork函数是从已经存在的进程中创建⼀个新进程。

那么,系统是如何创建⼀个新进程的呢?这就需要我们更深⼊的剖析fork 函数。

1.1 fork函数的返回值调⽤fork创建进程时,原进程为⽗进程,新进程为⼦进程。

运⾏man fork后,我们可以看到如下信息:#include <unistd.h>pid_t fork(void);fork函数有两个返回值,⼦进程中返回0,⽗进程返回⼦进程pid,如果创建失败则返回-1。

实际上,当我们调⽤fork后,系统内核将会做:分配新的内存块和内核数据结构(如task_struct)给⼦进程将⽗进程的部分数据结构内容拷贝⾄⼦进程添加⼦进程到系统进程列表中fork返回,开始调度1.2 写时拷贝在创建进程的过程中,默认情况下,⽗⼦进程共享代码,但是数据是各⾃私有⼀份的。

如果⽗⼦只需要对数据进⾏读取,那么⼤多数的数据是不需要私有的。

这⾥有三点需要注意:第⼀,为什么⼦进程也会从fork之后开始执⾏?因为⽗⼦进程是共享代码的,在给⼦进程创建PCB时,⼦进程PCB中的⼤多数数据是⽗进程的拷贝,这⾥⾯就包括了程序计数器(PC)。

由于PC中的数据是即将执⾏的下⼀条指令的地址,所以当fork返回之后,⼦进程会和⽗进程⼀样,都执⾏fork之后的代码。

第⼆,创建进程时,⼦进程需要拷贝⽗进程所有的数据吗?⽗进程的数据有很多,但并不是所有的数据都要⽴马使⽤,因此并不是所有的数据都进⾏拷贝。

⼀般情况下,只有当⽗进程或者⼦进程对某些数据进⾏写操作时,操作系统才会从内存中申请内存块,将新的数据拷写⼊申请的内存块中,并且更改页表对应的页表项,这就是写时拷贝。

原理如下图所⽰:第三,为什么数据要各⾃私有?这是因为进程具有独⽴性,每个进程的运⾏不能⼲扰彼此。

1.3 fork函数的⽤法及其调⽤失败的原因fork函数的⽤法:⼀个⽗进程希望复制⾃⼰,通过条件判断,使⽗⼦进程分流同时执⾏不同的代码段。

操作系统第二次实验报告——Linux创建进程及可执行文件结构分析

操作系统第二次实验报告——Linux创建进程及可执行文件结构分析

操作系统第⼆次实验报告——Linux创建进程及可执⾏⽂件结构分析0 个⼈信息张樱姿201821121038计算18121 实验⽬的熟练Linux创建进程fork操作。

2 实验内容在服务器上⽤VIM编写⼀个程序:⼀个进程创建两个⼦进程。

查看进程树查看进程相关信息3 实验报告 3.1编写程序创建两个⼦进程1 #include<sys/types.h>2 #include<stdio.h>3 #include<unistd.h>45int main(){6 pid_t cpid1 = fork(); //创建⼦进程178if(cpid1<0){9 printf("fork cd1 failed\n");10 }11else if(cpid1==0){12 printf("Child1:pid: %d, ppid: %d\n",getpid(),getppid());13 }14else{15 pid_t cpid2 = fork(); //创建⼦进程216if(cpid2<0){17 printf("fork cd2 failed\n");18 }19else if(cpid2==0){20 printf("Child2:pid: %d, ppid: %d\n",getpid(),getppid());21 }22else{23 printf("Parent: pid :%d\n",getpid());24 }25 }26 }编译运⾏后的结果:3.2打印进程树 添加sleep函数以挂起进程,⽅便打印进程树:1 #include<sys/types.h>2 #include<stdio.h>3 #include<unistd.h>45int main(){6 pid_t cpid1 = fork();78if(cpid1<0){9 printf("fork cd1 failed\n");10 }11else if(cpid1==0){12 printf("Child1:pid: %d, ppid: %d\n",getpid(),getppid());13 sleep(30); //挂起30秒14 }15else{16 pid_t cpid2 = fork();17if(cpid2<0){18 printf("fork cd2 failed\n");19 }20else if(cpid2==0){21 printf("Child2:pid: %d, ppid: %d\n",getpid(),getppid());22 sleep(30); //挂起30秒23 }24else{25 printf("Parent: pid :%d\n",getpid());26 sleep(60); //挂起60秒27 }28 }29 }pstree -p pid #打印进程树 3.3 解读进程相关信息 3.3.1 解释执⾏ps -ef后返回结果中每个字段的含义 ps -ef输出格式 :UID PID PPID C STIME TTY TIME CMDUID: User ID,⽤户ID。

linux的进程管理实验总结

linux的进程管理实验总结

linux的进程管理实验总结Linux的进程管理实验总结1. 引言Linux中的进程管理是操作系统的核心功能之一,在实际的系统运行中起着重要的作用。

进程管理能够有效地分配系统资源、管理进程的运行状态和优先级,以及监控进程的行为。

本文将以Linux的进程管理实验为主题,分步骤介绍实验过程及总结。

2. 实验目的本次实验的目的是理解Linux中进程的概念,掌握进程的创建、运行和终止的基本操作,以及进程的状态转换过程。

3. 实验环境本次实验使用的是Linux操作系统,可以选择使用虚拟机安装Linux或者使用Linux主机进行实验。

4. 实验步骤4.1 进程的创建在Linux中,可以使用系统调用fork()来创建一个新的子进程。

在实验中,可以编写一个简单的C程序来调用fork()系统调用,实现进程的创建。

具体步骤如下:(1)创建一个新的C程序文件,例如"process_create.c"。

(2)在C程序文件中,包含必要的头文件,如<stdio.h>和<unistd.h>。

(3)在C程序文件中,编写main()函数,调用fork()函数进行进程的创建。

(4)编译并运行该C程序文件,观察控制台输出结果。

实验中,可以通过观察控制台输出结果,判断新的子进程是否被成功创建。

4.2 进程的运行在Linux中,通过调用系统调用exec()可以用一个新的程序替换当前进程的执行。

可以使用exec()函数来实现进程的运行。

具体步骤如下:(1)创建一个新的C程序文件,例如"process_run.c"。

(2)在C程序文件中,包含必要的头文件和函数声明,如<stdio.h>和<unistd.h>。

(3)在C程序文件中,编写main()函数,调用execl()函数来执行一个可执行程序。

(4)编译并运行该C程序文件,观察控制台输出结果。

实验中,可以通过观察控制台输出结果,判断新的程序是否被成功执行。

Linux系统线程创建及同步互斥方法简要说明(供查考)

Linux系统线程创建及同步互斥方法简要说明(供查考)

Linux系统线程创建及同步互斥方法简要说明(供查考)1、.POSIX线程函数的定义在头文件pthread.h中,所有的多线程程序都必须通过使用#include<pthread.h>包含这个头文件2、用gcc编译多线程程序时,必须与pthread函数库连接。

可以使用以下两种方式编译(建议使用第一种)(1)gcc –D_REENTRANT -o 编译后的目标文件名源文件名-lpthread例如:gcc –D_REENTRANT -o pthread_create pthread_create.c -lpthread (执行该编译结果的方式为:./pthread_create)(2)gcc -pthread -o 编译后的文件名源文件名例如:gcc -pthread -o example example.c一、需要用到的函数的用法提示1、创建线程函数pthread_t a_thread; /*声明a_thread变量,用来存放创建的新线程的线程ID(线程标识符)*/int res=pthread_create(&a_thread,NULL,thread_function,NULL);/*创建一个执行函数thread_function的新线程,线程ID存放在变量a_thread */ 2、退出线程函数pthread_exit(NULL);/*那个线程在执行中调用了该方法,那个线程就退出*/创建和退出线程实例3、连接(等待)线程函数int error;int *exitcodeppthread_t tid; /*用来表示一个已经存在的线程*/error=pthread_join(tid,&exitcodep); /*执行该方法的线程将要一直等待,直到tid 表示的线程执行结束,exitcodep 存放线程tid退出时的返回值*/4、返回线程ID的函数pthread_t t/*声明表示线程的变量t */t=pthread_self( ) /*返回调用该方法的线程的线程ID*/5、判断两个线程是否相等的函数(pthread_equal)int pthread_equal(pthread_t t1, pthread_t t2);/*判断线程t1与线程t2是否线程ID相等*/二、线程同步1、使用互斥量同步线程(实现互斥)(1)互斥量的创建和初始化pthread_mutex_t a_mutex=PTHREAD_MUTEX_INITIALIZER/*声明a_mutex为互斥量,并且初始化为PTHREAD_MUTEX_INITIALIZER */ (2)锁定和解除锁定互斥量pthread_mutex_t a_mutex=PTHREAD_MUTEX_INITIALIZER/*声明互斥量a_mutex*/int rc=pthread_mutex_lock(&a_mutex) /*锁定互斥量a_mutex*/ ………………………………/*锁定后的操作*/int rd= pthread_mutex_unlock(&a_mutex) /*解除对互斥量a_mutex的锁定*/例子:利用互斥量来保护一个临界区pthread_mutex_t a_mutex=PTHREAD_MUTEX_INITIALIZER;pthread_mutex_lock(&a_mutex) /*锁定互斥量a_mutex*//*临界区资源*/pthread_mutex_unlock(&a_mutex) /*解除互斥量a_mutex的锁定*/(3)销毁互斥量Int rc=pthread_mutex_destory(&a_mutex) /*销毁互斥量a_mutex*/2、用条件变量同步线程(实现真正的同步)条件变量是利用线程间共享的全局变量进行同步的一种机制,主要包括两个动作:一个线程等待"条件变量的条件成立"而挂起;另一个线程使"条件成立"(给出条件成立信号)。

建立线程的实验报告(3篇)

建立线程的实验报告(3篇)

第1篇一、实验目的1. 理解线程的概念和作用;2. 掌握在C++中创建和使用线程的方法;3. 了解线程同步机制,如互斥锁、条件变量等;4. 分析线程间的通信和协作。

二、实验环境1. 操作系统:Windows 102. 编译器:Visual Studio 20193. 编程语言:C++三、实验内容本次实验主要分为以下几个部分:1. 线程的基本概念和作用;2. 创建和使用线程;3. 线程同步机制;4. 线程间的通信和协作。

四、实验步骤1. 线程的基本概念和作用线程是程序执行过程中的一个独立单位,它包含程序执行所需的基本信息,如程序计数器、寄存器等。

线程的主要作用是提高程序的执行效率,实现并发执行。

2. 创建和使用线程在C++中,可以使用`std::thread`类来创建线程。

以下是一个简单的例子:```cppinclude <iostream>void printNumber(int n) {for (int i = 0; i < n; ++i) {std::cout << i << std::endl;}}int main() {std::thread t1(printNumber, 10); // 创建线程,传入函数和参数std::thread t2(printNumber, 20);t1.join(); // 等待线程t1执行完毕t2.join(); // 等待线程t2执行完毕return 0;}```在上面的代码中,我们创建了两个线程`t1`和`t2`,分别执行`printNumber`函数。

使用`join`函数可以等待线程执行完毕。

3. 线程同步机制线程同步机制用于解决多线程在执行过程中可能出现的数据竞争、死锁等问题。

以下是一些常用的线程同步机制:(1)互斥锁(Mutex)互斥锁用于保护共享资源,确保同一时刻只有一个线程可以访问该资源。

以下是一个使用互斥锁的例子:```cppinclude <iostream>include <mutex>std::mutex mtx;void printNumber(int n) {mtx.lock(); // 获取互斥锁for (int i = 0; i < n; ++i) {std::cout << i << std::endl;}mtx.unlock(); // 释放互斥锁}int main() {std::thread t1(printNumber, 10);std::thread t2(printNumber, 20);t1.join();t2.join();return 0;}```(2)条件变量(Condition Variable)条件变量用于在线程间实现等待和通知机制。

  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。



父进程创建子进程后,子进程一般要执行不同的程 序.为了调用系统程序,我们可以使用系统调用 exec 族调用。 Exec函数可以把当前进程替换为一个新进程 exec函数包含以下五个函数:
#include<unistd.h> int execl(constchar* path, const char* arg, …); int execlp(constchar* file, const char* arg, …); int execle(constchar* path, const char* arg, char* const envp[]); int execv(constchar* path, char* const argv[]); int execvp(constchar* file, char* const argv[]);
4.11 Fibonacci序列是一组:0,1,1,2,3,5,8,…… fib0=0; fib1=1; fibn=fibn-1+fibn-2; 使用phtread多线程编程程序来生成Fibonacci序列。 程序应该这样工作:用户运行程序时,在命令行输 入要产生Fibonaci序列数,然后程序创建一个新的 线程来产生Fibonacci数,把这个序列放到线程共 享的数据中(数组可能是一种最方便的数据结构)。 当线程执行完成后,父线程将输出子线程产生的序 列。由于在子线程结束前,父线程不能开始输出 Fibonacci序列,因此,父线程需要等子线程结束
等待线程结束,收集线程信息 参数1.指定了将要等待结束的线程 参数2. 是一个指针,指向了一个指针,后者指向线 程的返回值
例子
#include <stdio.h> #include <unistd.h> #include <stdlib.h> #include <pthread.h> void *thread_function(void *arg); char message[] = "Hello world"; int main(){ int res; pthread_t a_thread; void *thread_result; res = pthread_create(&a_thread,NULL,thread_function,(void*)message); if(res !=0){perror("Thread creation failed"); exit(EXIT_FAILURE);} printf("Waiting for thread to finish..\n");
4/19/2013
Linux进程创建


Linux进程的族亲关系
Linux进程之间不是孤立存在的,它们之间有一定的族亲关系。
产生
进程1
进程2
父进程

子进程
一个进程的子进程可以多于一个。 一个进程只会有一个父进程。
Linux进程创建

系统加电启动后,系统中只有一个进程——初始化 进程,又称init进程,是所有进程的祖先进程,它 的进程pid=1 在linux中除了init进程是由系统启动时创建的外, 其他所有进程都是由当前进程使用系统调用fork() 创建的。 进程创建后父子进程在系统中并发执行。

4/19/2013
Linux 创建线程

线程:一个进程内部的一个控制序列 每个进程都至少有一个执行线程
创建线程与fork区别
fork: 当前进程的拷贝,有自己的变量和PID,时 间调度独立,执行几乎完全独立于父进程 新线程:有自己独立栈(有局部变量) 但与创建者共享全局变量,文件描述,信号句柄和 当前状态等
Linux下进程和线程的创建
吴国伟

学习Linux中创建进程的方法 学习系统调用Fork的使用方法 学习系统调用Exec族的使用方法

学习Linux中创建线程的方法 学习pthread_create使用方法

PID:进程号。 PPID:父进程号。


UID:其创建者的用户标识号,父子进程有相同 的UID值。
int pthread_create( pthread_t *thread, pthread_attr_t *attr, void *(*start_routine)(void*), void *arg) 参数 1. 该指针指向的变量中将被写入一个标识符,用 该 标识符来引用新线程 2. 设置线程属性,一般不用设为null 3. 表示线程将要启动执行的函数 4. 该函数需要的参数 调用成功返回值为0,否则返回错误代码
◦ 父进程:返回子进程的PID;
最初的进程
◦ 子进程:返回0;
◦ 出现错误:返回-1;
fork()
返回一个 新的pid 返 回 0
原进程继续 执行
新进程
#include <unistd.h> #include <sys/types.h> #include <stdio.h> int main() { int pid = fork(); if (pid == -1 ){ printf("error!\n"); } else if( pid == 0 ) { printf("This is the child process!\nchild process id = %d\n",getpid()); } else { printf("This is the parent process!\nparent process id = %d\n",getpid()); } return 0;}
res = pthread_join(a_thread, &thread_result); if(res !=0){perror("Thread join failed"); exit(EXIT_FAILURE);} printf("Thread joined, it returned %s\n",(char*)thread_result); printf("Message is now %s\n",message); exit(EXIT_SUCCESS); } void *thread_function(void *arg){ printf("thread_function is running. Argument was %s \n",(char *)arg); sleep(3); strcpy(message,"Bye!"); pthread_exit("Thank you for the CPU time"); }

此时一个进程“分裂”成两个进程:父进程和子进程。
区别:进程ID。利用getpid()来得到进程号。


通过fork创建一个新进程,系统复制当前进程,在进程表中创建一个新的表 项 新进程几乎与原进程相同,执行代码也相同 但有自己的数据空间,环境,文件描述符等


Fork()系统调用

fork调用的一个奇妙之处就是它仅仅被调用一次, 却能够返回两次,它可能有三种不同的返回值:

线程优缺点


优点: 一般而言,线程创建以及线程之间的切换需要操作 系统做的工作比进程少得多。 例子:编辑文档过程的多线程,一个线程负责用户 输入和编辑,一个线程负责字数统计


缺点: 多线程程序因变量共享可能引发潜在错误,调试困 难
pthread_creat



运行结果
Waiting for thread to finish... thread_function is running. Argument was Hello world Thread joined, it returned Thank you for the CPU time Message is now Bye!
#include<sys/types.h> #include<sys/wait.h> pid_t wait(int *stat_loc) 父进程调用
1. wait系统调用将暂停父进程直到它的子进程结束为止 2. 返回子进程的PID 3. 状态信息允许父进程了解子进程的退出状态,即子进程 main函数返回值或子进程中exit函数的退出码 4. 若stat_loc不为空指针,状态信息被写入他指定的为止
4/19/2013

其中只有execve是真正意义上的系统调用,其它 都是在此基础上经过包装的库函数。 #include<unistd.h> intexecve(constchar* path, char* const argv[], char* const envp[]);
◦ path:可执行文件路径。 ◦ argv[]:要执行的文件名或命令名。 ◦ envp[]:环境变量,可省略
4/19/2013

运行结果: This is the child process! child process id = 4800 This is the parent process! parent process id = 4799

4/19/2013
Fork()系统调用
等待一个进程结束
void pthread_exit(void *retval)

调用上面函数终止线程 返回一个指向某个对象的指针 注意:不能用它来返回一个指向局部变量的指针, 因为线程调用该函数后,这个局部变量就不存在了
pthread_join

int pthread_join( pthread_t th, void **thread_return)
相关文档
最新文档