1:有一个总任务A,分解为子任务A1 A2 A3 ...,任何一个子任务失败后要快速取消所有任务,请写程序模拟。

「请寻求最优解,不要只是粗暴wait()」

本题解题思路:Fork/Join

通常使用其更专门的类型之一 RecursiveTask(可以返回结果)或 RecursiveAction。

Oracle 官方文档:https://docs.oracle.com/javase/tutorial/essential/concurrency/forkjoin.html

主功能就是将一个大任务拆分成多个小任务进行处理。

处理过程中只要有个小任务失败报错,剩下的任务将可能被立即停止。

以下为代码实现:

1.1:实现代码

public class SonTask extends RecursiveAction {

    private static final Logger logger = LoggerFactory.getLogger(SonTask.class);

    /**
* 总共任务量
**/
private final int taskCount;
/**
* 当前task被分配的任务量
**/
private final int taskMete;
/**
* 当前task序号
**/
private int taskRank;
/**
* 每个task最大可处理任务量
**/
private final int maxTask = 1; public SonTask(int taskCount) {
this.taskCount = taskCount;
this.taskMete = taskCount;
} private SonTask(int taskCount, int taskMete, int taskRank) {
this.taskCount = taskCount;
this.taskMete = taskMete;
this.taskRank = taskRank;
} @Override
protected void compute() {
// 任务分配量是否满足处理条件,不满足则将任务再拆分
if (taskMete == maxTask) {
printSelf();
} else {
List<SonTask> sonTaskList = new ArrayList<>();
for (int i = 1; i <= taskCount; i++) {
sonTaskList.add(new SonTask(taskCount, 1, i));
}
// 执行所有任务
invokeAll(sonTaskList);
}
} /**
* task 1 正常结束 ->
* task 2 执行报错 ->
* task 3 直接终止
**/
private void printSelf() {
logger.info("SON TASK RANK [{}] START", taskRank);
try {
TimeUnit.SECONDS.sleep(taskRank * 3);
if (taskRank == 2) {
logger.error("eroor occured");
throw new RuntimeException("error");
} else {
logger.info("TASK [{}] OVER", taskRank);
}
} catch (InterruptedException e) {
e.printStackTrace();
}
} }

1.2:测试

public class StartMain {

    public static void main(String[] args) {
ForkJoinPool pool = new ForkJoinPool(10); SonTask sonTask = new SonTask(3); pool.invoke(sonTask);
}
}

在task 1结束后由于task 2报错了,task 3被取消执行。

看看ForkJoinTask#invokeAll(Collection tasks) 的源码注释中有这么一句话:

If any task encounters an exception, others may be cancelled.

    /**
* Forks all tasks in the specified collection, returning when
* {@code isDone} holds for each task or an (unchecked) exception
* is encountered, in which case the exception is rethrown. If
* more than one task encounters an exception, then this method
* throws any one of these exceptions. If any task encounters an
* exception, others may be cancelled. However, the execution
* status of individual tasks is not guaranteed upon exceptional
* return. The status of each task may be obtained using {@link
* #getException()} and related methods to check if they have been
* cancelled, completed normally or exceptionally, or left
* unprocessed.
*
* @param tasks the collection of tasks
* @param <T> the type of the values returned from the tasks
* @return the tasks argument, to simplify usage
* @throws NullPointerException if tasks or any element are null
*/
public static <T extends ForkJoinTask<?>> Collection<T> invokeAll(Collection<T> tasks) {
...
}

2:请用两个线程交替输出A1B2C3D4...,A线程输出字母,B线程输出数字,要求A线程首先执行,B线程其次执行!(多种同步机制的运用)

「请寻求最优解,不要简单的synchronized」

本题解题思路:ReentrantLock、Condtional

1:利用Conditon#await、Condition#signal 进行线程之间的通信,替代Object#wait、Object#notify。

2:勿使用Thread#join 这种阻塞主线程的方式,也达不到该题的需求。

Condition的类注释:

 /**
* {@code Condition} factors out the {@code Object} monitor
* methods ({@link Object#wait() wait}, {@link Object#notify notify}
* and {@link Object#notifyAll notifyAll}) into distinct objects to
* give the effect of having multiple wait-sets per object, by
* combining them with the use of arbitrary {@link Lock} implementations.
* Where a {@code Lock} replaces the use of {@code synchronized} methods
* and statements, a {@code Condition} replaces the use of the Object
* monitor methods.
* ...
*/

2.1:实现代码

public class StartMain {

    private static final Logger logger = LoggerFactory.getLogger(StartMain.class);
private static final ReentrantLock lock = new ReentrantLock();
private static final String[] arr = new String[]{"A1", "B2", "C3", "D4"};
private static final AtomicInteger index = new AtomicInteger(0); public static void main(String[] args) {
Condition conditionA = lock.newCondition();
Condition conditionB = lock.newCondition(); Thread threadA = new Thread(() -> {
while (index.get() < arr.length) {
try {
lock.lock();
logger.info(arr[index.get()]);
index.incrementAndGet();
conditionB.signal();
conditionA.await();
} catch (Exception e) {
e.printStackTrace();
} finally {
lock.unlock();
}
}
}, "thread-A"); Thread threadB = new Thread(() -> {
while (index.get() < arr.length) {
try {
lock.lock();
conditionB.await();
logger.info(arr[index.get()]);
index.incrementAndGet();
conditionA.signal();
} catch (Exception e) {
e.printStackTrace();
} finally {
lock.unlock();
}
}
}, "thread-B"); threadB.start(); // 为了使测试更加逼真,先让B开始
try {
TimeUnit.SECONDS.sleep(2);
} catch (InterruptedException e) {
e.printStackTrace();
} threadA.start();
} }

2.2:Condition#await

    /**
* Causes the current thread to wait until it is signalled or
* {@linkplain Thread#interrupt interrupted}.
*
* <p>The lock associated with this {@code Condition} is atomically
* released and the current thread becomes disabled for thread scheduling
* purposes and lies dormant until <em>one</em> of four things happens:
*
* ...
* @throws InterruptedException if the current thread is interrupted
* (and interruption of thread suspension is supported)
*/
void await() throws InterruptedException;

**The lock associated with this {@code Condition} is atomically released and the current thread becomes disabled **

for thread scheduling purposes and lies dormant until ...

调用了Condition.await()方法后该线程所持有的Lock锁会被释放掉,并且当前线程会变得不可用(阻塞),直到调用了Condtion.signal()方法。

3:华为面试题

「请寻求最优解,不要简单的生产者-消费者模式」

有一个生产奶酪的厂家,每天需要生产100000份奶酪卖给超市,通过一辆货车发货,送货车每次送100份。

厂家有一个容量为1000份的冷库,用于奶酪保鲜,生产的奶酪需要先存放在冷库,运输车辆从冷库取货。

厂家有三条生产线,分别是牛奶供应生产线,发酵剂制作生产线,奶酪生产线。

生产每份奶酪需要2份牛奶和一份发酵剂。

请设计生产系统?

本题解题思路: BlockingDeque阻塞队列、Atomic原子类

Oracle 官方文档:https://docs.oracle.com/javase/7/docs/api/java/util/concurrent/BlockingDeque.html

3.1:api注释

1:BlockingDeque#take()

    /**
* Retrieves and removes the head of the queue represented by this deque
* (in other words, the first element of this deque), waiting if
* necessary until an element becomes available.
*
* <p>This method is equivalent to {@link #takeFirst() takeFirst}.
*
* @return the head of this deque
* @throws InterruptedException if interrupted while waiting
*/
E take() throws InterruptedException;

该方法若从队列中取不到元素会造成当前线程阻塞,直到拿到元素为止。

2:BlockingDeque#put(E e)

    /**
* Inserts the specified element into the queue represented by this deque
* (in other words, at the tail of this deque), waiting if necessary for
* space to become available.
*
* <p>This method is equivalent to {@link #putLast(Object) putLast}.
*
* @param e the element to add
* @throws InterruptedException {@inheritDoc}
* @throws ClassCastException if the class of the specified element
* prevents it from being added to this deque
* @throws NullPointerException if the specified element is null
* @throws IllegalArgumentException if some property of the specified
* element prevents it from being added to this deque
*/
void put(E e) throws InterruptedException;

当队列容量达到上限时,其他元素无法入队且使当前线程阻塞,直到队有可用空间为止。

3.2:实现代码

public class ProductOnlineBus {

    private static final Logger logger = LoggerFactory.getLogger(ProductOnlineBus.class);
/**
* 生产奶酪数量
**/
private final int prodNum;
/**
* 牛奶=奶酪*2
**/
private final int milkMultiple = 2;
/**
* 发酵剂=奶酪*1
**/
private final int fjjMultiple = 1;
/**
* 奶酪仓库容量
**/
private final int cheeseCapacity = 1000;
/**
* 单词运输奶酪数量
**/
private final int truckCapacity = 100;
/**
* 总共需要运输多少次
**/
private final int needTruckTimes;
/**
* 生产线--阻塞队列
**/
private final BlockingDeque<MiikNode> milkNodeBlockingDeque;
private final BlockingDeque<FJJNode> fjjNodeBlockingDeque;
private final BlockingDeque<CheeseNode> cheeseNodeBlockingDeque;
/**
* 实际运输次数
**/
private final AtomicInteger trucked = new AtomicInteger(0);
/**
* 各生产线生产次数
**/
private final AtomicInteger milkProded = new AtomicInteger(0);
private final AtomicInteger fjjProded = new AtomicInteger(0);
private final AtomicInteger cheeseProded = new AtomicInteger(0); public ProductOnlineBus(int prodNum) {
if ((prodNum % truckCapacity) != 0) {
throw new RuntimeException("请输入truckCapacity的倍数");
}
this.prodNum = prodNum;
this.milkNodeBlockingDeque = new LinkedBlockingDeque<>(milkMultiple);
this.fjjNodeBlockingDeque = new LinkedBlockingDeque<>(fjjMultiple);
this.cheeseNodeBlockingDeque = new LinkedBlockingDeque<>(cheeseCapacity);
this.needTruckTimes = prodNum / truckCapacity;
} public void starter() {
new Thread(() -> {
int len = prodNum * milkMultiple;
for (int i = 0; i < len; i++) {
try {
milkNodeBlockingDeque.put(new MiikNode(i));
milkProded.incrementAndGet();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}, "MilkThread").start(); new Thread(() -> {
int len = prodNum * fjjMultiple;
for (int i = 0; i < len; i++) {
try {
fjjNodeBlockingDeque.put(new FJJNode(i));
fjjProded.incrementAndGet();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}, "FJJThread").start(); new Thread(() -> {
for (int i = 0; i < prodNum; i++) {
try {
for (int j = 0; j < milkMultiple; j++) {
milkNodeBlockingDeque.take();
}
for (int j = 0; j < fjjMultiple; j++) {
fjjNodeBlockingDeque.take();
}
cheeseNodeBlockingDeque.put(new CheeseNode(i));
cheeseProded.incrementAndGet();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}, "CheeseThread").start(); new Thread(() -> {
while (trucked.get() < needTruckTimes) {
try {
for (int i = 0; i < truckCapacity; i++) {
cheeseNodeBlockingDeque.take();
}
trucked.incrementAndGet();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
logger.info("over of->cheese:[{}],milk:[{}],fjj[{}],truck:[{}]",
cheeseProded.get(), milkProded.get(), fjjProded.get(), trucked.get());
}, "TruckThread").start();
} /**
* 牛奶
**/
private class MiikNode {
public MiikNode(int seq) {
logger.info("生产牛奶[{}]...", seq);
}
} /**
* 发酵剂
**/
private class FJJNode {
public FJJNode(int seq) {
logger.info("生产发酵剂[{}]...", seq);
}
} /**
* 奶酪
**/
private class CheeseNode {
public CheeseNode(int seq) {
logger.info("生产奶酪[{}]...", seq);
}
} }

3.3:运行

public class StartMain {

     public static void main(String[] args) {
ProductOnlineBus pb = new ProductOnlineBus(100000);
pb.starter();
}
}

答不上的JUC笔试题的更多相关文章

  1. 数字IC笔试题-芯源

    前言 由于最近开始找数字IC的工作,所以准备多练笔试题,下面贴上芯源笔试题,来源微信公众号<数字IC打工人> 参考资源: 1. mu_guang_ 2.  李锐博恩 3. 长弓的坚持 4. ...

  2. [笔试题]黑板上写下50个数字,选两个黑板上数字a和b,在黑板写|b-a|,剩下的数字?

    在黑板上写下50个数字:1至50.在接下来的49轮操作中,每次做如下操作:选取两个黑板上的数字a和b,擦去,在黑板上写|b-a|.请问最后一次动作之后剩下的数字可能是什么?为什么?(不用写代码,不写原 ...

  3. 北京Java笔试题整理

    北京Java笔试题整理 1.什么是java虚拟机?为什么ava被称作是"平台无关的编程语言? 答:Java虚拟机可以理解为一个特殊的"操作系统",只是它连接的不是硬件,而 ...

  4. C/C++ 笔试题

    /////转自http://blog.csdn.net/suxinpingtao51/article/details/8015147#userconsent# 微软亚洲技术中心的面试题!!! 1.进程 ...

  5. oracle 笔试题

    ORACLE笔试题一.单选题1.在Oracle中,以下不属于集合操作符的是( ). A. UNION B. SUM C. MINUS D. INTERSECT2.在Oracle中,执行下面的语句:SE ...

  6. NET出现频率非常高的笔试题

    又到了金三银四的跳槽季,许多朋友又开始跳槽了,这里我简单整理了一些出现频率比较高的.NET笔试题,希望对广大求职者有所帮助. 一..net基础 1.  a=10,b=15,请在不使用第三方变量的情况下 ...

  7. PHP笔试题(转载)

    整理了一份PHP高级工程师的笔试题,问题很全面.嗯,基本上这些题都答得不错,那么你应该可以胜任大部分互联网企业的PHP职位了.下面直接上题. 1. 基本知识点 HTTP协议中几个状态码的含义:503, ...

  8. 收藏所用C#技术类面试、笔试题汇总

    技术类面试.笔试题汇总 注:标明*的问题属于选择性掌握的内容,能掌握更好,没掌握也没关系. 下面的参考解答只是帮助大家理解,不用背,面试题.笔试题千变万化,不要梦想着把题覆盖了,下面的题是供大家查漏补 ...

  9. Android开发面试经——4.常见Android进阶笔试题(更新中...)

      Android开发(29)  版权声明:本文为寻梦-finddreams原创文章,请关注:http://blog.csdn.net/finddreams 关注finddreams博客:http:/ ...

随机推荐

  1. 【疯狂挖坑】linux服务器尝试中的问题(nohup等)

    实验环境:Ubuntu 16.0.4 首当其冲: 1. nohup使用及究极重要事项 我们用服务器肯定都是想在服务器存点什么.让服务器帮我们干点什么,这时候就需要nohup(no hang up)提供 ...

  2. 在Android用vulkan完成蓝绿幕扣像

    效果图(1080P处理) 因为摄像头开启自动曝光,画面变动时,亮度变化导致扣像在转动时如上. 源码地址vulkan_extratest 这个demo主要测试二点,一是测试ndk camera集成效果, ...

  3. Hadoop----hdfs dfs常用命令的使用

    用法    -mkdir    创建目录    Usage:hdfs dfs -mkdir [-p] < paths>    选项:-p    很像Unix mkdir -p,沿路径创建父 ...

  4. 掌握数位dp

    最近遇到了数位dp题目,于是就屁颠屁颠的跑过来学习数位dp了~ "在信息学竞赛中,有这样一类问题:求给定区间中,满足给定条件的某个D 进制数或此类数的数量.所求的限定条件往往与数位有关,例如 ...

  5. zjnu1730 PIRAMIDA(字符串,模拟)

    Description Sample Input 6 JANJETINA 5 1 J 1 A 6 N 6 I 5 E Sample Output 1 0 2 1 1 题意:给你一个长度小于等于10^6 ...

  6. Codeforces Round #646 (Div. 2) C、Game On Leaves

    题目链接:C.Game On Leaves 题意: 给你一个n个节点的无根树,你每次可以删除一个叶节点.如果谁先删除x号节点谁就赢了.两个人轮流操作 题解: 如果x号节点本身就是一个叶节点,那么谁先走 ...

  7. Codeforces Round #614 (Div. 1) A. NEKO's Maze Game (思维,模拟)

    题意:有一个\(2\)X\(n\)的矩阵,你想从\((1,1)\)走到\((2,n)\),每次可以向上下左右四个方向走,但在某些时间段某个点会被堵住,如果已经被堵住,那么即恢复正常,每次对某个点操作, ...

  8. Codeforces Round #649 (Div. 2)

    Codeforces Round #649 (Div. 2) -- WKL \(\mathcal{A}\)题: \(\mathrm{XXXXX}\) Greedy implementation *12 ...

  9. Codeforces Round #649 (Div. 2) A. XXXXX (贪心)

    题意:有一个长度为\(n\)的数组,找一段最长子数组,使得其元素和为\(x\),如果存在,输出子数组的长度,否则输出\(-1\). 题解:这题我们要从元素和\(sum\)来考虑,首先,如果原数组的所有 ...

  10. history-back

    ;!function(pkg, undefined){ //此声明函数在xback.js文件里有,在app.js里必须再声明一次,不然监听返回事件失败 var STATE = 'x-back'; va ...