3

我有一个控制台应用程序,它使用 ComImport 调用 Windows 搜索。虽然这在调试模式下工作正常......控制台应用程序在发布模式下崩溃。可能是什么问题呢?

[ComImport]
    [Guid("9DAA54E8-CD95-4107-8E7F-BA3F24732D95")]
    [ClassInterface(ClassInterfaceType.None)]
    [TypeLibType(TypeLibTypeFlags.FCanCreate)]
    public class WordBreaker : IWordBreaker
    {
        [MethodImpl(MethodImplOptions.InternalCall, MethodCodeType = MethodCodeType.Runtime)]
        public virtual extern bool Init([In] bool query, [In] uint maxTokenSize);

        [MethodImpl(MethodImplOptions.InternalCall, MethodCodeType = MethodCodeType.Runtime)]
        public virtual extern void BreakText([In, MarshalAs(UnmanagedType.LPStruct)] TEXT_SOURCE textSource,
            [In] IWordSink wordSink, [In] IPhraseSink phraseSink);

        [MethodImpl(MethodImplOptions.InternalCall, MethodCodeType = MethodCodeType.Runtime)]
        public virtual extern void ComposePhrase([In, MarshalAs(UnmanagedType.LPWStr)] string noun, [In] uint nounLen,
            [In, MarshalAs(UnmanagedType.LPWStr)] string modifier, [In] uint modifierLen,
            [In] uint attachmentType, [Out] out IntPtr phrase, [In, Out] ref uint phraseLen);

        [MethodImpl(MethodImplOptions.InternalCall, MethodCodeType = MethodCodeType.Runtime)]
        public virtual extern IntPtr GetLicenseToUse();
    }

访问 WordBreaker.BreakText 函数时,代码在发布模式下失败。

它正在我的代码中使用,如下所示

if (!string.IsNullOrWhiteSpace(text))
                try
                {
                    IWordBreaker breaker = new WordBreaker();
                    bool reqLicense = breaker.Init(query, 256);
                    if (reqLicense)
                    {
                        IntPtr lic = breaker.GetLicenseToUse();
                        string licText = Marshal.PtrToStringUni(lic);
                    }
                    TEXT_SOURCE source = new TEXT_SOURCE();
                    source.fillTextBuffer = FillTextBuffer;
                    source.buffer = text;
                    source.cur = 0;
                    source.end = (uint)(text.Length);
                    breaker.BreakText(source, new WordSink(result), null);
                }
                catch (Exception ex)
                {
                    Console.Out.WriteLine(ex.ToString());
                    //log4net.LogManager.GetLogger(typeof(WindowsIntegration)).Error("BreakText", ex);
                }
            var resultWithoutNoise = NoiseWord.Remove(result);
            return resultWithoutNoise;
        }

崩溃发生在应用程序崩溃之前,breaker.BreakTextBreakText函数被调用了很多次(500 到 7000 次之间)。

故障转储说明了以下有关异常信息的 The thread tried to read from or write to a virtual address for which it does not have the appropriate access. 说明:我没有在我的代码中使用任何线程。

4

3 回答 3

1

这看起来像是垃圾收集器将文本移动到另一个位置的情况,因此记录在 source.buffer 中的地址不再有效。您没有包含声明 TEXT_SOURCE 的来源,因此我无法对此发表评论。但是 BreakText 的声明是不正确的。MarshalAs(UnmanagedType.LPStruct) 不应该在那里使用。UnmanagedType.LPStruct 仅在一种特定情况下受支持:将 System.Guid 值类型视为具有额外间接级别的非托管 GUID。请参阅http://blogs.msdn.com/b/adam_nathan/archive/2003/04/23/56635.aspx如何将结构编组为指向结构的指针?.

于 2013-09-09T23:27:34.340 回答
1

该代码至少需要一个线程,这就是错误所指的内容。至于崩溃的类型,它本质上是访问冲突,要么是空指针/对象引用(在本机代码中),要么是缓冲区溢出。检查什么值。细绳。您在失败时传入的长度等。

于 2011-06-29T11:49:41.383 回答
0

如果您无法重现该问题并且它仅在发布时出现,我强烈建议您添加日志记录以缩小您的问题范围。

当您缩小问题范围后,我会编辑您上面的帖子,以包含有关崩溃的更详细信息。

于 2011-06-29T11:43:06.083 回答