1. 线程的概念

     1.1多进程与多线程

进程:一个正在执行的程序.每个进程执行都有一个执行顺序,该顺序是一个执行路径,或叫一个控制单元. 一个进程至少有一个线程.

    线程:就是进程中的一个独立的控制单元. 线程控制这进程的执行.

多进程的缺点:进程切换开销大;进程间的通信很不方便。

多线程: 指的是在单个程序中可以同时运行多个不同的线程,执行不同的任务,线程切换的开销小 。

     1.2线程的状态

Java语言使用Thread类及其子类的对象来表示线程,新建的线程在它的一个完整的生命周期通常要经历5种状态.

  

冻结状态:在sleep和wait时, 既没有运行资格,有没有执行权

阻塞状态: 具备运行资格, 没有执行权

1.3线程调度与优先级(多线程的特性:随机性)

Java采用抢占式调度策略,下面几种情况下,当前线程会放弃CPU: 
          (1)当前时间片用完; 
          (2)线程在执行时调用了yield() 或sleep() 方法主动放弃;  
          (3)进行I/O 访问,等待用户输入,导致线程阻塞;或者为等候一个条件变量,线程调用wait()方法;    
          (4)有高优先级的线程参与调度。 
     线程的优先级用数字来表示,范围从1~10。主线程的默认优先级为5
          Thread.MIN_PRIORITY=1
          Thread.NORM_PRIORITY=5

    Thread.MAX_PRIORITY=10


2. 多线程编程方法

     2.1 Thread类简介

Thread类综合了Java程序中一个线程需要拥有的属性和方法,其构造方法如下:

public Thread (ThreadGroup group,Runnable target,String name);

public Thread();

public Thread(Runnable target);

public Thread(Runnable target,String name);

public Thread(String name);

public Thread(ThreadGroup group,Runnable target);

public Thread(ThreadGroup group,String name);

Thread类的主要方法以及功能如表

2.2 继承Thread类实现多线程

    需要重写run方法实现线程的任务.需要注意的是,程序中不要直接调用此方法,而是调用线程对象的start()方法启动线程,让其进入可调度状态,线程获得调度自动执行run()方法.

2.3 实现Runnable接口编写多线程

    通过 Thread 类的构造函数public Thread(Runnable target)可以将一个Runnable 接口对象传递给线程,线程在调度时将自动调用Runnable 接口对象的run方法。

  实现方式和继承方法的区别:

    实现方式的好处:避免了单继承的局限性.在定义线程时,建议使用实现方式.


3. 线程资源的同步处理

3.1 临界资源问题

Java对于多线程的安全问题提供了专业的解决方式:  就是同步代码块

  synchronized(对象) {

    需要被同步的代码

  }

对象如同锁,持有锁的线程可以在同步中执行.

没有持有锁的线程及时获取cpu的执行权,也进不去,因为没有获取锁.

同步的前提:

1.必须要有两个或者以上的线程

2.必须要多个线程使用同一个锁

好处:解决了多线程的安全问题

弊端:多个线程需要判断锁,较为消耗资源.

同步函数使用的是哪个锁:

函数需要被对象调用,那么函数都有一个所属对象的引用.就是this.

所以同步函数使用的锁是this锁.静态函数使用的是该方法所在类的字节码文件对象.

   懒汉式增强  

 class Single{
private Single(){}
private static Single s = null;
public static Single getInstance(){
if(s == null){
synchronized(Single.class){
if(s == null)
s = new Single();
}
}
return s;
}
}

3.2 wait()和notify()

3.3 避免死锁

多个线程相互等待对方释放持有的锁,并且在得到对方锁之前不释放自己的锁.(自己能写一个死锁代码)

 /*
* 死锁: 同步中嵌套同步容易发生
*/ public class LockDemo { public static void main(String[] args) {
Test t = new Test(false);
Test t2 = new Test(true);
new Thread(t).start();
new Thread(t2).start(); } } class Test implements Runnable{
private boolean flag;
Test(boolean flag){
this.flag = flag;
}
public void run(){
if(flag){
try{Thread.sleep(30);}catch(Exception e){}
synchronized(KK.oa){
System.out.println("if a");
synchronized(KK.ob){
System.out.println("if b");
}
}
}
else{
synchronized(KK.ob){
System.out.println("else b");
synchronized(KK.oa){
System.out.println("else a");
}
}
}
}
} //自定义两个锁
class KK{
static Object oa = new Object();
static Object ob = new Object();
}

4. 线程间通信

思考1: wait(),notify(),nofifyAll()用来操作线程为什么定义在Object类中?

1.这些方法存在同步中,用来操作同步中的线程,必须要标志它们所操作线程的只有锁.

2.使用这些方法必须标识出所属同步的锁,只有同一个锁上被等待线程,可以被同一个锁上notify唤醒.不可以对不同锁中的线程进行唤醒.

3.锁可以是任意对象,所以任意对象调用的方法一定定义Object类中.

思考2:wait(),sleep()有什么区别?

wait():释放资源,释放锁

sleep():释放资源,不释放锁

 //线程通信: 取mike, 取丽丽
public class Demo2 { public static void main(String[] args) {
Person p = new Person();
new Thread(new Input(p)).start();
new Thread(new Output(p)).start();
} } class Person{
private String name ;
private String sex ;
private boolean flag = false; public synchronized void set(String name, String sex){
//flag为true,表示已存在
if(flag){
try {wait();}catch (InterruptedException e) {e.printStackTrace();}
}
//flag为false,表示没人可加
this.name = name;
this.sex = sex;
flag = true;
this.notify();
} public synchronized void out(){
//flag为false,表示无人没法取
if(!flag){
try {wait();}catch (InterruptedException e) {e.printStackTrace();}
}
//flag为true,表示有人可以取出
System.out.println(toString());
flag = false;
this.notify();
} public String toString(){
return "姓名:"+name+"性别:"+sex;
}
} class Input implements Runnable{
private Person p; Input(Person p){
this.p = p;
} public void run(){
boolean flag = false;
while(true){
if(flag){
p.set("likai","男");
flag = false;
}
else{
p.set("tangll","女");
flag = true;
}
}
}
} class Output implements Runnable{
private Person p; Output(Person p){
this.p = p;
} public void run(){
while(true){
p.out();
}
}
}

线程通信Demo1

 public class Demo3 {

     public static void main(String[] args) {
Bridge b = new Bridge();
for(int i=1;i<6;i++){
new Thread(new Person(b,"由北向南第"+i+"人")).start();
} for(int i=1;i<7;i++){
new Thread(new Person(b,"由南向北第"+i+"人")).start();
}
}
} class Bridge{
private String name;
private boolean flag;
public synchronized void UpBridge(){
if(flag){
try {
wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
flag = true; }
public synchronized void DownBridge(){
flag = false;
notifyAll();
}
} class Person implements Runnable{
private Bridge bridge;
private String pName; Person(Bridge bridge, String pName){
this.bridge = bridge;
this.pName = pName;
} public void run() {
bridge.UpBridge();
try{
Thread.sleep(80);
}catch(InterruptedException e){}
bridge.DownBridge();
System.out.println(pName);
} }

过桥问题

生产消费者问题

生产消费者问题--JDK1.5提供的多线程升级解决方法

将同步synchronized替换成Lock操作.

将Object中的wait,notify,notifyAll方法替换成Condition对象.

该对象可以Lock锁进行获取.

    该实例中,实现了本方只唤醒对方的操作.

停止线程

1.定义循环结束标记

因为线程运行代码一般都是循环,只是控制了循环即可.

2.使用interrupt(中断)方法: 该方法是强制结束线程的冻结状态,使线程回到运行状态中来.这样就可以操作标记结束.

(注)stop方法和suspend方法已过时不在再使用.

setDeamon()方法

标记为后台线程, 当所有前台前程结束后会Java虚拟机退出. 该方法必须在启动线程前调用.

join()方法

等待该线程终止, 其实是抢夺CPU执行权.

当A线程执行到B线程的.join()方法, A就会等待. 等B线程都执行完, A才会执行. join可以用来临时加入线程执行.

toString()

返回该线程的字符串表示形式,包括线程名称,优先级和线程组

yield()

暂停当前执行的线程对象,并执行其他线程

06_Java多线程、线程间通信的更多相关文章

  1. Java多线程——线程间通信

    Java多线系列文章是Java多线程的详解介绍,对多线程还不熟悉的同学可以先去看一下我的这篇博客Java基础系列3:多线程超详细总结,这篇博客从宏观层面介绍了多线程的整体概况,接下来的几篇文章是对多线 ...

  2. 多线程-线程间通信-多生产者多消费者问题(JDK1.5后Lock,Condition解决办法及开发中代码范例)

    1 package multithread4; 2 3 import java.util.concurrent.locks.Condition; 4 import java.util.concurre ...

  3. 多线程-线程间通信-多生产者多消费者问题解决(notifyAll)

    1 package multithread4; 2 3 /* 4 * 生产者,消费者. 5 * 6 * 多生产者,多消费者的问题. 7 * 8 * if判断标记,只有一次,会导致不该运行的线程运行了. ...

  4. 多线程 线程间通信 wait,notify

    1. 方法wait锁释放,notify()锁不释放

  5. 【原】iOS多线程之线程间通信和线程互斥

    线程间通信 1> 线程间通信分为两种 主线程进入子线程(前面的方法都可以) 子线程回到主线程 2> 返回主线程 3> 代码 这个案例的思路是:当我触摸屏幕时,会在子线程加载图片,然后 ...

  6. C++多线程编程(三)线程间通信

    多线程编程之三——线程间通讯 作者:韩耀旭 原文地址:http://www.vckbase.com/document/viewdoc/?id=1707 七.线程间通讯 一般而言,应用程序中的一个次要线 ...

  7. Java多线程基础——线程间通信

    在使用多线程的时候,经常需要多个线程进行协作来完成一件事情.在前面两章分析了Java多线程的基本使用以及利用synchronized来实现多个线程同步调用方法或者执行代码块.但上面两章的内容涉及到的例 ...

  8. Java多线程(二) —— 线程安全、线程同步、线程间通信(含面试题集)

    一.线程安全 多个线程在执行同一段代码的时候,每次的执行结果和单线程执行的结果都是一样的,不存在执行结果的二义性,就可以称作是线程安全的. 讲到线程安全问题,其实是指多线程环境下对共享资源的访问可能会 ...

  9. Java多线程:线程间通信之volatile与sychronized

    由前文Java内存模型我们熟悉了Java的内存工作模式和线程间的交互规范,本篇从应用层面讲解Java线程间通信. Java为线程间通信提供了三个相关的关键字volatile, synchronized ...

随机推荐

  1. mybatis 获取自增ID

    在Mybatis Mapper文件中添加属性“useGeneratedKeys”和“keyProperty”,其中keyProperty是Java对象的属性名,而不是表格的字段名. <inser ...

  2. vs2010:fatal error LNK1123: 转换到 COFF 期间失败

    解决方法: 项目\属性\配置属性\清单工具\输入和输出\嵌入清单:原来是“是”,改成“否”.

  3. 【linux】ps

    来源:http://blog.chinaunix.net/uid-25681671-id-3201927.html Linux下PS命令详解 要对系统中进程进行监测控制,查看状态,内存,CPU的使用情 ...

  4. orace 取昨天凌晨的日期

    sysdate 为现在时间sysdate-1为昨天trunc(sysdate-1)为昨天凌晨0:00trunc(sysdate-1)+20/24 为昨天晚上8点select trunc(sysdate ...

  5. 查看Linux内核

    方法一: 命令: uname -a 作用: 查看系统内核版本号及系统名称 方法二: 命令: cat /proc/version 作用: 查看目录"/proc"下version的信息 ...

  6. IOS - 唯一标识符的获得和更新

    苹果公司不可能让其他人获得个人终端的唯一标识符,所以一个终端给另一个终端发送消息,必须经过苹果的APNS(Apple Push Notification Service)....而且苹果为了防止苹果用 ...

  7. 26. Remove Duplicates from Sorted Array

    题目: Given a sorted array, remove the duplicates in place such that each element appear only once and ...

  8. 实现 Bootstrap 基本布局

    看到了一篇 20 分钟打造 Bootstrap 站点的文章,内容有点老,重新使用 Bootstrap3 实现一下,将涉及的内容也尽可能详细说明. 1. 创建基本的页面 我们先创建一个基本的 HTML ...

  9. 51nod1057(python2计算n!)

    题目链接:www.51nod.com/onlineJudge/questionCode.html#!problemId=1057 思路:直接for循环呗- 代码: n = int( raw_input ...

  10. 菜鸟学Linux命令:nohup命令启动程序

    在UNIX/LINUX中,普通进程用&符号放到后台运行,如果启动该程序的控制台logout,则该进程随即终止. 要实现守护进程,一种方法是按守护进程的规则去编程,比较麻烦:另一种方法是仍然用普 ...