线程范围变量

  我们知道线程在cpu上的使用权并不是长时间的,因为计算机的cpu只有一个,而在计算上运行的进程有很多,线程就更不用说了,所以cpu只能通过调度来上多个线程轮流占用cpu资源运行,且为了保障多个进程能够同时运行。cpu只能通过频繁调度线程来是的每个线程在几乎同一时间内都能得到cpu的使用权来执行代码来保证程序能够正常运行,所以cpu的调度频率是很快的。那么问题来了。如下代码:

 package cn.wz.traditional.wf;

 import javafx.scene.chart.PieChart;

 /**
* Created by WangZhe on 2017/5/4.
*/
public class ThraedLocalTest {
public static void main(String[] args) {
new Thread(new Runnable() {
public void run() {
A.age=5;
A.age+=10;
System.out.println(A.age);
}
}).start(); new Thread(new Runnable() {
public void run() {
A.age=7;
}
}).start();
}
static class A{
static Integer age=5;
private Integer data;
public A(Integer data){
this.data=data;
}
public void sayHi(){
data+=10;
System.out.println(data); }
public void sayHello(){
data=5;
}
}
}

demo

上面这段代码创建了两个线程,第一个线程将a的原始值加10后输出,第二个线程将a赋值为7,但在不改变代码的前提下多次执行后发现这段代码出现了两种运行结果。如图所示。

这种问题正是由于cpu调度线程所引起的,当其结果为15时使我们所预期的效果,那怎样来确保其结果和我们预期的效果一致呢?废话不多说了直接上代码

package cn.wz.traditional.wf;

import javafx.scene.chart.PieChart;

import java.util.HashMap;
import java.util.Map; /**
* Created by WangZhe on 2017/5/4.
*/
public class ThraedLocalTest {
public static void main(String[] args) {
new Thread(new Runnable() {
public void run() {
A a=new A(5);
a.sayHi(5);
}
}).start(); new Thread(new Runnable() {
public void run() {
A a=new A(5);
a.sayHello();
}
}).start();
}
static class A{
static Map<String,Integer> ages=new HashMap<String, Integer>();//创建一个键值集合一线程id为key值来存放线程范围的变量
private Integer data;
public A(Integer data){
this.data=data;
}
public void sayHi(Integer age){
ages.put(String.valueOf(Thread.currentThread().getId()),age);
Integer data= ages.get(String.valueOf(Thread.currentThread().getId()));
data+=10;
ages.put(String.valueOf(Thread.currentThread().getId()),data);
System.out.println(ages.get(String.valueOf(Thread.currentThread().getId()))); }
public void sayHello(){
ages.put(String.valueOf(Thread.currentThread().getId()),7);
}
}
}

在之前代码的基础上增加一个map集合来保存线程范围的变量数据,以当前线程的id为key值来确保每次取到的值都是当前线程的值。但是这种方法出现了大量的冗余代码,每次取值赋值都很费劲,其实Java API提供了Threadlocal类来保存线程范围的变量,下面是使用Threadlocal类之后的代码.

package cn.wz.traditional.wf;

import javafx.scene.chart.PieChart;

import java.util.HashMap;
import java.util.Map; /**
* Created by WangZhe on 2017/5/4.
*/
public class ThraedLocalTest {
public static void main(String[] args) {
new Thread(new Runnable() {
public void run() {
A a=new A();
a.sayHi(5);
}
}).start(); new Thread(new Runnable() {
public void run() {
A a=new A();
a.sayHello();
}
}).start();
}
static class A{
ThreadLocal<Integer> tl=new ThreadLocal<Integer>();
public void sayHi(Integer age){
tl.set(age);
Integer data = tl.get();
data+=10;
tl.set(data);
System.out.println(tl.get());
}
public void sayHello(){
tl.set(7);
}
}
}

使用Threadlocal之后的代码

如上面代码所示ThreadLocal类在赋值和获取值时比我们之前写的代码方便了许多并不需要提供key值就能进行操作,其实ThreadLocal类的实现和我们之前的实现差不多的差不多,下面介绍下threadLocal类的方法实现,

ThreadLocal 类方法详解

  ThreadLocal类提供了一下四种方法供程序员调用,下面为大家介绍每个方法的实现。

  

  InitialValue()方法,该方法在对象调用get方法时,之前并没有调用过set方法赋值的情况下调用该方法返回null值,如图所示,

  set()方法,该方法是为线程范围变量的赋值方法,其内容如下:

GetMap()方法内容

 CreateMap方法内容

get方法:给方法是获取线程范围变量的方法,我们已经知道了线程范围变量保存在一个以ThreadLocal实例为key值得map集合中所以对此这里不再进行赘述了,我们着重关注一下调用get之前并没有使用set方法为集合中该项赋值的情况,

如下所示:

 remove方法:该方法是移除当前线程中该ThreadLocal实例对应的线程范围变量值,注意,并不是当前线程的所有范围变量值,如图所示

总结:

  java API提供ThreadLocal类来保存线程范围变量,一个ThreadLocal对象在一个线程上只能绑定一个线程范围变量,但一个ThreadLocal对象可以为多个不同的线程绑定线程范围变量值。

java线程(二)的更多相关文章

  1. Java线程(二):线程同步synchronized和volatile

    上篇通过一个简单的例子说明了线程安全与不安全,在例子中不安全的情况下输出的结果恰好是逐个递增的(其实是巧合,多运行几次,会产生不同的输出结果),为什么会产生这样的结果呢,因为建立的Count对象是线程 ...

  2. java 线程二

    一.线程的优先级别 线程优先级别的使用范例: 1 package cn.galc.test; 2 3 public class TestThread6 { 4 public static void m ...

  3. Java线程专栏文章汇总(转)

    原文:http://blog.csdn.net/ghsau/article/details/17609747 JDK5.0之前传统线程        Java线程(一):线程安全与不安全 Java线程 ...

  4. Java线程专栏文章汇总

        转载自 http://blog.csdn.net/ghsau/article/details/17609747 JDK5.0之前传统线程        Java线程(一):线程安全与不安全 J ...

  5. Java线程池使用和分析(二) - execute()原理

    相关文章目录: Java线程池使用和分析(一) Java线程池使用和分析(二) - execute()原理 execute()是 java.util.concurrent.Executor接口中唯一的 ...

  6. java线程池技术(二): 核心ThreadPoolExecutor介绍

    版权声明:本文出自汪磊的博客,转载请务必注明出处. Java线程池技术属于比较"古老"而又比较基础的技术了,本篇博客主要作用是个人技术梳理,没什么新玩意. 一.Java线程池技术的 ...

  7. Java线程池ThreadPoolExecutor使用和分析(二) - execute()原理

    相关文章目录: Java线程池ThreadPoolExecutor使用和分析(一) Java线程池ThreadPoolExecutor使用和分析(二) - execute()原理 Java线程池Thr ...

  8. 性能测试三十二:监控之Java线程监控

    线程的五种状态 * 新建:new * 运行:runnable * 等待:waitting(无限期等待),timed waitting(限期等待) * 阻塞:blocked * 结束:terminate ...

  9. Java线程池详解(二)

    一.前言 在总结了线程池的一些原理及实现细节之后,产出了一篇文章:Java线程池详解(一),后面的(一)是在本文出现之后加上的,而本文就成了(二).因为在写完第一篇关于java线程池的文章之后,越发觉 ...

  10. 【Java并发专题之二】Java线程基础

    使用线程更好的提高资源利用率,但也会带来上下文切换的消耗,频繁的内核态和用户态的切换消耗,如果代码设计不好,可能弊大于利. 一.线程 进程是分配资源的最小单位,线程是程序执行的最小单位:线程是依附于进 ...

随机推荐

  1. IntelliJ Idea和IntelliJ webstrm 常用快捷键

    Ctrl+Shift + Enter,语句完成"!",否定完成,输入表达式时按 "!"键Ctrl+E,最近的文件Ctrl+Shift+E,最近更改的文件Shif ...

  2. 《Machine Learning》系列学习笔记之第三周

    第三周 第一部分 Classification and Representation Classification 为了尝试分类,一种方法是使用线性回归,并将大于0.5的所有预测映射为1,所有小于0. ...

  3. 用SSE指令计算点乘和累加

    void sse_mul_float:两段内存float数据点乘,结果覆盖第一组内存. float sse_acc_float:一组内存float值累加. 注: 1. 没有考虑中间的精确问题,结果会有 ...

  4. Tcl与Design Compiler (七)——环境、设计规则和面积约束

    本文属于原创手打(有参考文献),如果有错,欢迎留言更正:此外,转载请标明出处 http://www.cnblogs.com/IClearner/  ,作者:IC_learner 本文的主要内容是讲解( ...

  5. JS中new的自定义实现创建实例对象

    我们都知道在JS中通常通过对象字面量和new关键字来创建对象,那么今天我就来给大家讲讲new是怎么创建实例对象的:首先创建一个构造函数: function Person(name,age){ this ...

  6. SUSE linux 使用LVM安装系统和管理

    引出 在我们安装好linux系统后会发现在需要修改磁盘分区的时候会比较困难,系统安装的/目录的文件系统要更改基本不太可能,其他目录如/home目录也比较困难.但是系统安装时要是采用的LVM管理的方式安 ...

  7. java中的基本jdbc中mvc基本示例

    数据库: 包文件: Student.java 1 package com.model; 2 3 public class Student { 4 private int id; 5 private S ...

  8. 关于JDEV的连接问题

    在JDev中有两个连接数据哭库的地方,双击项目名称,里面的Business Components里面的Connection里面的链接,这个链接是Run页面时候的链接 第二个链接在Oracle Appl ...

  9. shell是什么,各种shell的初步认识,适用于初学者

    shell是什么?有什么用处?怎么用?我相信,这是大部分人刚接触到shell都有过的疑问.下面小编为大家讲解一下自己的讲解,希望能对大家有所帮助. 什么是shell? shell就是系统内核的一层壳, ...

  10. 20155206 2016-2017-2 《Java程序设计》第5周学习总结

    20155206 2016-2017-2 <Java程序设计>第5周学习总结 教材学习内容总结 Java中所有错误都会被打包为对象,运用try.catch,可以在错误发生时显示友好的错误信 ...