文档章节

java并发编程的艺术-第四章之Java并发基础

wusd1256
 wusd1256
发布于 2017/10/14 16:32
字数 1375
阅读 1886
收藏 54
点赞 3
评论 0

为什么要使用多线程

Java天生就是多线程程序,简单的一个main() 方法的执行,其实是由main线程和其它的线程共同执行的。当然使用多线程有如下的好处:

        1、利用更多的处理器核心

        2、更快的响应速度,如将数据一致性不强的操作交给其它的线程去操作

        3、更好的编程模型

线程简介

一、线程优先级

在线程执行过程中,线程优先级决定了需要分配处理器资源的多少,从而决定了获取时间片的多少。现在线程构建时,可以通过setPriority()方法来修改优先级,默认优先级是5.但是在不同的JVM以及操作系统上面,线程的规划有些不同,甚至有些操作系统会忽略对线程优先级的设定,都是采用默认值。

 

二、线程状态

线程状态

状态名称 说明
NEW 初始状态,线程被创建,还没有有调用start()方法
RUNNABLE 运行状态,java中就绪也属于运行状态
BLOCKED 阻塞状态,表示线程阻塞在进入synchronize修饰的方法或代码块(获取锁)时的状态
WAITING 等待状态,进入该状态表示当前线程需要等待其他的线程做出特定动作(通知、中断)才返回
TIME_WAITING 超时等待,不同于WAITING,它是等到指定时间到达自动返回
TERMINTED 终止状态,表示当前线程执行完成

 

Daemon 是一种支持线程,主要用来支持程序的后台的调度,所以当Java虚拟机中不存在非Daemon线程(main 线程)时,Java虚拟机将会退出。

import java.util.concurrent.TimeUnit;  
  
/** 
 * Created by shidong.wu on 2017/10/13. 
 */  
public class Daemon {  
    public static void main(String[] args) {  
        Thread thread = new Thread(new DaemonRunner(),"DaemonRunner");  
        thread.setDaemon(true);  
        thread.start();  
    }  
    static class DaemonRunner implements Runnable {  
        @Override  
        public void run() {  
            try {  
                TimeUnit.SECONDS.sleep(10);  
            }catch (InterruptedException e) {  
  
            }finally {  
                System.out.println("Daemon finally");  
            }  
        }  
    }  
}  

运行上面的程序终端没有任何输出,主要是因为非Daemon线程(main线程)运行结束后,Java虚拟机退出了,所以DaemRunner中的finally代码块没有执行。

 

三、中断

中断可以理解为线程的一个标识位属性,它标识一个线程是否其它的线程进行的了中断操作。其它的线程通过调用当前线程的interrupt()方法对其进行中断操作。线程通过方法isInterrupt()方法来检查自身是否被中断来进行响应。抛出InterruptException的线程的,他的中断标识会被清除。

 

线程间通信

一、volatile 和synchronized 关键字

在前面的已经讲过它们的内存含义了,

1、关键字volatile用来修饰一个字段(变量),就是告诉程序任意一个线程,对该变量的读取都要去共享内存中获取,而都变量的修改必须要同步刷新到共享内存里,volatile保证了所有线程的对变量的访问可见性。但是过多的使用volatile,会影响程序的执行效率。

        2、关键字synchronized用来修饰方法和代码块,它主要保证多个线程在同一时刻,只能有一个线程处于synchronized修饰的方法和代码块中,其本质是对一个对象监视器(Monitor)的获取,这个获取的过程是排他的,同一个时刻,只能有一个线程能获取大对象的监视器。没有获取到监视器的线程进入同步队列,等待其他线程释放锁。如下图:

 

二、等待/通知机制

等待 / 通知机制是指一个线程A调用了对象O的wait()方法,释放锁,进入等待状态,而线程B调用了对象O的notify()或者notifyAll()方法,线程A收到通知后从对象O的wait()方法返回,进而执行后面的操作。

等待 / 通知机制的经典范式,分为,消费者和生产者;

消费者遵循的规范:

1、获取对象的锁

2、如果条件不满足,调用对象的wait()方法,被通知后还要进行条检查

3、条件满足后执行对应的逻辑

对应的伪代码:

synchronized(对象){

while(条件不满足){

对象.wait();

   }

对应的逻辑;

}

 

 

生产者遵循的规范:

1、获取对象的锁

2、改变条件

3、通知所有等待在该对象的线程

对此的伪代码:

synchronized (对象) {

改变条件;

       对象.notifyAll();

 

 

}

 

下面我们就利用上面的等待 / 通知的范式实现一个线程的等待/ 通知的机制;


package com.concurrency;

import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.concurrent.TimeUnit;

/**
 * Created by shidong.wu on 2017/10/14.
 */
public class WaitNotify {

    private static boolean flag = true;

     private  static Object lock = new Object();

    public static void main(String[] args) throws Exception{
        Thread waitThread = new Thread(new Wait(),"waitThread");
        waitThread.start();
        TimeUnit.SECONDS.sleep(1);

        Thread notifyThread = new Thread(new Notify(),"notifyThread");
        notifyThread.start();
    }


    static class Wait implements Runnable {
        public void run() {
            // 加锁,lock对象加锁
            synchronized (lock) {
                while (flag){  // 条件不满足
                    try {
                        lock.wait();
                    }catch (InterruptedException e){
                    }
                }
                // 条件满足时,进行完成操作
                System.out.println(Thread.currentThread() + new SimpleDateFormat("HH:mm:ss").format(new Date()));
            }
        }
    }

    static class Notify implements Runnable {
        public void run() {
            // 加锁,lock对象加锁
            synchronized (lock) {
                lock.notifyAll(); // 通知
                flag = false;

            }
        }
    }
}


三、join() 使用

如果一个线程A调用了thread.join()表示线程A等待线程thread线程终止之后,才从threa.join()处返回。

 

四、ThreadLocal 使用

ThreadLocal 是一个线程变量,这个变量是附带在线程上面的。

 

 

 

 

© 著作权归作者所有

共有 人打赏支持
wusd1256
粉丝 28
博文 5
码字总数 4914
作品 0
浦东
程序员
读书笔记之《Java并发编程的艺术》-并发编程基础

读书笔记部分内容来源书出版书,版权归本书作者,如有错误,请指正。 欢迎star、fork,读书笔记系列会同步更新 git https://github.com/xuminwlt/j360-jdk module j360-jdk-thread/me.j360....

Hi徐敏
2015/11/11
0
8
读书笔记之《Java并发编程的艺术》-并发编程容器和框架(重要)

读书笔记部分内容来源书出版书,版权归本书作者,如有错误,请指正。 欢迎star、fork,读书笔记系列会同步更新 git https://github.com/xuminwlt/j360-jdk module j360-jdk-thread/me.j360....

Hi徐敏
2015/11/11
0
1
读书笔记之《Java并发编程的艺术》-线程池和Executor的子孙们

读书笔记部分内容来源书出版书,版权归本书作者,如有错误,请指正。 欢迎star、fork,读书笔记系列会同步更新 git https://github.com/xuminwlt/j360-jdk module j360-jdk-thread/me.j360....

Hi徐敏
2015/11/11
0
1
读书笔记之《Java并发编程的艺术》-java中的锁

读书笔记部分内容来源书出版书,版权归本书作者,如有错误,请指正。 欢迎star、fork,读书笔记系列会同步更新 git https://github.com/xuminwlt/j360-jdk module j360-jdk-thread/me.j360....

Hi徐敏
2015/11/11
0
0
Java程序员必读书单,家族又添新成员

点击关注异步图书,置顶公众号 每天与你分享IT好书 技术干货 职场知识 参与文末话题讨论,每日赠送异步图书。 ——异步小编 有些革命出其不意地吸引了全世界的眼球。Twitter、Linux操作系统和...

异步社区
05/09
0
0
安卓开发必备知识体系:Java篇

大家好我是张拭心,自从各位朋友帮点广X开始,我发现我每天更有奔头了,走起路来也更有劲了,说啥也得更新的勤快一点。不过放心,我一定推送有价值的内容给大家,还请朋友们照旧动动手指点点...

d29h1jqy3akvx
05/10
0
0
Java并发教程-1进程和线程

http://www.iteye.com/magazines/131 计算机的使用者一直以为他们的计算机可以同时做很多事情。他们认为当其他的应用程序在下载文件,管理打印队列或者缓冲音频的时候他们可以继续在文字处理...

noday
2014/04/25
0
0
JVM内存结构 VS Java内存模型 VS Java对象模型

Java作为一种面向对象的,跨平台语言,其对象、内存等一直是比较难的知识点。而且很多概念的名称看起来又那么相似,很多人会傻傻分不清楚。比如本文我们要讨论的JVM内存结构、Java内存模型和...

Java架构
07/11
0
0
书单丨5本Java后端技术书指引你快速进阶

一名Java开发工程师 不仅要对Java语言及特性有深层次的理解 而且需要掌握与Java相关的 框架、生态及后端开发知识 本文涉及多种后端开发需要掌握的技能 对于帮助提高开发能力非常有帮助 NO.1...

Java高级架构
05/30
0
0
Java 使用 happen-before 规则实现共享变量的同步操作

前言 熟悉 Java 并发编程的都知道,JMM(Java 内存模型) 中的 happen-before(简称 hb)规则,该规则定义了 Java 多线程操作的有序性和可见性,防止了编译器重排序对程序结果的影响。按照官方的...

stateIs0
01/20
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

微服务架构下的安全认证与鉴权

微服务下常见认证解决方案; OAuth认证与授权; JWT认证介绍; Spring Cloud的OAuth2实现; 单体应用转变为分布式应用 单体应用转变为分布式应用在架构方式上存在较大区别,单体应用下的简单...

Java大蜗牛
27分钟前
0
0
前端面试题汇总

最近在复习,准备找工作了,特此总结一下前端的相关知识。 1.获取浏览器URL中查询字符的参数: function getQuery(name){    var reg = new RegExp("(^|&)"+name+"=([^&]*)"(&|$));...

凛冬来袭
27分钟前
0
0
可持续发展的学习道路

与其要求别人,不如提升自己 内心渴望进步 经常做出改变现有模式,不断学习 寻找资源,整合资源,不断熟练这种模式 渠道很重要 先打开新世界的航路

狮子狗
32分钟前
0
0
apollox-lua开源项目 示例codepen2

今天在示例上增加了几个功能, 首先添加js array的标准库。 所有js array的方法目前都支持了。 添加查看code模式。 点击查看code可以看到生成的lua代码。默认web模式需要把标准库连接进来, ...

钟元OSS
47分钟前
0
0
javascript性能优化之避免重复工作

javascript最重要也最根本的性能优化标准之一是避免工作,避免工作又包括两点,第一,不做不必要的工作,第二,不做重复的已经完成的工作。第一部分可以通过代码重构完成,第二部分不做重复的...

老韭菜
59分钟前
0
0
缓存穿透、并发和雪崩那些事

0 题记 缓存穿透、缓存并发和缓存雪崩是常见的由于并发量大而导致的缓存问题,本文讲解其产生原因和解决方案。 缓存穿透通常是由恶意攻击或者无意造成的;缓存并发是由设计不足造成的;缓存雪...

Java填坑之路
今天
1
0
项目jar包管理构建工具---Maven

一、what is Maven? 我们来寻找一下官网,里面介绍了maven到底是什么?下面一句话就有讲解到:Apache Maven is a software project management and comprehension tool. Based on the conc...

一看就喷亏的小猿
今天
0
0
JVM学习手册(一):查看堆内存使用情况以及排错

平时出现内存溢出以及死锁,一般处理方式都是查看日志,找到抛出异常的代码行,然后本地分析代码,但是这样对于线上排查十分糟糕,这段时间在研究JVM发现了几个比较好的工具和指令. 1.针对频繁GC和...

勤奋的蚂蚁
今天
1
0
17.TCP:传输控制协议

介绍 TCP和UDP使用同一网络层(IP),但TCP提供了面向连接、可靠的传输层服务 TCP传输给IP层的信息单位称为报文段或段 TCP通过如下方式保证可靠性: 应用数据被分割成TCP认为最合适发送的数据...

loda0128
今天
0
0
重装Oracle时出现environment variable "PATH"错误的解决办法

在win7 64位下重新安装oracle 11g,一直报environment variable "PATH"的错误,按说明将path里多余的路径删除,但没办法解决。选择忽略错误继续安装,装一半会报CRC错误,还是安装失败。最好...

良言
今天
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部