JAVA同步synchronized的问题

qq_38941327 2018-05-16 05:22:54
我编写了一个客户端和服务器的程序,其中服务器要实现多线程,为了测试是否真的能“同步”,我在操作中加入了synchronized关键字,并加入了Thread.sleep(1000),看看到底能不能同步:
我同步部分代码基本结构如下:synchronized(this){
打印进入该部分代码的时间;
操作;(包含Thread.sleep(1000))
打印即将离开该部分代码的时间
}
然后我运行了两个客户端,一个让服务器的一个线程修改数据,然后让另一个客户端查询修改后的数据,但是总是查不到(我感觉并没有将这部分代码上锁),之所以打印该时间也是为了验证这个:服务器的时间打印很奇怪:
修改进入时间
2018-05-16 05:19:43
查询进入时间
2018-05-16 05:19:44
修改离开时间
2018-05-16 05:19:43
查询离开时间
2018-05-16 05:19:44
我的问题:既然修改操作在查询前面,虽然时间上修改在查询前面,但是正常的打印顺序不应该是:修改时间,修改时间,查询时间,查询时间吗?而且后来我让线程sleep(10000),这么长的时间(我明显感觉到操作非常的慢),输出的修改和查询仍然是这样的形式,并且结果仍然是错误的(修改后的信息仍然查不到);
另外非常有意思的一件事:我把Thread.sleep(1000)写的位置是删除、修改操作之间(也就是说,修改操作之前必须线程沉睡,而删除操作之后线程才会沉睡,我在先用一个客户端删除操作时,明显能感觉到操作非常快(按钮很快就恢复了,要是选择修改操作很慢按钮才恢复),这之后再查询,果不其然,查询不到已经删除的信息,虽然这个操作是对的我很欣慰,但是怎么删除操作不需要睡一会?就因为它的代码在Thread.sleep(1000)的前面?可是他们明明在一个同步代码块里;
这部分的代码如下:
synchronized(this) {
Date d = new Date();
String s = null;
if(info.get(0).indexOf("修改")!=-1) {
System.out.println("修改进入时间");
DateFormat sdf2 = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss");
s = sdf2.format(d);
System.out.println(s);
}
if(info.get(0).indexOf("查询")!=-1) {
System.out.println("查询进入时间");
DateFormat sdf2 = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss");
s = sdf2.format(d);
System.out.println(s);
}
info1=this.search(info);
if(info.get(0).indexOf("删除")!=-1) {
//操作
}
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
if(info.get(0).indexOf("修改")!=-1) {
//操作
}
if(info.get(0).indexOf("添加")!=-1) {
//操作
}
toFile();//写回文件
if(info.get(0).indexOf("查询")!=-1||info.get(0).indexOf("更新")!=-1||info.get(0).indexOf("无条件")!=-1) {
//操作
}
if(info.get(0).indexOf("修改")!=-1) {
System.out.println("修改离开时间");
DateFormat sdf3 = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss");
s = sdf3.format(d);
System.out.println(s);
}
if(info.get(0).indexOf("查询")!=-1) {
System.out.println("查询离开时间");
DateFormat sdf3 = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss");
s = sdf3.format(d);
System.out.println(s);
}
}
...全文
941 5 打赏 收藏 转发到动态 举报
写回复
用AI写文章
5 条回复
切换为时间正序
请发表友善的回复…
发表回复
淡定的蜗牛 2019-01-11
  • 打赏
  • 举报
回复
看这个就够了,拿走拿走,不客气 http://mp.weixin.qq.com/s?__biz=MzU3MDc5NjU2OA==&mid=2247483762&idx=1&sn=d36886881204870e193ea37888226467&chksm=fce8be05cb9f3713aaacae5d91c0651a0ebf604e0a68bb9e06616f90404ec13aece4034bebde#rd
qq_38941327 2018-06-29
  • 打赏
  • 举报
回复
谢谢楼上,因为我这个使用的多线程不是一个对象,所以用一个this是不好使的,非常感谢
  • 打赏
  • 举报
回复
代码描述太多,就不解释了, 至于你说的synchronized (this.pets)就可以了,其中pets是public static的,所有线程共享一份 那就说明之前错误的原因是this。给this加锁是当前对象加锁,二个线程去执行的时候用的不是同一把锁就不会实现同步。
qq_38941327 2018-05-16
  • 打赏
  • 举报
回复
算了,虽然我这个加锁了和没加没区别,但是我也有办法修改到正确,我就是不明白,为啥加了锁和没加一样,我又设置了同步代码区的打印: Thread-0 get in Thread-1 get in Thread-0 get out Thread-1 get out 明显看出是错误的,线程1在线程0还没有出来的时候进入了该部分代码块; 我稍加修改,改成这样:synchronized (this.pets)就可以了,其中pets是public static的,所有线程共享一份,我不明白为什么之前不行,以下是正确的打印 Thread-1 get in Thread-1 get out Thread-0 get in Thread-0 get out
qq_38941327 2018-05-16
  • 打赏
  • 举报
回复
请大家帮帮忙,虽然只是没有Thread.sleep都是对的(因为正常操作都是对的),但是这样为什么不对?
主要特性Java 语言是简单的:Java 语言的语法与 C 语言和 C++ 语言很接近,使得大多数程序员很容易学习和使用。另一方面,Java 丢弃了 C++ 中很少使用的、很难理解的、令人迷惑的那些特性,如操作符重载、多继承、自动的强制类型转换。特别地,Java 语言不使用指针,而是引用。并提供了自动分配和回收内存空间,使得程序员不必为内存管理而担忧。Java 语言是面向对象的:Java 语言提供类、接口和继承等面向对象的特性,为了简单起见,只支持类之间的单继承,但支持接口之间的多继承,并支持类与接口之间的实现机制(关键字为 implements)。Java 语言全面支持动态绑定,而 C++语言只对虚函数使用动态绑定。总之,Java语言是一个纯的面向对象程序设计语言。Java语言是分布式的:Java 语言支持 Internet 应用的开发,在基本的 Java 应用编程接口中有一个网络应用编程接口(java net),它提供了用于网络应用编程的类库,包括 URL、URLConnection、Socket、ServerSocket 等。Java 的 RMI(远程方法激活)机制也是开发分布式应用的重要手段。Java 语言是健壮的:Java 的强类型机制、异常处理、垃圾的自动收集等是 Java 程序健壮性的重要保证。对指针的丢弃是 Java 的明智选择。Java 的安全检查机制使得 Java 更具健壮性。Java语言是安全的:Java通常被用在网络环境中,为此,Java 提供了一个安全机制以防恶意代码的攻击。除了Java 语言具有的许多安全特性以外,Java 对通过网络下载的类具有一个安全防范机制(类 ClassLoader),如分配不同的名字空间以防替代本地的同名类、字节代码检查,并提供安全管理机制(类 SecurityManager)让 Java 应用设置安全哨兵。Java 语言是体系结构中立的:Java 程序(后缀为 java 的文件)在 Java 平台上被编译为体系结构中立的字节码格式(后缀为 class 的文件),然后可以在实现这个 Java 平台的任何系统中运行。这种途径适合于异构的网络环境和软件的分发。Java 语言是可移植的:这种可移植性来源于体系结构中立性,另外,Java 还严格规定了各个基本数据类型的长度。Java 系统本身也具有很强的可移植性,Java 编译器是用 Java 实现的,Java 的运行环境是用 ANSI C 实现的。Java 语言是解释型的:如前所述,Java 程序在 Java 平台上被编译为字节码格式,然后可以在实现这个 Java 平台的任何系统中运行。在运行时,Java 平台中的 Java 解释器对这些字节码进行解释执行,执行过程中需要的类在联接阶段被载入到运行环境中。Java 是高性能的:与那些解释型的高级脚本语言相比,Java 的确是高性能的。事实上,Java 的运行速度随着 JIT(Just-In-Time)编译器技术的发展越来越接近于 C++。Java 语言是多线程的:在 Java 语言中,线程是一种特殊的对象,它必须由 Thread 类或其子(孙)类来创建。通常有两种方法来创建线程:其一,使用型构为 Thread(Runnable) 的构造子类将一个实现了 Runnable 接口的对象包装成一个线程,其二,从 Thread 类派生出子类并重写 run 方法,使用该子类创建的对象即为线程。值得注意的是 Thread 类已经实现了 Runnable 接口,因此,任何一个线程均有它的 run 方法,而 run 方法中包含了线程所要运行的代码。线程的活动由一组方法来控制。Java 语言支持多个线程的同时执行,并提供多线程之间的同步机制(关键字为 synchronized)。Java 语言是动态的:Java 语言的设计目标之一是适应于动态变化的环境。Java 程序需要的类能够动态地被载入到运行环境,也可以通过网络来载入所需要的类。这也有利于软件的升级。另外,Java 中的类有一个运行时刻的表示,能进行运行时刻的类型检查。

50,523

社区成员

发帖
与我相关
我的任务
社区描述
Java相关技术讨论
javaspring bootspring cloud 技术论坛(原bbs)
社区管理员
  • Java相关社区
  • 小虚竹
  • 谙忆
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧