我从 System.Thread.Timer 线程池得到了这个(上面标题中的错误)所以我有我的 TimerWrapper 包装 System.Thread.Timer 将实际执行移动到 System.Thread.ThreadPool 并且我仍然得到它,所以我将它移动到一个新的 Thread(callback).Start() 并且我仍然得到它。当我将它放在一个全新的线程上时,它如何调度输入同步调用???
这是一个非常非常小的原型(prototype)应用程序,我在其中所做的就是触发一个正在执行此操作的计时器...
IEnumerable swc = SHDocVw.ShellWindows()
HashSet<WindowInfo> windows = new HashSet<WindowInfo>();
foreach (SHDocVw.InternetExplorer ie in swc)
{
if (!ie.FullName.ToLower().Contains("iexplore.exe"))
continue;
IntPtr hwnd;
IEPlugin.IOleWindow window = ie.Document as IEPlugin.IOleWindow;
window.GetWindow(out hwnd);
WindowInfo info = new WindowInfo();
info.handle = hwnd;
info.extraInfo = ie;
windows.Add(info);
}
最佳答案
恭喜;您已经设法偶然发现了我最喜欢的 COM 怪癖之一,在本例中,是对 IOleWindow 的 GetWindow 方法的一个令人愉快的模糊限制 - 以及一条错误消息,让您对正在发生的事情一无所知。这里的根本问题是 GetWindow() 方法被标记为 [input_sync] - 来自 SDK 中的 include\oleidl.idl 文件:
interface IOleWindow : IUnknown
{
...
[input_sync]
HRESULT GetWindow
(
[out] HWND *phwnd
);
不幸的是,IOleWindow 的文档没有提到这个属性,但是一些其他的文档,比如IOleDocumentView::SetRect()。做:
This method is defined with the [input_sync] attribute, which means that the view object cannot yield or make another, non input_sync RPC call while executing this method.
此属性背后的想法是向调用方(可以是 Word 或其他 OLE 控件主机等应用程序)保证它可以安全地调用这些方法而不必担心重入。
事情变得棘手的是 COM 决定强制执行此操作:如果它认为可能违反这些约束,它将拒绝对 [input_sync] 方法的跨单元调用。因此,IIRC,如果您在 SendMessage() 中,则无法进行跨公寓 [input_sync] 调用 - 这种情况下错误消息有点暗示。并且 - 这是让您来到这里的原因 - 您不能从 MTA 线程调用跨单元 [input_sync] 方法。也许 COM 在这里的强制执行有点过分热心,但这就是您无论如何都必须处理的问题。
(关于 MTA 与 STA 线程的简要评论:在 COM 中,线程和对象要么是 STA,要么是 MTA。STA,Single-Threaded-Aparment,是 Windows UI 的工作方式;单个线程拥有 UI 和与之关联的所有对象它,并且那些对象期望被该线程单独调用。MTA,或多线程公寓,更像是一个免费的;对象可以期望在任何时间从任何线程调用,所以需要做它们自己的同步是线程安全的。MTA 线程通常用于辅助任务和后台任务。因此您可以在单个 STA 线程上管理 UI,但在后台使用一个或多个 MTA 线程下载一堆文件。COM 确实一堆工作允许两者相互操作并试图隐藏一些复杂性。这里的部分问题是你混合了这些隐喻:线程池与后台工作相关联,MTA 也是如此,但 IOleWindow 是 UI-以中心为中心,STA 也是如此 - 而 GetWindow 恰好是一种对 en 非常严格的方法强制这个。)
长话短说,您不能从 ThreadPool thead 调用此方法,因为它们是 MTA 线程。此外,默认情况下,新线程是 MTA,因此仅创建一个新线程来完成工作是不够的。
相反,创建新线程,但在启动之前使用 tempThread.SetApartmentState(ApartmentState.STA);,这将为您提供一个 STA 线程。您实际上可能需要将处理 shell COM 对象的所有代码放在该 STA 线程中,而不仅仅是对 GetWindow() 的单个调用 - 我不记得具体的细节,但是如果您最终在 MTA 线程池线程上获取原始 COM 对象(此处似乎是 ShellWindows 对象),即使您尝试从 STA 调用它,它仍将与该 MTA 保持关联。
如果您可以改为从 STA 线程而不是从 ThreadPool 的 MTA 线程完成所有工作,那就更好了,这将首先避免这种情况。与其使用专为后台/非 UI 代码设计的 System.Threading.Timer,不如尝试使用以 UI 为中心的 System.Windows.Forms.Timer反而。这确实需要一个消息循环 - 如果您的应用程序中已经有窗口和表单,那么您已经有一个,但如果没有,在测试代码中执行此操作的最简单方法是在相同的位置执行 MessageBox()主线代码等待退出的地方(通常使用 Sleep 或 Console.ReadLine 或类似的)。
关于c# - 无法进行传出调用,因为应用程序正在调度输入同步调用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/8839195/
很好奇,就使用rubyonrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提
我需要在客户计算机上运行Ruby应用程序。通常需要几天才能完成(复制大备份文件)。问题是如果启用sleep,它会中断应用程序。否则,计算机将持续运行数周,直到我下次访问为止。有什么方法可以防止执行期间休眠并让Windows在执行后休眠吗?欢迎任何疯狂的想法;-) 最佳答案 Here建议使用SetThreadExecutionStateWinAPI函数,使应用程序能够通知系统它正在使用中,从而防止系统在应用程序运行时进入休眠状态或关闭显示。像这样的东西:require'Win32API'ES_AWAYMODE_REQUIRED=0x0
我在从html页面生成PDF时遇到问题。我正在使用PDFkit。在安装它的过程中,我注意到我需要wkhtmltopdf。所以我也安装了它。我做了PDFkit的文档所说的一切......现在我在尝试加载PDF时遇到了这个错误。这里是错误:commandfailed:"/usr/local/bin/wkhtmltopdf""--margin-right""0.75in""--page-size""Letter""--margin-top""0.75in""--margin-bottom""0.75in""--encoding""UTF-8""--margin-left""0.75in""-
在控制台中反复尝试之后,我想到了这种方法,可以按发生日期对类似activerecord的(Mongoid)对象进行分组。我不确定这是完成此任务的最佳方法,但它确实有效。有没有人有更好的建议,或者这是一个很好的方法?#eventsisanarrayofactiverecord-likeobjectsthatincludeatimeattributeevents.map{|event|#converteventsarrayintoanarrayofhasheswiththedayofthemonthandtheevent{:number=>event.time.day,:event=>ev
对于具有离线功能的智能手机应用程序,我正在为Xml文件创建单向文本同步。我希望我的服务器将增量/差异(例如GNU差异补丁)发送到目标设备。这是计划:Time=0Server:hasversion_1ofXmlfile(~800kiB)Client:hasversion_1ofXmlfile(~800kiB)Time=1Server:hasversion_1andversion_2ofXmlfile(each~800kiB)computesdeltaoftheseversions(=patch)(~10kiB)sendspatchtoClient(~10kiBtransferred)Cl
Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack
我想用ruby编写一个小的命令行实用程序并将其作为gem分发。我知道安装后,Guard、Sass和Thor等某些gem可以从命令行自行运行。为了让gem像二进制文件一样可用,我需要在我的gemspec中指定什么。 最佳答案 Gem::Specification.newdo|s|...s.executable='name_of_executable'...endhttp://docs.rubygems.org/read/chapter/20 关于ruby-在Ruby中编写命令行实用程序
我对最新版本的Rails有疑问。我创建了一个新应用程序(railsnewMyProject),但我没有脚本/生成,只有脚本/rails,当我输入ruby./script/railsgeneratepluginmy_plugin"Couldnotfindgeneratorplugin.".你知道如何生成插件模板吗?没有这个命令可以创建插件吗?PS:我正在使用Rails3.2.1和ruby1.8.7[universal-darwin11.0] 最佳答案 随着Rails3.2.0的发布,插件生成器已经被移除。查看变更日志here.现在
我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当
我构建了两个需要相互通信和发送文件的Rails应用程序。例如,一个Rails应用程序会发送请求以查看其他应用程序数据库中的表。然后另一个应用程序将呈现该表的json并将其发回。我还希望一个应用程序将存储在其公共(public)目录中的文本文件发送到另一个应用程序的公共(public)目录。我从来没有做过这样的事情,所以我什至不知道从哪里开始。任何帮助,将不胜感激。谢谢! 最佳答案 无论Rails是什么,几乎所有Web应用程序都有您的要求,大多数现代Web应用程序都需要相互通信。但是有一个小小的理解需要你坚持下去,网站不应直接访问彼此