java线程(二)
线程范围变量
我们知道线程在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线程(二)的更多相关文章
- Java线程(二):线程同步synchronized和volatile
上篇通过一个简单的例子说明了线程安全与不安全,在例子中不安全的情况下输出的结果恰好是逐个递增的(其实是巧合,多运行几次,会产生不同的输出结果),为什么会产生这样的结果呢,因为建立的Count对象是线程 ...
- java 线程二
一.线程的优先级别 线程优先级别的使用范例: 1 package cn.galc.test; 2 3 public class TestThread6 { 4 public static void m ...
- Java线程专栏文章汇总(转)
原文:http://blog.csdn.net/ghsau/article/details/17609747 JDK5.0之前传统线程 Java线程(一):线程安全与不安全 Java线程 ...
- Java线程专栏文章汇总
转载自 http://blog.csdn.net/ghsau/article/details/17609747 JDK5.0之前传统线程 Java线程(一):线程安全与不安全 J ...
- Java线程池使用和分析(二) - execute()原理
相关文章目录: Java线程池使用和分析(一) Java线程池使用和分析(二) - execute()原理 execute()是 java.util.concurrent.Executor接口中唯一的 ...
- java线程池技术(二): 核心ThreadPoolExecutor介绍
版权声明:本文出自汪磊的博客,转载请务必注明出处. Java线程池技术属于比较"古老"而又比较基础的技术了,本篇博客主要作用是个人技术梳理,没什么新玩意. 一.Java线程池技术的 ...
- Java线程池ThreadPoolExecutor使用和分析(二) - execute()原理
相关文章目录: Java线程池ThreadPoolExecutor使用和分析(一) Java线程池ThreadPoolExecutor使用和分析(二) - execute()原理 Java线程池Thr ...
- 性能测试三十二:监控之Java线程监控
线程的五种状态 * 新建:new * 运行:runnable * 等待:waitting(无限期等待),timed waitting(限期等待) * 阻塞:blocked * 结束:terminate ...
- Java线程池详解(二)
一.前言 在总结了线程池的一些原理及实现细节之后,产出了一篇文章:Java线程池详解(一),后面的(一)是在本文出现之后加上的,而本文就成了(二).因为在写完第一篇关于java线程池的文章之后,越发觉 ...
- 【Java并发专题之二】Java线程基础
使用线程更好的提高资源利用率,但也会带来上下文切换的消耗,频繁的内核态和用户态的切换消耗,如果代码设计不好,可能弊大于利. 一.线程 进程是分配资源的最小单位,线程是程序执行的最小单位:线程是依附于进 ...
随机推荐
- Java程序员应当知道的10个面向对象设计原则
面向对象设计原则是OOPS编程的核心, 但我见过的大多数Java程序员热心于像Singleton (单例) . Decorator(装饰器).Observer(观察者) 等设计模式,而没有把足够多的注 ...
- RabbitMQ集群搭建
准备三个节点,系统为CentOS7 Node IP rabbitmq01 172.50.0.64 rabbitmq02 172.50.0.65 rabbitmq03 172.50.0.66 这里把no ...
- JS——函数、事件
1.函数字符串函数 var s=new string(); var ss="hello world"; var sss=""HELLO, WORLD" ...
- IOS开发创建开发证书及发布App应用(一)——流程说明
之前在自己做的博客网站上面发布了这个系列的文章,当时还是不错的,帮助了很多跟我一样的新手朋友,不过由于服务器出现问题,丢失了一年了,现在终于找到了,所以发到博客园给大家共享一下,也是为我自己做个参考 ...
- ArcGIS API for JavaScript 4.2学习笔记[20] 使用参数查询要素(油井和地震关系)
这个例子相当复杂.我先简单说说这个例子是干啥的. 在UI上,提供了一个下拉框.两个滑动杆,以确定三个参数,使用这三个参数进行空间查询.这个例子就颇带空间查询的意思了. 这个例子是干嘛的呢?第一个参数是 ...
- sublime Text3快捷键使用大全
Ctrl+D 选中光标所占的文本,继续操作则会选中下一个相同的文本.Alt+F3 选中文本按下快捷键,即可一次性选择全部的相同文本进行同时编辑.举个栗子:快速选中并更改所有相同的变量名.函数名等.Ct ...
- 635B. Bear and Compressing
B. Bear and Compressing time limit per test 2 seconds memory limit per test 256 megabytes input stan ...
- Redis基础学习(五)—Redis的主从复制
一.概述 Redis的主从复制策略是通过其持久化的rdb文件来实现的,其过程是先dump出rdb文件,将rdb文件全量传输给slave,然后再将dump后的操作实时同步到slave中.让从服务 ...
- MySQL学习笔记(二)—查询
一.多表连接查询 新建两张表t_user.t_order. 1.内连接 返回满足条件的所有记录. (1)显式内连接 使用inner join关键字,在on ...
- python 面向对象编程(二)
在上一篇文章中谈到了类的基本定义和使用方法,这只体现了面向对象编程的三大特点之一:封装. 下面就来了解一下另外两大特征:继承和多态. 在Python中,如果需要的话,可以让一个类去继承一个类,被继承的 ...