java类属性线程安全_1.java面试问题 之 基础类型属性(int)能否线程安全?

脑海第一感觉 static int 公告的属性肯定是非线程安全的。int直接公告的属性难道也是非线程安全吗?(疑问)。

通过题面意思就能感觉到面试官的用意,他就是想让你说是非线程安全的。而后他好问为什么。结果我直接说不知道。说实话真拿不准,于是自己通过实践验证得出了少量结论并记录下来。加申印象。private static int value = 1;private int value = 1;以下想通过实践证实几点:

1.两种公告方式能否线程安全。

2.总结两种方式的区别。

第一两种公告方式能否线程安全。

证实1:private static int value = 1; 非线程安全/** * 证实static int 公告属性为非线程安全的类 */class TTT { static int value = 1; //注释1:Integer value = new Integer(1) 同样 public int get1() throws InterruptedException { Thread.sleep(10); //注释2:值越大重复值越多 return value++; } }/** * 测试类 */public class Test2 { public static void main(String[] args) { TTT t = new TTT(); //注释3:实例化一个对象,并通过多个线程调用 get1 方法。 for(int i=1; i<=1000; i++) { new Thread(new Runnable() { @Override public void run() { try { System.out.println(t.get1()); } catch (Exception e) { } } }).start(); } }}期望结果:1 - 1000

实际结果:1 - x (<1000) 实际输出结果中存在重复值

将已上代码片段稍作调整再次验证。

注释3处,实例化对象挪到线程run方法体内/** * 证实static int 公告属性为非线程安全的类 */class TTT { static int value = 1; //注释1:Integer value = new Integer(1) 同样 public int get1() throws InterruptedException { Thread.sleep(10); //注释2:值越大重复值越多 return value++; } }/** * 测试类 */public class Test2 { public static void main(String[] args) { for(int i=1; i<=1000; i++) { new Thread(new Runnable() { @Override public void run() { TTT t = new TTT(); //注释3:实例化1000个对象,并调用 get1 方法。 try { System.out.println(t.get1()); } catch (Exception e) { } } }).start(); } } }期望结果:1 - 1000

实际结果:1 - x (<1000) 实际输出结果中存在重复值

结论:已上两种情况相同针对 private static int value = 1; 都是非线程安全的。

那么都知道通过synchronized关键字可以将get1方法改为线程安全的。分别在两段代码片段中的get1方法加上synchronized关键字,但是结果却又不同了第一段代码 实例化一个对象,并通过1000个线程调用 get1 方法,synchronized关键字起作用的。

第二段代码 通过1000个线程实例化1000个对象,并调用 get1 方法,synchronized关键字不起作用。

补充:synchronized关键字在多线程情况下针对同一个实例(对象Object)是起作用的。

证实2:private int value = 1; 非线程安全/** * 证实 int 公告属性为非线程安全的类 */class TT { private int value = 1; //Integer value = new Integer(1) 同样 public int get1() throws Exception { Thread.sleep(10); //值越大重复值越多 return value++; } }/** * 测试类 */public class Test { public static void main(String[] args) throws Exception { TT t = new TT(); //注释3:实例化一个对象,并通过多个线程调用 get1 方法。 for(int i=1; i<=1000; i++) { new Thread(new Runnable() { @Override public void run() { try { System.out.println(t.get1()); } catch (Exception e) { e.printStackTrace(); } } }).start(); } } }期望结果:1 - 1000

实际结果:1 - x (<1000) 实际输出结果中存在重复值

同样将已上代码片段稍作调整再次验证。

注释3处,实例化对象挪到线程run方法体内/** * 证实 int 公告属性为非线程安全的类 */class TT { private int value = 1; //Integer value = new Integer(1) 同样 public int get1() throws Exception { Thread.sleep(10); //值越大重复值越多 return value++; } }/** * 测试类 */public class Test { public static void main(String[] args) throws Exception { for(int i=1; i<=1000; i++) { new Thread(new Runnable() { @Override public void run() { TT t = new TT(); //注释3:实例化1000个对象,并调用 get1 方法。 try { System.out.println(t.get1()); } catch (Exception e) { e.printStackTrace(); } } }).start(); } } }实际结果:输出的一律是 1

结论:针对 private int value = 1;第一段代码 实例化一个对象,并通过1000个线程调用 get1 方法,value值非线程安全。

第二段代码 通过1000个线程实例化1000个对象,并调用 get1 方法,value值线程安全。

补充:在多线程情况下针对同一个实例(对象Object)内的基础类型公告的属性 进行调用是非线程安全的。

总结两种方式的区别。静态属性相对于类(class)是非线程安全的。如上结论无论实例化一个对象,并通过多线程调用方法。还是通过多线程实例化多个对象,调用方法结果是一样的。

一般属性相对于对象(object)是非线程安全的。如上结论,实例化一个对象,并通过多个线程调用方法获取属性值,值是不可靠的。而实通过线程实例化多个对象,并调用方法获取属性值,值是可靠的。

具体理论可参考jvm实战第二章内存管理。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
面试题包括以下十九部分:Java 基础、容器、多线程、反射、对象拷贝、Java Web 模块、异常、网络、设计模式、Spring/Spring MVC、Spring Boot/Spring Cloud、Hibernate、Mybatis、RabbitMQ、Kafka、Zookeeper、MySql、Redis、JVM 。 目录: 一、Java 基础 1.JDK 和 JRE 有什么区别? 2.== 和 equals 的区别是什么? 3.两个对象的 hashCode()相同,则 equals()也一定为 true,对吗? 4.final 在 java 中有什么作用? 5.java 中的 Math.round(-1.5) 等于多少? 6.String 属于基础的数据类型吗? 7.java 中操作字符串都有哪些?它们之间有什么区别? 8.String str="i"与 String str=new String(“i”)一样吗? 9.如何将字符串反转? 10.String 的常用方法都有那些? 11.抽象必须要有抽象方法吗? 12.普通和抽象有哪些区别? 13.抽象能使用 final 修饰吗? 14.接口和抽象有什么区别? 15.java 中 IO 流分为几种? 16.BIO、NIO、AIO 有什么区别? 17.Files的常用方法都有哪些? 二、容器 18.java 容器都有哪些? 19.Collection 和 Collections 有什么区别? 20.List、Set、Map 之间的区别是什么? 21.HashMap 和 Hashtable 有什么区别? 22.如何决定使用 HashMap 还是 TreeMap? 23.说一下 HashMap 的实现原理? 24.说一下 HashSet 的实现原理? 25.ArrayList 和 LinkedList 的区别是什么? 26.如何实现数组和 List 之间的转换? 27.ArrayList 和 Vector 的区别是什么? 28.Array 和 ArrayList 有何区别? 29.在 Queue 中 poll()和 remove()有什么区别? 30.哪些集合线程安全的? 31.迭代器 Iterator 是什么? 32.Iterator 怎么使用?有什么特点? 33.Iterator 和 ListIterator 有什么区别? 34.怎么确保一个集合不能被修改? 三、多线程 35.并行和并发有什么区别? 36.线程和进程的区别? 37.守护线程是什么? 38.创建线程有哪几种方式? 39.说一下 runnable 和 callable 有什么区别? 40.线程有哪些状态? 41.sleep() 和 wait() 有什么区别? 42.notify()和 notifyAll()有什么区别? 43.线程的 run()和 start()有什么区别? 44.创建线程池有哪几种方式? 45.线程池都有哪些状态? 46.线程池中 submit()和 execute()方法有什么区别? 47.在 java 程序中怎么保证多线程的运行安全? 48.多线程锁的升级原理是什么? 49.什么是死锁? 50.怎么防止死锁? 51.ThreadLocal 是什么?有哪些使用场景? 52.说一下 synchronized 底层实现原理? 53.synchronized 和 volatile 的区别是什么? 54.synchronized 和 Lock 有什么区别? 55.synchronized 和 ReentrantLock 区别是什么? 56.说一下 atomic 的原理? 四、反射 57.什么是反射? 58.什么是 java 序列化?什么情况下需要序列化? 59.动态代理是什么?有哪些应用? 60.怎么实现动态代理? 五、对象拷贝 61.为什么要使用克隆? 62.如何实现对象克隆? 63.深拷贝和浅拷贝区别是什么? 六、Java Web 64.jsp 和 servlet 有什么区别? 65.jsp 有哪些内置对象?作用分别是什么? 66.说一下 jsp 的 4 种作用域? 67.session 和 cookie 有什么区别? 68.说一下 session 的工作原理? 69.如果客户端禁止 cookie 能实现 session 还能用吗? 70.spring mvc 和 struts 的区别是什么? 71.如何避免 sql 注入? 72.什么是 XSS 攻击,如何避免? 73.什么是 CSRF 攻击,如何避免? 七、异常 74.throw 和 throws 的区别? 75.final、finally、finalize 有什么区别? 76.try-catch-finally 中哪个部分

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值