线程同步&线程池

线程同步

线程不同步会出现的问题:

  • 当多个线程操作同一资源时,会出现重复操作和和操作不存在的资源的问题,为了规避这一问题就需要线程的同步操作来实现资源的共同使用。

线程同步:

  • 当多个线程操作同一资源时,给操作该资源的代码加上一把锁,当有一个线程拿到这把锁后,其他线程都不能操作带锁的资源代码,直至拿到锁的线程释放锁。

线程同步实现的3种方式:

  • 同步代码块

    synchornized(obj){

    //涉及操作同一资源的代码

    }

    注: 没有静态的同步代码块

    obj: 任意类型的对象,相当于一把锁,操作同一资源的而线程必须使用同一把锁。

    如何保证多个线程使用同一把锁?

    在设置线程任务,即定义线程对象时在成员变量的位置上声明一个对象,将声明的变量放在sychornized代码块中

    /**

    *模拟买票功能,开启多个线程,同时买一百张票

    /

    public class SaleTicket implements Runnable{

    /


    不同对象使用同一个变量

    */

    private static int ticket=100 ;

        /**
    *不同线程使用同一个锁对象
    */
    private Object obj = new Object();
    @Override
    public void run() {
    while(true) {
    /**
    * 使用synchornized(obj) {} 代码块
    */
    synchronized (obj) {
    if (ticket > 0) {
    try {
    Thread.sleep(500);
    } catch (InterruptedException e) {
    e.printStackTrace();
    }
    System.out.println(Thread.currentThread().getName() + "sale=====" + ticket);
    ticket--;
    }
    }
    }
    }
    } public class SaleTicketDemoTest {
    public static void main(String[] args) {
    SaleTicket sale = new SaleTicket();
    //开启3个线程操作同一对象,从而操作同一资源
    Thread th1 = new Thread(sale);
    Thread th2 = new Thread(sale);
    Thread th3 = new Thread(sale);
    th1.start();
    th2.start();
    th3.start();
    }
    }
  • 同步方法

    public synchornied void 方法名(){

    //涉及操作同一资源的方法

    }

    同步方法的锁:

    同步方法的锁使用的是this对象,即new Thread(Xxx) 的Xxx对象

    package com.test.java;

    /**
    *模拟买票功能,开启多个线程,同时买一百张票
    */ public class SaleTicket implements Runnable{
    /*
    不同对象使用同一个变量
    */
    private static int ticket=100 ; /**
    *不同线程使用同一个锁对象
    */
    private Object obj = new Object();
    @Override
    public void run() {
    while(true) {
    saleTicket();
    }
    }
    /**
    * 定义个个同步方法 public snychonized void XXX(){}
    */
    public synchronized void saleTicket() {
    if (ticket > 0) {
    try {
    Thread.sleep(500); } catch (InterruptedException e) {
    e.printStackTrace();
    }
    System.out.println(Thread.currentThread().getName() + "sale=====" + ticket);
    ticket--;
    }
    }
    }

    静态的同步方法

    public static sychornized xxx(){

    //涉及操作共同资源的代码

    }

    静态同步方法的锁是类名.class对象

    package com.test.java;

    /**
    *模拟买票功能,开启多个线程,同时买一百张票
    */ public class SaleTicket implements Runnable{
    /*
    不同对象使用同一个变量
    */
    private static int ticket=100 ; /**
    *不同线程使用同一个锁对象
    */
    private Object obj = new Object();
    @Override
    public void run() {
    while(true) {
    saleTicket();
    } /**
    * 静态同步方法
    */ public static synchronized void saleTicket() {
    if (ticket > 0) {
    try {
    Thread.sleep(500); } catch (InterruptedException e) {
    e.printStackTrace();
    }
    System.out.println(Thread.currentThread().getName() + "sale=====" + ticket);
    ticket--;
    }
    }
    }
  • 锁机制

    Jdk1.5后在java.util.concurrent.locks

    lock的好处:解决了当共享资源的锁一直未被释放,其他线程无限等待的问题。

    当多个线程都只是进行读操作,所以当一个线程在进行读操作时,其他线程只能等待无法进行读操 作。因此就需要一种机制来使得多个线程都只是进行读操作时,线程之间不会发生冲突,通过Lock就可以办到

    Lock和synchornized的区别:

    1. Lock不是Java语言内置的,synchronized是Java语言的关键字,因此是内置特性。Lock是一个类,通过这 个类可以实现同步访问;

      2. Lock和synchronized有一点非常大的不同,采用synchronized不需要用户去手动释放锁,当synchronized方法或者synchronized代码块执行完之后,系统会自动让线程释放对锁的占用;而Lock则必须要用户去手动释放锁,如果没有主动释放锁,就有可能导致出现死锁现象。

    lock的特点:

    1. 采用Lock,必须主动去释放锁,并且在发生异常时,不会自动释放锁。因此一般来说,使用Lock必须在try{}catch{}块中进行,并且将释放锁的操作放在finally块中进行,以保证锁一定被被释放,防止死锁的发生。通常使用Lock来进行同步的话

线程池

线程同步&线程池的更多相关文章

  1. 线程:主线程、子线程 同步线程、异步线程 单线程、多线程 System.Threading与System.Windows.Threading

    入门-------------------------------------------------------------------------------- 概述与概念    一个C#程序开始 ...

  2. Python并发编程-线程同步(线程安全)

    Python并发编程-线程同步(线程安全) 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 线程同步,线程间协调,通过某种技术,让一个线程访问某些数据时,其它线程不能访问这些数据,直 ...

  3. Qt QThread 线程创建,线程同步,线程通信 实例

    1.  继承QThread, 实现run()方法, 即可创建线程. 2. 实例1 代码 myThread.h #ifndef MYTHREAD_H #define MYTHREAD_H #includ ...

  4. C#线程同步--线程通信

    问题抽象:当某个操作的执行必须依赖于另一个操作的完成时,需要有个机制来保证这种先后关系.线程通信方案:ManualResetEventSlim.ManualResetEvent.AutoResetEv ...

  5. 细说C#多线程那些事 - 线程同步和多线程优先级

    上个文章分享了一些多线程的一些基础的知识,今天我们继续学习. 一.Task类 上次我们说了线程池,线程池的QueueUserWorkItem()方法发起一次异步的线程执行很简单 但是该方法最大的问题是 ...

  6. C#并行编程(6):线程同步面面观

    理解线程同步 线程的数据访问 在并行(多线程)环境中,不可避免地会存在多个线程同时访问某个数据的情况.多个线程对共享数据的访问有下面3种情形: 多个线程同时读取数据: 单个线程更新数据,此时其他线程读 ...

  7. Linux多线程编程——多线程与线程同步

    多线程 使用多线程好处: 一.通过为每种事件类型的处理单独分配线程,可以简化处理异步事件的代码,线程处理事件可以采用同步编程模式,启闭异步编程模式简单 二.方便的通信和数据交换 由于进程之间具有独立的 ...

  8. java SE学习之线程同步(详细介绍)

           java程序中可以允许存在多个线程,但在处理多线程问题时,必须注意这样一个问题:               当两个或多个线程同时访问同一个变量,并且一些线程需要修改这个变量时,那么这个 ...

  9. Java之线程,常用方法,线程同步,死锁

    1, 线程的概念 进程与线程 进程:每个进程都有独立的代码和数据空间(进程上下文),进程间的切换会有较大的开销,一个进程包含1--n个线程.(进程是资源分配的最小单位) 线程:同一类线程共享代码和数据 ...

随机推荐

  1. 3. Rsync-远程同步(下)-企业案例

    已知3台服务器主机名分别为web01.backup .nfs主机信息见下表: 角色 外网IP(NAT) 内网IP(LAN) 主机名 WEB eth0:10.0.0.7 eth1:172.16.1.7 ...

  2. angular之模块开发二

    一.模块化规范 1.服务器端规范 CommonJS--node.js 2.浏览器端规范 AMD--RequireJS 国外相对流行 CMD--SeaJS 国内相对流行 3.模块化框架实现 CMD实现- ...

  3. 百万年薪python之路 -- 小数据池和代码块

    1.小数据池和代码块 # 小数据池 -- 缓存机制(驻留机制) # == 判断两边内容是否相等 # a = 10 # b = 10 # print(a == b) # is 是 # a = 10 # ...

  4. MySQL操作(一)用户及权限

    一.mysql 里的所有用户都是存储在数据库mysql的user表里 二.创建普通用户.赋权.撤销权限 的操作 1.创建用户(需要先用root进去mysql)格式:create  user  '用户名 ...

  5. 面试又被 Java 基础难住了?推荐你看看这篇文章。

    本文已经收录自 JavaGuide (59k+ Star):[Java学习+面试指南] 一份涵盖大部分Java程序员所需要掌握的核心知识. 1. 面向对象和面向过程的区别 面向过程 :面向过程性能比面 ...

  6. Just 5分钟!使用k3s部署轻量Kubernetes集群快速教程

    大小仅有40MB的k3s为想要节省开销进行开发和测试的企业提供了一个很好的选择.本文将用一种极为简洁的方式,教你在5分钟之内使用k3s部署轻量Kubernetes集群. Kubernetes已经改变了 ...

  7. ReoGrid.Mvvm:ReoGrid绑定模型

    ReoGrid 是 C# 编写的.NET 电子表格控件(类似 Excel).支持单元格合并,边框样式,图案背景颜色,数据格式,冻结,公式,宏和脚本执行,表格事件等.支持 Winform\WPF. Re ...

  8. JVM概述和类加载器

    JVM概述 1.Java虚拟机所管理的内存包括以下几个运行时数据区域:   ①.程序计数器     程序计数器是一块较小的内存空间,可以看做是当前线程所执行的字节码的行号指示器,字节码解释器工作时就是 ...

  9. Mysql数据库(三)Mysql表结构管理

    一.MySQL数据类型 1.数字类型 (1)整数数据类型包括TINYINT/BIT/BOOL/SMALLINT/MEDIUMINT/INT/BIGINT (2)浮点数据类型包括FLOAT/DOUBLE ...

  10. NetworkManager网络通讯_networkReader/Writer(六)

    unet客户端和服务端进行消息发送时可以采用上一节中方法,也可以直接用networkReader/Writer类进行发送 (一)服务端/客户端注册消息 ; m_Server.RegisterHandl ...