Java多线程是Java开发中的基础内容,但是涉及到高并发就有很深的研究可做了。

最近看了下《Java并发实战》,发先有些地方,虽然可以理解,但是自己在应用中很难下手。

  所以还是先回顾一下基础知识:

Java中的线程

  线程的相关知识,了解操作系统的基本都能有所了解。

  线程有5中状态,基本变化如图所示:

如何在Java代码中创建线程

  众所周知,Java创建线程有两种方式:

  1 实现Runable接口

  2 继承Thread类

  那么这两种方式有什么区别呢?

  1 Runable属于接口,所以可以有多个实现;Thread只有一个。

  2 实现Runable的线程类,可以被多个线程实例共享数据。

  举个简单的例子,火车站售票处一共有3个售票口,但是只剩下5张票:

  如果单纯使用Thread实现3个售票口的售票过程:

package com.imooc.test;
class MyThread extends Thread{ private int ticketsCount = 5;
private String name; public MyThread(String name){
this.name = name;
} @Override
public void run() {
while(ticketsCount > 0){
ticketsCount--;
System.out.println(name+" 卖了一张票,还剩下:"+ticketsCount);
}
}
}
public class TicketsTestThread {
public static void main(String[] args) {
MyThread mt1 = new MyThread("窗口1");
MyThread mt2 = new MyThread("窗口2");
MyThread mt3 = new MyThread("窗口3"); mt1.start();
mt2.start();
mt3.start();
}
}

  执行结果如下:

窗口1 卖了一张票,还剩下:4
窗口2 卖了一张票,还剩下:4
窗口1 卖了一张票,还剩下:3
窗口1 卖了一张票,还剩下:2
窗口1 卖了一张票,还剩下:1
窗口1 卖了一张票,还剩下:0
窗口2 卖了一张票,还剩下:3
窗口2 卖了一张票,还剩下:2
窗口2 卖了一张票,还剩下:1
窗口2 卖了一张票,还剩下:0
窗口3 卖了一张票,还剩下:4
窗口3 卖了一张票,还剩下:3
窗口3 卖了一张票,还剩下:2
窗口3 卖了一张票,还剩下:1
窗口3 卖了一张票,还剩下:0

  可以看到每个线程拥有自己的5张票,其实是重复了!

  那么如果使用Runnable,则不会出现这种情况:

package com.imooc.test;
class MyRunnable implements Runnable{ private int ticketsCount = 5; @Override
public void run() {
while(ticketsCount > 0){
ticketsCount--;
System.out.println(Thread.currentThread().getName()+" 卖了一张票,还剩下:"+ticketsCount);
}
}
}
public class TicketsTestRunnable {
public static void main(String[] args) {
MyRunnable mr = new MyRunnable();
Thread th1 = new Thread(mr,"窗口1");
Thread th2 = new Thread(mr,"窗口2");
Thread th3 = new Thread(mr,"窗口3"); th1.start();
th2.start();
th3.start();
}
}

  执行结果:

窗口1 卖了一张票,还剩下:4
窗口3 卖了一张票,还剩下:2
窗口3 卖了一张票,还剩下:1
窗口3 卖了一张票,还剩下:0
窗口2 卖了一张票,还剩下:3

  这是因为创建Thread实例时,使用的是同一个MyRunnable类对象,所以会共享其中的数据。

用户线程与守护线程

  在Java线程中,共有两类线程:

  1 用户线程:用户代码生成

  2 守护线程:用于特定的功能,当用户线程都结束时,守护线程会随着JVM的停止而停止,因此守护线程不能用于IO操作。

  那么下面一个简单的守护线程的例子:

  创建一个守护线程,持续不断的向文件中写入数据。主线程中启动该线程,然后主线程在一定时间后,退出。

  观察守护线程的状态!

  代码如下:

package com.imooc.test;

import java.io.File;
import java.io.FileOutputStream;
import java.io.OutputStream;
import java.util.Scanner; class DaemonThread implements Runnable{
public void run() {
System.out.println("进入守护线程:"+Thread.currentThread().getName());
try {
Write2File();
} catch (Exception e) {
e.printStackTrace();
}
System.out.println("退出守护线程:"+Thread.currentThread().getName());
}
public void Write2File() throws Exception{
File filename = new File("d:"+File.separator+"daemon.txt");
OutputStream os = new FileOutputStream(filename,true);
int count = 0;
while(count < 999){
os.write(("\r\nword "+count).getBytes());
System.out.println("守护线程"+Thread.currentThread().getName()+
"写入了 "+count);
count++;
Thread.sleep(1000);
}
}
}
public class DaemonTest {
public static void main(String[] args) {
System.out.println("进入主线程"+Thread.currentThread().getName());
DaemonThread dt = new DaemonThread();
Thread th = new Thread(dt);
th.setDaemon(true);
th.start(); try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
} System.out.println("退出主线程"+Thread.currentThread().getName());
}
}

  当主线程睡眠了5秒后,便结束。此时JVM中没有其他的用户线程,于是守护线程也直接退出。

  执行结果如下:

进入主线程main
进入守护线程:Thread-0
守护线程Thread-0写入了 0
守护线程Thread-0写入了 1
守护线程Thread-0写入了 2
守护线程Thread-0写入了 3
守护线程Thread-0写入了 4
退出主线程main

  可以看到守护线程直接就中断退出了!

  鉴于守护线程的这种特性,常用于实时监控系统状态。比如数据库,JVM等等。

查看线程快照

  通过使用Jstack.exe程序,可以帮助用户查看线程状态。

  使用方法:

  1 查询线程PID

  2 在cmd中输入jstack -l pid

C:\Users\Administrator>jstack -l 5028
2015-04-01 17:43:30
Full thread dump Java HotSpot(TM) Client VM (24.60-b09 mixed mode, sharing): "Thread-0" daemon prio=6 tid=0x00928800 nid=0x2798 waiting on condition [0x03d4f
000]
java.lang.Thread.State: TIMED_WAITING (sleeping)
at java.lang.Thread.sleep(Native Method)
at com.imooc.test.DaemonThread.Write2File(DaemonTest.java:27)
at com.imooc.test.DaemonThread.run(DaemonTest.java:12)
at java.lang.Thread.run(Unknown Source) Locked ownable synchronizers:
- None "Service Thread" daemon prio=6 tid=0x008de000 nid=0xd64 runnable [0x00000000]
java.lang.Thread.State: RUNNABLE Locked ownable synchronizers:
- None "C1 CompilerThread0" daemon prio=10 tid=0x008dc400 nid=0x2158 waiting on conditi
on [0x00000000]
java.lang.Thread.State: RUNNABLE Locked ownable synchronizers:
- None "Attach Listener" daemon prio=10 tid=0x008f4800 nid=0x13e0 waiting on condition
[0x00000000]
java.lang.Thread.State: RUNNABLE Locked ownable synchronizers:
- None "Signal Dispatcher" daemon prio=10 tid=0x00905800 nid=0x1c7c runnable [0x0000000
0]
java.lang.Thread.State: RUNNABLE Locked ownable synchronizers:
- None "Finalizer" daemon prio=8 tid=0x00875800 nid=0x2460 in Object.wait() [0x03e0f000
]
java.lang.Thread.State: WAITING (on object monitor)
at java.lang.Object.wait(Native Method)
- waiting on <0x23800fc8> (a java.lang.ref.ReferenceQueue$Lock)
at java.lang.ref.ReferenceQueue.remove(Unknown Source)
- locked <0x23800fc8> (a java.lang.ref.ReferenceQueue$Lock)
at java.lang.ref.ReferenceQueue.remove(Unknown Source)
at java.lang.ref.Finalizer$FinalizerThread.run(Unknown Source) Locked ownable synchronizers:
- None "Reference Handler" daemon prio=10 tid=0x00870800 nid=0x21c8 in Object.wait() [0
x03b6f000]
java.lang.Thread.State: WAITING (on object monitor)
at java.lang.Object.wait(Native Method)
- waiting on <0x23800db0> (a java.lang.ref.Reference$Lock)
at java.lang.Object.wait(Object.java:503)
at java.lang.ref.Reference$ReferenceHandler.run(Unknown Source)
- locked <0x23800db0> (a java.lang.ref.Reference$Lock) Locked ownable synchronizers:
- None "main" prio=6 tid=0x0099c000 nid=0x14b8 waiting on condition [0x0052f000]
java.lang.Thread.State: TIMED_WAITING (sleeping)
at java.lang.Thread.sleep(Native Method)
at com.imooc.test.DaemonTest.main(DaemonTest.java:40) Locked ownable synchronizers:
- None "VM Thread" prio=10 tid=0x0086f000 nid=0x22dc runnable "VM Periodic Task Thread" prio=10 tid=0x00927000 nid=0x131c waiting on condition JNI global references: 111 C:\Users\Administrator>

  其中详细的描述了线程的名字,是否为守护线程,以及状态等等。

参考

  【1】慕课网Thread VS Runnable:http://www.imooc.com/learn/312

Java多线程之Runable与Thread的更多相关文章

  1. Java多线程之Runnable与Thread

    Java多线程之Thread与Runnable 一.Thread VS Runnable 在java中可有两种方式实现多线程,一种是继承Thread类,一种是实现Runnable接口:Thread类和 ...

  2. Java多线程之this与Thread.currentThread()的区别——java多线程编程核心技术

      package mythread; public class CountOperate extends Thread{ public CountOperate(){ System.out.prin ...

  3. JAVA多线程之wait/notify

    本文主要学习JAVA多线程中的 wait()方法 与 notify()/notifyAll()方法的用法. ①wait() 与 notify/notifyAll 方法必须在同步代码块中使用 ②wait ...

  4. JAVA多线程之volatile 与 synchronized 的比较

    一,volatile关键字的可见性 要想理解volatile关键字,得先了解下JAVA的内存模型,Java内存模型的抽象示意图如下: 从图中可以看出: ①每个线程都有一个自己的本地内存空间--线程栈空 ...

  5. java多线程之yield,join,wait,sleep的区别

    Java多线程之yield,join,wait,sleep的区别 Java多线程中,经常会遇到yield,join,wait和sleep方法.容易混淆他们的功能及作用.自己仔细研究了下,他们主要的区别 ...

  6. JAVA多线程之UncaughtExceptionHandler——处理非正常的线程中止

    JAVA多线程之UncaughtExceptionHandler——处理非正常的线程中止 背景 当单线程的程序发生一个未捕获的异常时我们可以采用try....catch进行异常的捕获,但是在多线程环境 ...

  7. java多线程之wait和notify协作,生产者和消费者

    这篇直接贴代码了 package cn.javaBase.study_thread1; class Source { public static int num = 0; //假设这是馒头的数量 } ...

  8. Java多线程之ConcurrentSkipListMap深入分析(转)

    Java多线程之ConcurrentSkipListMap深入分析   一.前言 concurrentHashMap与ConcurrentSkipListMap性能测试 在4线程1.6万数据的条件下, ...

  9. Java——多线程之Lock锁

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

随机推荐

  1. CentOS添加新硬盘到新的分区(xfs/ext4) 或者添加新分区

    CentOs添加新硬盘到新的分区(xfs/ext4)  添加新分区 转载请注明:http://www.cnblogs.com/juandx/p/5618162.html 这篇文章介绍怎么添加一块新的硬 ...

  2. ADO.Net(三)——数据库操作类

    操作数据类 避免代码重用.造对象太多.不能分工开发 利用面向对象的方法,把数据访问的方式优化一下,利用封装类 一般封装成三个类: 1.数据连接类 提供数据连接对象 需要引用命名空间: using Sy ...

  3. SQL Server 2000 :选择许可模式及更改

    在SQL Server企业版中,有一个许可模式概念,有两种许可模式:“处理器许可证”和“每客户”模式.“处理器许可证”模式表示允许几个CPU运行SQL Server,“每客户”决定的是客户端连接数. ...

  4. 烂泥:学习ubuntu之快速搭建LNMP环境

    本文由秀依林枫提供友情赞助,首发于烂泥行天下 现在公司使用的都是ubuntu系统,这几天由于个别项目需要,需要搭建一个LNMP环境.为了快速搭建这个环境,我使用是apt-get方式进行安装.具体的操作 ...

  5. linux批量删除进程

    在虚拟机用脚本跑了几十个client程序用来测试服务器,然后发现参数设置错误,得重来,就傻眼了,不知道怎么关这么多client进程,总不能一个一个关.还好,学习一下,想出了以下的命令.   ps -e ...

  6. mybatis 插入数据时返回主键

    在使用MyBatis做持久层时,insert语句默认是不返回记录的主键值,而是返回插入的记录条数:显然,假如主键是你生成后插入的,自然你已经有主键了,显然不需要我们再去获得,所以我们这里处理的是当主键 ...

  7. stanford coursera 机器学习编程作业 exercise 6(支持向量机-support vector machines)

    在本练习中,先介绍了SVM的一些基本知识,再使用SVM(支持向量机 )实现一个垃圾邮件分类器. 在开始之前,先简单介绍一下SVM ①从逻辑回归的 cost function 到SVM 的 cost f ...

  8. VIJOS1240 朴素的网络游戏[DP]

    描述 佳佳最近又迷上了某款类似于虚拟人生的网络游戏.在游戏中,佳佳是某旅行团的团长,他需要安排客户住进旅馆.旅馆给了佳佳的旅行团一个房间数的限制.每一个房间有不同的容纳人数和价钱(这个价格是房间的总价 ...

  9. AC日记——鬼谷子的钱袋 codevs 2998

    2998 鬼谷子的钱袋 2006年省队选拔赛湖南  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 大师 Master    题目描述 Description 鬼谷子非常聪明,正 ...

  10. Unity官网教程之Tips

    前言 翻译整理unity官网教程的tips部分,原文:http://unity3d.com/cn/learn/tutorials/topics/tips Snap 按住Ctrl键,并用鼠标拖动Game ...