线程和进程

166 阅读5分钟

几乎所有的操作系统都支持进程(process) 的概念。所有运行中的任务通常对应一个进程。当程序进入内存运行时,就会变成一个进程。进程时处于运行过程中的程序。进程是系统进行资源调度和分配的一个独立单位。

进程的三个特征

独立性:进程是系统中独立的实体,拥有自己独立的资源,每个进程都有自己私有的地址空间,一个进程不可以独立访问其他系统的地址空间。 动态性:进程和程序的区别,程序是一堆静态指令的集合,进程是系统中活动的指令集合。进程中加入了时间的概念。进程中有自己的生命周期和各个不同的状态,程序是不具备的。 并发性:多个进程在单个处理器上运行,多个进程之间不互相影响。

实际上:一个CPU 在某个时间节点只能运行一个程序,也就是只能运行一个进程。CPU在这些进程之间不断的进行切换执行。CPU的切换越快,人的感知度越低。

多线程扩展了多进程的概念,使得同一个进程可以并发处理多个任务。THREAD 线程被称作轻量级进程LightWeight Process。 线程是进程的执行单元。 等同于 进程在操作系统中的地位。

线程是独立的、并发的执行流,进程初始化后,线程就被创建出来了,大多数进程都要一个主线程,也可以创建其他的执行流,也就是线程。每个线程也是相互独立的。

一个线程必须有一个父进程。线程拥有自己的堆栈、程序计数器、局部变量。多个线程共享进程的全部资源。

线程是独立运行的,它并不知道是否有其他线程的存在。线程的执行是抢占式的,即 当前运行的线程在任何时刻都有可能被挂起,以便另一个线程可以运行。

一个线程可以创建和撤销另一个线程。一个进程中多个线程可以并发进行。

总而言之,一个程序后台至少运行一个进程,一个进程可以包含多个线程,但至少一个线程。

#线程的运行效率大于进程,多个线程共享同一个进程的虚拟空间。 线程共享的环境包括:代码段、进程的公有数据。利用共享数据,线程之间很容易通信。

注意:进程之间不能共享内存。线程之间共享内存非常容易。 创建进程需要操作系统从新分配资源,相比之下,创建线程代价小很多。

应用场景

浏览器: 一个进程,支持同时下载多个图片(多个线程)

image.png

服务器:一个进程,支持影响多个用户请求(多线程)

image.png

创建进程

第一种方式是通过Runtime.exec()方法来创建一个进程 Process process = new ProcessBuilder(cmd).start();


Process类是一个抽象类


public abstract class Process
{
     
    abstract public OutputStream getOutputStream();   //获取进程的输出流
       
    abstract public InputStream getInputStream();    //获取进程的输入流
  
    abstract public InputStream getErrorStream();   //获取进程的错误流
  
    abstract public int waitFor() throws InterruptedException;   //让进程等待
   
    abstract public int exitValue();   //获取进程的退出标志
  
    abstract public void destroy();   //摧毁进程
}

ProcessImpl类具体实现了 抽象类 Process 的start方法
    
final class ProcessImpl extends Process {
  
    // System-dependent portion of ProcessBuilder.start()
    static Process start(String cmdarray[],
             java.util.Map<String,String> environment,
             String dir,
             boolean redirectErrorStream)
    throws IOException
    {
    String envblock = ProcessEnvironment.toEnvironmentBlock(environment);
    return new ProcessImpl(cmdarray, envblock, dir, redirectErrorStream);
    }
 ....

}
    
    
  ProcessBuilder是一个final类,它有两个构造器:


public final class ProcessBuilder
{
    private List<String> command;
    private File directory;
    private Map<String,String> environment;
    private boolean redirectErrorStream;
  
   //将命令参数放进List
    public ProcessBuilder(List<String> command) {
    if (command == null)
        throw new NullPointerException();
    this.command = command;
    }
    //不定长字符串
    public ProcessBuilder(String... command) {
    this.command = new ArrayList<String>(command.length);
    for (String arg : command)
        this.command.add(arg);
    }
    
    
    
public Process start() throws IOException {
       String[] cmdarray = command.toArray(new String[command.size()]);
         for (String arg : cmdarray)
         if (arg == null)  throw new NullPointerException();

        String prog = cmdarray[0];
       SecurityManager security = System.getSecurityManager();
      if (security != null)
        security.checkExec(prog);

    String dir = directory == null ? null : directory.toString();

    try {
        return ProcessImpl.start(cmdarray,
                     environment,
                     dir,
                     redirectErrorStream);
    } catch (IOException e) {
        // It's much easier for us to create a high-quality error
        // message than the low-level C code which found the problem.
        throw new IOException(
        "Cannot run program \"" + prog + "\""
        + (dir == null ? "" : " (in directory \"" + dir + "\")")
        + ": " + e.getMessage(),
        e);
    }
}
    ....

}

    

事实上通过ProcessBuilder的start方法创建的是一个ProcessImpl对象。

/**
 * 实现进程的方式1:通过ProcessBuild创建进程
 * @author chenyr
 * @date 2016.03.28
 */
public class ProcesssorDemo1 {
    public static void main(String[] args) throws IOException {
    //通过ProcessBuilder来启动一个进程打开cmd,并获取ip地址信息
        ProcessBuilder pb = new ProcessBuilder("/bin/sh", "-c","ifconfig");
        Process process = pb.start();
        Scanner scanner = new Scanner(process.getInputStream());
        while (scanner.hasNextLine()) {
            System.out.println(scanner.nextLine());
        }
        scanner.close();
    }
}


第二种方法是 通过Runtime的exec方法来创建进程

Runtime中采用了单例模式,即只会产生一个虚拟机实例:



public class Runtime {
    private static Runtime currentRuntime = new Runtime();
  
    /**
     * Returns the runtime object associated with the current Java application.
     * Most of the methods of class <code>Runtime</code> are instance
     * methods and must be invoked with respect to the current runtime object.
     *
     * @return  the <code>Runtime</code> object associated with the current
     *          Java application.
     */
    public static Runtime getRuntime() {
    return currentRuntime;
    }
  
  
    /** Don't let anyone else instantiate this class */
    **private** Runtime() {}
    
    
    
 public Process exec(String[] cmdarray, String[] envp, File dir)
   throws IOException {
   return new ProcessBuilder(cmdarray)
       .environment(envp)
       .directory(dir)
       .start();
   }
    ...
 }
 

 
/**
 * 实现进程的方式2:通过Runtime获取
 * @author chenyr
 * @date 2016.03.28
 */
public class ProcesssorDemo2 {
    public static void main(String[] args) throws IOException {
        Process process = Runtime.getRuntime().exec("/bin/sh -c ifconfig");
        Scanner scanner = new Scanner(process.getInputStream());
        while (scanner.hasNextLine()) {
            System.out.println(scanner.nextLine());
        }
        scanner.close();
    }
}



Java创建Nohup进程

$ cat /tmp/test.sh 
#!/bin/bash

for sig in SIGINT SIGTERM SIGHUP; do
  trap "echo Caught $sig" $sig
done

echo Traps registered, sleeping
sleep 10
echo Done sleeping, exiting

当我通过以下方式调用它时:**

new ProcessBuilder("/tmp/test.sh").inheritIO().start().waitFor(1, TimeUnit.SECONDS);