- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
TLDR:我想知道从后台线程使用时如何影响基于运行循环的UndoManager
自动撤消分组,而我的最佳选择是什么。
我在具有iOS和macOS目标的自定义Swift框架中使用UndoManager
(以前称为NSUndoManager
)。
在该框架内,后台GCD串行队列上进行了大量工作。我知道UndoManager
在每个运行循环周期自动将顶级已注册的撤消操作分组,但是我不确定不同的线程情况将如何影响这种情况。
我的问题:
UndoManager
的已注册撤消操作的运行循环分组产生什么影响(如果有)? methodCausingUndoRegistration()
和
anotherMethodCausingUndoRegistration()
都不花哨,并从调用它们的线程中调用
UndoManager.registerUndo
而不进行任何调度。
// Assume this runs on main thread
methodCausingUndoRegistration()
// Other code here
anotherMethodCausingUndoRegistration()
// Also assume every other undo registration in this framework takes place inline on the main thread
UndoManager
。上面的两个撤消注册都将在相同的运行循环周期中进行,因此将被放置在同一撤消组中。
// Assume this runs on an arbitrary background thread, possibly managed by GCD.
// It is guaranteed not to run on the main thread to prevent deadlock.
DispatchQueue.main.sync {
methodCausingUndoRegistration()
}
// Other code here
DispatchQueue.main.sync {
anotherMethodCausingUndoRegistration()
}
// Also assume every other undo registration in this framework takes place
// by syncing on main thread first as above
// Assume this runs from an unknown context. Might be the main thread, might not.
DispatchQueue.main.async {
methodCausingUndoRegistration()
}
// Other code here
DispatchQueue.main.async {
anotherMethodCausingUndoRegistration()
}
// Also assume every other undo registration in this framework takes place
// by asyncing on the main thread first as above
// Assume this runs from an unknown context. Might be the main thread, might not.
backgroundSerialDispatchQueue.async {
methodCausingUndoRegistration()
// Other code here
anotherMethodCausingUndoRegistration()
}
// Also assume all other undo registrations take place
// via async on this same queue, and that undo operations
// that ought to be grouped together would be registered
// within the same async block.
UndoManager
仅在同一后台队列中使用即可。但是,我担心可能会有一些因素使分组未定义,特别是因为我认为GCD队列(或其托管线程)并不总是(如果有的话)会得到运行循环。
最佳答案
TLDR:从后台线程使用UndoManager
时,最简单的选择是简单地禁用通过groupsByEvent
进行自动分组并手动进行。上述情况均无法按预期工作。如果您确实要在后台进行自动分组,则需要避免使用GCD。
我将添加一些背景来解释期望,然后根据我在Xcode Playground中所做的实验,讨论每种情况下实际发生的情况。
自动撤消分组
Apple的Cocoa Application Competencies for iOS指南的“撤消管理器”一章指出:
NSUndoManager通常在运行循环的周期内自动创建撤消组。第一次要求在周期中记录撤消操作时,它将创建一个新组。然后,在循环结束时,它将关闭组。您可以创建其他嵌套的撤消组。
通过将自己注册为NotificationCenter
和NSUndoManagerDidOpenUndoGroup
的观察者NSUndoManagerDidCloseUndoGroup
,可以轻松在项目或Playground中观察到此行为。通过观察这些通知并将结果打印到包括undoManager.levelsOfUndo
的控制台,我们可以实时准确地看到分组的情况。
该指南还指出:
撤消管理器收集在运行循环(例如应用程序的主事件循环)的单个周期内发生的所有撤消操作。
这种语言将表明UndoManager
是唯一可以观察到的主运行循环。因此,很可能UndoManager
观察到代表CFRunLoop
实例发送的通知,该实例在记录第一个撤消操作并打开组时是当前的。
GCD和运行循环
即使Apple平台上运行循环的一般规则是“每个线程一个运行循环”,但该规则也有例外。具体来说,通常公认的是,Grand Central Dispatch不会总是(如果有的话)将标准CFRunLoop
与它的调度队列或其关联线程一起使用。实际上,似乎唯一具有相关联的CFRunLoop
的调度队列似乎是主队列。
苹果的Concurrency Programming Guide指出:
主调度队列是全局可用的串行队列,可在应用程序的主线程上执行任务。此队列与应用程序的运行循环(如果存在)一起工作,以使排队任务的执行与附加到运行循环的其他事件源的执行交织在一起。
主应用程序线程不一定总会有一个运行循环(例如命令行工具)是有道理的,但是如果有,似乎可以保证GCD将与运行循环协调。对于其他调度队列似乎不存在这种保证,并且似乎没有任何 public API或文档记录的方式将任意调度队列(或其基础线程之一)与CFRunLoop
关联。
通过使用以下代码可以观察到这一点:
DispatchQueue.main.async {
print("Main", RunLoop.current.currentMode)
}
DispatchQueue.global().async {
print("Global", RunLoop.current.currentMode)
}
DispatchQueue(label: "").async {
print("Custom", RunLoop.current.currentMode)
}
// Outputs:
// Custom nil
// Global nil
// Main Optional(__C.RunLoopMode(_rawValue: kCFRunLoopDefaultMode))
RunLoop.currentMode
的文档指出:
CFRunLoop
(这是
RunLoop
背后的基础机制)。因此,除非我们要调度到主队列,否则
UndoManager
将没有 Activity 的
RunLoop
可供观察。这对于情况4及以后的情况非常重要。
PlaygroundPage.current.needsIndefiniteExecution = true
)和通知观察机制来观察每种情况。
UndoManager
期望使用的方式(基于文档)。观察撤消通知显示,正在创建一个撤消组,其中包含两个撤消。
async
时,一个简单的测试揭示了与情况1相同的行为。似乎是因为两个块都在有机会被run循环实际运行之前被分派到了主线程,所以run循环同时执行同一周期中的块。因此,两个撤消注册都位于同一组中。
sync
和
async
中引入了细微的差别。因为
sync
会阻塞当前线程直到完成,所以运行循环必须在返回之前开始(和结束)一个循环。当然,然后,运行循环将无法在同一周期中运行另一个块,因为在运行循环开始并查找消息时,它们不会在那儿。但是,对于
async
,直到两个块都已排队后,运行循环才可能发生,因为
async
在工作完成之前就返回了。
sleep(1)
调用之间插入
async
调用来模拟情况3内的情况2。这样,运行循环就有机会在发送第二个块之前开始其循环。实际上,这将导致创建两个撤消组。
backgroundSerialDispatchQueue
是GCD自定义串行队列,则在第一次撤销注册之前立即创建一个撤销组,但是永远不会关闭它。如果我们考虑上面关于GCD和运行循环的讨论,这是有道理的。仅仅由于我们调用了
registerUndo
而创建了撤消组,所以还没有顶级组。但是,它从未关闭过,因为它从未收到有关运行循环结束其循环的通知。它从未收到该通知,因为后台GCD队列未获得与之关联的功能性
CFRunLoop
,因此
UndoManager
很可能甚至从未能够观察到运行循环。
UndoManager
,则以上两种情况都不是理想的(第一种情况除外,这不满足在后台触发的要求)。似乎有两个选择可行。两者都假定
UndoManager
将仅在相同的后台队列/线程中使用。毕竟,
UndoManager
不是线程安全的。
undoManager.groupsByEvent
轻松关闭。然后可以像这样实现手动分组:
undoManager.groupsByEvent = false
backgroundSerialDispatchQueue.async {
undoManager.beginUndoGrouping() // <--
methodCausingUndoRegistration()
// Other code here
anotherMethodCausingUndoRegistration()
undoManager.endUndoGrouping() // <--
}
UndoManager
的行为时确实找到了一种替代方法。
UndoManager
无法观察自定义GCD队列,因为它们似乎没有关联的
CFRunLoop
。但是,如果我们创建了自己的
Thread
并设置了相应的
RunLoop
,该怎么办。从理论上讲,这应该起作用,并且下面的代码演示:
// Subclass NSObject so we can use performSelector to send a block to the thread
class Worker: NSObject {
let backgroundThread: Thread
let undoManager: UndoManager
override init() {
self.undoManager = UndoManager()
// Create a Thread to run a block
self.backgroundThread = Thread {
// We need to attach the run loop to at least one source so it has a reason to run.
// This is just a dummy Mach Port
NSMachPort().schedule(in: RunLoop.current, forMode: .commonModes) // Should be added for common or default mode
// This will keep our thread running because this call won't return
RunLoop.current.run()
}
super.init()
// Start the thread running
backgroundThread.start()
// Observe undo groups
registerForNotifications()
}
func registerForNotifications() {
NotificationCenter.default.addObserver(forName: Notification.Name.NSUndoManagerDidOpenUndoGroup, object: undoManager, queue: nil) { _ in
print("opening group at level \(self.undoManager.levelsOfUndo)")
}
NotificationCenter.default.addObserver(forName: Notification.Name.NSUndoManagerDidCloseUndoGroup, object: undoManager, queue: nil) { _ in
print("closing group at level \(self.undoManager.levelsOfUndo)")
}
}
func doWorkInBackground() {
perform(#selector(Worker.doWork), on: backgroundThread, with: nil, waitUntilDone: false)
}
// This function needs to be visible to the Objc runtime
@objc func doWork() {
registerUndo()
print("working on other things...")
sleep(1)
print("working on other things...")
print("working on other things...")
registerUndo()
}
func registerUndo() {
let target = Target()
print("registering undo")
undoManager.registerUndo(withTarget: target) { _ in }
}
class Target {}
}
let worker = Worker()
worker.doWorkInBackground()
UndoManager
之所以能够观察到周期,是因为与GCD不同,
Thread
使用的是
RunLoop
。
关于ios - UndoManager运行循环分组将如何在不同的线程上下文中受到影响?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47988403/
我将 Bootstrap 与 css 和 java 脚本结合使用。在不影响前端代码的情况下,我真的很难在css中绘制这个背景。在许多问题中,人们将宽度和高度设置为 0%。但是由于我的导航栏,我不能使用
我正在用 c 编写一个程序来读取文件的内容。代码如下: #include void main() { char line[90]; while(scanf("%79[^\
我想使用 javascript 获取矩阵数组的所有对 Angular 线。假设输入输出如下: input = [ [1,2,3], [4,5,6], [7,8,9], ] output =
可以用pdfmake绘制lines,circles和other shapes吗?如果是,是否有documentation或样本?我想用jsPDF替换pdfmake。 最佳答案 是的,有可能。 pdfm
我有一个小svg小部件,其目的是显示角度列表(参见图片)。 现在,角度是线元素,仅具有笔触,没有填充。但是现在我想使用一种“内部填充”颜色和一种“笔触/边框”颜色。我猜想line元素不能解决这个问题,
我正在为带有三角对象的 3D 场景编写一个非常基本的光线转换器,一切都工作正常,直到我决定尝试从场景原点 (0/0/0) 以外的点转换光线。 但是,当我将光线原点更改为 (0/1/0) 时,相交测试突
这个问题已经有答案了: Why do people write "#!/usr/bin/env python" on the first line of a Python script? (22 个回
如何使用大约 50 个星号 * 并使用 for 循环绘制一条水平线?当我尝试这样做时,结果是垂直(而不是水平)列出 50 个星号。 public void drawAstline() { f
这是一个让球以对角线方式下降的 UI,但球保持静止;线程似乎无法正常工作。你能告诉我如何让球移动吗? 请下载一个球并更改目录,以便程序可以找到您的球的分配位置。没有必要下载足球场,但如果您愿意,也可以
我在我的一个项目中使用 Jmeter 和 Ant,当我们生成报告时,它会在报告中显示 URL、#Samples、失败、成功率、平均时间、最短时间、最长时间。 我也想在报告中包含 90% 的时间线。 现
我有一个不寻常的问题,希望有人能帮助我。我想用 Canvas (android) 画一条 Swing 或波浪线,但我不知道该怎么做。它将成为蝌蚪的尾部,所以理想情况下我希望它的形状更像三角形,一端更大
这个问题已经有答案了: Checking Collision of Shapes with JavaFX (1 个回答) 已关闭 8 年前。 我正在使用 JavaFx 8 库。 我的任务很简单:我想检
如何按编号的百分比拆分文件。行数? 假设我想将我的文件分成 3 个部分(60%/20%/20% 部分),我可以手动执行此操作,-_-: $ wc -l brown.txt 57339 brown.tx
我正在努力实现这样的目标: 但这就是我设法做到的。 你能帮我实现预期的结果吗? 更新: 如果我删除 bootstrap.css 依赖项,问题就会消失。我怎样才能让它与 Bootstrap 一起工作?
我目前正在构建一个网站,但遇到了 transform: scale 的问题。我有一个按钮,当用户将鼠标悬停在它上面时,会发生两件事: 背景以对 Angular 线“扫过” 按钮标签颜色改变 按钮稍微变
我需要使用直线和仿射变换绘制大量数据点的图形(缩放图形以适合 View )。 目前,我正在使用 NSBezierPath,但我认为它效率很低(因为点在绘制之前被复制到贝塞尔路径)。通过将我的数据切割成
我正在使用基于 SVM 分类的 HOG 特征检测器。我可以成功提取车牌,但提取的车牌除了车牌号外还有一些不必要的像素/线。我的图像处理流程如下: 在灰度图像上应用 HOG 检测器 裁剪检测到的区域 调
我有以下图片: 我想填充它的轮廓(即我想在这张图片中填充线条)。 我尝试了形态学闭合,但使用大小为 3x3 的矩形内核和 10 迭代并没有填满整个边界。我还尝试了一个 21x21 内核和 1 迭代,但
我必须找到一种算法,可以找到两组数组之间的交集总数,而其中一个数组已排序。 举个例子,我们有这两个数组,我们向相应的数字画直线。 这两个数组为我们提供了总共 7 个交集。 有什么样的算法可以帮助我解决
简单地说 - 我想使用透视投影从近裁剪平面绘制一条射线/线到远裁剪平面。我有我认为是使用各种 OpenGL/图形编程指南中描述的方法通过单击鼠标生成的正确标准化的世界坐标。 我遇到的问题是我的光线似乎
我是一名优秀的程序员,十分优秀!