博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
第21章 java线程(1)-线程初步
阅读量:6591 次
发布时间:2019-06-24

本文共 4771 字,大约阅读时间需要 15 分钟。

java线程(1)-线程初步

1.并行和并发

并行和并发是即相似又有区别:

并行:指两个或者多个事件在同一时刻点发生。
并发:指两个或多个事件在同一时间段内发生
在操作系统中,并发性是指在一段事件内宏观上有多个程序在同时运行,但是单CPU系统中,每一时刻仅能有一道程序执行。故微观上这些程序只能是分时交替执行
单核处理器的计算机肯定是不能并行的处理多个任务的,只能是多个任务在单个CPU上并发运行,同理,线程也是一样的,从宏观角度上理解是并行运行的,但是从微观角度上分析却是串行运行的,即一个线程一个线程的去运行,当系统只有一个CPU时,线程会以某种顺序执行多个线程,我们把这种情况称之为线程调度
790334-20161209155107022-452915489.png

2.理解进程和线程

进程和线程:

进程是指一个内存中运行中的应用程序,每一个进程都有自己独立的一块内存空间,一个应用程序可以同启动多个进程。
大多数的时候,操作系统都不需要一个进程方法其他进程的空间,也就是说进程之间的通信是很不方便的。此时我们引入“线程”这门技术,来解决这个问题
线程是指进程中一个执行任务,一个进程可以同时并发运行多个线程,如:多线程下载软件。
多任务系统:该系统可以运行多个进程,一个进程可以执行多个任务,一个进程可以包含多个线程

一个进程至少有一个线程,为了提高效率,可以在一个进程中开启多个执行任务,即多线程

多进程:操作系统中同时运行的多个程序

多线程:在同一个进程中同时运行的多个任务

在操作系统中允许多个任务,每一个任务就是一个进程,每一个进程也可以同时执行多个任务,每一个任务就是线程

进程和线程的区别:

进程:有独立的内存空间,进程中的数据存放空间(堆空间和栈空间)是独立的,至少有一个线程
线程:堆空间是共享的,栈空间是独立的,线程消耗的资源也比进程小,相互之间可以影响的,又称为轻型进程或进程元
因为一个进程中的多个线程是并发的,那么从微观角度来考虑也是有先后顺序的,那么哪一个线程执行完全取决于CPU调度器(JVM来调度),程序员是控制不了的
我们可以把线程并发性看做是多个线程在瞬间抢CPU资源,谁抢到了资源谁就运行,这也就造成了多线程的随机性
java程序的进程里至少包含主线程和垃圾回收线程(后台线程)

线程调度:

计算机通常只有一个CPU时,在任意时刻只能执行一条计算机指令,每一个进程只有获取CPU的使用权才能执行指令。所谓多进程并发性,从宏观上来看,其实就是各个进程轮流获得CPU的使用权,分别执行各自的任务
那么,在可运行池中,会有多个线程处理就绪状态等到CPU,JVM就负责了线程的调度
JVM采用的是抢占式调度,没有采用分时调度,因此可能造成多线程执行结果的随机性。

3.多线程的优势

多线程作为一种多任务,并发的工作方法,当然有其存在的优势:

1.进程之前不能共享内存,而线程之间共享内存(堆内存)则很简单
2.系统创建进程时需要为该进程重新分配系统资源,创建线程则代价小的多,因此实现多任务并发时,多线程效率更高
3.java语言本身内置多线程功能的支持,而不是单纯作为底层系统的调度方式,从而简化了多线程编程

4.创建一个简单的多线程

在java代码中如何去运行一个进程:

方式1:Runtime类的exec方法
方式2:ProcessBuilder的start方法

//在java中如何开启一个进程:运行记事本程序public class ProcessDemo{    public static voild main(String[] args) throws Exception{    //方式1:使用Runtime类的exec方法    Runtime runtime = Runtime.getRuntime();    runtime.exce("notepad");    //方式2:使用ProcessBuilder的start方法    ProcessBuilder pb = new ProcessBuilder("notepad");    pb.start();}}

5.使用继承方式和实现方式创建并启动线程

创建和启动线程,传统有两种方法:

方式1:继承Thread类
方式2:实现Runnable接口
线程类:Thread类(java.lang.Thread)和Thread的子类才能称之为线程类(API有详细例子)
方式1:继承Thread类
步骤:
1.定义一个类A继承于java.lang.Thread类
2.在A类中覆盖Thread类中的run方法
3.我们在run方法中编写需要执行的操作->run方法里的线程执行体
4.在main方法(线程)中,创建线程对象,并启动线程

创建线程类: A类 a = new A类();

调用线程对象的start方法,a.start();//启动一个线程

注意:千万不要调用run方法,如果调用run方法,好比是对象调用方法,依然还是只有一个线程,并没有开启新的线程.

方式1:继承Thread类

//计算大于某一规定值的质数的线程可以写成://1.定义Thread的集成类class PrimeThread extends Thread{    long mainPrime;    public PrimeThread(long mainPrime){        this.mainPrime = mainPrime;    }    public void run(){        //TODO    }}//2.调用线程类.创建并启动一个线程PrimeThread p = new PrimeThread(143);p.start();//注意是调用start方法

方式2:实现Runnable接口

步骤:
1.定义一个类A继承于java.lang.Thread类
2.在A类中覆盖Thread类中的run方法
3.我们在run方法中编写需要执行的操作->run方法里的线程执行体
4.在main方法(线程)中,创建线程对象,并启动线程

创建线程类: Thread(Runnable target);//分配新的Thread对象

调用线程对象的start方法,a.start();//启动一个线程

//方式2,是声明实现 Runnable 接口的类class PrimeRun implements Runable{    long mainPrime;    public PrimeRun(long mainPrime){        this.mainPrime = mainPrime;    }    public void run(){        //TODO    }}//创建并启动一个线程:PrimeRun p = new PrimeRun(143);Thread t = new Thread(p);t.start();//每个线程都有一个标识名,多个线程可以同名。如果线程创建时没有指定标识名,就会为其生成一个新名称。

6.使用匿名内部类来创建线程

只适用于某一个类只适用一次的情况。

匿名内部类其实就是把5中步骤1,2,3放在4中写,例如创建一个Runnable接口的匿名内部类,上面的改成:

new Thread(new Runnable(){    public void run(){        //TODO    }}).start();

通过继承方式的内部类也是可以这样写的,写的时候就不用声明继承了,直接在{}里面写run()方法就可以了。

new Thread(){    public void run(){        //TODO    }}.start();

这种方式用的比较少

7.案例

我们写一个案例来具体说明:

需求:有50个苹果,请三个小朋友来吃,A,B,C三个人可以同时吃苹果,用多线程来实现
分析:
可以先定义线程对象,然后启动线程
方法:
可以用我们上面的两种方法来实现
方式1:使用继承Thread方法
但是下面类型创造出来多线程会每个线程都吃50个苹果

/** * 用继承方法来完成吃苹果的多线程需求 *///创建一个线程类class Person extends Thread{    private int num = 50;    //构造器,调用Thread的方法赋值    public Person(String name){        super(name);    }    public void run(){        for (int i = 0; i < 50; i++) {            if (num > 0){                System.out.println(super.getName()+"吃了编号为"+ num-- +"的苹果");            }        }    }}public class AppleExtends {    public static void main(String[] args) {        //创建桑个线程        new Person("小A").start();        new Person("小B").start();        new Person("小C").start();    }}

方式2:使用实现Runnable方式来实现

class Apple implements Runnable{    private int num = 50;    public void run(){        for (int i = 0; i < 50; i++) {            if (num > 0){                System.out.println(Thread.currentThread().getName()+"吃了编号为"+ num-- +"的苹果");            }        }    }}public class appleImplements {    public static void main(String[] args) {        //创建桑个线程        Apple a = new Apple();        new Thread(a,"小A").start();        new Thread(a,"小B").start();        new Thread(a,"小C").start();    }}

两种方式的区别分析:

继承方式:
1.java中类是单继承的,如果继承了Thread了,该类就不能再有其他的直接父类了
2.从操作上分析,继承方式更简单,获取线程名字也简单(操作上,更简单)
3.从多线程共享一个资源上分析,继承方式不能做到共享

实现接口方式:

1.java中类可以多实现接口,此时该类还可以继承其他类,并且还可以实现其他接口(设计上更优雅)
2.从操作上分析,实现方式稍微复杂点,获取线程名字也稍微复杂,得使用Thread.currentThread()来获取当前线程的引用,使用Thread.currentThread().getname()来获取当前线程的名字
3.从多线程共享一个资源分析,实现方式可以做到共享

转载于:https://www.cnblogs.com/cenyu/p/6149838.html

你可能感兴趣的文章
哈默尔的核心竞争力--《可以量化的管理学》
查看>>
Unity中关于作用力方式ForceMode的功能注解
查看>>
view生命周期的一个找父类的控件的方法
查看>>
今天晚上完成期中考试的视频
查看>>
物理读之LRU(最近最少被使用)的深入解析
查看>>
写给将要毕业的学弟学妹们的感言
查看>>
mybatis-ehcache 用法配置备忘
查看>>
Python2.7升级到3.0 HTMLTestrunner报错解决方法
查看>>
Redis介绍以及安装(Linux)
查看>>
FreeBSD下php-mbstring的安装
查看>>
去掉VS2012中的红色波浪下划线
查看>>
[文档]关于接口文档的写法
查看>>
一次tensorflow的尝试
查看>>
建立Git版本库管理框架例子
查看>>
nginx防止部分DDOS攻击
查看>>
编写一个截取字符串的函数,输入为一个字符串和字节数,输出为按字节截取的字符串。 但是要保证汉字......
查看>>
number_format() 函数定义和用法
查看>>
Java8中聚合操作collect、reduce方法详解
查看>>
查看记录
查看>>
mybatis报ORA-00911: 无效字符
查看>>