多线程基础及实例(java)

前言:

每个正在系统上运行的程序都是一个进程。每个进程包含一到多个线程。线程是一组指令的集合,或者是程序的特殊段,它可以在程序里独立执行。也可以把它理解为代码运行的上下文。所以线程基本上是轻量级的进程,它负责在单个程序里执行多任务。通常由操作系统负责多个线程的调度和执行。

使用线程可以把占据时间长的程序中的任务放到后台去处理,程序的运行速度可能加快,在一些等待的任务实现上如用户输入、文件读写和网络收发数据等,线程就比较有用了。在这种情况下可以释放一些珍贵的资源如内存占用等等。

如果有大量的线程,会影响性能,因为操作系统需要在它们之间切换,更多的线程需要更多的内存空间,线程的中止需要考虑其对程序运行的影响。通常块模型数据是在多个线程间共享的,需要防止线程死锁情况的发生。

正文:

两种实现多线程的方式,继承Thread类或者实现Runnable接口。

继承Thread类 :

package com.zsz.thread;

public class MyThread extends Thread {

private int index;

public MyThread( int index){

this. index = index;

}

public void run () {

forint i=0;i<5;i++){

System. out.println( index+ ":"+i);

}

}

public static void main(String[] args){

MyThread MyThread = new MyThread(1);

MyThread MyThread2 = new MyThread(2);

MyThread.start();

MyThread2.start();

}

}

可能的运行结果:

1:0

2:0

1:1

2:1

1:2

2:2

1:3

2:3

1:4

2:4

由于执行CPU占用时间被切换,线程状态的切换,导致执行顺序的不同。

实现Runnable接口:

package com.zsz.thread;

public class MyRunnable implements Runnable{

private int index;

public MyRunnable( int index){

this. index = index;

}

@Override

public void run() {

forint i=0;i<5;i++){

System. out.println( index+ ":"+i);

}

}

public static void main(String[] args){

MyRunnable MyRunnable1 = new MyRunnable(1);

Thread Thread1 = new Thread(MyRunnable1);

MyRunnable MyRunnable2 = new MyRunnable(2);

Thread Thread2 = new Thread(MyRunnable2);

Thread1.start();

Thread2.start();

}

}

可能的结果:

1:0

1:1

1:2

2:0

2:1

2:2

2:3

2:4

1:3

1:4

场景应用:

一、车站多个窗口买票,车票总数是一定的,实现Runnable可以共享总票数。

package com.zsz.thread;

class SaleTicket  implements Runnable{

private int ret = 5; //剩下票数

private int num;            //一次买票张数

public SaleTicket( int num){

this. num = num;

}

@Override

public void run() {

synchronizedthis){

if( num> ret){

System. out.println( "余票不足");

return;

}

ret = ret - num;

System. out.println( "出票"+num+"张成功,剩余票数:" +ret );  //出票成功

}

}

}

public class MyRunnable extends Thread{

public static void main(String[] args){

SaleTicket saleTicket1 = new SaleTicket(1);

new Thread(saleTicket1).start();

new Thread(saleTicket1).start();

new Thread(saleTicket1).start();

new Thread(saleTicket1).start();

new Thread(saleTicket1).start();

new Thread(saleTicket1).start();

new Thread(saleTicket1).start();

}

}

可能的执行结果:

出票1张成功,剩余票数:4

出票1张成功,剩余票数:3

出票1张成功,剩余票数:2

出票1张成功,剩余票数:1

出票1张成功,剩余票数:0

余票不足

余票不足

注:考虑数据同步和线程安全,synchronized (this)确保同步,确保一个时刻只有一个线程占用synchronized 程序块,否则会出现线程不安全的情况,。

实现Runnable接口相比继承Thread类:
1):适合多个相同的程序代码的线程去处理同一个资源
2):可以避免java中的单继承的限制
3):增加程序的健壮性,代码可以被多个线程共享,代码和数据独立。

main方法其实也是一个线程。

线程其他情况:

线程休眠:Thread.sleep(2000);

线程中断:new Thread(new Runnable()).interrupt();

线程优先级:new Thread(new Runnable()).setPriority(8);

 


线程的状态及说明:

1、新建状态(New):新创建了一个线程对象。 
2、就绪状态(Runnable):线程对象创建后,其他线程调用了该对象的start()方法。该状态的线程位于可运行线程池中,变得可运行,等待获取CPU的使用权。

3、运行状态(Running):就绪状态的线程获取了CPU,执行程序代码。 
4、阻塞状态(Blocked):阻塞状态是线程因为某种原因放弃CPU使用权,暂时停止运行。直到线程进入就绪状态,才有机会转到运行状态。阻塞的情况分三种: 

  4.1、等待阻塞:运行的线程执行wait()方法,JVM会把该线程放入等待池中。 

  4.2、同步阻塞:运行的线程在获取对象的同步锁时,若该同步锁被别的线程占用,则JVM会把该线程放入锁池中。 
  4.3、其他阻塞:运行的线程执行sleep()或join()方法,或者发出了I/O请求时,JVM会把该线程置为阻塞状态。当sleep()状态超时、join()等待线程终止或者超时、或者I/O处理完毕时,线程重新转入就绪状态。 
5、死亡状态(Dead):线程执行完了或者因异常退出了run()方法,该线程结束生命周期。

 

关键字volatile

用volatile修饰的变量,线程在每次使用变量的时候,都会读取变量修改后的最的值。volatile很容易被误用,用来进行原子性操作。

volatile不用做原子性操作的原因是:线程在运行时会在线程内存块中变量副本,之后,主内存中的变量与线程内的变量不联系,当运行结束时,线程内的变量会将值同步给主内存,因而会有可能出现线程不安全。

 有相关问题,可以提出来一起讨论。后续将会张贴一些进阶的多线程、线程池内容。

 博客园原文地址:http://www.cnblogs.com/zhongshengzhen

时间: 2024-10-27 12:24:46

多线程基础及实例(java)的相关文章

[转]Java多线程干货系列—(一)Java多线程基础

Java多线程干货系列—(一)Java多线程基础 字数7618 阅读1875 评论21 喜欢86 前言 多线程并发编程是Java编程中重要的一块内容,也是面试重点覆盖区域,所以学好多线程并发编程对我们来说极其重要,下面跟我一起开启本次的学习之旅吧. 正文 线程与进程 1 线程:进程中负责程序执行的执行单元线程本身依靠程序进行运行线程是程序中的顺序控制流,只能使用分配给程序的资源和环境 2 进程:执行中的程序一个进程至少包含一个线程 3 单线程:程序中只存在一个线程,实际上主方法就是一个主线程 4

Java多线程总结(一)多线程基础

本文转载地址:             http://www.cnblogs.com/zrtqsk/p/3776328.html 多线程是Java学习的非常重要的方面,是每个Java程序员必须掌握的基本技能.本文只是多线程细节.本质的总结,并无代码例子入门,不适合初学者理解.初学者学习多线程,建议一边看书.看博文,以便写代码尝试. 一.进程 进程是操作系统结构的基础:是一次程序的执行:是一个程序及其数据在处理机上顺序执行时所发生的活动.操作系统中,几乎所有运行中的任务对应一条进程(Process

Java多线程基础(一)

线程与进程 1 线程:进程中负责程序执行的执行单元线程本身依靠程序进行运行线程是程序中的顺序控制流,只能使用分配给程序的资源和环境 2 进程:执行中的程序一个进程至少包含一个线程 3 单线程:程序中只存在一个线程,实际上主方法就是一个主线程 4 多线程:在一个程序中运行多个任务目的是更好地使用CPU资源 5  在Java语言中,引入对象互斥锁的概念,保证共享数据操作的完整性. 每个对象都对应于一个可称为"互斥锁"的标记,这个标记保证在任一时刻,只能有一个线程访问对象用关键字synchr

Java多线程基础

1. 前言 这篇文章,是对Java多线程编程的基础性介绍. 文章将介绍Java语言为支持多线程编程提供的一些特性.通过这篇文章,您将了解到如何通过Java语言创建一个线程,如何通过内置的锁来实现线程间的同步,如何在线程间进行通信以及线程的中断机制. 2. 什么是线程 线程是操作系统调度的最小单位,在一个进程中,一般至少有一个线程在运行.一个进程中包含的多个线程,在多核处理器中,操作系统可以将多个线程调度到不同的CPU核心上运行,多个线程可以并行运行. 在同一个进程中的多个线程,共享同一个进程空间

Java基础笔记(四:多线程基础及生命周期)

一.多线程基础 编写线程程序主要是构造线程类.构造线程类的方式主要有两种,一种是通过构造类java.lang.Thread的子类,另一种是通过构造方法实现接口java.lang.Runnable的类.因为类java.lang.Thread实际上也是实现了接口java.lang.Runnable的类,所以上面两种构造线程类的方法从本质上都是构造实现接口java.lang.Runnable的类.下面将具体介绍着两种方法. (1)通过类Thread的子类构造线程 类java.lang.Thread的每

Java多线程基础总结

背景 Java采用多线程方式实现并行计算,当然并行计算也可以采用多进程方式实现,但是进程切换耗费比较高.而且进程间是隔离的,进程间通信机制比较麻烦,最后JVM本身在操作系统中就一个进程,由它再启动一个进程不太合适,所以Java采用多线程方式实现并行计算. Java从诞生之初,多线程就围绕的是Runnable接口和Thread类展开的.它的底层采用的是c的p线程方式,而且由于多线程的复杂性,p线程的很多概念知识被延伸到了Java层面,这对Java开发者来说算是一个不幸的消息.但是由于多线程的复杂性

Java多线程基础(二)定时器类:Timer类和TimerTask类

Java多线程基础(二)定时器类:Timer类和TimerTask类 Timer类和TimerTask类是jdk实现定时器功能的早期方法,jdk1.5以前就支持Timer类和TimerTask类.JDK1.5之后引入了新的机制,将在后续博文中研究. 1 指定时间间隔后执行任务 import java.util.Date; import java.util.Timer; import java.util.TimerTask; public class TraditionalTimerTest {

Java基础16:Java多线程基础最全总结

Java基础16:Java多线程基础最全总结 Java中的线程 Java之父对线程的定义是: 线程是一个独立执行的调用序列,同一个进程的线程在同一时刻共享一些系统资源(比如文件句柄等)也能访问同一个进程所创建的对象资源(内存资源).java.lang.Thread对象负责统计和控制这种行为. 每个程序都至少拥有一个线程-即作为Java虚拟机(JVM)启动参数运行在主类main方法的线程.在Java虚拟机初始化过程中也可能启动其他的后台线程.这种线程的数目和种类因JVM的实现而异.然而所有用户级线

Java多线程基础(四)Java传统线程同步通信技术

Java多线程基础(四)Java传统线程同步通信技术 编写代码实现以下功能 子线程循环10次,接着主线程循环100次,接着又回到子线程循环10次,接着再回到主线程又循环100次,如此循环50次. 分析 1)子线程循环10次与主线程循环100次必须是互斥的执行,不能出现交叉,下面代码中通过synchronized关键字实现此要求: 2)子线程与主线程必须交替出现,可以通过线程同步通信技术实现,下面代码中通过bShouldSub变量实现此要求: 其他需要注意的地方 1)其中business变量必须声