- 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的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
最近做一个项目,由于是在别人框架里开发app,导致了很多限制,其中一个就是不能直接引用webservice 。 我们都知道,调用webserivice 最简单的方法就是在 "引用"
这是SDL2代码的一部分 SDL主函数 int main(int argc,char *argv[]) { ... ... bool quit=false; S
c 中的函数: PHPAPI char *php_pcre_replace(char *regex, int regex_len, ch
我有以下映射: public class SecurityMap : ClassMap { public SecurityMap() {
我在vue-lic3中使用了SCSS,但是有一个奇怪的错误,使用/ deep /会报告错误,我不想看到它。 代码运行环境 vue-cli3 + vant + scss 的CSS /deep/ .van
我在深入阅读 C# 时遇到了这个我能理解的内容: 当它被限制为引用类型时,执行的比较类型完全取决于类型参数被限制为什么。 但是不能理解这个: 如果进一步限制派生自重载 == 和 != 运算符的特定类型
Closed. This question is opinion-based。它当前不接受答案。 想改善这个问题吗?更新问题,以便editing this post用事实和引用来回答。 3年前关闭。
有人可以详细介绍关于自赋值的运算符重载中的 *this 和 const 例如: Class& Class::operator=(const Class& other) { a = other.
在向树中插入新节点时,如何填充闭包表的深度/长度列? ancestor 和 descendant 中的值是来自另一个表的 ID,表示要以树结构排列的页面。 关闭表: ancestor desce
现在我正在阅读“深入了解 C#”。缺少的一件事是完成一章后我可以解决的一系列问题。那会帮助我理解我刚刚学到的概念。 哪里可以找到适合 C#3.0 的问题集? 谢谢 最佳答案 你可以试试LINQ 101
TypeScript 给 JavaScript 扩展了类型的语法,我们可以给变量加上类型,在编译期间会做类型检查,配合编辑器还能做更准确的智能提示。此外,TypeScript 还支持了高级类型用
是否有一个单行代码来获取生成器并生成该生成器中的所有元素?例如: def Yearly(year): yield YEARLY_HEADER for month in range(1, 13)
所以我阅读了一些与“什么是方法组”相关的 StackOverflow 问题以及其他互联网文章,它们在底线都说了同样的话——方法组是“一组重载方法” ". 但是,在阅读 Jon Skeet 的“C# 深
有什么方法可以从子组件中获取子组件吗? 想象一下以下组件树: 应用程序 问题 问题选项(包含复选框) 问题选项(包含复选框) 问题选项(包含复选框) 我想从 App 访问问题选项以选中所有复选框。 参
class_eval 和 instance_eval 在定义方法等情况下是完全可以预测的。我也理解类的实例和类的单例(又名特征类)之间的区别。 但是 我无法弄清楚以下唯一的事情:比方说,出于某些策略目
我想出了如何将符号 rwx 部分读取/转换为 421 个八进制部分,这非常简单。但是当涉及到特殊字符时,我感到非常困惑。我们知道 -r-xr---wx 转换为 0543,但 -r-sr---wt 或
我怀疑我系统的 Java 版本有问题。某些应用程序出现段错误或内存不足或存在链接错误。如果我从源代码安装了 JDK,我会做类似“make test”的事情,看看哪些测试失败了。但是,看起来从源代码构建
如何克隆一个 repo(使用 libgit2 ) 我想做什么git clone确实,但有 libgit2 .我可能要问的是什么 git clone确实很深入。 这是我目前正在做的: 初始化一个repo
00、头痛的JS闭包、词法作用域? 被JavaScript的闭包、上下文、嵌套函数、this搞得很头痛,这语言设计的,感觉比较混乱,先勉强理解总结一下😂😂😂.
我开始玩 lubridate R 中的包。我注意到 now(tzone="EST")计算为: [1] "2015-08-25 13:01:08 EST" 而 now(tzone="PST")导致警告:
我是一名优秀的程序员,十分优秀!