public class ThreadTest implements Callable<String> { public String call() throws Exception { // TODO Auto-generated method stub wait(10000); return "hello"; } }调用代码:
public static void main(String[] args) { System.out.println("開始启动线程"+Thread.currentThread().getName()); ExecutorService exe = Executors.newCachedThreadPool(); ThreadTest t= new ThreadTest(); Future<String> f = exe.submit(t); Future<String> f2 = exe.submit(t); try { t.notify(); System.out.println(f.get()); System.out.println(f2.get()); System.out.println("主线程"+Thread.currentThread().getName()); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (ExecutionException e) { // TODO Auto-generated catch block e.printStackTrace(); } }
在学习并发是遇到了例如如下错误html
java.lang.IllegalMonitorStateException: current thread not ownerjava
个人代码例如如下:编程
经过查找资料找到缘由即解决之道。并发
《java编程思想》第四版一书中有描写叙述到:“线程操做的wait()、notify()、notifyAll()方法仅仅能在同步控制方法或同步控制块内调用。学习
假设在非同步控制方法或控制块里调用。程序能经过编译,但运行的时候,将获得 IllegalMonitorStateException 异常。并伴随着一些含糊信息,比方 ‘当前线程不是拥有者’。spa
事实上异常的含义是 调用wait()、notify()、notifyAll()的任务在调用这些方法前必须 ‘拥有’(获取)对象的锁。线程
”code
JAVA JDK API文档中也有描写叙述例如如下:htm
wait()、notify()、notifyAll()法仅仅应由做为此对象监视器的所有者的线程来调用。对象
经过下面三种方法之中的一个,线程可以成为此对象监视器的所有者:
- 经过运行此对象的同步 (Sychronized) 实例方法。
- 经过运行在此对象上进行同步的
synchronized
语句的正文。 - 对于
Class
类型的对象,可以经过运行该类的同步静态方法。
依据以上分析,个人代码改动例如如下:
一、call方法前添加synchronizedkeyword,这样调用wait()方法就不会出错了。
二、在t.notify()方法调用时使用同步上下文。改成例如如下:
synchronized (t) {
t.notify();
}
再次运行就正常了。