java多线程编程技术详解和实例代码

 java多线程编程技术详解和实例代码

1.   Java和他的API都可以使用并发。

可以指定程序包含不同的执行线程,每个线程都具有自己的方法调用堆栈和程序计数器,使得线程在与其他线程并发地执行能够共享程序范围内的资源,比如共享内存,这种能力被称为多线程编程(multithreading),在核心的C和C++语言中并不具备这种能力,尽管他们影响了JAVA的设计。

2.   线程的生命周期

新线程的生命周期从“新生”状态开始。程序启动线程前,线程一直是“新生”状态;程序启动线程后,线程进入“可运行”状态。“可运行”状态的线程,被认为是正在执行他的任务。

在程序启动线程之前,线程一直处于“等待”状态,只有当另一个线程通知正在等待的线程继续执行时,这个线程才会从“等待”状态恢复到“可运行”状态。

“可运行”状态的线程可以进入“定时等待”状态,等待一个指定的时间段。当时间到达或线程正在等待的某个事件发生时,该线程就会返回“可运行”状态。即使处理器可以使用,处于“定时等待”状态和“等待”状态的线程也不能用它。当处于“可运行”状态的线程正在等待另一个线程执行任务时,如果它提供了可选的等待时间段,则这个线程会进入“定时等待”状态。当另一个线程通知了这个线程,或者当定时的时间段到达时(以先满足的为准),这个线程就会返回到“可运行”状态.。使线程进入“定时等待”状态的另一方法是是处于“可运行”状态的线程睡眠。睡眠线程会在“定时等待”状态维持一个指定的时间段(称为睡眠时间段),过了这段时间,它会返回到“可运行”状态。当线程没有工作要执行时,它会立即睡眠。;例
当线程试图执行某个任务,而任务又不能立即完成,线程就从“可运行”状态转到“阻塞”状态。;例。即使有处理器可供使用,“阻塞”状态的线程也不能使用它。

线程成功完成任务,或者(由于出错)终止了时,“可运行”线程就会进入“终止”状态(有时称“停滞”状态)。
在操作系统级别,JAVA的“可运行”状态通常包含两个独立的状态。当线程首先从“新生”状态转到“可运行”状态,线程处于“就绪”状态。当操作系统将线程给处理器时,线程就从“就绪”状态进入“运行”状态(即开始执行),这也被称为“调度线程”。大多数操作系统中,每个线程被赋予一小段处理器时间(时间片)来执行任务。当时间片到达时,线程就会返回到“就绪”状态,而操作系统将另一个线程给予处理器。

3.   线程优先级与线程调度

JAVA的线程优先级范围为MIN_PRIORITY(常量1)到MAX_PRIORITY(常量10),默认是NORM_PRIORITY(常量5)

4.   创建并执行线程

创建线程推介实现Runnable接口

(1)Runnable与Thread类

// Fig. 4.1: PrintTask.java 

// PrintTask class sleeps for a random time from 0 to 5 seconds 

import java.util.Random; 

public class PrintTask implements Runnable  

{ 

  private final int sleepTime; // random sleep time for thread 

  private final String taskName; // name of task 

  private final static Random generator = new Random(); 

  public PrintTask( String name ) 

  { 

   taskName = name; // set task name 

   // pick random sleep time between 0 and 5 seconds 

   sleepTime = generator.nextInt( 5000 ); // milliseconds 

  } // end PrintTask constructor 

  // method run contains the code that a thread will execute 

  public void run() 

  { 

   try // put thread to sleep for sleepTime amount of time  

   { 

     System.out.printf( "%s going to sleep for %d milliseconds.\n",  

      taskName, sleepTime ); 

     Thread.sleep( sleepTime ); // put thread to sleep 

   } // end try     

   catch ( InterruptedException exception ) 

   { 

     System.out.printf( "%s %s\n", taskName, 

      "terminated prematurely due to interruption" ); 

   } // end catch 

   // print task name 

   System.out.printf( "%s done sleeping\n", taskName );  

  } // end method run 

} // end class PrintTask
// Fig. 4.2 ThreadCreator.java 

// Creating and starting three threads to execute Runnables. 

import java.lang.Thread; 

public class ThreadCreator 

{ 

  public static void main( String[] args ) 

  { 

   System.out.println( "Creating threads" ); 

   // create each thread with a new targeted runnable 

   Thread thread1 = new Thread( new PrintTask( "task1" ) ); 

   Thread thread2 = new Thread( new PrintTask( "task2" ) ); 

   Thread thread3 = new Thread( new PrintTask( "task3" ) ); 

   System.out.println( "Threads created, starting tasks." ); 

   // start threads and place in runnable state 

   thread1.start(); // invokes task1抯 run method 

   thread2.start(); // invokes task2抯 run method 

   thread3.start(); // invokes task3抯 run method 

   System.out.println( "Tasks started, main ends.\n" ); 

  } // end main 

} // end class RunnableTester

   (2)线程管理与Executor框架

5为显示的创建线程,但推介使用Executor接口,用来管理Runnable对象的执行。Executor对象创建并管理一组Runnable对象的线程,这组线程就做线程池(thread pool).优点是Executor对象能复用了已经有的线程,减少为每个任务创建新线程的开销,提高性能。

Executor接口只声明了一个名称为execute的方法,接收一个Runnable实参。Executor会将传递给他的execute方法的每个Runnable对象赋予线程池中可以用的线程。如果没有可以用的线程,则Executor会创建一个新线程,或等待某个线程会成为可用的,并会将这个线程赋予传递给execute方法的Runnable对象。

ExecutorService接口扩展了Executor接口。

// Fig. 4.3: TaskExecutor.java
// Using an ExecutorService to execute Runnables.
import java.util.concurrent.Executors;
import java.util.concurrent.ExecutorService; 

public class TaskExecutor
{
  public static void main( String[] args )
  {
   // create and name each runnable
   PrintTask task1 = new PrintTask( "task1" );
   PrintTask task2 = new PrintTask( "task2" );
   PrintTask task3 = new PrintTask( "task3" ); 

   System.out.println( "Starting Executor" ); 

   // create ExecutorService to manage threads
   ExecutorService threadExecutor = Executors.newCachedThreadPool(); 

   // start threads and place in runnable state
   threadExecutor.execute( task1 ); // start task1
   threadExecutor.execute( task2 ); // start task2
   threadExecutor.execute( task3 ); // start task3 

   // shut down worker threads when their tasks complete
   threadExecutor.shutdown();  

   System.out.println( "Tasks started, main ends.\n" );
  } // end main
} // end class TaskExecutor

5.       线程同步 

(1)线程同步(thread synchronization),协调多个并发线程对共享数据的访问。这种方式同步多个线程,就可以保证访问共享对象的每个线程都能同步地将其他所有线程排除在外,这被称为“互斥”。
另一个方法,使用JAVA内置的监控器(monitor)。每个对象都有一个监控器和监控锁(或内置锁)。监控器保证任何时候监控锁由具有最大可能的唯一一个线程持有。

(2)同步的数据共享:执行原子操作。

// Adds integers to an array shared with other Runnables 

import java.lang.Runnable; 

public class ArrayWriter implements Runnable 

{ 

  private final SimpleArray sharedSimpleArray; 

  private final int startValue; 

  public ArrayWriter( int value, SimpleArray array ) 

  { 

   startValue = value; 

   sharedSimpleArray= array; 

  } // end constructor 

  public void run() 

  { 

   for ( int i = startValue; i < startValue + 3; i++ ) 

   { 

     sharedSimpleArray.add( i ); // add an element to the shared array 

   } // end for 

  } // end method run 

} // end class ArrayWrite
// Fig 5.2: SharedArrayTest.java 

// Executes two Runnables to add elements to a shared SimpleArray. 

import java.util.concurrent.Executors; 

import java.util.concurrent.ExecutorService; 

import java.util.concurrent.TimeUnit; 

public class SharedArrayTest 

{ 

  public static void main( String[] arg ) 

  { 

   // construct the shared object 

   SimpleArray sharedSimpleArray = new SimpleArray( 6 ); 

   // create two tasks to write to the shared SimpleArray 

   ArrayWriter writer1 = new ArrayWriter( 1, sharedSimpleArray ); 

   ArrayWriter writer2 = new ArrayWriter( 11, sharedSimpleArray ); 

   // execute the tasks with an ExecutorService 

   ExecutorService executor = Executors.newCachedThreadPool(); 

   executor.execute( writer1 ); 

   executor.execute( writer2 ); 

   executor.shutdown(); 

   try 

   { 

     // wait 1 minute for both writers to finish executing 

     boolean tasksEnded = executor.awaitTermination(  

      1, TimeUnit.MINUTES ); 

     if ( tasksEnded ) 

      System.out.println( sharedSimpleArray ); // print contents 

     else 

      System.out.println(  

        "Timed out while waiting for tasks to finish." ); 

   } // end try 

   catch ( InterruptedException ex ) 

   { 

     System.out.println(  

      "Interrupted while wait for tasks to finish." ); 

   } // end catch 

  } // end main 

} // end class SharedArrayTest
// Fig.5.3 : SimpleArray.java 

// Class that manages an integer array to be shared by multiple  

// threads with synchronization. 

import java.util.Random; 

public class SimpleArray 

{ 

  private final int array[]; // the shared integer array 

  private int writeIndex = 0; // index of next element to be written 

  private final static Random generator = new Random(); 

  // construct a SimpleArray of a given size 

  public SimpleArray( int size ) 

  { 

   array = new int[ size ]; 

  } // end constructor 

  // add a value to the shared array 

  public synchronized void add( int value ) 

  { 

   int position = writeIndex; // store the write index 

   try 

   { 

     // put thread to sleep for 0-499 milliseconds 

     Thread.sleep( generator.nextInt( 500 ) );  

   } // end try 

   catch ( InterruptedException ex ) 

   { 

     ex.printStackTrace(); 

   } // end catch 

   // put value in the appropriate element 

   array[ position ] = value; 

   System.out.printf( "%s wrote %2d to element %d.\n",  

     Thread.currentThread().getName(), value, position ); 

   ++writeIndex; // increment index of element to be written next 

   System.out.printf( "Next write index: %d\n", writeIndex ); 

  } // end method add 

  // used for outputting the contents of the shared integer array 

  public String toString() 

  { 

   String arrayString = "\nContents of SimpleArray:\n"; 

   for ( int i = 0; i < array.length; i++ ) 

     arrayString += array[ i ] + " "; 

   return arrayString; 

  } // end method toString 

} // end class SimpleArray 

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

(0)

相关推荐

  • Java多线程编程中的两种常用并发容器讲解

    ConcurrentHashMap并发容器 ConcurrentHashMap可以做到读取数据不加锁,并且其内部的结构可以让其在进行写操作的时候能够将锁的粒度保持地尽量地小,不用对整个ConcurrentHashMap加锁. ConcurrentHashMap的内部结构 ConcurrentHashMap为了提高本身的并发能力,在内部采用了一个叫做Segment的结构,一个Segment其实就是一个类Hash Table的结构,Segment内部维护了一个链表数组,我们用下面这一幅图来看下Con

  • Java多线程编程之读写锁ReadWriteLock用法实例

    读写锁:分为读锁和写锁,多个读锁不互斥,读锁与写锁互斥,这是由jvm自己控制的,你只要上好相应的锁即可.如果你的代码只读数据,可以很多人同时读,但不能同时写,那就上读锁:如果你的代码修改数据,只能有一个人在写,且不能同时读取,那就上写锁.总之,读的时候上读锁,写的时候上写锁! 三个线程读数据,三个线程写数据示例: 可以同时读,读的时候不能写,不能同时写,写的时候不能读. 读的时候上读锁,读完解锁:写的时候上写锁,写完解锁. 注意finally解锁. package com.ljq.test.th

  • 详解Java多线程编程中互斥锁ReentrantLock类的用法

    0.关于互斥锁 所谓互斥锁, 指的是一次最多只能有一个线程持有的锁. 在jdk1.5之前, 我们通常使用synchronized机制控制多个线程对共享资源的访问. 而现在, Lock提供了比synchronized机制更广泛的锁定操作, Lock和synchronized机制的主要区别: synchronized机制提供了对与每个对象相关的隐式监视器锁的访问, 并强制所有锁获取和释放均要出现在一个块结构中, 当获取了多个锁时, 它们必须以相反的顺序释放. synchronized机制对锁的释放是

  • java的多线程用法编程总结

    一.进程与线程 1.进程是什么? 狭义定义:进程是正在运行的程序的实例(an instance of a computer program that is being executed). 广义定义:进程是一个具有一定独立功能的程序关于某个数据集合的一次运行活动.它是操作系统动态执行的基本单元,在传统的操作系统中,进程既是基本的分配单元,也是基本的执行单元. 2.线程是什么? 线程,有时被称为轻量级进程(Lightweight Process,LWP),是程序执行流的最小单元.一个标准的线程由线

  • java多线程编程之使用Synchronized块同步变量

    下面的代码演示了如何同步特定的类方法: 复制代码 代码如下: package mythread; public class SyncThread extends Thread{ private static String sync = ""; private String methodType = ""; private static void method(String s) {  synchronized (sync)  {sync = s;System.out

  • java多线程编程制作电子时钟

    模拟一个电子时钟,它可以在任何时候被启动或者停止,并可以独立的运行. 1.定义一个Clock类.它继承Label类,并实现Runnable接口.这个类中有一个Thread类型的clocker域,以及start()和run()方法.在run()方法中,每隔一秒就把系统时间显示为label的文本. class Clock extends Label implements Runnable { //定义Thread类型的clocker域 public Thread clocker=null; publ

  • Java多线程编程之访问共享对象和数据的方法

    多个线程访问共享对象和数据的方式有两种情况: 1.每个线程执行的代码相同,例如,卖票:多个窗口同时卖这100张票,这100张票需要多个线程共享. 2.每个线程执行的代码不同,例如:设计四个线程,其中两个线程每次对j增加1,另外两个线程每次对j减少1. a.如果每个线程执行的代码相同,可以使用同一个Runnable对象,这个对象中有共享数据.卖票就可以这样做,每个窗口都在做卖票任务,卖的票都是同一个数据(点击查看具体案例). b.如果每个线程执行的代码不同,就需要使用不同的Runnable对象,有

  • 详解Java多线程编程中CountDownLatch阻塞线程的方法

    直译过来就是倒计数(CountDown)门闩(Latch).倒计数不用说,门闩的意思顾名思义就是阻止前进.在这里就是指 CountDownLatch.await() 方法在倒计数为0之前会阻塞当前线程. CountDownLatch是一个同步辅助类,在完成一组正在其他线程中执行的操作之前,它允许一个或多个线程一直等待. CountDownLatch 的作用和 Thread.join() 方法类似,可用于一组线程和另外一组线程的协作.例如,主线程在做一项工作之前需要一系列的准备工作,只有这些准备工

  • java多线程编程之为什么要进行数据同步

    Java中的变量分为两类:局部变量和类变量.局部变量是指在方法内定义的变量,如在run方法中定义的变量.对于这些变量来说,并不存在线程之间共享的问题.因此,它们不需要进行数据同步.类变量是在类中定义的变量,作用域是整个类.这类变量可以被多个线程共享.因此,我们需要对这类变量进行数据同步.数据同步就是指在同一时间,只能由一个线程来访问被同步的类变量,当前线程访问完这些变量后,其他线程才能继续访问.这里说的访问是指有写操作的访问,如果所有访问类变量的线程都是读操作,一般是不需要数据同步的.那么如果不

  • java多线程编程技术详解和实例代码

     java多线程编程技术详解和实例代码 1.   Java和他的API都可以使用并发. 可以指定程序包含不同的执行线程,每个线程都具有自己的方法调用堆栈和程序计数器,使得线程在与其他线程并发地执行能够共享程序范围内的资源,比如共享内存,这种能力被称为多线程编程(multithreading),在核心的C和C++语言中并不具备这种能力,尽管他们影响了JAVA的设计. 2.   线程的生命周期 新线程的生命周期从"新生"状态开始.程序启动线程前,线程一直是"新生"状态:

  • Java中初始化块详解及实例代码

    Java中初始化块详解 在Java中,有两种初始化块:静态初始化块和非静态初始化块. 静态初始化块:使用static定义,当类装载到系统时执行一次.若在静态初始化块中想初始化变量,那仅能初始化类变量,即static修饰的数据成员. 非静态初始化块:在每个对象生成时都会被执行一次,可以初始化类的实例变量. 非静态初始化块会在构造函数执行时,且在构造函数主体代码执行之前被运行. 括号里的是初始化块,这里面的代码在创建Java对象时执行,而且在构造器之前执行! 其实初始化块就是构造器的补充,初始化块是

  • Java Process类的详解及实例代码

    Java Process类的详解 前言: 今天用了下Java.lang.Process类,只是初步的学习,并没有深入实践,因为感觉它的用途并不是很大,偶尔才可能用上,如果要经常使用它的人可以自行参考JDk文档. 对Process类的简要说明: Process类是一个抽象类,方法都是抽象的,它封装了一个进程,也就是一个可执行的程序  该类提供进程的输入.执行输出到进程.等待进程的完成和检查进程的退出状态及销毁进程的方法 ProcessBuilder.start()和Runtime.exec方法创建

  • Java 垃圾回收机制详解及实例代码

     Java 垃圾回收机制详解 乍一看,垃圾回收所做的事情应当恰如其名--查找并清除垃圾.事实上却恰恰相反.垃圾回收会跟踪所有仍在使用的对象,然后将剩余的对象标记为垃圾.牢记了这点之后,我们再来深入地了解下这个被称为"垃圾回收"的自动化内存回收在JVM中到底是如何实现的. 手动管理内存 在介绍现代版的垃圾回收之前,我们先来简单地回顾下需要手动地显式分配及释放内存的那些日子.如果你忘了去释放内存,那么这块内存就无法重用了.这块内存被占有了却没被使用.这种场景被称之为内存泄露. 下面是用C写

  • java DataInputStream和DataOutputStream详解及实例代码

    java DataInputStream和DataOutputStream详解 操作基本数据类型的流 DataInputStream DataOutputStream import java.io.DataInputStream; import java.io.DataOutputStream; import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.IOException; public c

  • java 嵌套类的详解及实例代码

    java 嵌套类 到现在为止,我们都是在Java文件中直接定义类.这样的类出现在包(package)的级别上.Java允许类的嵌套定义. 这里将讲解如何在一个类中嵌套定义另一个类. 内部类 Java允许我们在类的内部定义一个类.如果这个类是没有static修饰符,那么这样一个嵌套在内部的类称为内部类(inner class). 内部类被认为是外部对象的一个成员.在定义内部类时,我们同样有访问权限控制(public, private, protected). 在使用内部类时,我们要先创建外部对象.

  • Java 回调机制(CallBack) 详解及实例代码

     Java 回调机制 概要: 最近学习java,接触到了回调机制(CallBack).初识时感觉比较混乱,而且在网上搜索到的相关的讲解,要么一言带过,要么说的比较单纯的像是给CallBack做了一个定义.当然了,我在理解了回调之后,再去看网上的各种讲解,确实没什么问题.但是,对于初学的我来说,缺了一个循序渐进的过程.此处,将我对回调机制的个人理解,按照由浅到深的顺序描述一下,如有不妥之处,望不吝赐教! 开始之前,先想象一个场景:幼稚园的小朋友刚刚学习了10以内的加法. 第1章. 故事的缘起 幼师

  • Java 线程池ExecutorService详解及实例代码

    Java 线程池ExecutorService 1.线程池 1.1什么情况下使用线程池 单个任务处理的时间比较短. 将需处理的任务的数量大. 1.2使用线程池的好处 减少在创建和销毁线程上所花的时间以及系统资源的开销. 如果不使用线程池,有可能造成系统创建大量线程而导致消耗系统内存以及"过度切换"; 2.ExecutorService和Executors 2.1简介 ExecutorService是一个接口,继承了Executor, public interface ExecutorS

  • Java 调用天气Webservice详解及实例代码

    Java调用天气Webservice的小应用 废话不多说,直接贴代码: CityReq.java package com.weather; import javax.xml.bind.annotation.XmlElement; import javax.xml.bind.annotation.XmlRootElement; @XmlRootElement(name="getWeatherbyCityName",namespace="http://WebXml.com.cn/

  • Java Serializable和Parcelable详解及实例代码

    对 Serializable和Parcelable理解 1.首先他们两个接口都是为了实现对象的序列化,使之可以传递,所谓序列化就是将对象信息装换成可以存储的介质的过程. 2.Serializable是jdk所提供的序列化接口,该接口存在于io包下,可想用于输入输出,使用非常简单,只要让你的类实现此接口就ok了:可以使用transient关键字修饰你不想序列化的属性. 3.Parcelable是sdk所提供的序列化接口,使用较上者麻烦,实现此接口后,需要重写writeToParcel方法,将需要序

随机推荐