一、线程安全问题:

    当我们使用多个线程操作统一方法内的局部变量的时候,每个局部变量在当前线程里都有自己的副本,这种情况是不会出现线程安全问题的。当我们两个线程同时操作全局变量的时候,有可能会引发线程安全的问题。
 
①.业务类

  1. package com.multiThread.bean;
  2. publicclassAservise{
  3. privateString name;
  4. publicvoid doBusiness(String name){
  5. this.name = name;
  6. System.out.println("大家好,我是"+this.name);
  7. }
  8. }
②.线程类
  1. package com.multiThread.thread;
  2. import com.multiThread.bean.Aservise;
  3. publicclassUnSafeThreadimplementsRunnable{
  4. privateAservise aServise;
  5. privateString name;
  6. publicUnSafeThread(Aservise aServise,String name){
  7. this.aServise = aServise;
  8. this.name = name;
  9. }
  10. @Override
  11. publicvoid run(){
  12. aServise.doBusiness(this.name);
  13. }
  14. }
③.测试类
  1. package com.multiThread.test.common;
  2. import com.multiThread.bean.Aservise;
  3. import com.multiThread.thread.UnSafeThread;
  4. publicclassUnSafeThreadTest{
  5. publicstaticvoid main(String[] args){
  6. Aservise aService =newAservise();
  7. UnSafeThread unSafeThreadZhang =newUnSafeThread(aService,"张三");
  8. UnSafeThread unSafeThreadLi =newUnSafeThread(aService,"李四");
  9. Thread zhang =newThread(unSafeThreadZhang);
  10. Thread li =newThread(unSafeThreadLi);
  11. zhang.start();
  12. li.start();
  13. }
  14. }
预期输出结果
  1. 大家好,我是张三
  2. 大家好,我是李四
多次运行实际输出结果
  1. 大家好,我是李四
  2. 大家好,我是李四
  3. 大家好,我是张三
  4. 大家好,我是李四
  5. 大家好,我是张三
  6. 大家好,我是张三
这个例子很好的解释了多个线程同时操作全局变量会存在线程安全的问题。
那么这种问题该如何解决呢?在这里我们只讨论服务器为单节点的情况,不考虑集群模式。
解决的方式是加锁,只要保证这两段程序不同时变更全局变量就OK。
 
 
将业务类的doBusiness方法更改为使用synchronized关键字修饰就可以:
1.在方法上声明,给当前对象加锁(非静态方法):
  1. publicsynchronizedvoid doBusiness(String name){
  2. this.name = name;
  3. System.out.println("大家好,我是"+this.name);
  4. }
 
2.synchronized不仅仅可以在方法上声明,也可以在方法内部声明(参数是Object类型的值,写成this代表给当前对象上锁)。这种情况叫做同步代码块:
  1. publicvoid doBusiness(String name){
  2. synchronized(this){
  3. this.name = name;
  4. System.out.println("大家好,我是"+this.name);
  5. }
  6. }
对象监视器:
    synchronize修饰方法或者synchronized(this),对象监视器监视当前类的对象。
    synchronize(其他对象),对象监视器给其他对象上锁。
需要注意的问题:
    一般不会使用字符串作为监视对象,因为字符串有个常量池的概念,在不同的地方操作可能锁的是同一个对象。
synchronize作用:
    1.给对象监视器监视的对象上锁
    2.保证同一时间只有一个对象可以获得此对象监视器上的锁。
 
注意:
    ①.synchronize关键字如果在非静态方法上声明或者在非静态代码块上声明synchronized(this),代表给当前对象上锁,当一个线程获得此锁的同时,其他线程调用synchronize修饰的方法、代码块均需等待。当此线程执行完毕或者主动释放锁时,根据cpu调度看哪个线程能获得此锁。
    ②.如果synchronize(其他对象),则代表给参数中的对象上锁,所有参数对象共用同一把锁,只有获得锁的线程可以执行此代码块。
    ③.如果是静态方法,则代表是给整个类上锁,那么整个类共用同一把锁,只有获得锁的线程可以执行同步方法、代码块。
 
使用jstack查看死锁:
    
    ①.cd到jdk/bin目录下,执行jps命令,得到正在运行的进程id,这块我笔记本上运行不出来。我也不知道哪个是进程id。
    ②.执行jstack - l 进程id
    ③.得到jstack后分析一下程序哪个地方设计有bug,修改程序。
 
二、Thread类相关API操作:
    
好了,现在我们已经对多线程和线程安全有了一定的认识。下面我们通过具体代码来了解一下java.lang.Thread类相关的API的常用操作。
 
currentThread():获取当前执行当前线程的线程对象
isAlive():当前线程是否存活(正在执行)
sleep():使当前线程睡眠一段时间,参数单位毫秒
getId():获取当前线程的唯一标识
yield():释放当前线程的控制权,将控制权交由CPU调度。
interrupt():停止线程执行(只调用这个方法无法中止线程的执行,需要配合interrupted()才能中断线程。线程在sleep的情况下中断会抛异常并且清除停止状态的值,使之变为false
interrupted():测试当前线程是否已经是中断状态,执行后具有将状态标志清除为false的功能。
                        此方法多次调用会有问题。比如第一次中断线程,再次调用则会清除中断状态。
isInterrupted():测试线程是否已中断。
setPriority ():设置线程的优先级,优先级越高,优先执行的几率越大。取值范围是1~10
 
jdk中不推荐使用的过期的方法以及原因
stop():强制中断当前线程。 强制停止某个线程可能会造成某些清理操作无法完成。而且会将对象的锁给清除掉,有可能会造成数据不一致的问题。
suspend():暂停当前线程的执行,此操作并不会释放锁,有锁独占的问题。
                另外如果在System.out.println()中暂停了线程,同步锁并未释放。其他有System.out.println()的地方就无法执行。
                因为Sysem.out.println()本身底层的实现也是基于synchronized的。一个对象占着锁不放,这边就一直得不到执行。
System.out.println()的实现:
  1. publicvoid println(String x){
  2. synchronized(this){
  3. print(x);
  4. newLine();
  5. }
  6. }
resume():恢复当前线程的执行,此操作并不会释放锁,有锁独占的问题。
 
    

多线程(二)~Thread类相关的API介绍的更多相关文章

  1. (转)多线程——继承Thread 类和实现Runnable 接口的区别

    java中我们想要实现多线程常用的有两种方法,继承Thread 类和实现Runnable 接口,有经验的程序员都会选择实现Runnable接口 ,其主要原因有以下两点: 首先,java只能单继承,因此 ...

  2. 多线程:Thread类的Join()方法

    多线程:Thread类的Join()方法 http://blog.163.com/hc_ranxu/blog/static/3672318220095284513678/ 当我们在线程B中调用Thre ...

  3. 多线程学习笔记(四)---- Thread类的其他方法介绍

    一.wait和 sleep的区别 wait可以指定时间也可以不指定时间,而sleep必须指定时间: 在同步中时,对cpu的执行权和锁的处理不同: wait:释放执行权,释放锁:释放锁是为了别人noti ...

  4. Java多线程01(Thread类、线程创建、线程池)

    Java多线程(Thread类.线程创建.线程池) 第一章 多线程 1.1 多线程介绍 1.1.1 基本概念 进程:进程指正在运行的程序.确切的来说,当一个程序进入内存运行,即变成一个进程,进程是处于 ...

  5. 多线程之 Thread类

    一.多线程第一种方式的实现步骤(继承Thread类) 代码演示: 1.定义MyThread类,继承Thread类 2.重写了里面的run方法,在run方法中定义线程要执行的任务 public clas ...

  6. Java中实现多线程继承Thread类与实现Runnable接口的区别

    Java中线程的创建有两种方式: 1.  通过继承Thread类,重写Thread的run()方法,将线程运行的逻辑放在其中 2.  通过实现Runnable接口,实例化Thread类 在实际应用中, ...

  7. Java 多线程之 Thread 类 和 Runnable 接口初步使用

    目录 Thread 类 Thread之定义线程类 Thread之开启线程 Runnable 接口 Runnable 之定义线程类 Runnable 之开启线程 @ Thread 类 Thread 类是 ...

  8. 多线程——继承Thread类实现一个多线程

    继承Thread类实现一个多线程 Thread类部分源码: package java.lang; //该类实现了Runnable接口 public class Thread implements Ru ...

  9. Java 多线程 (Thread 类)

    1.多线程 2.卖票 1.多线程实现 两种方式可以实现多线程: 继承 Thread 类,重写 run 方法:定义对象,调用 start 方法 创建类实现 Runnable 接口,作为实参传递给 thr ...

随机推荐

  1. hexo 博客

    梦飞扬~ 个人网站:Mauger`s Blog 博客园 标签 新随笔 随笔 管理 Github 随笔 - 61  文章 - 1  评论 - 0 使用Node.js+Hexo+Github搭建个人博客 ...

  2. php 实现无限极分类

    原始数据 $array = array( array('id' => 1, 'pid' => 0, 'n' => '河北省'), array('id' => 2, 'pid' ...

  3. Linux 未安装vi如何编辑文件

    sed -i "s/搜索内容/替换内容/g" 文件名

  4. c# 委托访问listbox多线程操作

    c# 委托访问listbox多线程操作 using System;using System.Collections.Generic;using System.ComponentModel;using ...

  5. ZOJ Monthly, January 2019 I Little Sub and Isomorphism Sequences(set 妙用) ZOJ4089

    写这篇博客来证明自己的愚蠢 ...Orz  飞机 题意:给定你个数组,以及一些单点修改,以及询问,每次询问需要求得,最长的字串长度,它在其他位置存在同构 题解:经过一些奇思妙想后 ,你可以发现问题是传 ...

  6. HDU - 6188

    用vis表贪心异常方便 #include<bits/stdc++.h> #define rep(i,j,k) for(register int i=j;i<=k;i++) #defi ...

  7. java的Spring学习2- junit

    1.maven依赖 <?xml version="1.0" encoding="UTF-8"?> <project xmlns="h ...

  8. zookeeper 节点信息

    使用get命令获取指定节点的数据时, 同时也将返回该节点的状态信息, 称为Stat. 其包含如下字段: czxid. 节点创建时的zxid. mzxid. 节点最新一次更新发生时的zxid. ctim ...

  9. IAR使用技巧 之 快捷键批量更换指定字符(以及Keil的全局替换功能)

    使用IAR(或者Keil)写/移植程序时批量更换字符 作者:李剀 出处:https://www.cnblogs.com/kevin-nancy/p/10776712.html 或者 https://b ...

  10. 关于Json字符串"反序列化Error reading JObject from JsonReader. Current JsonReader item is not an object: StartArray. Path..."

    描述的很清楚就是说给它的不是一个对象,而是一个数组,所以他在建议你用JArray去解析,但是你明明就是给它的一个对象,并不是一个数组 这是我下意识的去把我的json字符串中的"[ ]&quo ...