zoukankan      html  css  js  c++  java
  • 解决异常:公共语言运行时检测到无效的程序

      我碰到这个问题比较奇怪,我写的一个方法(基于.NET 2.0)在win7、win2003下运行没有问题,在winxp下运行就抛异常:“公共语言运行时检测到无效的程序”,对应英文为:common language runtime detected an invalid program.

      抛异常的方法代码摘抄如下:

        private Control control = ...;
        public void ActionOnUI<T1>(bool showMessageBoxOnException, bool beginInvoke,  CbGeneric<T1> method, params object[] args)
        {
            if (this.control.InvokeRequired)
            {
                if (beginInvoke)
                {
                    this.control.BeginInvoke(new CbGeneric<bool, bool, CbGeneric<T1>, object[]>(this.ActionOnUI), showMessageBoxOnException, beginInvoke, method, args);
                }
                else
                {
                    this.control.Invoke(new CbGeneric<bool, bool, CbGeneric<T1>, object[]>(this.ActionOnUI), showMessageBoxOnException, beginInvoke, method, args);
                }
            }
            else
            {
                try
                {
                    method((T1)args[0]);
                }
                catch (Exception ee)
                {                if (showMessageBoxOnException)
                    {
                        MessageBox.Show(ee.Message);
                    }
                }
            }
        }

        方法的目的是对UI调用转发做一个封装,让使用者更方便的将调用转发到UI线程。

        但是,这个方法在执行时,异常在xp下发生了:

         Common Language Runtime detected an invalid program.

              at ESBasic.Helpers.UiSafeInvoker.ActionOnUI[T1](Boolean showMessageBoxOnException, Boolean beginInvoke, CbGeneric`1 method, Object[] args)

         我在网上搜了一些相关问题的解答,比较靠谱的一点是这样说的:

        “这种错误非常少见,是一个编译器错误,通常产生在将C#等托管语言生成为MSIL时候出的错,没有什么好的解决办法,现在可行的方法好像就是修改现在的程序结构,这样根据新的结构生成新的MSIL时不会出错就基本可以避免这个问题。”

        根据这个提示,我对方法的代码进行了各种修改尝试,最后终于得到了一种在xp下也不抛异常的结构,粘贴如下:

        private Control control = ...;
        public void ActionOnUI<T1>(bool showMessageBoxOnException, bool beginInvoke, CbGeneric<T1> method, T1 args)
        {
            if (this.control.InvokeRequired)
            {
                if (beginInvoke)
                {
                    this.control.BeginInvoke(new CbGeneric<bool, CbGeneric<T1>, T1>(this.Do_ActionOnUI<T1>), showMessageBoxOnException, method, args);
                    return;
                }
                this.control.Invoke(new CbGeneric<bool, CbGeneric<T1>, T1>(this.Do_ActionOnUI<T1>), showMessageBoxOnException, method, args);
                return;
            }
    
            this.Do_ActionOnUI<T1>(showMessageBoxOnException, method, args);
        }
    
        private void Do_ActionOnUI<T1>(bool showMessageBoxOnException, CbGeneric<T1> method, T1 args)
        {
            try
            {
                method(args);
            }
            catch (Exception ee)
            {            if (showMessageBoxOnException)
                {
                    MessageBox.Show(ee.Message);
                }
            }
        } 

      总结起来,改变的几点如下:

    (1)将真正执行的部分重构为一个方法Do_ActionOnUI,然后,转发调用Invoke都指向这个方法。

    (2)Invoke转发调用时,为指向的方法加上泛型参数,避免编译器自动去匹配。

    (3)将弱类型的参数object[]修改为强类型的参数T1。

      好吧,现在问题总算是解决了,好好折腾了一番啊~~

     

  • 相关阅读:
    SSH框架学习笔记(六)----hibernate关联映射注解
    1. Elasticsearch startup on local
    Oracle PL/SQL 入门
    单例模式的两种实现方式对比:DCL (double check idiom)双重检查 和 lazy initialization holder class(静态内部类)
    单例模式中用volatile和synchronized来满足双重检查锁机制
    Java反射机制可以动态修改实例中final修饰的成员变量吗?
    left join 条件区别
    Java应用程序项目的打包与发行(run.bat形式)
    Eclipse调试常用技巧(转)
    PhoneGap: Android 自定义组件
  • 原文地址:https://www.cnblogs.com/zhuweisky/p/4046501.html
Copyright © 2011-2022 走看看