JAVA多线程一
介绍
线程是操作系统的最小单位,一个进程可以创建多个线程。
线程有五种状态,分别是新建、就绪、运行、阻塞、死亡状态。

多线程可以提高执行效率,但是如果单线程可以完成的任务,使用多线程反而会增加不必要的开销,降低效率。例如将某个数加一百次,使用多线程反而会比单线程耗费的时间多。
创建线程
java创建线程有两种方法
- 继承Thread,重写run函数,调用start方法
package com.thread;
public class ExtendTreadTest extends Thread {
int i = 0;
public void run()
{
for(;i<100;i++){
System.out.println(getName()+" "+i);
try {
sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
public static void main(String[] args) {
new ExtendTreadTest().start();
new ExtendTreadTest().start();
}
}
- 实现Runnable接口,重写run函数,调用start方法
package com.thread;
import static java.lang.Thread.sleep;
public class ImplementRunnable implements Runnable {
int i = 0;
public void run()
{
for(;i<100;i++){
System.out.println(Thread.currentThread().getName()+" "+i);
try {
sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
public static void main(String[] args) {
Thread thread = new Thread(new ImplementRunnable(), "test1");
thread.start();
Thread thread1 = new Thread(new ImplementRunnable(), "test2");
thread1.start();
}
}
JAVA允许继承一个类,并实现多个接口,所以通常应用中实现Runnable比较好。
线程间通信
volatile和synchornized关键字
volatile
volatile 会确保线程在每一次使用变量之前都会从共享内存中读取变量的值,然后然后放入自己的工作内存进行处理,处理完成后将新的值立即同步到内存,在这个期间,可能会有其他的线程也对该变量进行处理。所以volatile并不是线程安全的。
package com.thread;
public class Test {
public volatile int inc = 0;
public void increase() {
inc++;
System.out.println(inc);
}
public static void main(String[] args) {
final Test test = new Test();
for(int i=0;i<10;i++){
new Thread(){
public void run() {
for(int j=0;j<10000;j++)
test.increase();
};
}.start();
}
while(Thread.activeCount()>1) //保证前面的线程都执行完
Thread.yield();
System.out.println(test.inc);
}
}
例如上面的方法,期望的实验结果是100000,但是实际情况偶尔会出现小于100000的情况。是因为volatile没法保证对变量操作的原子性,inc++不是一个原子操作,就会导致结果出现问题。
解决的办法有使用synchronized,锁,以及将变量变为原子操作AtomicInteger。
总体来说,如果能保证对变量的操作是原子性的,那么使用volatile会是一个较好的办法。
参考:http://www.cnblogs.com/dolphin0520/p/3920373.html
synchornized
确保多线程对临界资源的互斥性访问的一种方式是使用synchornized。
例如上面的例子:
public synchornized void increase() {
inc++;
System.out.println(inc);
}
或者
public void increase() {
synchornized {
inc++;
System.out.println(inc);
}
}
上面的两个都是对类的对象做同步,而不是对类本身进行同步。每个线程必须共用一个对象才能够达到同步效果。
public void increase() {
synchornized(Test.class) {
inc++;
System.out.println(inc);
}
}
上面是对类本身进行同步,对于Test类,它只有一个类定义,同时只有一个线程可以访问increase方法。
在JAVA中,任意的一个对象都有自己的监视器,当这个对象由同步块或者同步方法调用的时候,执行的线程必须获取该对象的监视器,然后再进入同步块(方法),否则就会进入一个阻塞队列,等待线程退出监视器。
wait和notify以及notifyAll
1.调用某个对象的wait()方法能让当前线程阻塞,并且当前线程必须拥有此对象的monitor(即锁)
2.调用某个对象的notify()方法能够唤醒一个正在等待这个对象的monitor的线程,如果有多个线程都在等待这个对象的monitor,则只能唤醒其中一个线程
3.调用notifyAll()方法能够唤醒所有正在等待这个对象的monitor的线程
package com.thread;
import static java.lang.Thread.sleep;
public class WaitNotify {
static boolean flag = true;
static Object lock = new Object();
public static void main(String[] args) {
Thread waitThread = new Thread(new Wait(), "waitThread");
waitThread.start();
try {
sleep(100);
} catch (Exception err) {
err.printStackTrace();
}
Thread notifyThread = new Thread(new Notify(), "notifyThread");
notifyThread.start();
}
static class Wait implements Runnable {
public void run() {
synchronized (lock) {
while (flag) {
try {
System.out.println(Thread.currentThread().getName() + " wait");
lock.wait();
} catch (InterruptedException e) {
}
}
}
System.out.println(Thread.currentThread().getName() + " run");
}
}
static class Notify implements Runnable {
public void run() {
synchronized (lock) {
try {
System.out.println(Thread.currentThread().getName() + " hold lock");
sleep(1000);
lock.notify();
flag = false;
} catch (InterruptedException e) {
}
}
try {
sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + " run");
}
}
}

图中,waitThread首先获取了对象的锁,然后调用对象的wait()方法,从而放弃了锁并进入等待队列WaitQueue中。NotifyThread随后获取了对象的锁,并调用对象的notify方法,将WaitThread从WaitQueue转移到了synchornizedQueue中,然后waitThread转变为了阻塞状态。NotifyThread释放了锁之后,WaitThread再次获得了锁并从wait()方法中返回并继续执行。
参考:JAVA并发编程的艺术
JAVA多线程一的更多相关文章
- 深入java多线程一
涉及到 1.线程的启动(start) 2.线程的暂停(suspend()和resume()) 3.线程的停止(interrupt与异常停止,interrupt与睡眠中停止,stop(),return) ...
- java 多线程一
java 多线程一 java 多线程二 java 多线程三 java 多线程四 java 多线程实现的几种方式: 1.extends Thread 2.implements Runnable 3.im ...
- (三十)java多线程一
我们通常在电脑中打开的应用称作进程,一个应用就是一个进程,而一个进程里边一般包含多个线程. 系统要为每一个进程分配独立的内存空间,而进程里的多个线程共用这些内存. 我们通常所写的main方法就是一个线 ...
- Java多线程之ConcurrentSkipListMap深入分析(转)
Java多线程之ConcurrentSkipListMap深入分析 一.前言 concurrentHashMap与ConcurrentSkipListMap性能测试 在4线程1.6万数据的条件下, ...
- 用“逐步排除”的方法定位Java服务线上“系统性”故障(转)
一.摘要 由于硬件问题.系统资源紧缺或者程序本身的BUG,Java服务在线上不可避免地会出现一些“系统性”故障,比如:服务性能明显下降.部分(或所 有)接口超时或卡死等.其中部分故障隐藏颇深,对运维和 ...
- JAVA多线程之wait/notify
本文主要学习JAVA多线程中的 wait()方法 与 notify()/notifyAll()方法的用法. ①wait() 与 notify/notifyAll 方法必须在同步代码块中使用 ②wait ...
- JAVA多线程之volatile 与 synchronized 的比较
一,volatile关键字的可见性 要想理解volatile关键字,得先了解下JAVA的内存模型,Java内存模型的抽象示意图如下: 从图中可以看出: ①每个线程都有一个自己的本地内存空间--线程栈空 ...
- java多线程之yield,join,wait,sleep的区别
Java多线程之yield,join,wait,sleep的区别 Java多线程中,经常会遇到yield,join,wait和sleep方法.容易混淆他们的功能及作用.自己仔细研究了下,他们主要的区别 ...
- Java多线程之Runnable与Thread
Java多线程之Thread与Runnable 一.Thread VS Runnable 在java中可有两种方式实现多线程,一种是继承Thread类,一种是实现Runnable接口:Thread类和 ...
随机推荐
- 自绘CProgressCtrl进度条控件,支持自定义显示文本和进程百分比信息
// CXProgressCtrl 头文件 #pragma once // CXProgressCtrl class CXProgressCtrl : public CProgressCtrl { D ...
- iOS:iPad和iPhone开发的异同(UIPopoverController、UISplitViewController)
iPad和iPhone开发的异同 1.iPhone和iPad: niPhone是手机,iPad.iPad Mini是平板电脑 iPhone和iPad开发的区别 屏幕的尺寸 \分辨率 UI元素的排布 \ ...
- 统计MySQL数据表大小
SELECT CONCAT(TRUNCATE(SUM(data_length)/1024/1024,2),'MB') AS data_size,CONCAT(TRUNCATE(SUM(max_data ...
- Java中的Enum的使用与分析
使用name()方法和valueOf(String)方法可以在枚举类型对象和字符串之间方便得转换.如果valueOf(String)方法的参数不是该枚举类型合法的字符串,则会抛出IllegalArgu ...
- Git工作流指南:Gitflow工作流 Comparing Workflows
Comparing Workflows The array of possible workflows can make it hard to know where to begin when imp ...
- 15_采用Pull解析器解析和生成XML内容
java还提供SAX和DOM用于解析XML Android还集成了Pull解析器——推荐 package cn.itcast.service; import java.io.InputStream; ...
- Java I/O操作学习笔记
书上写的有点乱,所以就自己总结了一下,主要参考:http://www.cnblogs.com/qianbi/p/3378466.html 1.从文件读出和写入: import java.io.*; i ...
- 一个日期Js文件。 2013年10月12日 星期六 癸巳年九月初八
1.简单用法 <div align="center"> <SCRIPT language=JavaScript src="js/calendar.js ...
- build.gradle(Project) 和 build.gradle(Module) 的区别
参考: http://stackoverflow.com/questions/28295933/difference-between-build-gradleproject-and-build-gra ...
- Oracle中用一个表的数据更新另一个表的数据
update tbl1 a set (a.col1, a.col2) = (select b.col1, b.col2 from tbl2 ...