- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个运行多个服务(长期)和工作(短期)线程的系统。它们都共享一个包含对象的状态。任何线程都可以通过一个名为 ObjectManager
的单例类随时请求一个对象。 .如果对象不可用,则需要加载它。
以下是对象加载现在的一些伪代码:
class ObjectManager {
getLoadinData(path) {
if (hasLoadingDataFor(path))
return whatWeHave()
else {
loadingData = createNewLoadingData();
loadingData.path = path;
pushLoadingTaskToLoadingThread(loadingData);
return loadingData;
}
}
// loads object and blocks until it's loaded
loadObjectSync(path) {
loadingData = getLoadinData(path);
waitFor(loadingData.conditionVar);
return loadingData.loadedObject;
}
// initiates a load and calls a callback when done
loadObjectAsync(path, callback) {
loadingData = getLoadinData(path);
loadingData.callbacks.add(callback);
}
// dedicated loading thread
loadingThread() {
while (running) {
loadingData = waitForLoadingData();
object = readObjectFromDisk(loadingData.path);
object.onLoaded(); // !!!!
loadingData.object = object;
// unblock cv waiters
loadingData.conditionVar.notifyAll();
// call callbacks
loadingData.callbacks.callAll(object);
}
}
}
object.onLoaded
行.我无法控制此功能。一些对象可能决定它们需要其他对象才能有效。所以在他们的
onLoaded
他们可能调用的方法
loadObjectSync
.哦哦!这(自然)死锁。它阻塞加载循环,直到加载循环进行更多迭代。
onLoaded
调用启动线程。这将改变
loadObjectSync
类似于:
loadObjectSync(path) {
loadingData = getLoadinData(path);
waitFor(loadingData.conditionVar);
if (loadingData.wasCreatedInThisThread()) {
object.onLoaded();
loadingData.onLoadedConditionVar.notifyAll();
loadingData.callbacks.callAll(object);
}
else {
// wait more
waitFor(loadingData.onLoadedConditionVar);
}
return loadingData.loadedObject;
}
loadSync
且仅适用于
loadAsync
或者只是
loadAsync
call 是第一个创建加载数据的,不会有任何人来完成对象。因此,为了完成这项工作,我必须引入另一个线程来完成对象,这些对象的 loadingData 是由 loadObjectAsync 创建的。
getLoadingData
反而。如果它这样做怎么办:
getLoadinData(path) {
if (hasLoadingDataFor(path))
return whatWeHave()
else {
loadingData = createNewLoadingData();
loadingData.path = path;
///
thread = spawnLoadingThread(loadingData);
thread.detach();
///
return loadingData;
}
}
readObjectFromDisk
的 I/O 成本相比,它可以忽略不计。
最佳答案
很有意思!这是我遇到过几次的问题,试图将同步接口(interface)添加到由服务线程执行的基本异步操作(即文件加载,或者在我的情况下是网络写入)。
我自己的偏好是不提供同步接口(interface)。为什么?因为它使代码在设计和实现中更简单,更容易推理——对于多线程来说总是很重要。
坚持单线程和仅异步的好处是您只有 1 个服务线程,因此资源增长不是问题,而且用户回调始终在同一线程上调用,这简化了 ObjectManager
用户的线程安全问题(如果您有多个回调线程,则每个用户回调都必须是线程安全的,因此这是一个重要的选择)。然而,只坚持一个异步接口(interface)确实意味着 ObjectManager
的用户还有更多的工作要做。
但是,如果您确实想保留同步接口(interface),那么我采取的另一种方法可能对您有用。您坚持使用单个服务线程,但在 loadObjectSync
的实现中您检查线程 ID 以确定调用者是服务线程还是任何其他线程。如果它是任何其他线程,您将请求排队并安全地阻止。但是如果是服务线程,你可以立即加载对象,比如调用一个新函数loadObjectImpl
.您需要在初始化期间获取服务线程的线程 ID 并将其存储在 ObjectManager
中。实例,并将其用于线程识别。您将需要一个新函数,它基本上只是 loadingThread
的内部范围。函数 -- 即一个名为 loadObjectImpl
的新函数.
关于multithreading - 为每个对象加载生成一个新线程,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58638997/
嘿。本周的一个教程,其中一个问题要求通过使用其他函数 formatLine 和 formatList 创建一个函数 formatLines,以格式化行列表。 我的代码是这样的; type Line =
我是一名优秀的程序员,十分优秀!