设为首页 收藏本站
查看: 1050|回复: 0

[经验分享] IIS中等信任环境下(medium),DbProviderFactories.GetFactory不能调用非微软数据库的简单替代方案。

[复制链接]

尚未签到

发表于 2015-8-13 11:24:20 | 显示全部楼层 |阅读模式
  本文讲述在“中等信任”环境的IIS下面创建一个简单但是实用的多数据库兼容方案。有时候我们换一下思路,可以得到更实用、间接的办法。 强调两个要点:

  • 中等信任环境
  • 多数据库兼容
  二者缺一不可。“多数据库兼容”中的“多数据库”是指 除了SQLServer和ACCESS之外,至少包含有MySQL、SQLite、DB2、FireBird中的一款到多款非微软公司开发的关系型数据库

如果您能够保证自己的程序今后永远都能够运行在“完全信任”等级的IIS中,本文不适合您,您完全可以去使用现成的第三方DBHelper和DbProviderFactories.GetFactory,它们更成熟稳定


记得我小学五年级的时候,语文老师(同时也是我班主任)经常布置一些生词给我们造句,就拿“简洁”这个词来说,很难琢磨出一句话来交作业,没办法,我承认我智商比较低。但是我总感觉应该有一种万能的方法,可以快速过关,以不变应万变的。 所以,当时我花了几天时间来研究了一下诸子百家和四库全书,最终得出这个万能方案:




public string Print(string word)
{
    Console.WriteLine("用\"{0}\"这个词来造句简直易如反掌。",word);
}  

怎么样?完成任务!这下无论她布置什么词语下来,我都可以瞬间作答

作业交上去,一天又一天过去了,她既不打勾,也不打叉。终于有一天,她拿着我的几篇作业,在班会上面宣读,全班哄堂而笑,他们笑什么我不知道,因为我照规章完成任务了,但是我也认识到,再这样下去是行不通了。


唉,看来她还是喜欢我们老老实实,只要谁有了点不规矩的行为,即使在道理上没有做错,她也要打压赶尽杀绝,唉,其实今天都还有这种事情,看看周围,算了不说了。


DbProviderFactories.GetFactory 是一个很方便的功能,能够动态获取我们需要的数据库连接对象,但是它有一个遗憾: 它的内部实现存在比较复杂的反射机制,导致用这种办法不能在中等信任权限的IIS上面创建Sqlserver和OLEDB之外的数据库连接对象,例如:MySql、SQLite, 典型的例子就是Godaddy的空间。


MySql自家的 Connector 组件虽然提供了DbProviderFactory,但是依然因为权限问题无法运行。

SQLite 则分为两种情况:



  • 普遍用得最多的一种是.net作为外壳,内部是WIN32的,这种DLL就不要妄想在IIS上面运行了,除非是自己的服务器。

  • 另一种方案是纯.Net编写的 CSharpSQLite , 它的情况和MySql的Connector一样,自身提供的 DbProviderFactory 只能够运行于完整信任模式。 中等信任的Godaddy免谈。

好了,下面进入解决方案:

我们知道,所有的Connection类都最终继承于 IDbConnection 接口 ,为此我们可以先建立一个IDbConnection 创建方法:




    public interface IDbFactory
    {
        IDbConnection CreateConnection();
    }


接下来的问题,怎么实现具体类呢,直接返回 new SqlConnection() ? 不,那是相当愚蠢的做法。通过泛型调用,我们可以这样子实现:




        public IDbConnection  CreateConnection<C>() where C : IDbConnection, new()
        {
            var temp = new C();
            temp.ConnectionString = &#8220;server=xxx,id=xxx,password=xxx&#8221;;
            return temp;
        }
  

好了,现在行为逻辑已经有了,但是怎么运用在实际程序里面呢, 如果我们直接使用:




var conn = CreateConnection<MySqlConnection>()  


这简直就是自欺欺人。还不如直接 var conn = new MySqlConnection() 算了。 现在我们急需一个封装,避免在客户端中使用实际的类型参数。我们先这样子实现:





   internal sealed class FactoryBuilder<C> : IDbFactory
        where C : IDbConnection, new()
    {
        string _connectionString;
        public FactoryBuilder(string connectionString)
        {
_connectionString = connectionString;
        }
        public IDbConnection CreateConnection()
        {
            var temp = new C();
            temp.ConnectionString = _connectionString;
            return temp;
        }
    }
  
  这里说一句题外话,在一个framework的设计和编码中,合理的使用 internal 和 sealed 来修饰一个class, 可以在&#8220;对自己开放&#8221;、&#8220;对客户封闭&#8221;这个矛盾之间达到合适的平衡。客户端没有必要调用的类,就尽量封闭,这样便越是稳定。
  
然后我们再建立一个全局的静态变量:

  


    public class Config
    {
       public  static readonly IDbFactory DefaultFactory  = new FactoryBuilder< OleDbConnection>(connectionString);
    }

在客户端中我们这样引用:




var conn = Config.DefaultFactory.CreateConnection();  
  
  大功告成!客户端中已经不存在类型参数了,放心用就是。
细心的读者此时可能还会发现代码中存在一个致命弱点:如果要把OleDbConnection换成SqlConnection怎么办?还是需要配置文件支持才可以修改的,还是需要web.config。

这时候我们又回到了问题的开始:如果使用web.config,只有两种办法:



  • 一是使用 Eval结合web.config中的字符串来制造动态代码,但是免谈了,那还不如一开始就 var conn = Eval(&#8220;new SqlConnection()&#8221;)。 或者用反射,但是反射的致命弱点:中等信任环境拒绝运行(ACCESS和SQLServer可以,别的数据库不行,也没有保证)。所以,反射动态代码都是行不通的。

  • 二是 switch判断, 先在代码中硬编码:

                string t = ConfigurationManager.AppSetting["dbtype"];
                IDbFactory fac;
    string connectionString = "server=xxx;uid=xxx;password=xxx";
                switch (t)
                {
                    case "oledb":
                        fac = new FactoryBuilder< OleDbConnection>(connectionString) ();
                        break;
                    case "sqlserver":
                        fac = new FactoryBuilder< SqlConnection>(connectionString) ();
                        break;
                    case "mysql":
                        conn = new FactoryBuilder< MySqlConnection>(connectionString) ();
                        break;
                        //.........其它雷同,省略........
                }
      

  
反正来来去取也就那么几个数据库,你把它们都写进去之后,就不存在&#8220;需求扩充&#8221;的问题了,这个办法看似解决了实际问题,但是会在每一次对象建立的时候都要判断一次,而我们的系统,也许一但定型下来,就再也不需要修改数据库类型了。 而运行时还需要作这样的判断是相当愚蠢的。所以,这个办法也免谈了。


一要避免无效计算,二要避免中等信任环境拒绝运行。再想一想,一定有办法解决的,就像10多年前一样:用&#8220;简洁&#8221;这个词来造句简直易如反掌。


&#8220;简洁&#8221;?灵光一闪:是不是尽量化简会更好?


我们是不是可以直接用cs文件来作为配置文件? PHP、JAVASCRIPT、ASP 不就是用它们自身的代码文件来直接些配置的吗? 一切化简回去,不是很好?凭什么写在多余的XML文件中?不用XML就犯法?还是会被抓去挖煤?  


ASP中可以通过 <!--# include=&#8221;xxx.asp&#8221; --> , javascript可以通过 script src=&#8221;xxx.js&#8221;, PHP中应该是什么我不知道,但是我配置wordpress的时候就是看它这么做的。


C#不存在这样的或者类似的导入办法,难道还要建立一个工厂或者什么抽象类、接口、再来一道多态,接这再封装隐藏一个泛型参数?为了一个Connection类再这样搞下去的话,和我那篇《回忆孔先生》还有何区别?再想想,有办法的。


partial class !



不说废话,直接改:




    //这一段封装为DLL文件
    public partial class Config
    {
        static IDbFactory _defaultFactory;
        static void SetDefault<C>(string connectionString) where C : IDbConnection, new()
        {
            _defaultFactory = new FactoryBuilder<C>(connectionString);
        }
        public static IDbFactory DefaultFactory
        {
            get
            {
                return _defaultFactory; //不需要null判断,静态构造函数会保证它一定经过初始化
            }
        }
    }

    //这一段分布类作为一个单独文件(并且是这个文件的全部内容),放到app_code目录中
    public partial class Config
    {
        static Config()
        {
            SetDefault<OleDbConnection>("Server=xxx;uid=xxx;password=xxx");
        }
    }
  

这时候我们可以在客户端里面这样子调用了:




    class Program
    {
        static void Main(string[] args)
        {
            var conn = Config.DefaultFactory.CreateConnection();
            conn.Open();
            //该做什么做什么了
            conn.Close();
        }
    }  

怎么样? 无反射、不存在多余判断、Godaddy顺利运行。 有了最关键的Connection,天得一以清,地得一以宁,接下来的Command、Datareader、Datatable、DataSet &#8230;&#8230;不用愁了。

其实不止Godaddy,很多国外空间都是中等信任的,而且出于&#8220;做好自己&#8221;的原则,我们应该保证自己的程序尽最大程度的兼容,而不是到时候埋怨&#8220;XXX权限不够,不关我的事&#8221;---这是推脱。我们不能保证别人会怎么样,只有做好自己。

为什么我要反复强调基于&#8220;国外空间&#8221;,你们懂的。


后话:写到一半的时候,想给一段代码加颜色,结果IE8假死,幸好最后是恢复了,代码段外面的内容换颜色没问题。


完整示例代码下载



运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-98411-1-1.html 上篇帖子: ASP.NET报错集合一----.net4.0创建项目后,在iis上部署项目,无法浏览,提示404 下篇帖子: 在IIS中创建虚拟目录 (C#)
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表