主线程wait等待子线程返回对象信息

需求:假设一个线程(JarThread)正在从网络连接(用System.in模拟)中读取一个JAR归档文件。这个归档文件中第一项是清单文件(用‘/’标识结束)。另一个线程(MainThread)可能对这个清单文件的内容感兴趣,即使归档文件的其余部分尚不可用。对清单文件感兴趣的线程会创建一个定制的ManifestFile对象,将这个对象的引用传递给将要读取JAR归档文件的线程,并等待这个对象。读取归档文件的线程首先用流中的项填写ManifestFile,然后通知ManifestFile,再继续读取JAR归档文件的其余部分。当阅读器线程通知ManifestFile时,原来的线程会被唤醒,按其计划处理现在已经完全准备就绪的ManifestFile对象

1.       JarThread

public class JarThread implements Runnable {

    Logger logger = LoggerFactory.getLogger(JarThread.class);

    ManifestFile manifestFile;
    InputStream in;

    public JarThread(ManifestFile m, InputStream in) {
        this.manifestFile = m;
        this.in = in;
    }

    @Override
    public void run() {
        int i;
        //与MainThread获取同一个对象锁
        synchronized (manifestFile) {
            //从流in中读入清单文件
            try {
                while ((i = in.read()) != -1) {
                    //如果字符/ 标志MainThread感兴趣的内容结束
                    if(i==47){
                        break;
                    }
                    manifestFile.add(i);
                }
            } catch (IOException e) {
                e.printStackTrace();
            }
            //释放对象锁
            manifestFile.notifyAll();
        }
        //读取流的其他部分
        try {
            while ((i = in.read()) != -1) {
                logger.info(String.valueOf((char)i));
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

2.      ManifestFile

public class ManifestFile {

    private StringBuffer content = new StringBuffer();

    public synchronized StringBuffer getContent() {
        return content;
    }

    public synchronized void add(int i) {
        content.append((char)(i));
    }
}

3.       MainThread

public class MainThread {
    final static Logger logger = LoggerFactory.getLogger(MainThread.class);
    public static void main(String[] args) throws IOException {
        InputStream in = System.in;
        //system.in如何判断读到了
        ManifestFile m = new ManifestFile();
        //启动读取线程 这里的需求比较特别
        //并不是从线程中返回 而是返回所需要的部分,而线程继续执行
        JarThread t = new JarThread(m, in);
        ExecutorService exec = Executors.newCachedThreadPool();

        //首先获取对象锁
        synchronized (m) {
            exec.execute(t);
            try {
                //挂起 释放锁 等待JarThread通知
                m.wait();
                //处理清单文件
                logger.info(m.getContent().toString());
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

4.       执行结果


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值