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线程基础
使用线程更好的提高资源利用率,但也会带来上下文切换的消耗,频繁的内核态和用户态的切换消耗,如果代码设计不好,可能弊大于利. 一.线程 进程是分配资源的最小单位,线程是程序执行的最小单位:线程是依附于进 ...
随机推荐
- ASP.NET Web服务(ASMX)学习和代理生成
第一步:按照http://www.c-sharpcorner.com/article/getting-started-with-asp-net-web-services-part-one/ 建立项目和 ...
- iOS 将对象的属性和属性值拆分成key、value,通过字符串key来获取该属性的值
这篇博客光看标题或许就会产生疑问,某个对象,只要它存在某个属性,且值不是空的,不就能直接用点方法获取吗,为什么要拆分成key和value多此一举呢?下面,我用一个例子告诉大家,既然这方法是存在的,那就 ...
- Appium手势密码滑动之Z字形走势(java篇)
1.直接使用负的偏移量appium会报错,在后面加上moveto(1,1)就行了 2.直接看图说话 废话少说看代码如: List<AndroidElement> element = dri ...
- 深度学习实践系列(2)- 搭建notMNIST的深度神经网络
如果你希望系统性的了解神经网络,请参考零基础入门深度学习系列,下面我会粗略的介绍一下本文中实现神经网络需要了解的知识. 什么是深度神经网络? 神经网络包含三层:输入层(X).隐藏层和输出层:f(x) ...
- Weblogic虚拟目录
p.MsoNormal,li.MsoNormal,div.MsoNormal { margin: 0cm; margin-bottom: .0001pt; text-align: justify; f ...
- POPTEST培训:web自动化测试之DOM
POPTEST培训:web自动化测试之DOM poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性能测试,测试工具开发等工作为目标.如果对课程感兴趣,请大家咨询qq ...
- AngularJS1.X学习笔记6-控制器和作用域
经过一番艰苦卓绝的鏖战,我终于来到了控制器和作用域部分.控制器作为MVC的C,其重要性不可谓不重要:作用域决定了你可以拿到哪些东西,亦是分外重要.现在就来学习一下两个东西.去看看$apply,$wat ...
- Jmeter函数组件开发
插件开发方法有两种: 一.在jmeter官网下载jmeter源码,在源码里面新加函数,然后导出jar: 二.不下载源码,直接导入jmeter相应的jar包,即可开发.(推荐) 下面介绍第二种开发方法: ...
- 打印pid,写着玩。
#include <stdio.h> #include <string.h> #include <dirent.h> #include <limits.h&g ...
- 用java写的一个程序,可以调用windows系统中自带的各种工具,例如截图,便签等
由于图片资源没有上传,所以运行后不会有图片,感兴趣的同学可以@我,我打包上传, package SmallPrograme; import java.awt.*; import java.awt.ev ...