`
duyufeng1
  • 浏览: 2719 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

java多线程简介

    博客分类:
  • java
阅读更多

 

 java多线程简介
     我们目前所写代码都是顺序往下执行,多个业务代码累加起来可能会非常耗时,并且拆分不易,比如同时读取数据库10个表的数据,我们假设每个表需要10s钟,那么一共需要100s钟;假如我们可以同时读取这十个表的数据,    
     在最理想情况下,一共也只需要花10s钟。
 
   多线程编程就是利用多个线程,每个线程执行对应的业务代码,来达到分拆任务,加快总体执行速度的编程方法。
 
   但是,多线程也存在其问题,那就是线程太多,线程之间来回切换也需要耗时;在对公共资源进行操作时,会经常遇到各种并发问题。
 
   Java 本身具有多线程机制,即使不涉及到具体的多线程程序编写,也会无时不刻在用它 (比如在调用main方法时已经启动了一个主线程 )Java 的线程机制是抢占式的 ,也就是说jvm 的调度机制会周期性的(时间极短)中断当
 前某个线程,切换到另外一个线程去执行。
 
 
1 ThreadRunnable
 
   在早期我们可以使用两种方式定义并启动线程:
   继承Thread ,实现其 run方法,然后调用start方法启动线程,比如:
    public class MyThread extends Thread {
               public void run(){
System.out.println("执行线程 ");
}
}
 
       public static void main(String[] args) {
MyThread mt=new MyThread();
//启动线程
mt.start();
}
 
          注意:run方法体是实现具体业务代码的地方。
 
      也可以让Runnable Thread配合使用, Runnable是一个描述任务的一个接口,而 Thread则用于启动这个任务,下面我们看看怎样使用它们编写一个线程程序,
       首先必须创建一个类来实现 Runnable接口
      public class MyRunnable implements Runnable {
                   @Override
                    public void run() {
                     System.out.println("runnable");
                   }
           }
         
          然后使用 Thread对象来启动:
         Thread thread=new Thread(new MyRunnable());
thread.start();
       
 
2 线程常用API

9.3.1 Thread.sleep(2000)

        该方法表示当前线程进入休眠状态,里面传入的参数表示将要休眠的时间,这个参数的单位是毫秒,也就是说此处应该休眠 2s,同理:Thread.sleep(3000) 表示休眠 3s
      

9.3.2 Thread.currentThread()

        该方法得到当前线程对象,在调试的时候我们往往需要打印出当前的线程信息,就可以使用这个 API
                Thread thread=Thread.currentThread();
                  System.out.println(thread.getName());
        getName() 方法可以打印出某个线程的名称

9.4 线程同步问题

    我们通常会把某个具体的业务分拆成多个业务并行执行,而这些业务往往会涉及到某些共享资源,处理好这些资源信息的同步,是我们编写多线程并发程序的一大阻碍和难点。
   
    那么,资源同步问题是怎样产生的?
               1 线程执行顺序的不确定性
    2 多个线程共享资源,并且在任务中对资源有更新和读取操作。
    下面我们通过实现一个多线程累加数值的功能来讲解线程同步问题。
   

3.1 线程同步示例

首先我们编写一个累加数值的类,这个类有一个 count属性:
     public class Const {
           private int count;
public int getCount() {
return count;
}
public void setCount(int count) {
this.count = count;
}
 
 
public void increament(){
System.out.println("加之前 "+this.count);
count++;
System.out.println("加之后 "+this.count);
}
}
然后新建一个 Runnable实现类,来调用递增方法:
public class IncreamentRunnable implements Runnable {
          private Const cnt;
public IncreamentRunnable(Const cnt){
this.cnt=cnt;
}
@Override
public  void run() {
cnt.increament();
}
 
}
这个类需要传入上面定义过的 Const类的对象,然后在run方法中调用递增方法。
 
下面我们启动多个线程来执行它:
 Const cnt=new Const();
for(int i=0;i<10;i++){
IncreamentRunnable ir=new IncreamentRunnable(cnt);
Thread thread2=new Thread(ir);
thread2.start();
}
 
注:这十个线程目前是公用的同一个 Const对象(共享资源)
 
 
我们会发现方法在执行的时候又被其他线程侵入并执行了,也就是说这个对象方法不是执行完成之后其他再执行的,而是同时都在执行,这就是所谓的线程不同步问题。
为了解决这个问题,我们可以使用 synchronized关键字使线程同步, synchronized既可以用在方法上,也可以作为方法体内的某个语句块。
     方法体上:
     public synchronized void increament()
 
 代码块:
     synchronized(this){}
   
    使用代码块级别的锁的好处是,可以自己控制锁的粒度。
  在方法上加上同步锁关键字 synchronized后,运行一下结果如下:
 
现在输出就正常了,每个方法都是执行完之后才会执行下一次。
 

3.2 wait()notifyAll()方法

     当线程得到某个对象锁时,可以调用此对象的 wait()来使当前线程进入等待状态;当另外的线程得到某个对象锁后,此对象假如调用 notifyAll()时,就会唤醒等待在此对象上的所有线程。
          synchronized(obj){
            obj.wait();
          }
   
          synchronized(obj){
            obj.notifyAll();
    }
 
    注意:wait notifyAll都是 Object的方法,并且只能在锁内调用。
 
4 java5并发包的使用
   Java5/6提供了并发包,可以简化很多多线程代码,并且提供更强劲,更细致的功能,并发包在 java.util.concurrent下面,它提供了非常简单的 API来实现线程池和细粒度的并发限制等功能。
  在之前的多线程程序中,我们会给每一个任务分配一个线程,当线程的任务执行完毕后,这个线程就被销毁,这种做法在技术上没有什么问题,但是有如下缺点:
 1 线程的创建和销毁是需要很大的开销的,有可能这个开销比执行任务的开销还要大
2 线程个数不固定,会随着任务的增多而增多,而每个线程本身也是需要消耗系统资源的,假如任务过多,服务器需要创建或维护大量的线程,可能会导致系统内存空间不足
而线程池就可以解决这些问题,线程池会预先创建多个工作线程,然后让他们不断的执行某些任务,当某个线程执行完一个任务时,就会继续执行下一个任务,这样的话,线
程池中的线程就可以被重复利用,减少了创建和销毁的开销,并且线程池可以设定线程个数,不至于造成内存空间不足。
 
      
java并发包中,创建线程池常用的大致有三种方式:
       一,根据执行的任务数自动设定线程池的大小  
            Executors.newCachedThreadPool();
 
二,根据传入的参数来设定线程大小
            Executors.newFixedThreadPool(5);
     
三,创建的线程池中只有一个线程
            Executors. SingleThreadExecutor();
 
 
Executors创建线程池时都会返回 ExecutorService 对象,我们就是通过这个执行对象来执行我们的任务。这里所说的任务,其实就是一个 Runnable实现类的对象,执行某个线程任务:
      exec.execute(Runnable 的实现类对象 )
     所以上面启动多线程程序的代码可以改为:
         Const cnt=new Const();
          ExecutorService es=Executors.newCachedThreadPool();
for(int i=0;i<10;i++){
IncreamentRunnable ir=new IncreamentRunnable(cnt);
es.execute(ir);
}
 
运行一下我们会发觉应用程序自己不会停掉,此时可以调用 shutdown()方法来停掉线程池:es.shutdown();
 
那么java 并发包是怎么处理线程同步问题的呢?
   除了之前所说的synchronized之外,可以使用并发包自带的 ReentrantLock对象,达到细粒度的锁控制
    ReentrantLock lock = new ReentrantLock(true);
    使用ReentrantLock lock()unlock() 来包含需要加锁的代码块。
一般来说,会把加锁 (lock)和解锁(unlock) 的代码放在 try{}finally{}里面:
try{
          // 加锁 
          lock.lock();
System.out.println("加之前 "+this.count);
count++;
System.out.println("加之后 "+this.count);
       }finally{
          // 解锁
           lock.unlock();
  }
这是最佳实践!
 
 
 
 
   5 总结
 
     java5 并发包之前,我们使用 ThreadRunnable 来定义并且启动线程。
 
使用synchronized 关键字来使线程同步,它既可以放在方法上,也可以放在代码块上。
 
并发包中,使用Executors来创建线程池,通过 ExecutorService对象来启动任务。
 
并发包中,可以使用ReentrantLock来创建一个锁,实现更细粒度的锁控制。

 

分享到:
评论

相关推荐

    Java多线程技术及其在网络编程中的应用.pdf

    #### Java多线程简介 Java中的多线程机制是其一大亮点。所谓线程,是指程序执行中的一个单独流程,它可以在操作系统层面并发执行。在一个进程中,可以同时存在多个线程,它们共享进程的资源,但独立执行不同的任务。...

    Java多线程设计模式上传文件

    Java多线程设计模式上传文件Java多线程设计模式上传文件Java多线程设计模式上传文件Java多线程设计模式上传文件Java多线程设计模式上传文件Java多线程设计模式上传文件Java多线程设计模式上传文件Java多线程设计模式...

    java多线程读取文件

    Java多线程读大文件 java多线程写文件:多线程往队列中写入数据

    java 多线程操作数据库

    ### Java多线程操作数据库:深入解析与应用 在当今高度并发的应用环境中,Java多线程技术被广泛应用于处理数据库操作,以提升系统的响应速度和处理能力。本文将基于一个具体的Java多线程操作数据库的应用程序,深入...

    java多线程ppt

    java多线程PPT 多线程基本概念 创建线程的方式 线程的挂起与唤醒 多线程问题

    Java多线程知识点总结

    Java多线程是Java编程语言中一个非常重要的概念,它允许开发者在一个程序中创建多个执行线程并行运行,以提高程序的执行效率和响应速度。在Java中,线程的生命周期包含五个基本状态,分别是新建状态(New)、就绪...

    java多线程的讲解和实战

    Java多线程是Java编程中的重要概念,尤其在如今的多核处理器环境下,理解并熟练掌握多线程技术对于提高程序性能和响应速度至关重要。本资料详细讲解了Java多线程的原理,并提供了丰富的实战代码,非常适合Java初学者...

    java多线程经典案例

    Java多线程是Java编程中的重要概念,它允许程序同时执行多个任务,极大地提升了程序的效率和性能。在Java中,实现多线程有两种主要方式:通过实现Runnable接口或者继承Thread类。本案例将深入探讨Java多线程中的关键...

    java多线程分页查询

    ### Java多线程分页查询知识点详解 #### 一、背景与需求分析 在实际的软件开发过程中,尤其是在处理大量数据时,如何高效地进行数据查询成为了一个关键问题。例如,在一个用户众多的社交平台上,当用户需要查看...

    java多线程Demo

    Java多线程是Java编程中的一个重要概念,它允许程序同时执行多个任务,提高了程序的效率和响应速度。在Java中,实现多线程有两种主要方式:继承Thread类和实现Runnable接口。 1. 继承Thread类: 当我们创建一个新...

    汪文君JAVA多线程编程实战(完整不加密)

    《汪文君JAVA多线程编程实战》是一本专注于Java多线程编程的实战教程,由知名讲师汪文君倾力打造。这本书旨在帮助Java开发者深入理解和熟练掌握多线程编程技术,提升软件开发的效率和质量。在Java平台中,多线程是...

    java多线程处理数据库数据

    在Java编程中,多线程处理是提升程序性能和效率的重要手段,特别是在处理大量数据库数据时。本主题将深入探讨如何使用Java的并发包(java.util.concurrent)来实现多线程对数据库数据的批量处理,包括增、删、改等...

    java多线程查询数据库

    综上所述,"java多线程查询数据库"是一个涉及多线程技术、线程池管理、并发控制、分页查询等多个方面的复杂问题。通过理解和掌握这些知识点,我们可以有效地提高数据库操作的效率和系统的响应速度。

    java 多线程并发实例

    在Java编程中,多线程并发是提升程序执行效率、充分利用多核处理器资源的重要手段。本文将基于"java 多线程并发实例"这个主题,深入探讨Java中的多线程并发概念及其应用。 首先,我们要了解Java中的线程。线程是...

    深入浅出 Java 多线程.pdf

    在本文中,我们将深入浅出Java多线程编程的世界,探索多线程编程的基本概念、多线程编程的优点、多线程编程的缺点、多线程编程的应用场景、多线程编程的实现方法等内容。 一、多线程编程的基本概念 多线程编程是指...

    JAVAJAVA多线程教学演示系统论文

    《JAVA多线程教学演示系统》是一篇深入探讨JAVA多线程编程的论文,它针对教育领域中的教学需求,提供了一种生动、直观的演示方式,帮助学生更好地理解和掌握多线程技术。这篇论文的核心内容可能包括以下几个方面: ...

    JAVA多线程练习题答案。

    JAVA多线程练习题答案详解 在本文中,我们将对 JAVA 多线程练习题的答案进行详细的解释和分析。这些题目涵盖了 JAVA 多线程编程的基本概念和技术,包括线程的生命周期、线程同步、线程状态、线程优先级、线程安全等...

    JAVA单线程多线程

    ### JAVA中的单线程与多线程概念解析 #### 单线程的理解 在Java编程环境中,单线程指的是程序执行过程中只有一个线程在运行。这意味着任何时刻只能执行一个任务,上一个任务完成后才会进行下一个任务。单线程模型...

    java多线程学习教程

    #### 一、Java多线程简介 Java多线程编程是指在Java应用程序中利用多线程技术提高程序的并发性和效率。相比于C或C++等语言,Java提供了更加高级的语言级支持,使得开发者能够更加方便地实现多线程功能。 #### 二、...

    JAVA多线程编程技术PDF

    这份“JAVA多线程编程技术PDF”是学习和掌握这一领域的经典资料,涵盖了多线程的全部知识点。 首先,多线程的核心概念包括线程的创建与启动。在Java中,可以通过实现Runnable接口或继承Thread类来创建线程。创建后...

Global site tag (gtag.js) - Google Analytics