说道多线程的安全问题,很多人想到就就是加锁。用到synchronized关键字。

那就要先说说synchronized问什么能够保证线程安全了。

首先要了解线程的工作方式:线程工作分为工作内存和主内存。主内存就是堆和静态区。当线程运行时,首先将主内存的数据拿到工作内存

然后在工作内存中运行,再将数据写回主内存。工作内存是私有的,但是主内存却是共享的。

那么线程不安全的主要根源就是不能线程读写主内存的共享数据。

那么判断要不要加锁,在什么位置加锁就有了依据——共享数据

下面看一个例子:

package code.thread;

public class SynchronizedDome extends Thread{
int a = 0;
Object obj = new Object();
@Override
public void run() {
synchronized(obj) {
for(int i=5;i>0;i--){
System.out.println(a);
a++;
try {
Thread.sleep(0);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
} public static void main(String[] args) {
SynchronizedDome dome = new SynchronizedDome();
SynchronizedDome dome2 = new SynchronizedDome();
SynchronizedDome dome3 = new SynchronizedDome();
System.out.println("Thread start:");
dome.start();
dome2.start();
dome3.start();
}
}

执行结果:

Thread start:
0
1
0
0
2
3
1
1
2
3
4
4
2
3
4

看到没,并没有发生错乱,与预想的输出结果一致

那么你可能会说,这是synchronized的功劳。真的是这样的么,稍微改动过一下在看看

package code.thread;

public class SynchronizedDome extends Thread{
int a = 0;
Object obj = new Object();
@Override
public void run() {
//synchronized(obj) {
{
for(int i=5;i>0;i--){
System.out.println(a);
a++;
try {
Thread.sleep(0);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
} public static void main(String[] args) {
SynchronizedDome dome = new SynchronizedDome();
SynchronizedDome dome2 = new SynchronizedDome();
SynchronizedDome dome3 = new SynchronizedDome();
System.out.println("Thread start:");
dome.start();
dome2.start();
dome3.start();
}
}

看到没,还是安全的,并没有因为没加锁而发生错乱

那么我没就要分析一下了,根据我们上面所说的,线程的不安全是因为数据的共享

这个例子中,分别new了三个线程对象。

每个对象在栈上有一个变量a,他们分别属于不同的对象。所以三个线程操作的都是属于自己本身类的数据。是对象私有的。

所以不存在数据的共享,那么就不用加锁了。

我没在看一个例子,让他们数据共享,将变量定义为静态变量

package code.thread;

public class SynchronizedDome2 {
public static void main(String[] args) {
Dome dome = new Dome();
Thread thread = new Thread(dome);
Thread thread2 = new Thread(dome); thread.start();
thread2.start(); }
} class Dome implements Runnable {
static int a = 0;
@Override
public void run() {
//synchronized(this){
{
for(int i=0;i<5;i++) {
System.out.println(Thread.currentThread().getName()+": "+a++);
}
}
}
}

输出结果:

Thread-1: 0
Thread-0: 1
Thread-1: 2
Thread-0: 3
Thread-1: 4
Thread-0: 5
Thread-1: 6
Thread-1: 8
Thread-0: 7
Thread-0: 9

这么换乱,而且我们侥幸没有得到错误的结果。如果多运行几次就会看到可能会出现错误的结果

那么下面用锁来解决,看看有什么不同。

程序就是将什么代码加锁注释去掉。

输出结果:

Thread-0: 0
Thread-0: 1
Thread-0: 2
Thread-0: 3
Thread-0: 4
Thread-1: 5
Thread-1: 6
Thread-1: 7
Thread-1: 8
Thread-1: 9

总结:总的来说线程不安全是由于共享数据的读写不同步引起的。当不涉及到共享数据,也就无不安全可说了。

synchronized关键字保证了操作的原子性和可见性。原子性就是说,一个执行步奏完整的执行完毕,不会再执行的过程中被其他线程打断。

可见性是说,当执行完锁定的代码块后,在解锁之前会把最新的数据写入到主内存中。并且清空其他线程工作内存中该数据的值。保证了该数据时最新的。

多线程简单实例(1)真的需要synchronized么?的更多相关文章

  1. Java 多线程 简单实例 (消费者与生成者)的关系

    PS::线程这套东西在PHP里完全是不存在的概念,有待进一步的学习: PS::这个实例是根据书本上的知识进行扩展的,理解程度50%左右吧! 1.定义生产消费环境 package second; pub ...

  2. Java 多线程 简单实例 (Runnable)

    1.多线程实例 package second; public class A implements Runnable { public char stat = '*'; public void run ...

  3. Java 多线程 简单实例 (Thread)

    package second; public class A extends Thread { public void run(){ for(int i = 1;i <= 10 ; i++){ ...

  4. Linux多线程及线程同步简单实例

    一.多线程基本概念 1. 线程的基本概念 ① 线程就是轻量级的进程 ②线程和创建他的进程共享代码段.数据段 ③线程拥有自己的栈 2. 在实际应用中,多个线程往往会访问同一数据或资源,为避免线程之间相互 ...

  5. Win32 API 多线程编程——一个简单实例(含消息参数传递)

    Win32 API进行程序设计具有很多优点:应用程序执行代码小,运行效率高,但是他要求程序员编写的代码较多,且需要管理所有系统提供给程序的资源,要求程序员对Windows系统内核有一定的了解,会占用程 ...

  6. java多线程Lock接口简介使用与synchronized对比 多线程下篇(三)

    前面的介绍中,对于显式锁的概念进行了简单介绍 显式锁的概念,是基于JDK层面的实现,是接口,通过这个接口可以实现同步访问 而不同于synchronized关键字,他是Java的内置特性,是基于JVM的 ...

  7. Java的多线程 简单入门

    Java的多线程 简单入门 首先能够先搞清楚什么是程序.进程.线程,以及它们之间的关系: 定义: 一 程序仅仅是一组指令的有序集合.它是静态的 二 进程是具有一定独立功能的程序关于某个数据集合上的一次 ...

  8. 简单的互斥同步方式——synchronized关键字详解

    目录 1. 关于synchronized关键字 2. synchronized的原理和实现细节 2.1 synchronized可以用在那些地方 2.2 synchronized是如何实现线程互斥访问 ...

  9. (转) C#多线程赛跑实例

    专于:http://blog.csdn.net/lidatgb/article/details/8363035 结合上篇<多线程的基础>,这次我们写一个多线程的赛跑实例,内容很简单:超人和 ...

随机推荐

  1. Java项目访问resources文件

    最近在对接支付宝支付的开发,需要取到支付的RSA公钥和私钥.于是把公钥和私钥加到resources文件夹里.但是不知道怎么读到这两个文件,也就是不知道路径怎么写.于是网上搜索了下如何获取工作路径,Sy ...

  2. Markdown 简单指北

    Markdown is intended to be as easy-to-read and easy-to-write as is feasible. Markdown 简介 Markdown是一种 ...

  3. matplotlib 填充颜色

    def huitu_host(nodes,total): x = np.arange(len(nodes)) plt.figure(figsize=(9,5)) plt.xticks(x,nodes) ...

  4. Kettle6.0表输入连接数据库

    kettle6.0表输入我们需要从数据中获取数据的时候,需要和数据库建立连接,简单的说下步骤: 1.双击表输入: 2.点击"新建",输入参数值: 注意:数据库用户名可以通过sql进 ...

  5. MySQL DNS反查导致连接缓慢

    场景 机器A上的一个模块连接机器B上的MySQL,在实验室网络环境下正常:同样A.B两台机器,网络环境切换为与外界隔离的一个小型局域网环境,A上的模块与B上MySQL建立连接非常慢.   环境 SuS ...

  6. HTTP协议Keep-Alive模式详解

    详见:http://blog.yemou.net/article/query/info/tytfjhfascvhzxcytp22 HTTP协议Keep-Alive模式详解 1.什么是Keep-Aliv ...

  7. 垃圾收集器Serial 、Parallel、CMS、G1

    详见:http://blog.yemou.net/article/query/info/tytfjhfascvhzxcyt378 这里介绍4个垃圾收集器,如果进行了错误的选择将会大大的影响程序的性能. ...

  8. 【C++小白成长撸】--(续)双偶数N阶魔阵

    原理: 把双偶数N阶魔阵均分为(N/4)^2个4阶魔阵(4*4) 每个魔阵的对角线都标为"-1",其余位置标为"0" 从第一个位置(a[0][0])从左到右,从 ...

  9. Springboot与Thymeleaf模板引擎整合基础教程(附源码)

    前言 由于在开发My Blog项目时使用了大量的技术整合,针对于部分框架的使用和整合的流程没有做详细的介绍和记录,导致有些朋友用起来有些吃力,因此打算在接下来的时间里做一些基础整合的介绍,当然,可能也 ...

  10. 前端js优化方案(连续更新)

    最近在读<高性能javascript>,在这里记录一下读后的一些感受,顺便加上自己的一些理解,如果有兴趣的话可以关注的我的博客http://www.bloggeng.com/,我会不定期发 ...