java 多线程 并发实例_Java 多线程(并发)
线程释义
使用java.lang.Thread类或者java.lang.Runnable接口编写代码来定义、实例化和启动新线程。
一个Thread类实例只是一个对象,像Java中的任何其他对象一样,具有变量和方法,生死于堆上。
Java中,每个线程都有一个调用栈,即使不在程序中创建任何新的线程,线程也在后台运行着。
一个Java应用总是从main()方法开始运行,mian()方法运行在一个线程内,它被称为主线程。
一旦创建一个新的线程,就产生一个新的调用栈。
线程总体分两类:用户线程和守候线程。
守护线程与用户线程的区别是:
守护线程就是main同生共死,当main退出,它将终止,而普通线程是在任务执行结束才停止。守护线程则是用来服务用户线程的,如果没有其他用户线程在运行,那么就没有可服务对象,也就没有理由继续下去。
Java虚拟机在它所有非守护线程已经离开后自动离开。
但是daemon Thread实际应用在那里呢?举个例子,web服务器中的Servlet,容器启动时后台初始化一个服务线程,即调度线程,负责处理http请求,然后每个请求过来调度线程从线程池中取出一个工作者线程来处理该请求,从而实现并发控制的目的。
线程与进程
线程是进程中的最小执行单元。也就是说,进程包含线程,一个进程可以完成多个任务,每个任务就可以看作是一个线程。
那什么又是进程呢。
程序的执行过程是进程。
持有资源和线程的是进程。
使用线程的目的是辅助写出CPU最大利用率的高效程序,使环境异步。
线程实现
实现Runnable接口的多线程例子
classDoSomethingimplements Runnable{
privateStringname;
publicDoSomething(Stringname){
this.name =name;
}
publicvoidrun(){
for(inti =0;i <5;i++){
for(longk =0;k <10;k++)
System.out.println(name +": "+i);
}
}
}
publicclassTestRunnable{
publicstaticvoidmain(String[]args){
DoSomethingds1 =newDoSomething("阿三");
DoSomethingds2 =newDoSomething("李四");
Threadt1 =newThread(ds1);
Threadt2 =newThread(ds2);
t1.start();
t2.start();
}
}
扩展Thread类实现的多线程例子
/**
* 测试扩展Thread类实现的多线程程序
*/
publicclassTestThreadextends Thread{
publicTestThread(Stringname){
super(name);
}
publicvoidrun(){
for(inti =0;i<5;i++){
for(longk=0;k <10;k++)
System.out.println(this.getName()+" :"+i);
}
}
publicstaticvoidmain(String[]args){
Threadt1 =newTestThread("阿三");
Threadt2 =newTestThread("李四");
t1.start();
t2.start();
}
}
对于上面的多线程程序代码来说,输出的结果是不确定的。其中的一条语句for(long k= 0; k <10;k++);是用来模拟一个非常耗时的操作的。
两者比较
继承方法目的是用多线程完成不同的任务,实现方法目的是用多线程完成相同的任务。
没有必要用run之外的方法的时候(除非是要加强或者修改),不要继承Thread类。
启动线程
在线程的Thread对象上调用start()方法,而不是run()或者别的方法。
在调用start()方法之前:线程处于新状态中,新状态指有一个Thread对象,但还没有一个真正的线程。
在调用start()方法之后:发生了一系列复杂的事情
启动新的执行线程(具有新的调用栈);
该线程从新状态转移到可运行状态;
当该线程获得机会执行时,其目标run()方法将运行。
注意:对Java来说,run()方法没有任何特别之处。像main()方法一样,它只是新线程知道调用的方法名称(和签名)。因此,在Runnable上或者Thread上调用run方法是合法的。但并不启动新的线程。
同步和锁定
线程的同步是为了防止多个线程访问一个数据对象时,对数据造成的破坏。
需要完成两个操作:
把竞争访问的资源类变量标识为private。
同步哪些修改变量的代码,使用synchronized关键字同步方法或代码。
锁
Java中每个对象都有一个内置锁
当程序运行到非静态的synchronized同步方法上时,自动获得与正在执行代码类的当前实例(this实例)有关的锁。获得一个对象的锁也称为获取锁、锁定对象、在对象上锁定或在对象上同步。
当程序运行到synchronized同步方法或代码块时才该对象锁才起作用。
一个对象只有一个锁。所以,如果一个线程获得该锁,就没有其他线程可以获得锁,直到第一个线程释放(或返回)锁。这也意味着任何其他线程都不能进入该对象上的synchronized方法或代码块,直到该锁被释放。
一个线程可以获得多个锁
释放锁是指持锁线程退出了synchronized同步方法或代码块。
volatile关键字
该关键字可以直接让被声明的变量直接在原始内存地址进行存取,而不是寄存器。虽然这样会失去优化性能的机会。
关于锁和同步,有一下几个要点:
1)、只能同步方法,而不能同步变量和类;
2)、每个对象只有一个锁;当提到同步时,应该清楚在什么上同步?也就是说,在哪个对象上同步?
3)、不必同步类中所有的方法,类可以同时拥有同步和非同步方法。
4)、如果两个线程要执行一个类中的synchronized方法,并且两个线程使用相同的实例来调用方法,那么一次只能有一个线程能够执行方法,另一个需要等待,直到锁被释放。也就是说:如果一个线程在对象上获得一个锁,就没有任何其他线程可以进入(该对象的)类中的任何一个同步方法。
5)、如果线程拥有同步和非同步方法,则非同步方法可以被多个线程自由访问而不受锁的限制。
6)、线程睡眠时,它所持的任何锁都不会释放。
7)、线程可以获得多个锁。比如,在一个对象的同步方法里面调用另外一个对象的同步方法,则获取了两个对象的同步锁。
8)、同步损害并发性,应该尽可能缩小同步范围。同步不但可以同步整个方法,还可以同步方法中一部分代码块。
9)、在使用同步代码块时候,应该指定在哪个对象上同步,也就是说要获取哪个对象的锁。例如:
public int fix(int y) {
synchronized (this) {
x = x - y;
}
return x;
}
当然,同步方法也可以改写为非同步方法,但功能完全一样的,例如:
public synchronized int getX() {
return x++;
}
与
public int getX() {
synchronized (this) {
return x;
}
}
效果是完全一样的。
静态方法同步
要同步静态方法,需要一个用于整个类对象的锁,这个对象是就是这个类(XXX.class)。
例如:
public static synchronized int setName(String name){
Xxx.name = name;
}
等价于
public static int setName(String name){
synchronized(Xxx.class){
Xxx.name = name;
}
}
死锁
一个线程访问一个对象内的一个同步块时,对象内其余的同步块将被锁定。因为访问那些块也需要锁,但是锁已经被拿走了。
线程间的交互
先看这三个方法
void notify()
唤醒在此对象监视器上等待的单个线程。
void notifyAll()
唤醒在此对象监视器上等待的所有线程。
void wait()
导致当前的线程等待,直到其他线程调用此对象的 notify()方法或 notifyAll()方法。
当然,wait()还有另外两个重载方法:
void wait(long timeout)
导致当前的线程等待,直到其他线程调用此对象的 notify()方法或 notifyAll()方法,或者超过指定的时间量。
void wait(long timeout, int nanos)
导致当前的线程等待,直到其他线程调用此对象的 notify()方法或 notifyAll()方法,或者其他某个线程中断当前线程,或者已超过某个实际时间量。
以上这些方法是帮助线程传递线程关心的时间状态。
关于等待/通知,要记住的关键点是:
必须从同步环境内调用wait()、notify()、notifyAll()方法。线程不能调用对象上等待或通知的方法,除非它拥有那个对象的锁。
wait()、notify()、notifyAll()都是Object的实例方法。与每个对象具有锁一样,每个对象可以有一个线程列表,他们等待来自该信号(通知)。线程通过执行对象上的wait()方法获得这个等待列表。从那时候起,它不再执行任何其他指令,直到调用对象的notify()方法为止。如果多个线程在同一个对象上等待,则将只选择一个线程(不保证以何种顺序)继续执行。如果没有线程等待,则不采取任何特殊操作。
sleep();不放锁,这是它与wait的主要区别。
publicclassMyThreadPrinterimplements Runnable{
privateStringname;
privateObjectprev;
privateObjectself;
privateMyThreadPrinter2(Stringname,Objectprev,Objectself){
this.name =name;
this.prev =prev;
this.self =self;
}
@Override
publicvoidrun(){
intcount =10;
while(count >0){
synchronized (prev){
synchronized (self){
System.out.print(name);
count--;
self.notify();
}
try{
prev.wait();
}catch(InterruptedExceptione){
e.printStackTrace();
}
}
}
}
publicstaticvoidmain(String[]args)throws Exception{
Objecta =newObject();
Objectb =newObject();
Objectc =newObject();
MyThreadPrinter2pa =newMyThreadPrinter2("A",c,a);
MyThreadPrinter2pb =newMyThreadPrinter2("B",a,b);
MyThreadPrinter2pc =newMyThreadPrinter2("C",b,c);
newThread(pa).start();
newThread(pb).start();
newThread(pc).start();}
}
java 多线程 并发实例_Java 多线程(并发)相关推荐
- java中thread实例_Java多线程并发执行demo代码实例
主类:MultiThread,执行并发类 package java8test; import java.util.ArrayList; import java.util.List; import ja ...
- java多线程并发实例_JAVA多线程的并发控制|java多线程并发实例
java的多线程实现主要有两种,一种是继承Thread,一种是实现Runnable接口,这个是java最基本的多线程知识.这里要补充一下,runnable接口中的run方法是不返回任何内容的,如果想返 ...
- java中thread实例_Java多线程2:Thread中的实例方法
Thread类中的方法调用方式: 学习Thread类中的方法是学习多线程的第一步.在学习多线程之前特别提出一点,调用Thread中的方法的时候,在线程类中,有两种方式,一定要理解这两种方式的区别: 1 ...
- java join使用实例_Java多线程中关于join方法的使用实例解析
先上代码 新建一个Thread,代码如下: package com.thread.test; public class MyThread extends Thread { private String ...
- java 多线程并发 问题_JAVA多线程和并发基础面试问答
原文链接 译文连接作者:Pankaj 译者:郑旭东 校对:方腾飞 多线程和并发问题是Java技术面试中面试官比较喜欢问的问题之一.在这里,从面试的角度列出了大部分重要的问题,但是你仍然应该牢固的掌 ...
- java 锁旗标_Java多线程
Java多线程 1. 多线程存在的意义 多线程最大的作用就是能够创建"灵活响应"的桌面程序,而编写多线程最大的困难就是不同线程之间共享资源的问题,要使这些资源不会同时被多个线程访问 ...
- java多线程 cpu分配_java多线程总结(转载)
Java 多线程编程总结 --------------------------------------------------------------------------------------- ...
- java线程代码实现_Java 多线程代码实现讲解
作为一个完全面向对象的语言,Java提供了类 java.lang.Thread 来方便多线程编程,这个类提供了大量的方法来方便我们控制自己的各个线程.那么如何提供给 Java 我们要线程执行的代码呢? ...
- java线程怎么用_Java多线程基本使用
一.概念 1.进程 1.1进程:是一个正在进行中的程序,每一个进程执行都有一个执行顺序,该顺序是一个执行路径,或者叫一个控制单元. 1.2线程:就是进程中一个独立的控制单元,线程在控制着进程的执行,一 ...
最新文章
- Interview:算法岗位面试—上海某公司算法岗位(偏数据分析,互联网行业)技术面试考点之特征工程考察点
- ubuntu makefile交叉编译报错:file not recognized: file format not recognized
- html鼠标自动向下滑动,基于JavaScript实现鼠标向下滑动加载div的代码
- 三菱fx2n做从站的modbus通讯_三菱FX2N与昆仑通态(MCGS)无线编程口通讯例程
- linux 备份配置文件
- 2019年最好的7个人工智能聊天机器人
- 你距离哈佛学霸到底有多远?实力证明,真正的学霸精神不是智商,而是。。。
- 【Spring练习】Spring+SpringMVC+JdbcTemplate简单练习用户管理
- 如何写一份具有业务价值的分析报告
- 2020-10-14Go语言数据类型和变量常量
- 通过access口加vlan标签吗_VLAN标签处理过程
- springboot报错!!!
- linux提交任务执行时间,Linux之任务计划
- PMP证书,项目经理事业进步的阶梯
- 用java实现原神自动弹琴(附谱)
- 计算机机房建设标准.doc,计算机机房建设标准.doc
- 智能家居前装好还是后装好?哪个才是全屋智能更好的选择?
- 矩阵旋转(转置,顺时针,逆时针)
- flutter 自定义进度条progress
- 研二师弟拿下微信 offer