`
剑锋无刃
  • 浏览: 34338 次
  • 性别: Icon_minigender_1
  • 来自: 长沙市
最近访客 更多访客>>
社区版块
存档分类
最新评论

多线程

    博客分类:
  • java
阅读更多

 

多线程:

进程与线程:

进程:同一个操作系统中执行的一个子程序,包含了三部分虚拟CPU、代码、数据

多进程:同一个操作系统中执行的多个并行的子程序。可以提高cpu的使用率

线程:在同一个进程当中执行的子程序流

多线程:同一个进程当中并发执行的多个子程序流。可以提高cpu的使用率

进程与线程的区别:

进程有独立的进程空间,进程中的数据存放空间(堆空间和栈空间)是独立的。

线程的堆空间是共享的,栈空间是独立的,线程消耗的资源也比进程小,相互之间可以影响的。

java中如何调进程:

调用本地程序的两个类

Runtime

Runtime.getRuntime.exec(...); //执行一个程序

其返回值就是Process类型

Process

注意:

只有运行状态的线程才有机会执行代码,主线程的中止不会影响其他的正在运行中的线程,主线程中止也就是main()方法退出了。只有进程中的所有线程都中止时,进程(JVM进程)才会退出,只要有线程没有中止,进程就不会退出。

操作系统决定线程是否有优先级,独占式的操作系统中系统会有优先级的概念,共享式的操作系统则不会有优先级的。

在线程的内部,程序依然顺序执行

线程编程的两种方法:

写一个类,继承Thread类,覆盖Thread类中继承来的run()方法,这样就写好了自定义的线程类。

继承java.lang.Thread类:

class MyThread extends Thread{

  public void run(){ //覆盖run(),线程体方法,自身其实就是普通的方法

  .......

}

}

启动线程:

public class TestThread{

  public static void main(){

  Thread t1=new Mythread();

  T1.start(); //调用start()来启动线程,线程启动方法,向线程调度器说明当前线程已经准备好了,是一种可运行状态

  }

}

写一个类,实现Runable接口,实现其中的run()方法。这种方法写好的类的对象需要作为线程类创建对象时构造方法的参数。

实现java.lang.Runnable接口:

Class MyThread  implements Runnable{

  public void run(){

  }

}

启动线程:

public class TestThread{

public static void main(){

Runnable myThread = new MyThread();

Thread t = new Thread(myThread);

t.start();

}

}

Thread中的一些方法:

currentThread() 

          返回对当前正在执行的线程对象的引用(实现接口方式时使用)

sleep(long millis) 

          在指定的毫秒数内让当前正在执行的线程休眠(暂停执行)。

本线程不会去抢,除非sleep结束。

多个线程之间都会去抢执行权限,不会考虑优先级。

yield() 

          暂停当前正在执行的线程对象,并执行其他线程。

          只给本类或者优先级大于本类优先级的线程去抢。

          join() 

          等待该线程终止。

          放在start()前面则没有用处。

          setDaemon(boolean on) 

          将该线程标记为守护线程,守护线程需要依赖其他线程,会在虚拟机停止的时候停止。

         

线程的生命周期:

1)初始状态:此时线程只是处于JVM进程中,只是创建了一个线程对象,并没有真正开始运行。

2)可动行状态:调用线程对象的start()方法,此时线程才真正的被创建,进入可运行状态,等待CPU的调度。“万事俱备,只欠CPU”。

3)运行状态:正在运行的线程,此时它拥有CPU的执行权。

4)阻塞状态:运行状态中的线程,如果正在等待用户输入或调用了sleep()和join()等方法都会导致线程进入阻塞状态,注意从阻塞状态出来的线程不一定马上回到运行状态,而是重新回到可运行状态,等待CPU的再次调度。

5)等待队列状态:一个线程调用一个对象的wait()会自动放弃该对象的锁标记,进入等待队列状态,只有当有另外一线程调用临界资源的notify()或notifyAll()方法,建议多使用notifyAll(),才会将等待队列中的线程释放,此线程进入锁池状态。

6)锁池状态:每个对象都有互斥锁标记,以防止对临界资源的访问造成数据的不一致,和数据的不完整性。一个线程拥有一个对象的锁标记后,另一线程想访问该对象,必须在锁池中等待。由系统决定哪个线程拿到锁标记并运行。注意从锁池状态出来的线程不是马上回到运行状态,而是重新回到可运行状态,等待CPU的再次调度。

7)终止状态:一个线程运行结束后称为终止状态,一个进程中只有所有的线程退出后才会终止。

 

 

多线程:

多线程的同步:

多线程并发访问同一个对象(临界资源),如果不对线程进行同步控制,破坏了原子操作(不可再分的操作),则会造成临界资源(两个线程同时访问的资源)的数据不一致。   

 

每一个对象都有一个互斥的锁标记和一个锁池。当线程拥有这个对象的锁标记时才能访问这个资源,没有锁标记便进入锁池,保证在同步代码块中只有一个线程,解决了多线程同步控制的问题。

关键字:synchronized //线程在同步代码中必须采用串行访问

synchronized修饰代码块:对括号内的对象object加锁,只有拿到对象锁标记的线程才能进入该代码块。

  public void push(char c){ 

        synchronized(object){ //object只要是对象就可以,但必须保证是同一对象

        ……

        同步代码 

        ……

       

  }

 

synchronized修饰方法:在整个方法范围内对当前对象的加锁,只有拿到对象锁标记的线程才能执行该方法。尽可能的少用

  public synchronized void push(char c) {

……

同步代码 

……

}

     

一个线程可以同时拥有多个对象的锁标记,锁标记如果过多,就会出现线程等待其他线程释放锁标记,而又都不释放自己的锁标记供其他线程运行的状况,造成死锁。

 

静态方法可以是同步方法:但是它所锁的并不是当前对象,是类对象。

抽象方法不能是synchronized同步的方法。

构造方法不能是synchronized同步的方法。

线程因为未拿到锁标记而发生阻塞进入锁池(lock pool)。每个对象都有自己的一个锁池的空间,用于放置等待运行的线程。由系统决定哪个线程拿到锁标记并运行

利用Collections类中的synchronizedXxxx(Xxxx ss)方法可以得到相应集合的线程安全的集合

注意:

在同步语句块中不能直接操作对象锁正在使用的对象。

对象与锁一一对应。

同步依赖对象锁,锁对象相同,同步语句串行,锁对象不同,同步语句并行。

顺序锁,不要回调,反向打开。

能不用同步就不用同步,有数据共享冲突时才使用同步。

等待通知机制:

线程间通信使用的空间称之为对象的等待对列(wait pool),该队列也是属于对象的空间的。

使用Object类中wait()的方法,在运行状态中,线程调用wait(),此时表示线程将释放自己所有的锁标记和CPU的占用,同时进入这个对象的等待池。等待池的状态也是阻塞状态,只不过线程释放自己的锁标记。只有在对该对象加锁的同步代码块里,才能掉用该对象的wait(),表示线程将会释放所有锁标记,进入等待队列,线程将进入等待队列状态。

一个线程进入了一个对对象加锁的同步代码块,并对该对象调用了wait()方法,释放自己拥有的所有锁标记,进入该对象等待队列,另一个线程获得了该对象的锁标记,进入代码块对该对象调用了notify()方法,就会从等待队列里释放出一线程,释放出的这个线程要继续运行就还要进入那个同步代码块,因为得不到要访问代码块对象的锁标记,而进入该对象的锁池,等待锁标记释放。

什么情况下释放锁:

同类代码执行完毕。

异常未处理,错误退出。

调用wait()。

相关方法:

1) wait():交出锁和CPU的占用; 

2) notify():将从对象的等待池中移走一个任意的线程,并放到锁池中,那里的对象一直在等待,直到可以获得对象的锁标记。 

3) notifyAll(): 将从等待池中移走所有等待那个对象的线程并放到锁池中,只有锁池中的线程能获取对象的锁标记,锁标记允许线程从上次因调用wait()而中断的地方开始继续运行

注意:

用notifyAll()取代notify(),因为在调用notify()方法时,是由系统决定释放出哪个线程。

只能对加锁的资源进行wait()和notify()。

判断是否进行等待wait()时,用while代替if来进行判断。

 

分享到:
评论

相关推荐

    c++多线程

    一个进程中可以同时运行多个线程,每个线程可以执行不同的任务,这就是所谓的多线程。同一进程中的多个线程将共享该进程中的全部系统资源,如虚拟地址空间、文件描述符和信号处理等,但是同一个进程中的多个线程都有...

    stm32单片机多线程实例

    多线程是实现复杂任务并发执行的关键技术,能够提高资源利用率,优化系统响应时间。在STM32上实现多线程,通常会借助实时操作系统(RTOS)如RT-Thread。 RT-Thread是一个轻量级、开源的实时操作系统,它为STM32等微...

    多线程测试

    在计算机科学领域,多线程是一种程序设计技术,它允许应用程序同时执行多个任务或子任务。这极大地提高了软件的效率和响应性,特别是在现代多核处理器的环境下。本主题将深入探讨多线程的实现方式及其与单线程的对比...

    大漠多线程模板_大漠_大漠多线程_

    "大漠多线程模板"是一个专门针对C#开发的多线程处理框架,它为开发者提供了便捷的方式来管理和优化多线程应用。这个框架由知名开发者"大漠"创建,旨在简化复杂的并发编程,提高代码的可读性和可维护性。 多线程允许...

    安卓多线程

    安卓多线程,安卓高级开发中,必备技能,项目开发中都要使用

    qt QTcpServer多线程

    QTcpServer多线程 每个客户端连接的tcpSocket分别分配一个专门的线程来处理。 核心思想:继承并重写QTcpServer的incomingConnection函数去自己实现tcpsocket连接的建立和分配。 incomingConnection函数说明: 当...

    多线程介绍

    多线程(英语:multithreading),是指从软件或者硬件上实现多个线程并发执行的技术。具有多线程能力的计算机因有硬件支持而能够在同一时间执行多于一个线程,进而提升整体处理性能。具有这种能力的系统包括对称多...

    多线程基础与基于多线程的简单聊天室

    在IT行业中,多线程是程序设计中的一个重要概念,尤其在Java编程中,它被广泛应用于提高应用程序的并发性能和响应速度。本压缩包“多线程基础与基于多线程的简单聊天室”提供了对多线程技术的实践理解和二次开发的...

    多线程示例

    在IT领域,多线程是程序设计中的一个重要概念,尤其在现代计算机系统中,它能够提升应用程序的效率和响应性。MFC(Microsoft Foundation Classes)是微软提供的一个C++类库,用于简化Windows应用程序的开发,包括...

    qt websocket 客户端 多线程使用

    qt 使用QWebSocket 创建websocket客户端来读取数据,异步链接,并且放入到线程中去执行,线程池的基础,代码使用两个用户,放入到一个线程中执行,同理,可以多个用户放入到多个线程中执行,为线程池执行websocket ...

    java深入理解多线程

    Java多线程是并发编程中的一个重要概念,它允许程序在同一时刻执行多个任务。以下是对Java多线程的深入理解: 线程概述 基本概念:线程是操作系统能够进行运算调度的最小单位,一个进程可以包含多个线程。 特性:...

    WPF多线程实例

    在Windows Presentation Foundation(WPF)开发中,多线程是一个重要的技术,特别是在处理大量数据或进行耗时操作时,为了保持用户界面(UI)的响应性,通常会使用多线程来实现非UI任务。本实例是关于如何在WPF应用...

    多线程导入excel 数据

    在Java编程中,多线程导入Excel数据是一项常见的任务,特别是在大数据处理和高并发场景下。这个场景通常涉及到性能优化和资源管理,以确保系统稳定性和数据一致性。下面将详细阐述多线程导入Excel数据的核心知识点。...

    python多线程测试代码

    使用一个简单的数字递减案例,来模拟多线程下的工作逻辑

    易语言关闭多线程句柄方法

    易语言关闭多线程句柄方法 易语言是一种功能强大且灵活的编程语言,多线程编程是其重要特性之一。然而,在多线程编程中,如何正确地关闭线程句柄是非常重要的。今天,我们将分享易语言关闭多线程句柄方法的相关知识...

    Java多线程详解(超详细)_狂神说笔记完整版_项目代码_适合小白随课程学习

    Java多线程详解 在Java编程中,多线程是一种重要的技术,它使得程序能够同时执行多个任务,提高系统的效率和响应性。本教程将详细讲解Java中的多线程概念,包括线程的创建、状态、同步以及高级主题,旨在帮助初学者...

    Java多线程知识点总结

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

    详解易语言启动多线程

    易语言多线程启动详解 易语言多线程启动是指在易语言中使用多线程相关的API、支持库或模块来实现多线程编程的技术。多线程编程可以极大地提高程序的执行效率和响应速度,特别是在需要执行大量计算或I/O操作的场景下...

    多线程多线程.xmind

    该文档是笔者在学习李刚老师《Java疯狂讲义》中有关多线程的用法而总结出来的笔记,其中主要的内容包括线程创建和启动、线程的生命周期、控制线程、线程同步、线程通信线程池等基本内容。对Java多线程有详细的介绍。

    C#多线程开发之并发编程经典实例.zip

    在C#编程中,多线程是一个至关重要的概念,尤其对于开发高性能、高并发的应用程序而言。本资源“C#多线程开发之并发编程经典实例”提供了丰富的实例,旨在帮助C#开发者深入理解并掌握多线程技术。以下是关于C#多线程...

Global site tag (gtag.js) - Google Analytics