- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章深入讲解java线程与synchronized关键字由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
我们将会从以下的几点理解java线程的一些概念:
1、线程的基本概念 。
在计算机中有进程和线程这么两个概念,进程中可以有多个线程,它们是从属关系,进程往往更像是资源的占有者,线程才是程序的执行者,多个线程之间共享着进程中的资源。一个cpu同时只能运行一个线程,每个线程都有一个时间片,时间片用完了就会被阻塞并让出CPU的控制权,交给下一个线程使用。这样在计算机中就可以实现多任务的假象,其实CPU在不断的切换线程,好像多个任务在同时运行.
使用线程的优势毋庸置疑,可以增加CPU的执行效率,一旦某个线程需要等待某种资源(例如:等待打印机),就可以将它阻塞释放CPU让CPU执行别的线程,而不需要让CPU和此线程一起等待某种资源从而提高系统效率,另外一点就是可以用这种假象增加用户体验度。但是,CPU在切换不同线程之间所要花费的代价也是不可忽视的,在较为复杂的程序中这种劣势可能九流一毛,但是如果在简单的程序中就会显得尤为突出.
2、创建一个线程 。
接下来我们看看如何在java中创建一个线程来实现多个线程同时运行。第一种方式,java 中有一个类Thread,我们只要继承这个类并重写他的run方法,调用start方法就可以启动一个新的线程了。(没见过的同学可能不能理解以下代码,下面我会解释) 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
/*声明自己的一个线程类*/
public
class
Test_thread
extends
Thread {
//重写Thread类中的run方法
public
void
run(){
System.out.println(
"i am the thread"
);
}
}
public
class
Test_Class {
public
static
void
main(String[] args){
Test_thread thread =
new
Test_thread();
thread.start();
}
}
|
输出结果:i am the thread 。
首先我们先了解一下,一个线程被创建之后,怎么才能启动运行,我们调用thread.start();方法启动一个线程,首先就会执行我们重写的run方法(如果没有重写就会调用Thread类的run方法,什么也不做,这也是我们重写run方法的原因),也就是说run方法是一个线程的开始。有个疑问,为什么调用start方法,却执行了run方法了?其实调用start方法就是为线程的启动做准备操作,分配线程私有的堆栈资源,然后执行run方法.
下面我们看创建一个线程的第二种方式,实现接口Runnable,并重写其中的run方法.
1
2
3
4
5
6
7
8
9
10
11
12
|
public
class
Test_thread
implements
Runnable {
public
void
run(){
System.out.println(
"i am the thread"
);
}
}
public
class
Test_Class {
public
static
void
main(String[] args){
Test_thread thread =
new
Test_thread();
thread.start();
//编译错误
}
}
|
我们会发现虽然重写了run方法,但是在调用start方法的时候却编译错误,我们进入到Runnable接口的源代码中可以看到,只有一个抽象方法run,所以没有启动线程的start方法,所以我们还是要借助Tread类.
1
2
3
4
5
6
|
public
class
Test_Class {
public
static
void
main(String[] args){
Thread thread =
new
Thread(
new
Test_thread());
thread.start();
}
}
|
因为Thread类中有start方法,所以可以使用Thread的一个构造函数传入一个实现了Runnable接口的类型,构建一个Thread类.
3、线程的属性和状态 。
在一个多线程的程序中我们使用线程的一些属性来区别和辨认它们:
1
2
3
4
5
6
|
private
long
tid;
private
volatile
char
name[];
public
static
native
Thread currentThread()
private
boolean
daemon =
false
;
private
volatile
int
threadStatus =
0
;
public
final
static
int
NORM_PRIORITY =
5
;
|
每个线程会有一个tid指定此线程的在所有线程中的排序,这个值是递增的,还有一个name指定该线程的名字,也可以使用setName设置一个优雅的线程名字,Thread类还提供了一个方法返回当前正在占用CPU的线程对象。每个线程在某个时刻都是有状态的,属性threadStatus记录了当前对象线程的状态是什么,默认情况下,所有的线程的优先级都被置为5,如果有特殊需要可以修改线程的优先级,使得某个线程可以优先得到运行。下面介绍线程的几种不同的状态.
下面具体的说说不同状态下的线程的一些操作,首先看看new,线程从此被创建,只是离运行状态还需要一些准备。Runnable表示线程是可运行,至于是否已经处于运行状态还要看系统给的时间片是否用完,如果用完了就会将此线程放置在可运行线程队列的尾部,等待下次分配时间片,如果时间片没有用完,就是处于运行状态的(这也是为什么叫做Runnable而不是Running的原因)。接下来的两种状态Blocked和waiting都表示线程阻塞,需要让出CPU。只是导致它们处于这种状态的原因不一样,具体的在我们介绍完synchronized关键字之后就会不言而喻了.
4、关键字synchronized 。
先看一段代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
public
class
Test_thread
extends
Thread{
public
static
int
count =
0
;
public
void
run(){
try
{
Thread.sleep((
int
) (Math.random() *
100
));
}
catch
(InterruptedException e){
}
count++;
}
}
public
class
Test_Class {
public
static
void
main(String[] args){
Test_thread[] thread =
new
Test_thread[
1000
];
for
(
int
a=
0
;a<
1000
;a++){
thread[a] =
new
Test_thread();
thread[a].start();
}
for
(
int
a=
0
;a<
1000
;a++){
try
{
thread[a].join();
}
catch
(InterruptedException e){
}
}
System.out.println(Test_thread.count);
}
}
|
按照直觉,创建1000个线程,每个线程随机睡觉并将公共变量增一,main线程等待所有线程执行结束之后,输出公共变量的值。按照直觉答案应该是1000,但是我们运行的结果每次都是不一样的,接近1000但每次都不一样。这是为什么呢?这其实就是简单的模拟了高并发,可能有几个线程睡了相同的时间,同时醒来获取的count值是相同的,这就导致这几个线程对count的操作被覆盖了.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
public
class
Test_thread
extends
Thread{
public
static
int
count =
0
;
public
void
run(){
try
{
Thread.sleep((
int
) (Math.random() *
100
));
}
catch
(InterruptedException e){
}
/*使用关键字*/
synchronized
(Test_thread.
class
){
count++;
}
}
}
|
一个简单的关键字就可以轻松解决这样的高并发的问题。至于为什么这么写?在介绍完synchronized关键字之后,想必你就会知道了.
首先我们需要知道,每个对象都有一把内部锁。所以被synchronized关键字修饰的方法,其实是被加了内部对象锁。我们看代码:
1
2
3
4
5
6
7
8
|
public
class
Counter{
private
int
count;
/*为实例方法加此关键字*/
public
synchronized
int
getCount(){
return
count;
}
}
|
为实例方法添加关键字,实际上就是给当前的对象加锁;括号中就是要加锁的对象.
1
2
3
4
5
6
7
8
|
public
class
Counter{
private
int
count;
/*为实例方法加此关键字*/
synchronized
(
this
){
return
count;
}
}
|
对于静态方法,实际上就是为这个类加上锁; 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
public
class
Counter{
private
static
int
count;
public
static
synchronized
int
getCount(){
return
count;
}
}
/*实际上给这个类加上锁*/
public
class
Counter{
private
int
count;
synchronized
(Counter.
class
){
return
count;
}
}
|
5、深入理解synchronized的一些特性 。
第一个性质是:可重入性。被synchronized修饰的方法中的所有操作都是原子操作,但是当我们需要在其中访问另外的一些需要锁的代码时候,可以直接获取别的锁。也就是当前的对象是可以获得多个锁的.
第二个性质是:内存的可见性。在我们的计算机中,其实有很多的操作并不是很"干脆"的,比如你向数据库中存数据时,其实很多时候一些待存入的数据积累在缓存中等到一定数据量的时候才会统一的存入数据库,但是在我们看来这些数据其实应该早就存入数据库了。这就导致了一个内存的不可见性问题。当然我们也是可以使用关键字synchronized来保证每次取出的数据都是最新的。在释放锁的时候会立即将数据协会内存,获得锁的时候会去读取最新的内容.
1
2
3
4
5
6
7
8
|
public
class
Counter{
private
int
count;
public
synchronized
int
getCount(){
return
count;
}
}
/*每次获得该对象的锁之后,去获取最新的count数值*/
|
其实,如果仅仅是要保证内存的不可见性,使用synchronized关键字可能代价有点高,在这种情况下,我们可以使用关键字volatile.
1
2
3
4
5
6
7
8
9
|
public
class
Counter{
/*使用关键字volatile*/
private volatile int count;
public int getCount(){
return count;
}
}
/*此关键字保证每次使用count的时候数据都是最新的*/
|
总结 。
以上就是这篇文章的全部内容了,还是希望大家发现其中错误直接指出,方便我纠正错误,更新知识。java并发系列文章,希望大家多多关注.
原文链接:http://www.cnblogs.com/yangming1996/p/6515830.html 。
最后此篇关于深入讲解java线程与synchronized关键字的文章就讲到这里了,如果你想了解更多关于深入讲解java线程与synchronized关键字的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我正在编写一个具有以下签名的 Java 方法。 void Logger(Method method, Object[] args); 如果一个方法(例如 ABC() )调用此方法 Logger,它应该
我是 Java 新手。 我的问题是我的 Java 程序找不到我试图用作的图像文件一个 JButton。 (目前这段代码什么也没做,因为我只是得到了想要的外观第一的)。这是我的主课 代码: packag
好的,今天我在接受采访,我已经编写 Java 代码多年了。采访中说“Java 垃圾收集是一个棘手的问题,我有几个 friend 一直在努力弄清楚。你在这方面做得怎么样?”。她是想骗我吗?还是我的一生都
我的 friend 给了我一个谜语让我解开。它是这样的: There are 100 people. Each one of them, in his turn, does the following
如果我将使用 Java 5 代码的应用程序编译成字节码,生成的 .class 文件是否能够在 Java 1.4 下运行? 如果后者可以工作并且我正在尝试在我的 Java 1.4 应用程序中使用 Jav
有关于why Java doesn't support unsigned types的问题以及一些关于处理无符号类型的问题。我做了一些搜索,似乎 Scala 也不支持无符号数据类型。限制是Java和S
我只是想知道在一个 java 版本中生成的字节码是否可以在其他 java 版本上运行 最佳答案 通常,字节码无需修改即可在 较新 版本的 Java 上运行。它不会在旧版本上运行,除非您使用特殊参数 (
我有一个关于在命令提示符下执行 java 程序的基本问题。 在某些机器上我们需要指定 -cp 。 (类路径)同时执行java程序 (test为java文件名与.class文件存在于同一目录下) jav
我已经阅读 StackOverflow 有一段时间了,现在我才鼓起勇气提出问题。我今年 20 岁,目前在我的家乡(罗马尼亚克卢日-纳波卡)就读 IT 大学。足以介绍:D。 基本上,我有一家提供簿记应用
我有 public JSONObject parseXML(String xml) { JSONObject jsonObject = XML.toJSONObject(xml); r
我已经在 Java 中实现了带有动态类型的简单解释语言。不幸的是我遇到了以下问题。测试时如下代码: def main() { def ks = Map[[1, 2]].keySet()
一直提示输入 1 到 10 的数字 - 结果应将 st、rd、th 和 nd 添加到数字中。编写一个程序,提示用户输入 1 到 10 之间的任意整数,然后以序数形式显示该整数并附加后缀。 public
我有这个 DownloadFile.java 并按预期下载该文件: import java.io.*; import java.net.URL; public class DownloadFile {
我想在 GUI 上添加延迟。我放置了 2 个 for 循环,然后重新绘制了一个标签,但这 2 个 for 循环一个接一个地执行,并且标签被重新绘制到最后一个。 我能做什么? for(int i=0;
我正在对对象 Student 的列表项进行一些测试,但是我更喜欢在 java 类对象中创建硬编码列表,然后从那里提取数据,而不是连接到数据库并在结果集中选择记录。然而,自从我这样做以来已经很长时间了,
我知道对象创建分为三个部分: 声明 实例化 初始化 classA{} classB extends classA{} classA obj = new classB(1,1); 实例化 它必须使用
我有兴趣使用 GPRS 构建车辆跟踪系统。但是,我有一些问题要问以前做过此操作的人: GPRS 是最好的技术吗?人们意识到任何问题吗? 我计划使用 Java/Java EE - 有更好的技术吗? 如果
我可以通过递归方法反转数组,例如:数组={1,2,3,4,5} 数组结果={5,4,3,2,1}但我的结果是相同的数组,我不知道为什么,请帮助我。 public class Recursion { p
有这样的标准方式吗? 包括 Java源代码-测试代码- Ant 或 Maven联合单元持续集成(可能是巡航控制)ClearCase 版本控制工具部署到应用服务器 最后我希望有一个自动构建和集成环境。
我什至不知道这是否可能,我非常怀疑它是否可能,但如果可以,您能告诉我怎么做吗?我只是想知道如何从打印机打印一些文本。 有什么想法吗? 最佳答案 这里有更简单的事情。 import javax.swin
我是一名优秀的程序员,十分优秀!