中山高端网站建设公司,企业直播解决方案,wordpress 插件汉化教程,微网页制作软件手机版之前写了两篇文章#xff0c;都是针对Linux这个系统的#xff0c;为什么?我为什么这么喜欢写这个系统的知识#xff0c;可能就是为了今天的内容多线程系列#xff0c;现在多线程不是一个面试重点 啊#xff0c;那如果你能深入系统内核回答这个知识点#xff0c;面试官会…之前写了两篇文章都是针对Linux这个系统的为什么?我为什么这么喜欢写这个系统的知识可能就是为了今天的内容多线程系列现在多线程不是一个面试重点 啊那如果你能深入系统内核回答这个知识点面试官会怎么想?你会不会占据面试的主动权(我不会说今天被一个面试者惊艳到了的)今天我就开始一个系列的内容多线程--高并发深入的给大家讲解我就不信讲不明白这么个小东西有问题的地方希望大家能够指出谢谢大家一起成长今天我们将第一个知识点进程Linux 内核如何描述一个进程?1. Linux 的进程进程的术语是 process是 Linux 最基础的抽象另一个基础抽象是文件。最简单的理解进程就是执行中 (executing, 不等于running) 的程序。更准确一点的理解进程包括执行中的程序以及相关的资源 (包括cpu状态、打开的文件、挂起的信号、tty、内存地址空间等)。一种简洁的说法进程 n*执行流 资源n1。Linux 进程的特点通过系统调用 fork() 创建进程fork() 会复制现有进程来创建一个全新的进程。内核里并不严格区分进程和线程。从内核的角度看调度单位是线程 (即执行流)。可以把线程看做是进程里的一条执行流1个进程里可以有1个或者多个线程。内核里常把进程称为 task 或者 thread这样描述更准确因为许多进程就只有1条执行流。内核通过轻量级进程 (lightweight process) 来支持多线程。1个轻量级进程就对应1个线程轻量级进程之间可以共享打开的文件、地址空间等资源。2. Linux 的进程描述符2.1 task_struct内核里通过 task_struct 结构体来描述一个进程称为进程描述符 (process descriptor)它保存着支撑一个进程正常运行的所有信息。每一个进程即便是轻量级进程(即线程)都有1个 task_struct。 (include\linux) struct task_struct { struct thread_info thread_info; volatile long state; void *stack; [...] struct mm_struct *mm; [...] pid_t pid; [...] struct task_struct *parent; [...] char comm[TASK_COMM_LEN]; [...] struct files_struct *files; [...] struct signal_struct *signal; } 这是一个庞大的结构体不仅有许多进程相关的基础字段还有许多指向其他数据结构的指针。它包含的字段能完整地描述一个正在执行的程序包括 cpu 状态、打开的文件、地址空间、挂起的信号、进程状态等。作为初学者先简单地了解部分字段就好struct thread_info thread_info: 进程底层信息平台相关下面会详细描述。long state: 进程当前的状态下面是几个比较重要的进程状态以及它们之间的转换流程。void *stack: 指向进程内核栈下面会解释。struct mm_struct *mm: 与进程地址空间相关的信息都保存在一个叫内存描述符 (memory descriptor) 的结构体 (mm_struct) 中。pid_t pid: 进程标识符本质就是一个数字是用户空间引用进程的唯一标识。struct task_struct *parent: 父进程的 task_struct。 char comm[TASK_COMM_LEN]: 进程的名称。 struct files_struct *files: 打开的文件表。 struct signal_struct *signal: 信号处理相关。 其他字段等到有需要的时候再回过头来学习。 当发生系统调用或者进程切换时内核如何找到 task_struct ?对于 ARM 架构答案是通过内核栈 (kernel mode stack)。为什么要有内核栈?因为内核是可重入的在内核中会有多条与不同进程相关联的执行路径。因此不同的进程处于内核态时都需要有自己私有的进程内核栈 (process kernel stack)。当进程从用户态切换到内核态时所使用的栈会从用户栈切换到内核栈。至于是如何切换的关键词是系统调用这不是本文关注的重点先放一边学习内核要懂得恰当的时候忽略细节。当发生进程切换时也会切换到目标进程的内核栈。同上关键词是硬件上下文切换 (hardware context switch)忽略具体实现。无论何时只要进程处于内核态就会有内核栈可以使用否则系统就离崩溃不远了。ARM 架构的内核栈和 task_struct 的关系如下内核栈的长度是 THREAD_SIZE对于 ARM 架构一般是 2 个页框的大小即 8KB。内核将一个较小的数据结构 thread_info 放在内核栈的底部它负责将内核栈和 task_struct 串联起来。thread_info 是平台相关的在 ARM 架构中的定义如下// (arch\arm\include\asm) struct thread_info { unsigned long flags; /* low level flags */ int preempt_count; /* 0 preemptable, 0 bug */ mm_segment_t addr_limit; /* address limit */ struct task_struct *task; /* main task structure */ [...] struct cpu_context_save cpu_context; /* cpu context */ [...] }; thread_info 保存了一个进程能被调度执行的最底层信息(low level task data)例如struct cpu_context_save cpu_context 会在进程切换时用来保存/恢复寄存器上下文。内核通过内核栈的栈指针可以快速地拿到 thread_info// (include\linux) static inline struct thread_info *current_thread_info(void) { // current_stack_pointer 是当前进程内核栈的栈指针 return (struct thread_info *) (current_stack_pointer ~(THREAD_SIZE - 1)); } 然后通过 thread_info 找到 task_struct: // current.h (include\asm-generic) #define current (current_thread_info()-task) 内核里通过 current 宏可以获得当前进程的 task_struct。2.3 task_struct 的分配和初始化当上层应用使用 fork() 创建进程时内核会新建一个 task_struct。进程的创建是个复杂的工作可以延伸出无数的细节。这里我们只是简单地了解一下 task_struct 的分配和部分初始化的流程。fork() 在内核里的核心流程dup_task_struct() 做了什么?至于设置内核栈里做了什么涉及到了进程的创建与切换不在本文的关注范围内以后再研究了。3. 实验打印 task_struct / thread_info / kernel mode stack实验目的梳理 task_struct / thread_info / kernel mode stack 的关系。实验代码实验代码 #include linux/ #include linux/ #include linux/ static void print_task_info(struct task_struct *task) { printk(KERN_NOTICE %10s %5d task_struct (%p) / stack(%p~%p) / thread_info-task (%p), task-comm, task-pid, task, task-stack, ((unsigned long *)task-stack) THREAD_SIZE, task_thread_info(task)-task); } static int __init task_init(void) { struct task_struct *task current; printk(KERN_INFO task module init\n); print_task_info(task); do { task task-parent; print_task_info(task); } while (task-pid ! 0); return 0; } module_init(task_init); static void __exit task_exit(void) { printk(KERN_INFO task module exit\n ); } module_exit(task_exit); 运行效果task module init insmod 3123 task_struct (edb42580) / stack(ed46c000~ed474000) / thread_info-task (edb42580) bash 2393 task_struct (eda13e80) / stack(c9dda000~c9de2000) / thread_info-task (eda13e80) sshd 2255 task_struct (ee5c9f40) / stack(c9d2e000~c9d36000) / thread_info-task (ee5c9f40) sshd 543 task_struct (ef15f080) / stack(ee554000~ee55c000) / thread_info-task (ef15f080) systemd 1 task_struct (ef058000) / stack(ef04c000~ef054000) / thread_info-task (ef058000) 在程序里我们通过 task_struct 找到 stack然后通过 stack 找到 thread_info最后又通过 thread_info-task 找到 task_struct。到这里不知道你对进程的概念是不是有了一个清晰的理解但是上面是通过Linux进行了线程的展示在日常的工作中代码的实现和编写我们还是以Java为主那我们来看一下Java进程进程的创建Java提供了两种方法用来启动进程或其它程序使用Runtime的exec()方法使用ProcessBuilder的start()方法 ProcessBuilderProcessBuilder类是J2SE 1.5在中新添加的一个新类此类用于创建操作系统进程它提供一种启动和管理进程(也就是应用程序)的方法。在J2SE 1.5之前都是由Process类处来实现进程的控制管理。每个 ProcessBuilder 实例管理一个进程属性集。start() 方法利用这些属性创建一个新的 Process 实例。start() 方法可以从同一实例重复调用以利用相同的或相关的属性创建新的子进程。每个进程生成器管理这些进程属性命令 是一个字符串列表它表示要调用的外部程序文件及其参数(如果有)。在此表示有效的操作系统命令的字符串列表是依赖于系统的。例如每一个总体变量通常都要成为此列表中的元素但有一些操作系统希望程序能自己标记命令行字符串——在这种系统中Java 实现可能需要命令确切地包含这两个元素。环境 是从变量 到值 的依赖于系统的映射。初始值是当前进程环境的一个副本(请参阅 ())。工作目录。默认值是当前进程的当前工作目录通常根据系统属性 来命名。redirectErrorStream 属性。最初此属性为 false意思是子进程的标准输出和错误输出被发送给两个独立的流这些流可以通过 () 和 () 方法来访问。如果将值设置为 true标准错误将与标准输出合并。这使得关联错误消息和相应的输出变得更容易。在此情况下合并的数据可从 () 返回的流读取而从 () 返回的流读取将直接到达文件尾。修改进程构建器的属性将影响后续由该对象的 start() 方法启动的进程但从不会影响以前启动的进程或 Java 自身的进程。大多数错误检查由 start() 方法执行。可以修改对象的状态但这样 start() 将会失败。例如将命令属性设置为一个空列表将不会抛出异常除非包含了 start()。注意此类不是同步的。如果多个线程同时访问一个 ProcessBuilder而其中至少一个线程从结构上修改了其中一个属性它必须 保持外部同步。构造方法摘要ProcessBuilder(List command) 利用指定的操作系统程序和参数构造一个进程生成器。 ProcessBuilder(String... command) 利用指定的操作系统程序和参数构造一个进程生成器。 方法摘要List command() 返回此进程生成器的操作系统程序和参数。 ProcessBuilder command(List command) 设置此进程生成器的操作系统程序和参数。 ProcessBuilder command(String... command) 设置此进程生成器的操作系统程序和参数。 File directory() 返回此进程生成器的工作目录。 ProcessBuilder directory(File directory) 设置此进程生成器的工作目录。 Map environment() 返回此进程生成器环境的字符串映射视图。 boolean redirectErrorStream() 通知进程生成器是否合并标准错误和标准输出。 ProcessBuilder redirectErrorStream(boolean redirectErrorStream) 设置此进程生成器的 redirectErrorStream 属性。 Process start() 使用此进程生成器的属性启动一个新进程。 1.2 Runtime每个 Java 应用程序都有一个 Runtime 类实例使应用程序能够与其运行的环境相连接。可以通过 getRuntime 方法获取当前运行时。应用程序不能创建自己的 Runtime 类实例。但可以通过 getRuntime 方法获取当前Runtime运行时对象的引用。一旦得到了一个当前的Runtime对象的引用就可以调用Runtime对象的方法去控制Java虚拟机的状态和行为。Java代码 收藏代码void addShutdownHook(Thread hook) 注册新的虚拟机来关闭挂钩。 int availableProcessors() 向 Java 虚拟机返回可用处理器的数目。 Process exec(String command) 在单独的进程中执行指定的字符串命令。 Process exec(String[] cmdarray) 在单独的进程中执行指定命令和变量。 Process exec(String[] cmdarray, String[] envp) 在指定环境的独立进程中执行指定命令和变量。 Process exec(String[] cmdarray, String[] envp, File dir) 在指定环境和工作目录的独立进程中执行指定的命令和变量。 Process exec(String command, String[] envp) 在指定环境的单独进程中执行指定的字符串命令。 Process exec(String command, String[] envp, File dir) 在有指定环境和工作目录的独立进程中执行指定的字符串命令。 void exit(int status) 通过启动虚拟机的关闭序列终止当前正在运行的 Java 虚拟机。 long freeMemory() 返回 Java 虚拟机中的空闲内存量。 void gc() 运行垃圾回收器。 InputStream getLocalizedInputStream(InputStream in) 已过时。 从 JDK 开始将本地编码字节流转换为 Unicode 字符流的首选方法是使用 InputStreamReader 和 BufferedReader 类。 OutputStream getLocalizedOutputStream(OutputStream out) 已过时。 从 JDK 开始将 Unicode 字符流转换为本地编码字节流的首选方法是使用 OutputStreamWriter、BufferedWriter 和 PrintWriter 类。 static Runtime getRuntime() 返回与当前 Java 应用程序相关的运行时对象。 void halt(int status) 强行终止目前正在运行的 Java 虚拟机。 void load(String filename) 加载作为动态库的指定文件名。 void loadLibrary(String libname) 加载具有指定库名的动态库。 long maxMemory() 返回 Java 虚拟机试图使用的最大内存量。 boolean removeShutdownHook(Thread hook) 取消注册某个先前已注册的虚拟机关闭挂钩。 void runFinalization() 运行挂起 finalization 的所有对象的终止方法。 static void runFinalizersOnExit(boolean value) 已过时。 此方法本身具有不安全性。它可能对正在使用的对象调用终结方法而其他线程正在操作这些对象从而导致不正确的行为或死锁。 long totalMemory() 返回 Java 虚拟机中的内存总量。 void traceInstructions(boolean on) 启用/禁用指令跟踪。 void traceMethodCalls(boolean on) 启用/禁用方法调用跟踪。 1.3 Process不管通过哪种方法启动进程后都会返回一个Process类的实例代表启动的进程该实例可用来控制进程并获得相关信息。Process 类提供了执行从进程输入、执行输出到进程、等待进程完成、检查进程的退出状态以及销毁(杀掉)进程的方法void destroy() 杀掉子进程。 一般情况下该方法并不能杀掉已经启动的进程不用为好。 int exitValue() 返回子进程的出口值。 只有启动的进程执行完成、或者由于异常退出后exitValue()方法才会有正常的返回值否则抛出异常。 InputStream getErrorStream() 获取子进程的错误流。 如果错误输出被重定向则不能从该流中读取错误输出。 InputStream getInputStream() 获取子进程的输入流。 可以从该流中读取进程的标准输出。 OutputStream getOutputStream() 获取子进程的输出流。 写入到该流中的数据作为进程的标准输入。 int waitFor() 导致当前线程等待如有必要一直要等到由该 Process 对象表示的进程已经终止。 2.多进程编程实例一般我们在java中运行其它类中的方法时无论是静态调用还是动态调用都是在当前的进程中执行的也就是说只有一个java虚拟机实例在运行。而有的时候我们需要通过java代码启动多个java子进程。这样做虽然占用了一些系统资源但会使程序更加稳定因为新启动的程序是在不同的虚拟机进程中运行的如果有一个进程发生异常并不影响其它的子进程。在Java中我们可以使用两种方法来实现这种要求。最简单的方法就是通过Runtime中的exec方法执行java classname。如果执行成功这个方法返回一个Process对象如果执行失败将抛出一个IOException错误。下面让我们来看一个简单的例子。// 文件 import .*; public class Test { public static void main(String[] args) { FileOutputStream fOut new FileOutputStream(c:\\); fOut.close(); System.out.println(被调用成功!); } } // public class Test_Exec { public static void main(String[] args) { Runtime run (); Process p run.exec(java test1); } } 通过java Test_Exec运行程序后发现在C盘多了个文件但在控制台中并未出现被调用成功!的输出信息。因此可以断定Test已经被执行成功但因为某种原因Test的输出信息未在Test_Exec的控制台中输出。这个原因也很简单因为使用exec建立的是Test_Exec的子进程这个子进程并没有自己的控制台因此它并不会输出任何信息。如果要输出子进程的输出信息可以通过Process中的getInputStream得到子进程的输出流(在子进程中输出在父进程中就是输入)然后将子进程中的输出流从父进程的控制台输出。具体的实现代码如下如示// import .*; public class Test_Exec_Out { public static void main(String[] args) { Runtime run (); Process p run.exec(java test1); BufferedInputStream in new BufferedInputStream(()); BufferedReader br new BufferedReader(new InputStreamReader(in)); String s; while ((s ()) ! null) System.out.println(s); } } 从上面的代码可以看出在中通过按行读取子进程的输出信息然后在Test_Exec_Out中按每行进行输出。 上面讨论的是如何得到子进程的输出信息。那么除了输出信息还有输入信息。既然子进程没有自己的控制台那么输入信息也得由父进程提供。我们可以通过Process的getOutputStream方法来为子进程提供输入信息(即由父进程向子进程输入信息而不是由控制台输入信息)。我们可以看看如下的代码// 文件 import .*; public class Test { public static void main(String[] args) { BufferedReader br new BufferedReader(new InputStreamReader(System.in)); System.out.println(由父进程输入的信息 ()); } } // import .*; public class Test_Exec_In { public static void main(String[] args) { Runtime run (); Process p run.exec(java test2); BufferedWriter bw new BufferedWriter(new OutputStreamWriter(())); (向子进程输出信息); (); bw.close(); // 必须得关闭流否则无法向子进程中输入信息 // System.in.read(); } } 从以上代码可以看出Test1得到由Test_Exec_In发过来的信息并将其输出。当你不加()和()时信息将无法到达子进程也就是说子进程进入阻塞状态但由于父进程已经退出了因此子进程也跟着退出了。如果要证明这一点可以在最后加上.read()然后通过任务管理器(在windows下)查看java进程你会发现如果加上()和()只有一个java进程存在如果去掉它们就有两个java进程存在。这是因为如果将信息传给Test2在得到信息后Test2就退出了。在这里有一点需要说明一下exec的执行是异步的并不会因为执行的某个程序阻塞而停止执行下面的代码。因此可以在运行test2后仍可以执行下面的代码。exec方法经过了多次的重载。上面使用的只是它的一种重载。它还可以将命令和参数分开如exec()可以写成exec(java, test2)。exec还可以通过指定的环境变量运行不同配置的java虚拟机。除了使用Runtime的exec方法建立子进程外还可以通过ProcessBuilder建立子进程。ProcessBuilder的使用方法如下// import .*; public class Test_Exec_Out { public static void main(String[] args) { ProcessBuilder pb new ProcessBuilder(java, test1); Process p (); … … } } 在建立子进程上ProcessBuilder和Runtime类似不同的ProcessBuilder使用start()方法启动子进程而Runtime使用exec方法启动子进程。得到Process后它们的操作就完全一样的。ProcessBuilder和Runtime一样也可设置可执行文件的环境信息、工作目录等。下面的例子描述了如何使用ProcessBuilder设置这些信息。ProcessBuilder pb new ProcessBuilder(Command, arg2, arg2, ); // 设置环境变量 MapString, String env (); (key1, value1); (key2); (key2, (key1) _test); (..\abcd); // 设置工作目录 Process p (); // 建立子进程 【编辑推荐】【责任编辑未丽燕 TEL(010)68476606】点赞 0