入门教程,最近开支品种中用到的编码小技巧汇总表达

一.暗中认可EF生成的接连字符串相比较的长和新奇,若想利用普通的连接字符串来连接EF,则足以因而创设分项目,天公地道写3个构造函数,在构造函数中通过动态拼接EntityConnectionString获得EF所需的连日字符串,具代达成代码如下:

一.私下认可EF生成的连年字符串相比较的长和新奇,若想行使普通的总是字符串来连接EF,则足以因而制造分档次,同样爱慕写一个构造函数,在构造函数中通过动态拼接EntityConnectionString获得EF所需的延续字符串,具代完成代码如下:

前不久支出项目中用到的编码小技巧汇总表明,开发品种小技巧

一.暗许EF生成的总是字符串相比较的长和新奇,若想利用普通的连接字符串来连接EF,则能够因而创立分连串,一视同仁写一个构造函数,在构造函数中通过动态拼接EntityConnectionString得到EF所需的连日字符串,具代达成代码如下:

    public partial class DataEntities
    {
        private static ConcurrentDictionary<string, string> entityConnStrings = new ConcurrentDictionary<string, string>();

        public DataEntities(string connName)
            : base(BuildEntityConnectionString(connName))
        {

        }

        private static string BuildEntityConnectionString(string connName)
        {

            if (!entityConnStrings.ContainsKey(connName))
            {
                var connStrSetting = System.Configuration.ConfigurationManager.ConnectionStrings[connName];

                EntityConnectionStringBuilder entityConnStrBuilder = new EntityConnectionStringBuilder();
                entityConnStrBuilder.Provider = connStrSetting.ProviderName;
                entityConnStrBuilder.ProviderConnectionString = EncryptUtility.DesDecrypt("XXXXX", connStrSetting.ConnectionString);
                entityConnStrBuilder.Metadata = "res://*/Data.csdl|res://*/Data.ssdl|res://*/Data.msl";
                string entityConnString = entityConnStrBuilder.ToString();
                entityConnStrings.AddOrUpdate(connName, entityConnString, (key, value) => entityConnString);
            }
            return entityConnStrings[connName];
        }
    }

瞩目上边的类是一个分部类:partial,同时BuildEntityConnectionString方法是二个静态方法,在BuildEntityConnectionString方法中ProviderConnectionString
= EncryptUtility.DesDecrypt(“XXXXX”,
connStrSetting.ConnectionString);是主要,作者那边是对config中的连接字符串
也都开始展览了加密,故此处作者须要解密,若无那几个要求能够平昔:ProviderConnectionString
=connStrSetting.ConnectionString即可。后续实例化EF上下文对象时,请使用:DataEntities(string
connName)那一个结构涵数即可,DataEntities是切实可行的EF上下文对象,我们的EF上下文类名均大概不均等。

二.帮衬三个通用对象的XML体系化(即:2个类中有可变类型属性成员,需求分化的系列结果及变更不一样的行列成分名称),具体贯彻代码如下:

三个亟需被体系化成XML的类:当中供给扭转的XML元素detail必需有子成分,且子成分名称及子成分内部属性依照项指标不等而各异(即:detail成分下的子成分是可变的)

    [XmlRootAttribute("master")]
    public class DemoMaster<T> where T : class
    {
        [XmlElement("attr")]
        public string DemoAttr { get; set; }

        [XmlElement("detail")]
        public DemoDetail<T> DemoDetail { get; set; } //关键点在这里,该属性元素为:detail,但其子元素根据T不同而不同

    }
    public class DemoDetail<T> : IXmlSerializable where T : class
    {
        public T body { get; set; }

        public System.Xml.Schema.XmlSchema GetSchema()
        {
            return null;
        }

        public void ReadXml(System.Xml.XmlReader reader)
        {
            string bodyStr = reader.ReadInnerXml();
            this.body = XmlHelper.XmlDeserialize<T>(bodyStr, Encoding.UTF8);
        }

        public void WriteXml(System.Xml.XmlWriter writer)
        {
            writer.WriteRaw(XmlHelper.XmlSerialize(this.body, Encoding.UTF8, true));
        }
    }

    [XmlTypeAttribute("list-a", AnonymousType = false)]
    public class DemoDetailA
    {
        public string Apro1 { get; set; }

        public string Apro2 { get; set; }

        public string Apro3 { get; set; }
    }

    [XmlTypeAttribute("list-b", AnonymousType = false)]
    public class DemoDetailB
    {
        public string Bpro1 { get; set; }

        public string Bpro2 { get; set; }

        public string Bpro3 { get; set; }
    }

    [XmlTypeAttribute("list-c", AnonymousType = false)]
    public class DemoDetailC
    {
        public string Cpro1 { get; set; }

        public string Cpro2 { get; set; }

        public string Cpro3 { get; set; }
    }

小心上边代码中,必要关切:德姆oDetail属性及德姆oDetail<T>类,DemoDetail属性仅是为了生成detail成分节点,而子节点则由DemoDetail<T>类来进展转变,DemoDetail<T>是贯彻了IXmlSerializable接口,在XML种类化时,德姆oDetail<T>类仅将body属性对应的T类型实例内容展开系列化(WriteRaw),而反体系化时,则先反体系化body属性对应的T类型实例,然后赋值给body属性,那也是优秀纷呈之处,德姆oDetail<T>类自个儿并从未真的参预到系列化中,故体系化的字符串也看不到德姆oDetail<T>类相关的要素,德姆oDetail<T>类仅仅是四个XML连串化格式生成的中介。系列化的XML结果如下:

种类化代码:

            var demo1 = new DemoMaster<DemoDetailA>()
            {
                DemoAttr = "demo1",
                DemoDetail = new DemoDetail<DemoDetailA>() { body = new DemoDetailA() { Apro1 = "demoA1", Apro2 = "demoA2", Apro3 = "demoA3" } }
            };

            var demo2 = new DemoMaster<DemoDetailB>()
            {
                DemoAttr = "demo2",
                DemoDetail = new DemoDetail<DemoDetailB>() { body = new DemoDetailB() { Bpro1 = "demoB1", Bpro2 = "demoB2", Bpro3 = "demoB3" } }
            };

            var demo3 = new DemoMaster<DemoDetailC>()
            {
                DemoAttr = "demo3",
                DemoDetail = new DemoDetail<DemoDetailC>() { body = new DemoDetailC() { Cpro1 = "demoC1", Cpro2 = "demoC2", Cpro3 = "demoC3" } }
            };

            textBox1.Text = XmlHelper.XmlSerialize(demo1, Encoding.UTF8);

            textBox1.Text += "\r\n" + XmlHelper.XmlSerialize(demo2, Encoding.UTF8);

            textBox1.Text += "\r\n" + XmlHelper.XmlSerialize(demo3, Encoding.UTF8);

类别化的XML:

<?xml version="1.0" encoding="utf-8"?>
<master>
    <attr>demo1</attr>
    <detail><list-a>
    <Apro1>demoA1</Apro1>
    <Apro2>demoA2</Apro2>
    <Apro3>demoA3</Apro3>
</list-a></detail>
</master>

<?xml version="1.0" encoding="utf-8"?>
<master>
    <attr>demo2</attr>
    <detail><list-b>
    <Bpro1>demoB1</Bpro1>
    <Bpro2>demoB2</Bpro2>
    <Bpro3>demoB3</Bpro3>
</list-b></detail>
</master>

<?xml version="1.0" encoding="utf-8"?>
<master>
    <attr>demo3</attr>
    <detail><list-c>
    <Cpro1>demoC1</Cpro1>
    <Cpro2>demoC2</Cpro2>
    <Cpro3>demoC3</Cpro3>
</list-c></detail>
</master>

三.winform DataGridView
完成钦点列选拔密码框格局展现与编写制定,以及列绑定到复合属性(即:绑定到多层次属性),具体落实代码如下:

            dataGridView1.CellFormatting += new DataGridViewCellFormattingEventHandler(dataGridView1_CellFormatting);
            dataGridView1.EditingControlShowing += new DataGridViewEditingControlShowingEventHandler(dataGridView1_EditingControlShowing);


        public string EvaluateValue(object obj, string property)
        {
            string retValue = string.Empty;
            string[] names = property.Split('.');

            for (int i = 0; i < names.Count(); i++)
            {
                try
                {
                    var prop = obj.GetType().GetProperty(names[i]);
                    var result = prop.GetValue(obj, null);
                    if (result != null)
                    {
                        obj = result;
                        retValue = result.ToString();
                    }
                    else
                    {
                        break;
                    }
                }
                catch (Exception)
                {
                    throw;
                }
            }

            return retValue;
        }


        private void dataGridView1_CellFormatting(object sender, DataGridViewCellFormattingEventArgs e)
        {

            if (dataGridView1.Columns[e.ColumnIndex].DataPropertyName.Contains("."))
            {
                e.Value = EvaluateValue(dataGridView1.Rows[e.RowIndex].DataBoundItem, dataGridView1.Columns[e.ColumnIndex].DataPropertyName);
            }


            if (dataGridView1.Columns[e.ColumnIndex].Name == "KeyCode")
            {
                if (e.Value != null && e.Value.ToString().Length > 0)
                {
                    e.Value = new string('*', e.Value.ToString().Length);
                }
            }
        }

        private void dataGridView1_EditingControlShowing(object sender, DataGridViewEditingControlShowingEventArgs e)
        {
            int i = this.dataGridView1.CurrentCell.ColumnIndex;
            bool usePassword = false;
            if (dataGridView1.Columns[i].Name == "KeyCode")
            {
                usePassword = true;
            }
            TextBox txt = e.Control as TextBox;
            if (txt != null)
            {
                txt.UseSystemPasswordChar = usePassword;
            }
        }

//示例:绑定的源数据类定义
    public class DemoBindClass
    {
        public string Attr { get; set; }

        public string KeyCode { get; set; }

        public DemoDetailA Detail { get; set; }
    }


    public class DemoDetailA
    {
        public string Apro1 { get; set; }

        public string Apro2 { get; set; }

        public string Apro3 { get; set; }

        public DemoDetailB DetailChild { get; set; }
    }


    public class DemoDetailB
    {
        public string Bpro1 { get; set; }

        public string Bpro2 { get; set; }

        public string Bpro3 { get; set; }
    }

绑定到数据源:

            var demo = new[] {
                new DemoBindClass()
                    {
                        Attr = "demo",
                        KeyCode="a123456789b",
                        Detail = new DemoDetailA()
                        {
                            Apro1 = "demoA1",
                            Apro2 = "demoA2",
                            Apro3 = "demoA3",
                            DetailChild = new DemoDetailB()
                            {
                                Bpro1 = "demoB1",
                                Bpro2 = "demoB2",
                                Bpro3 = "demoB3"
                            }
                        }
                    }
            };


            dataGridView1.AutoGenerateColumns = false;
            dataGridView1.DataSource = demo;

福寿无疆钦赐列采用密码框形式呈现与编写制定,以及列绑定到复合属性均要求订阅DataGridView的CellFormatting及艾德itingControlShowing事件,并在当中写转换当前Cell的Value,达成列绑定到复合属性,关键点在:伊娃luateValue方法,该格局逻辑很简短,就是依据绑定的性质层级(.分隔)层层遍历获取属性的值,直到遍历完或为空时截至,最终获得的结果便是绑定的属性的值。最后促成的效益如下图示:

美高梅开户网址 1

  

 

1.私下认可EF生成的连年字符串比较的长和新奇,若想行使普通的总是字符串来连…

前言

Contoso 大学以身作则 Web 应用程序演示怎么样选取实体框架(EF)Core 贰.0 和
Visual Studio 20一柒 创制 ASP.NET Core 2.0 MVC Web 应用程序。
示范应用程序是捏造的Contoso高校的网址。
它回顾学生入学,课程成立和导师任务等效用。
那是壹种类教程中的第三章,介绍如何早先创设Contoso大学示例应用程序。
下载或查看已做到的应用程序 –
官方。

本种类小说翻译索引目录

EF Core 2.0 是EF的摩登版本,但还未曾包蕴富有的 EF 陆.x 成效。 有关 EF 陆.x
和 EF Core 之间怎么挑选的音信,请参阅 EF Core vs.
EF6.x。
如若您接纳 EF 陆.x
,请参阅本学科连串的先前版本。

注意事项

  • 对于本课程的 ASP.NET Core 壹.一 版本,请参阅PDF格式的本课程的 VS
    20壹柒 Update
    二版本。
  • 对于本学科的 Visual Studio 20一伍 版本,请参阅PDF格式的 VS 2015
    版本的 ASP.NET Core
    文档。
    public partial class DataEntities
    {
        private static ConcurrentDictionary<string, string> entityConnStrings = new ConcurrentDictionary<string, string>();

        public DataEntities(string connName)
            : base(BuildEntityConnectionString(connName))
        {

        }

        private static string BuildEntityConnectionString(string connName)
        {

            if (!entityConnStrings.ContainsKey(connName))
            {
                var connStrSetting = System.Configuration.ConfigurationManager.ConnectionStrings[connName];

                EntityConnectionStringBuilder entityConnStrBuilder = new EntityConnectionStringBuilder();
                entityConnStrBuilder.Provider = connStrSetting.ProviderName;
                entityConnStrBuilder.ProviderConnectionString = EncryptUtility.DesDecrypt("XXXXX", connStrSetting.ConnectionString);
                entityConnStrBuilder.Metadata = "res://*/Data.csdl|res://*/Data.ssdl|res://*/Data.msl";
                string entityConnString = entityConnStrBuilder.ToString();
                entityConnStrings.AddOrUpdate(connName, entityConnString, (key, value) => entityConnString);
            }
            return entityConnStrings[connName];
        }
    }
    public partial class DataEntities
    {
        private static ConcurrentDictionary<string, string> entityConnStrings = new ConcurrentDictionary<string, string>();

        public DataEntities(string connName)
            : base(BuildEntityConnectionString(connName))
        {

        }

        private static string BuildEntityConnectionString(string connName)
        {

            if (!entityConnStrings.ContainsKey(connName))
            {
                var connStrSetting = System.Configuration.ConfigurationManager.ConnectionStrings[connName];

                EntityConnectionStringBuilder entityConnStrBuilder = new EntityConnectionStringBuilder();
                entityConnStrBuilder.Provider = connStrSetting.ProviderName;
                entityConnStrBuilder.ProviderConnectionString = EncryptUtility.DesDecrypt("XXXXX", connStrSetting.ConnectionString);
                entityConnStrBuilder.Metadata = "res://*/Data.csdl|res://*/Data.ssdl|res://*/Data.msl";
                string entityConnString = entityConnStrBuilder.ToString();
                entityConnStrings.AddOrUpdate(connName, entityConnString, (key, value) => entityConnString);
            }
            return entityConnStrings[connName];
        }
    }

支出条件

设置如下工具:

  • .NET Core 2.0.0
    SDK
    或更新版本。
  • Visual Studio
    2017
    V15.三 或更高版本,安装 ASP.NET 和 Web 开发工具。

在意上边的类是八个分部类:partial,同时BuildEntityConnectionString方法是二个静态方法,在BuildEntityConnectionString方法中ProviderConnectionString
= EncryptUtility.DesDecrypt(“XXXXX”,
connStrSetting.ConnectionString);是必不可少,我那边是对config中的连接字符串
也都开始展览了加密,故此处笔者必要解密,若无那几个要求能够直接:ProviderConnectionString
=connStrSetting.ConnectionString即可。后续实例化EF上下文对象时,请使用:DataEntities(string
connName)那些结构涵数即可,DataEntities是实际的EF上下文对象,大家的EF上下文类名均可能不等同。

瞩目上边的类是3个分部类:partial,同时BuildEntityConnectionString方法是二个静态方法,在BuildEntityConnectionString方法中ProviderConnectionString
= EncryptUtility.DesDecrypt(“XXXXX”,
connStrSetting.ConnectionString);是重要,作者那里是对config中的连接字符串
也都进行了加密,故此处小编索要解密,若无这几个需要能够一直:ProviderConnectionString
=connStrSetting.ConnectionString即可。后续实例化EF上下文对象时,请使用:DataEntities(string
connName)这一个结构涵数即可,DataEntities是现实性的EF上下文对象,大家的EF上下文类名均也许不平等。

故障排除

若是蒙受标题,您无法化解,平时能够经过将代码与成就的项目开始展览相比来找到消除方案。
有关常见错误和解决方法的列表,请参阅本体系最终2个课程的故障排除有的。
固然你未有找到所需的内容,您也得以在 StackOverflow.com 上发问。

小贴士

本连串涵盖拾3个科目,每个教程都建立在早期教程中的基础之上。
在成功做到每一个教程之后,请思考保存项指标副本。
然后,假若遇到难点,您能够从上三个学科重新起初,无需从头开始。

二.支撑贰个通用对象的XML系列化(即:三个类中有可变类型属性成员,须要不一样的类别结果及变化分化的行列成分名称),具体达成代码如下:

贰.协理二个通用对象的XML种类化(即:1个类中有可变类型属性成员,须求差异的行列结果及变化不一致的队列成分名称),具体贯彻代码如下:

Contoso 大学网址应用

课程中创设的是多少个大致的大学网址。
用户能够查看和换代学生,课程和教师职员和工人消息。 下边是你将要创立的部分页面。

美高梅开户网址 2

index

美高梅开户网址 3

edit

网址的 UI 风格与内置模板生成的 UI 风格保持壹致,本课程首要关切怎么样运用
Entity Framework。

3个急需被种类化成XML的类:在那之中供给转变的XML成分detail必需有子成分,且子成分名称及子成分内部属性依照项指标例外而不一样(即:detail元素下的子成分是可变的)

多个须要被体系化成XML的类:个中要求扭转的XML成分detail必需有子成分,且子成分名称及子元素内部属性根据项目标不及而差别(即:detail成分下的子成分是可变的)

成立 ASP.NET Core MVC 网站应用

开拓 Visual Studio 并创建名称叫 “ContosoUniversity” 的新 ASP.NET Core C#
web 项目。

  • 从文件菜单中,选取新建>项目。
  • 从左窗格中选用 已安装 -> Visual C# -> Web 。
  • 当中窗格选用 ASP.NET Core Web 应用程序。
  • 输入 ContosoUniversity 作为项目名称,然后单击鲜明。
![](https://upload-images.jianshu.io/upload_images/4235187-1c7a131da4c70365.png)

newProject
  • 入门教程,最近开支品种中用到的编码小技巧汇总表达。等待 “新建 ASP.NET Core Web 应用程序” 对话框出现
  • 选料 ASP.NET Core 二.0 和 Web应用程序(模型视图控制器)模板。
  • 小心:本课程供给 ASP.NET Core 2.0 和 EF Core 2.0 或更高版本 –
    确定保障未选拔ASP.NET Core 一.一。
  • 保障认证设置为“不举行身份验证”。
  • 单击 “确定” 按钮。

美高梅开户网址 4

newCore

    [XmlRootAttribute("master")]
    public class DemoMaster<T> where T : class
    {
        [XmlElement("attr")]
        public string DemoAttr { get; set; }

        [XmlElement("detail")]
        public DemoDetail<T> DemoDetail { get; set; } //关键点在这里,该属性元素为:detail,但其子元素根据T不同而不同

    }
    public class DemoDetail<T> : IXmlSerializable where T : class
    {
        public T body { get; set; }

        public System.Xml.Schema.XmlSchema GetSchema()
        {
            return null;
        }

        public void ReadXml(System.Xml.XmlReader reader)
        {
            string bodyStr = reader.ReadInnerXml();
            this.body = XmlHelper.XmlDeserialize<T>(bodyStr, Encoding.UTF8);
        }

        public void WriteXml(System.Xml.XmlWriter writer)
        {
            writer.WriteRaw(XmlHelper.XmlSerialize(this.body, Encoding.UTF8, true));
        }
    }

    [XmlTypeAttribute("list-a", AnonymousType = false)]
    public class DemoDetailA
    {
        public string Apro1 { get; set; }

        public string Apro2 { get; set; }

        public string Apro3 { get; set; }
    }

    [XmlTypeAttribute("list-b", AnonymousType = false)]
    public class DemoDetailB
    {
        public string Bpro1 { get; set; }

        public string Bpro2 { get; set; }

        public string Bpro3 { get; set; }
    }

    [XmlTypeAttribute("list-c", AnonymousType = false)]
    public class DemoDetailC
    {
        public string Cpro1 { get; set; }

        public string Cpro2 { get; set; }

        public string Cpro3 { get; set; }
    }
    [XmlRootAttribute("master")]
    public class DemoMaster<T> where T : class
    {
        [XmlElement("attr")]
        public string DemoAttr { get; set; }

        [XmlElement("detail")]
        public DemoDetail<T> DemoDetail { get; set; } //关键点在这里,该属性元素为:detail,但其子元素根据T不同而不同

    }
    public class DemoDetail<T> : IXmlSerializable where T : class
    {
        public T body { get; set; }

        public System.Xml.Schema.XmlSchema GetSchema()
        {
            return null;
        }

        public void ReadXml(System.Xml.XmlReader reader)
        {
            string bodyStr = reader.ReadInnerXml();
            this.body = XmlHelper.XmlDeserialize<T>(bodyStr, Encoding.UTF8);
        }

        public void WriteXml(System.Xml.XmlWriter writer)
        {
            writer.WriteRaw(XmlHelper.XmlSerialize(this.body, Encoding.UTF8, true));
        }
    }

    [XmlTypeAttribute("list-a", AnonymousType = false)]
    public class DemoDetailA
    {
        public string Apro1 { get; set; }

        public string Apro2 { get; set; }

        public string Apro3 { get; set; }
    }

    [XmlTypeAttribute("list-b", AnonymousType = false)]
    public class DemoDetailB
    {
        public string Bpro1 { get; set; }

        public string Bpro2 { get; set; }

        public string Bpro3 { get; set; }
    }

    [XmlTypeAttribute("list-c", AnonymousType = false)]
    public class DemoDetailC
    {
        public string Cpro1 { get; set; }

        public string Cpro2 { get; set; }

        public string Cpro3 { get; set; }
    }

安装网址样式

简易修改多少个职位,设置站点菜单,布局和主页。
打开 Views/Shared/_Layout.cshtml 文件,实行以下改变:

  • 将三处 “ContosoUniversity” 文字修改为“Contoso University”。
  • 添加 学生、课程、教师和单位菜单,删除联系人菜单。

重在的修改如下

<html>
......
    <title>@ViewData["Title"] - Contoso University</title>
......
                class="navbar-brand">Contoso University</a>
            <div class="navbar-collapse collapse">
                <ul class="nav navbar-nav">
                    <li><a asp-area="" asp-controller="Home" asp-action="Index">Home</a></li>
                    <li><a asp-area="" asp-controller="Home" asp-action="About">About</a></li>
                    <li><a asp-area="" asp-controller="Students" asp-action="Index">Students</a></li>
                    <li><a asp-area="" asp-controller="Courses" asp-action="Index">Courses</a></li>
                    <li><a asp-area="" asp-controller="Instructors" asp-action="Index">Instructors</a></li>
                    <li><a asp-area="" asp-controller="Departments" asp-action="Index">Departments</a></li>
                </ul>
            </div>
        </div>
    </nav>
    <div class="container body-content">
        @RenderBody()
        <hr />
        <footer>
            <p>© 2017 - Contoso University</p>
        </footer>
    </div>
</body>
</html>

在 Views/Home/Index.cshtml 文件,使用以下代码替换文件的内容:

@{
    ViewData["Title"] = "Home Page";
}

<div class="jumbotron">
    <h1>Contoso University</h1>
</div>
<div class="row">
    <div class="col-md-4">
        <h2>Welcome to Contoso University</h2>
        <p>
            Contoso University is a sample application that
            demonstrates how to use Entity Framework Core in an
            ASP.NET Core MVC web application.
        </p>
    </div>
    <div class="col-md-4">
        <h2>Build it from scratch</h2>
        <p>You can build the application by following the steps in a series of tutorials.</p>
        <p><a class="btn btn-default" href="https://docs.asp.net/en/latest/data/ef-mvc/intro.html">See the tutorial »</a></p>
    </div>
    <div class="col-md-4">
        <h2>Download it</h2>
        <p>You can download the completed project from GitHub.</p>
        <p><a class="btn btn-default" href="https://github.com/aspnet/Docs/tree/master/aspnetcore/data/ef-mvc/intro/samples/cu-final">See project source code »</a></p>
    </div>
</div>

按 CT瑞鹰L+F伍 运维品种或从菜单中接纳 调试-> 起头施行(不调试),
您将在浏览器中看出本课程中实现的首页。

美高梅开户网址 5

image.png

在意下面代码中,须要关怀:德姆oDetail属性及德姆oDetail<T>类,德姆oDetail属性仅是为着生成detail成分节点,而子节点则由德姆oDetail<T>类来实行转变,DemoDetail<T>是完毕了IXmlSerializable接口,在XML连串化时,德姆oDetail<T>类仅将body属性对应的T类型实例内容展开类别化(WriteRaw),而反体系化时,则先反体系化body属性对应的T类型实例,然后赋值给body属性,这也是抢眼之处,德姆oDetail<T>类自身并从未真的加入到连串化中,故连串化的字符串也看不到DemoDetail<T>类相关的成分,德姆oDetail<T>类仅仅是一个XML系列化格式生成的中介。连串化的XML结果如下:

瞩目下边代码中,需求关心:德姆oDetail属性及德姆oDetail<T>类,德姆oDetail属性仅是为着生成detail成分节点,而子节点则由德姆oDetail<T>类来进展转变,德姆oDetail<T>是贯彻了IXmlSerializable接口,在XML体系化时,德姆oDetail<T>类仅将body属性对应的T类型实例内容展开类别化(WriteRaw),而反体系化时,则先反类别化body属性对应的T类型实例,然后赋值给body属性,那也是高强之处,德姆oDetail<T>类本人并从未真的参预到类别化中,故连串化的字符串也看不到德姆oDetail<T>类相关的要素,德姆oDetail<T>类仅仅是一个XML种类化格式生成的中介。类别化的XML结果如下:

Entity Framework Core NuGet packages

翻译注: 此标题不翻译好过翻译
要在档次中添加 EF Core 支持,要求设置相应的数据库完成。本学科使用 SQL
Server 数据库,所急需的顺序包 Microsoft.EntityFrameworkCore.SqlServer
已经停放于 Microsoft.AspNetCore.All 包中,由此我们未来怎么样都无需做。

这些程序包 (Microsoft.EntityFrameworkCore) 及其正视项
(Microsoft.EntityFrameworkCore.Relational) 提供了EF运转时补助。在稍后的
”数据库迁移“教程中,你将会学习添加2个工具包。

有关可用来 Entity Framework Core 的任何数据库帮助程序的音讯,请参阅
Data
Providers。

体系化代码:

系列化代码:

创造数据模型

接下去,您将为Contoso学院应用程序创设实体课程。 您将从以下七个实体早先。

美高梅开户网址 6

class diagram

在 Student 与 Enrollement 实体间是1个一对多的关联, 在 Course 与
Enrollment 间也存在一些多关乎。
换句话说,学生可以参预任意数量的教程,课程能够有自由数量的学生登记。

在以下部分中,您将为种种实体创设一个类。

            var demo1 = new DemoMaster<DemoDetailA>()
            {
                DemoAttr = "demo1",
                DemoDetail = new DemoDetail<DemoDetailA>() { body = new DemoDetailA() { Apro1 = "demoA1", Apro2 = "demoA2", Apro3 = "demoA3" } }
            };

            var demo2 = new DemoMaster<DemoDetailB>()
            {
                DemoAttr = "demo2",
                DemoDetail = new DemoDetail<DemoDetailB>() { body = new DemoDetailB() { Bpro1 = "demoB1", Bpro2 = "demoB2", Bpro3 = "demoB3" } }
            };

            var demo3 = new DemoMaster<DemoDetailC>()
            {
                DemoAttr = "demo3",
                DemoDetail = new DemoDetail<DemoDetailC>() { body = new DemoDetailC() { Cpro1 = "demoC1", Cpro2 = "demoC2", Cpro3 = "demoC3" } }
            };

            textBox1.Text = XmlHelper.XmlSerialize(demo1, Encoding.UTF8);

            textBox1.Text += "\r\n" + XmlHelper.XmlSerialize(demo2, Encoding.UTF8);

            textBox1.Text += "\r\n" + XmlHelper.XmlSerialize(demo3, Encoding.UTF8);
            var demo1 = new DemoMaster<DemoDetailA>()
            {
                DemoAttr = "demo1",
                DemoDetail = new DemoDetail<DemoDetailA>() { body = new DemoDetailA() { Apro1 = "demoA1", Apro2 = "demoA2", Apro3 = "demoA3" } }
            };

            var demo2 = new DemoMaster<DemoDetailB>()
            {
                DemoAttr = "demo2",
                DemoDetail = new DemoDetail<DemoDetailB>() { body = new DemoDetailB() { Bpro1 = "demoB1", Bpro2 = "demoB2", Bpro3 = "demoB3" } }
            };

            var demo3 = new DemoMaster<DemoDetailC>()
            {
                DemoAttr = "demo3",
                DemoDetail = new DemoDetail<DemoDetailC>() { body = new DemoDetailC() { Cpro1 = "demoC1", Cpro2 = "demoC2", Cpro3 = "demoC3" } }
            };

            textBox1.Text = XmlHelper.XmlSerialize(demo1, Encoding.UTF8);

            textBox1.Text += "\r\n" + XmlHelper.XmlSerialize(demo2, Encoding.UTF8);

            textBox1.Text += "\r\n" + XmlHelper.XmlSerialize(demo3, Encoding.UTF8);

Student 实体

在 Models 文件夹中,创设一个名称为 Student.cs
的类公事,并行使以下代码替换模板代码。

using System;
using System.Collections.Generic;

namespace ContosoUniversity.Models
{
    public class Student
    {
        public int ID { get; set; }
        public string LastName { get; set; }
        public string FirstMidName { get; set; }
        public DateTime EnrollmentDate { get; set; }

        public ICollection<Enrollment> Enrollments { get; set; }
    }
}

ID 属性将变成对应数据表的主键。暗许情形下,Entity Framework 将名字为 ID
或 {类名}ID 的属性解释为主键。
Enrollments
属性是导航属性。导航属性用于关联别的实体。对于3个学员实体(数据)来说,当中的
Enrollments 属性包罗全数与该学生相关联的 Enrollment
实体(数据)。也正是说,若是数据库中的3个学员行数据涉嫌七个注册行数据(一对多,在
Enrollment 表中外键关联StudentID 值为该学生的主键值),则 Student
实体中的 Enrollments 导航属性将涵盖那八个 Enrollment 实体。

设若导航属性可以包容多少个实体(在多对多或壹对多涉及中),则其体系必须是能够添加,删除和翻新条指标列表,例如ICollection
<T>。您能够钦定ICollection <T>或项目,如List
<T>或HashSet <T>。如若钦点ICollection
<T>,EF暗许成立三个HashSet <T>集合。

连串化的XML:

系列化的XML:

Enrollment 实体

美高梅开户网址 7

Enrollment

在 Models 文件夹中,创建3个名字为 Student.cs
的类公事,并动用以下代码替换模板代码。

namespace ContosoUniversity.Models
{
    public enum Grade
    {
        A, B, C, D, F
    }

    public class Enrollment
    {
        public int EnrollmentID { get; set; }
        public int CourseID { get; set; }
        public int StudentID { get; set; }
        public Grade? Grade { get; set; }

        public Course Course { get; set; }
        public Student Student { get; set; }
    }
}

EnrollmentID 属性将成为主键。本实体使用 {类名}ID 格局代表在 Studnet
实体中动用的 ID 形式。 平日你会只选拔一种形式,并在整个数据模型中使用。
在此处,区别的情势是为了演示,表明您能够利用任1形式。
在后面包车型客车教程中,您将看到什么样运用未有类名的 ID
能够更易于地在数据模型中达成持续。

Grade (等级) 属性是3个枚举类型。 Grade 类型注明后的 ? 表示
可为空类型。 1个空的等级和3个值为0的等级是不相同的 —
空表示等级未知大概尚未被赋值。

StudentID 属性是外键,相应的领航属性是 Student。 一个 Enrollment
实体与贰个 Student 实体相关联,由此该属性只好拥有保留单个 Studnet
实体(与您事先看到的能够包蕴多少个登记实体的 Student.Enrollments
导航属性不相同)。

CourseID 属性是外键, 对应的导航属性是 Course。 一个 Enrollment
实体与2个 Course 实体相关联。

当二个属性名称符合形式 <导航属性名><主键名> , EF
将质量解析为外键属性(例如,StudentID 对应 Student 导航属性,因为
Student 实体的主键是 ID美高梅开户网址 ,)。 外键属性也能够省略地使用
<主键属性名称>(例如,CourseID,因为课程实体的主键是
CourseID)。

<?xml version="1.0" encoding="utf-8"?>
<master>
    <attr>demo1</attr>
    <detail><list-a>
    <Apro1>demoA1</Apro1>
    <Apro2>demoA2</Apro2>
    <Apro3>demoA3</Apro3>
</list-a></detail>
</master>

<?xml version="1.0" encoding="utf-8"?>
<master>
    <attr>demo2</attr>
    <detail><list-b>
    <Bpro1>demoB1</Bpro1>
    <Bpro2>demoB2</Bpro2>
    <Bpro3>demoB3</Bpro3>
</list-b></detail>
</master>

<?xml version="1.0" encoding="utf-8"?>
<master>
    <attr>demo3</attr>
    <detail><list-c>
    <Cpro1>demoC1</Cpro1>
    <Cpro2>demoC2</Cpro2>
    <Cpro3>demoC3</Cpro3>
</list-c></detail>
</master>
<?xml version="1.0" encoding="utf-8"?>
<master>
    <attr>demo1</attr>
    <detail><list-a>
    <Apro1>demoA1</Apro1>
    <Apro2>demoA2</Apro2>
    <Apro3>demoA3</Apro3>
</list-a></detail>
</master>

<?xml version="1.0" encoding="utf-8"?>
<master>
    <attr>demo2</attr>
    <detail><list-b>
    <Bpro1>demoB1</Bpro1>
    <Bpro2>demoB2</Bpro2>
    <Bpro3>demoB3</Bpro3>
</list-b></detail>
</master>

<?xml version="1.0" encoding="utf-8"?>
<master>
    <attr>demo3</attr>
    <detail><list-c>
    <Cpro1>demoC1</Cpro1>
    <Cpro2>demoC2</Cpro2>
    <Cpro3>demoC3</Cpro3>
</list-c></detail>
</master>

Course 实体

美高梅开户网址 8

Course

在 Models 文件夹中,成立贰个名字为 Course.cs
的类公事,并接纳以下代码替换模板代码。

using System.Collections.Generic;
using System.ComponentModel.DataAnnotations.Schema;

namespace ContosoUniversity.Models
{
    public class Course
    {
        [DatabaseGenerated(DatabaseGeneratedOption.None)]
        public int CourseID { get; set; }
        public string Title { get; set; }
        public int Credits { get; set; }

        public ICollection<Enrollment> Enrollments { get; set; }
    }
}

Enrollments 属性是导航属性。多少个 Course 实体可以提到到任意多个
Enrollment 实体。

作者们将在本类别的继承教程中详尽介绍 DatabaseGenerated 个性。
此个性允许你钦命 Course 的主键名,而不是让数据库生成它。

三.winform DataGridView
完成钦赐列选取密码框情势展现与编写制定,以及列绑定到复合属性(即:绑定到多层次属性)
,具体贯彻代码如下:

三.winform DataGridView
完结内定列选拔密码框格局显示与编写制定,以及列绑定到复合属性(即:绑定到多层次属性),具体实现代码如下:

开创数据库上下文 Database Context

将数据模型与 Entity Framework 效率协同工作的重中之重类是数据库上下文类。
通过从 Microsoft.EntityFrameworkCore.DbContext 类派生来创立此类。
在代码中,可以内定数据模型中含有怎么样实体。 还足以自定义某个 Entity
Framework 行为。 在那一个体系中,该类被命名称为 SchoolContext

在项目文件夹中,创立二个名叫Data的文本夹。
在 Data 文件夹中创建贰个名称为 SchoolContext.cs
的新类,并用于下代码替换模板代码:

using ContosoUniversity.Models;
using Microsoft.EntityFrameworkCore;

namespace ContosoUniversity.Data
{
    public class SchoolContext : DbContext
    {
        public SchoolContext(DbContextOptions<SchoolContext> options) : base(options)
        {
        }

        public DbSet<Course> Courses { get; set; }
        public DbSet<Enrollment> Enrollments { get; set; }
        public DbSet<Student> Students { get; set; }
    }
}
            dataGridView1.CellFormatting += new DataGridViewCellFormattingEventHandler(dataGridView1_CellFormatting);
            dataGridView1.EditingControlShowing += new DataGridViewEditingControlShowingEventHandler(dataGridView1_EditingControlShowing);


        public string EvaluateValue(object obj, string property)
        {
            string retValue = string.Empty;
            string[] names = property.Split('.');

            for (int i = 0; i < names.Count(); i++)
            {
                try
                {
                    var prop = obj.GetType().GetProperty(names[i]);
                    var result = prop.GetValue(obj, null);
                    if (result != null)
                    {
                        obj = result;
                        retValue = result.ToString();
                    }
                    else
                    {
                        break;
                    }
                }
                catch (Exception)
                {
                    throw;
                }
            }

            return retValue;
        }


        private void dataGridView1_CellFormatting(object sender, DataGridViewCellFormattingEventArgs e)
        {

            if (dataGridView1.Columns[e.ColumnIndex].DataPropertyName.Contains("."))
            {
                e.Value = EvaluateValue(dataGridView1.Rows[e.RowIndex].DataBoundItem, dataGridView1.Columns[e.ColumnIndex].DataPropertyName);
            }


            if (dataGridView1.Columns[e.ColumnIndex].Name == "KeyCode")
            {
                if (e.Value != null && e.Value.ToString().Length > 0)
                {
                    e.Value = new string('*', e.Value.ToString().Length);
                }
            }
        }

        private void dataGridView1_EditingControlShowing(object sender, DataGridViewEditingControlShowingEventArgs e)
        {
            int i = this.dataGridView1.CurrentCell.ColumnIndex;
            bool usePassword = false;
            if (dataGridView1.Columns[i].Name == "KeyCode")
            {
                usePassword = true;
            }
            TextBox txt = e.Control as TextBox;
            if (txt != null)
            {
                txt.UseSystemPasswordChar = usePassword;
            }
        }

//示例:绑定的源数据类定义
    public class DemoBindClass
    {
        public string Attr { get; set; }

        public string KeyCode { get; set; }

        public DemoDetailA Detail { get; set; }
    }


    public class DemoDetailA
    {
        public string Apro1 { get; set; }

        public string Apro2 { get; set; }

        public string Apro3 { get; set; }

        public DemoDetailB DetailChild { get; set; }
    }


    public class DemoDetailB
    {
        public string Bpro1 { get; set; }

        public string Bpro2 { get; set; }

        public string Bpro3 { get; set; }
    }
            dataGridView1.CellFormatting += new DataGridViewCellFormattingEventHandler(dataGridView1_CellFormatting);
            dataGridView1.EditingControlShowing += new DataGridViewEditingControlShowingEventHandler(dataGridView1_EditingControlShowing);


        public string EvaluateValue(object obj, string property)
        {
            string retValue = string.Empty;
            string[] names = property.Split('.');

            for (int i = 0; i < names.Count(); i++)
            {
                try
                {
                    var prop = obj.GetType().GetProperty(names[i]);
                    var result = prop.GetValue(obj, null);
                    if (result != null)
                    {
                        obj = result;
                        retValue = result.ToString();
                    }
                    else
                    {
                        break;
                    }
                }
                catch (Exception)
                {
                    throw;
                }
            }

            return retValue;
        }


        private void dataGridView1_CellFormatting(object sender, DataGridViewCellFormattingEventArgs e)
        {

            if (dataGridView1.Columns[e.ColumnIndex].DataPropertyName.Contains("."))
            {
                e.Value = EvaluateValue(dataGridView1.Rows[e.RowIndex].DataBoundItem, dataGridView1.Columns[e.ColumnIndex].DataPropertyName);
            }


            if (dataGridView1.Columns[e.ColumnIndex].Name == "KeyCode")
            {
                if (e.Value != null && e.Value.ToString().Length > 0)
                {
                    e.Value = new string('*', e.Value.ToString().Length);
                }
            }
        }

        private void dataGridView1_EditingControlShowing(object sender, DataGridViewEditingControlShowingEventArgs e)
        {
            int i = this.dataGridView1.CurrentCell.ColumnIndex;
            bool usePassword = false;
            if (dataGridView1.Columns[i].Name == "KeyCode")
            {
                usePassword = true;
            }
            TextBox txt = e.Control as TextBox;
            if (txt != null)
            {
                txt.UseSystemPasswordChar = usePassword;
            }
        }

//示例:绑定的源数据类定义
    public class DemoBindClass
    {
        public string Attr { get; set; }

        public string KeyCode { get; set; }

        public DemoDetailA Detail { get; set; }
    }


    public class DemoDetailA
    {
        public string Apro1 { get; set; }

        public string Apro2 { get; set; }

        public string Apro3 { get; set; }

        public DemoDetailB DetailChild { get; set; }
    }


    public class DemoDetailB
    {
        public string Bpro1 { get; set; }

        public string Bpro2 { get; set; }

        public string Bpro3 { get; set; }
    }

应用数据上下文 – 使用注重注入

ASP.NET Core 暗中认可使用依赖注入技术。
服务(如EF数据库上下文)在应用程序运转时期通过信赖注入注册实例。
那几个急需利用服务的机件通过构造函数参数获得劳动的实例。
稍后大家得以看来控制器构造函数获取上下文实例的代码。

要将 SchoolContext 注册为劳动,请打开 Startup.cs,并依据如下代码修改
ConfigureServices 方法。

//Startup.cs
public void ConfigureServices(IServiceCollection services)
{
    services.AddDbContext<SchoolContext>(options =>
        options.UseSqlServer(Configuration.GetConnectionString("DefaultConnection")));

    services.AddMvc();
}

因而调用 DbContextOptionsBuilder
对象上的法子将再而三字符串的名号传递给上下文。 对于本地开发,ASP.NET Core
配置种类从 appsettings.json 文件读取连接字符串。

打开appsettings.json文件并充分一个再三再四字符串,如下例所示。

{
  "ConnectionStrings": {
    "DefaultConnection": "Server=(localdb)\\mssqllocaldb;Database=ContosoUniversity1;Trusted_Connection=True;MultipleActiveResultSets=true"
  },
  "Logging": {
    "IncludeScopes": false,
    "LogLevel": {
      "Default": "Warning"
    }
  }
}

绑定到数据源:

绑定到数据源:

SQL Server Express LocalDB

连天字符串钦点 SQL Server LocalDB 数据库。 LocalDB 是 SQL Server Express
数据库引擎的轻量级版本,意在用于应用程序开发,而不是生育用途。 LocalDB
按需运行并以用户方式运作,由此未有复杂的配备。 默许意况下,LocalDB在
C:/Users/<user> 目录中成立 .mdf 数据库文件。

            var demo = new[] {
                new DemoBindClass()
                    {
                        Attr = "demo",
                        KeyCode="a123456789b",
                        Detail = new DemoDetailA()
                        {
                            Apro1 = "demoA1",
                            Apro2 = "demoA2",
                            Apro3 = "demoA3",
                            DetailChild = new DemoDetailB()
                            {
                                Bpro1 = "demoB1",
                                Bpro2 = "demoB2",
                                Bpro3 = "demoB3"
                            }
                        }
                    }
            };


            dataGridView1.AutoGenerateColumns = false;
            dataGridView1.DataSource = demo;
            var demo = new[] {
                new DemoBindClass()
                    {
                        Attr = "demo",
                        KeyCode="a123456789b",
                        Detail = new DemoDetailA()
                        {
                            Apro1 = "demoA1",
                            Apro2 = "demoA2",
                            Apro3 = "demoA3",
                            DetailChild = new DemoDetailB()
                            {
                                Bpro1 = "demoB1",
                                Bpro2 = "demoB2",
                                Bpro3 = "demoB3"
                            }
                        }
                    }
            };


            dataGridView1.AutoGenerateColumns = false;
            dataGridView1.DataSource = demo;
累加代码,使用测试数据开始化数据库

EF 将为您创设四个空数据库。
在本节中,您将编辑3个创建数据库后调用的不二法门,以便利用测试数据开始展览填空。

在此地,您将使用 EnsureCreated 方法自动创造数据库。
在末端的学科中,您将见到如何使用 Code First Migration (代码优先迁移)
来更改数据库架构而不是删除和再度成立数据库来处理框架结构更改。

Data 文件夹中,创制一个名字为 DbInitializer.cs
的新类文件,并应用以下代码替换模板代码,这个代码将在须要时创设数据库,并将测试数据加载到新数据库中。

//DbInitializer.cs

using ContosoUniversity.Models;
using System;
using System.Linq;

namespace ContosoUniversity.Data
{
    public static class DbInitializer
    {
        public static void Initialize(SchoolContext context)
        {
            context.Database.EnsureCreated();

            // Look for any students.
            if (context.Students.Any())
            {
                return;   // DB has been seeded
            }

            var students = new Student[]
            {
            new Student{FirstMidName="Carson",LastName="Alexander",EnrollmentDate=DateTime.Parse("2005-09-01")},
            new Student{FirstMidName="Meredith",LastName="Alonso",EnrollmentDate=DateTime.Parse("2002-09-01")},
            new Student{FirstMidName="Arturo",LastName="Anand",EnrollmentDate=DateTime.Parse("2003-09-01")},
            new Student{FirstMidName="Gytis",LastName="Barzdukas",EnrollmentDate=DateTime.Parse("2002-09-01")},
            new Student{FirstMidName="Yan",LastName="Li",EnrollmentDate=DateTime.Parse("2002-09-01")},
            new Student{FirstMidName="Peggy",LastName="Justice",EnrollmentDate=DateTime.Parse("2001-09-01")},
            new Student{FirstMidName="Laura",LastName="Norman",EnrollmentDate=DateTime.Parse("2003-09-01")},
            new Student{FirstMidName="Nino",LastName="Olivetto",EnrollmentDate=DateTime.Parse("2005-09-01")}
            };
            foreach (Student s in students)
            {
                context.Students.Add(s);
            }
            context.SaveChanges();

            var courses = new Course[]
            {
            new Course{CourseID=1050,Title="Chemistry",Credits=3},
            new Course{CourseID=4022,Title="Microeconomics",Credits=3},
            new Course{CourseID=4041,Title="Macroeconomics",Credits=3},
            new Course{CourseID=1045,Title="Calculus",Credits=4},
            new Course{CourseID=3141,Title="Trigonometry",Credits=4},
            new Course{CourseID=2021,Title="Composition",Credits=3},
            new Course{CourseID=2042,Title="Literature",Credits=4}
            };
            foreach (Course c in courses)
            {
                context.Courses.Add(c);
            }
            context.SaveChanges();

            var enrollments = new Enrollment[]
            {
            new Enrollment{StudentID=1,CourseID=1050,Grade=Grade.A},
            new Enrollment{StudentID=1,CourseID=4022,Grade=Grade.C},
            new Enrollment{StudentID=1,CourseID=4041,Grade=Grade.B},
            new Enrollment{StudentID=2,CourseID=1045,Grade=Grade.B},
            new Enrollment{StudentID=2,CourseID=3141,Grade=Grade.F},
            new Enrollment{StudentID=2,CourseID=2021,Grade=Grade.F},
            new Enrollment{StudentID=3,CourseID=1050},
            new Enrollment{StudentID=4,CourseID=1050},
            new Enrollment{StudentID=4,CourseID=4022,Grade=Grade.F},
            new Enrollment{StudentID=5,CourseID=4041,Grade=Grade.C},
            new Enrollment{StudentID=6,CourseID=1045},
            new Enrollment{StudentID=7,CourseID=3141,Grade=Grade.A},
            };
            foreach (Enrollment e in enrollments)
            {
                context.Enrollments.Add(e);
            }
            context.SaveChanges();
        }
    }
}

代码检查数据库中是或不是有学生,假使未有,则只要数据库是新的,并且要求利用测试数据实行种子。它将测试数据加载到数组而不是
List <T> 集合来优化质量。

在Program.cs中,修改Main方法在应用程序运行时执行以下操作:

  • 从依赖注入容器获取数据库上下文实例。
  • 调用种子方法,传递给它的上下文。
  • 种子方法成功时销毁上下文。

public static void Main(string[] args)
{
    var host = BuildWebHost(args);

    using (var scope = host.Services.CreateScope())
    {
        var services = scope.ServiceProvider;
        try
        {
            var context = services.GetRequiredService<SchoolContext>();
            DbInitializer.Initialize(context);
        }
        catch (Exception ex)
        {
            var logger = services.GetRequiredService<ILogger<Program>>();
            logger.LogError(ex, "An error occurred while seeding the database.");
        }
    }

    host.Run();
}

在较旧的教程中,您或许会在Startup.cs中的Configure方法中来看类似的代码。
我们建议您仅使用Configure方法来安装请求管道。
应用程序运营代码属于Main方法。

第一遍运转应用程序时,将创立数据库并植入测试数据。
无论几时转移数据模型,都能够去除数据库,更新种子方法,并以新的数据库重新初阶重复启航。
在后头的学科中,您将看到在数据模型更改时怎么样修改数据库,而不删除和重复创立它。

兑现钦赐列采纳密码框情势呈现与编辑,以及列绑定到复合属性均要求订阅DataGridView的CellFormatting及艾德itingControlShowing事件,并在里面写转换当前Cell的Value,完成列绑定到复合属性,关键点在:伊娃luateValue方法,该措施逻辑很简短,正是依照绑定的性格层级(.分隔)层层遍历获取属性的值,直到遍历完或为空时甘休,最后取得的结果就是绑定的品质的值。最后兑现的作用如下图示:

贯彻内定列选择密码框形式呈现与编辑,以及列绑定到复合属性均供给订阅DataGridView的CellFormatting及艾德itingControlShowing事件,并在个中写转换当前Cell的Value,达成列绑定到复合属性,关键点在:伊娃luateValue方法,该措施逻辑非常的粗略,正是基于绑定的属性层级(.分隔)层层遍历获取属性的值,直到遍历完或为空时停止,最终得到的结果就是绑定的习性的值。最后落实的功效如下图示:

成立控制器和视图

接下去,您将使用 Visual Studio 脚手架增加 MVC 控制器和视图,并选用 EF
来询问和保存数据。

电动成立CRUD操作方法和视图称为脚手架。
脚手架与代码生成器分化之处在于,脚手架代码只是基础代码,您能够依照本人的内需展开改动,而常见情况下,您不会修改生成器生成的代码。
当您须要自定义生成器生成的代码,能够动用部分类,可能在状态发生转移时时重新生成代码。

  • 右键单击消除方案财富管理器中的 Controllers 文件夹,然后选用 添加
    -> 控制器。
  • 在“添加基架”对话框中,选择“视图使用 Entity Framework 的 MVC
    控制器”,点击“添加”
  • 在“添加控制器”对话框中:
    • 模型类选拔 Student
    • 多少上下文类选取 SchoolContext
    • 点击 “添加”。

美高梅开户网址 9

new Controller

当您单击添加时,Visual Studio 脚手架引擎创立三个 StudentsController.cs
文件和1组与控制器壹起行使的视图(.cshtml文件)。

(脚手架引擎还足以为您成立数据库上下文,要是你不像从前在本教程中那样手动创制它。
您可以由此单击数据上下文类左侧的加号在“添加控制器”框中钦点新的左右文类。
然后Visual Studio将开创您的DbContext类以及控制器和视图。)

你会专注到控制器将多个 SchoolContext 作为二个构造函数参数。

namespace ContosoUniversity.Controllers
{
    public class StudentsController : Controller
    {
        private readonly SchoolContext _context;

        public StudentsController(SchoolContext context)
        {
            _context = context;
        }

ASP.NET 注重注入负责将 SchoolContext 的3个实例传递到控制器中。
前文中,已经
在 Startup.cs 文件中布局 SchoolContext 的依靠注入。

控制器包蕴三个 Index 方法,用于显示数据库中的全体学员。
该方法通过读取数据库上下文实例的 Students
属性获取学生实体集中的学习者列表:

public async Task<IActionResult> Index()
{
    return View(await _context.Students.ToListAsync());
}

稍后将介绍此代码中的异步编制程序知识。

视图 Views/Students/Index.cshtml 使用 HTML 表格展现学生列表。
(此处未对脚手架生成的代码进行任何修改,不再贴代码占用作品篇幅。 )

按 CT奥德赛L + F5 运转项目或从菜单中采用 调节和测试 -> 起头实行(不调节和测试)。

单击 Student 链接,能够看来 DbInitializer.Initialize
方法中插入的测试数据。 依照浏览器窗口的狭隘程度,您会看出页面顶部的
Student
链接,也有极大可能率您必须单击右上角的导航空图标才能观察隐藏菜单中的链接。

美高梅开户网址 10

narrow Page

美高梅开户网址 11

sdudent index

美高梅开户网址 12

美高梅开户网址 13

查阅数据库

当你运维应用程序时,DbInitializer.Initialize 方法调用 EnsureCreated 。
EF 看到没有数据库,所以它创立了2个,然后 Initialize
方法代码的别的部分用数据填充数据库。 在 Visual Studio 中,您能够采纳 SQL
Server 对象财富管理器(SSOX)查看数据库。

假定 SSOX 窗口未有打开,在 Visual Studio 中,点击菜单 “视图” -> “SQL
Server 对象财富管理器”。
在 SSOX 中,单击(localdb)\ MSSQLLocalDB > 数据库,然后单击
ContosoUniversity壹,也正是我们后面在 appsettings.json
文件中安装的连年字符串中数据库名称。
开始展览“表”节点以查看数据库中的表。

美高梅开户网址 14

ssox

右键单击 Student 表,然后单击 “查看数据”
以查看已创立的列和插入到表中的数据行。

美高梅开户网址 15

student table

.mdf 和.ldf 数据库文件位于C:\Users<你的用户名> 文件夹中。
因为你在应用程序运营时运营的开端化程序方法中调用 EnsureCreated
,所以今后得以更改 Student
类,删除数据库,再一次运营应用程序,并自动重新创建数据库以万分您的转移。
例如,要是你将 EmailAddress 属性添加到 Student
类,则会在重复创建的表中看到3个新的EmailAddress 列。

四.运用BCP(sqlbulkcopy)来落到实处三个区别数据库之间实行多少差距传输(即:数据同步)

四.利用BCP(sqlbulkcopy)来落到实处四个例外数据库之间举行数量差异传输(即:数据同步)

约定

传说约定优于配备的原则,Entity Framework
营造二个数据库时,你所需书写的代码很少。

  • DbSet 属性的名称作为表名。
    对于未由DbSet属性引用的实业,实体类名用作表名。

  • 实业性质名称用于列名。

  • 名叫 ID 或 classnameID 的实体性质被识别为主键属性。

  • 动用 导航属性名+实体主键名 命名的品质,会被自动识别为外键,例如:
    StudentID 由 Student (导航属性) + ID (Student实体主键名
    )组成。外键也足以总结只行使实体主键名,例如 EnrollmentID (外键) 与
    EnrollmentID (Enrollment 实体的主键)。

预定能够被掩盖。例如,你能够显式钦定表名,如本学科前边所见到的。
您能够设置列名称并将别的性质设置为主键或外键,那将在后头的科目中谈到。

TransferBulkCopy作用:达成七个分歧数据库之间开始展览多少差异传输,BuildInsertOrUpdateToDestTableSql功能:依据指标表及一时表生成更新与插入记录的SQL语句,以此达成:若同步的数据已存在,则更新,不设有,则插入。

TransferBulkCopy效用:达成五个例外数据库之间展开数量差别传输,BuildInsertOrUpdateToDestTableSql作用:依照目标表及一时表生成更新与插入记录的SQL语句,以此完成:若同步的数额已存在,则更新,不设有,则插入。

异步代码

ASP.NET Core和EF Core的默许使用异步编制程序。

Web
服务器的可用线程数量少于,在高负荷景况下,全部可用线程都或许都在使用。
当爆发那种情况时,服务器无法处理新的请求,直到线程被保释。
使用同步代码时,许八线程或许会被绑定,而实际上它们并不曾做其余工作,因为它们正在守候
I/O 落成。 使用异步代码,当进程正在等候I/O
完毕时,其线程将被假释,供服务器用于拍卖任何请求。
因而,异步代码可以更使得地应用服务器能源,并且使服务器能够无延迟地拍卖更加多流量。

异步代码在运转时引入了少量的开支,不过对于低流量情状,品质下跌能够忽略不计,而对此高流量意况,潜在的性子升高是惊天动地的。
在偏下代码中,async 关键字, Task<T> 重回值,await 关键字和
ToListAsync 方法共同组成异步执行代码。

public async Task<IActionResult> Index()
{
    return View(await _context.Students.ToListAsync());
}
  • async 关键字告诉编写翻译器为格局体生成回调函数,并活动创造重回的
    Task <IActionResult> 对象。

  • 重回类型 Task<IActionResult> 表示正在开始展览的劳作,其结果类型为
    IActionResult

  • await 关键字告诉编写翻译器将该格局分为两有的。
    第1有的以异步运营的操作截至。
    第一片段被放入回调方法,该操作在操作达成时被调用。

  • ToListAsyncToList 扩充方法的异步版本。

当您编写使用实体框架的异步代码时,供给注意的有个别作业:

  • 唯有会吸引查询或将指令发送到数据库的语句才需求异步执行。 这包蕴诸如
    ToListAsyncSingleOrDefaultAsyncSaveChangesAsync
    它不应当包蕴,例如,只是改变IQueryable的口舌,类似
    var students = context.Students.Where(s => s.LastName == "Davolio")
    那样的言辞。

  • EF上下文不是线程安全的:不要品味并行执行七个操作。 当您调用任何异步
    EF 方法时,请始终使用 await 关键字。

  • 倘使你想选取异步代码的品质优势,请确定保障您正在使用的别的库包(例如用于分页)也应用异步,如若她们调用任何导致查询发送到数据库的艺术。

有关.NET中异步编制程序的更加多新闻,请参阅 Async
Overview。

        /// <summary>
        /// 通用数据传输方法(采用SqlBulkCopy快速批量插入,然后再进行处理)
        /// </summary>
        /// <param name="sourceSelectSql"></param>
        /// <param name="sourceConn"></param>
        /// <param name="destTableName"></param>
        /// <param name="destConn"></param>
        /// <param name="colMapFunc"></param>
        /// <param name="lastSaveAction"></param>
        public void TransferBulkCopy(string sourceSelectSql, SqlConnection sourceConn, string destTableName, SqlConnection destConn, Func<DataTable, Dictionary<string, string>> colMapFunc,
                                    Func<string, DataTable, SqlConnection, SqlConnection, bool> lastSaveAction, bool closeConnection = true)
        {
            DataTable srcTable = new DataTable();
            SqlDataAdapter srcAdapter = new SqlDataAdapter(sourceSelectSql, sourceConn);
            srcAdapter.AcceptChangesDuringUpdate = false;
            SqlCommandBuilder srcCmdBuilder = new SqlCommandBuilder(srcAdapter);
            srcAdapter.Fill(srcTable);

            if (srcTable != null && srcTable.Rows.Count > 0)
            {
                string tempDestTableName = "#temp_" + destTableName;
                ClsDatabase.gExecCommand(destConn, string.Format("select top 0 * into {0}  from  {1}", tempDestTableName, destTableName), false);
                List<string> mapDestColNameList = new List<string>();
                using (SqlBulkCopy sqlBulkCopy = new SqlBulkCopy(destConn))
                {
                    sqlBulkCopy.DestinationTableName = tempDestTableName;
                    foreach (var map in colMapFunc(srcTable))
                    {
                        sqlBulkCopy.ColumnMappings.Add(map.Key, map.Value);
                        mapDestColNameList.Add(map.Value);
                    }
                    sqlBulkCopy.WriteToServer(srcTable);
                }
                srcTable.ExtendedProperties.Add(MapDestColNames_String, mapDestColNameList);
                bool needUpdate = lastSaveAction(tempDestTableName, srcTable, destConn, sourceConn);

                if (needUpdate)
                {
                    if (srcTable.Columns.Contains("TranFlag"))
                    {
                        foreach (DataRow row in srcTable.Rows)
                        {
                            row["TranFlag"] = true;
                        }
                    }
                    srcAdapter.Update(srcTable);
                }

            }

            if (closeConnection)
            {
                DisposeConnections(sourceConn, destConn);
            }
        }
        /// <summary>
        /// 通用数据传输方法(采用SqlBulkCopy快速批量插入,然后再进行处理)
        /// </summary>
        /// <param name="sourceSelectSql"></param>
        /// <param name="sourceConn"></param>
        /// <param name="destTableName"></param>
        /// <param name="destConn"></param>
        /// <param name="colMapFunc"></param>
        /// <param name="lastSaveAction"></param>
        public void TransferBulkCopy(string sourceSelectSql, SqlConnection sourceConn, string destTableName, SqlConnection destConn, Func<DataTable, Dictionary<string, string>> colMapFunc,
                                    Func<string, DataTable, SqlConnection, SqlConnection, bool> lastSaveAction, bool closeConnection = true)
        {
            DataTable srcTable = new DataTable();
            SqlDataAdapter srcAdapter = new SqlDataAdapter(sourceSelectSql, sourceConn);
            srcAdapter.AcceptChangesDuringUpdate = false;
            SqlCommandBuilder srcCmdBuilder = new SqlCommandBuilder(srcAdapter);
            srcAdapter.Fill(srcTable);

            if (srcTable != null && srcTable.Rows.Count > 0)
            {
                string tempDestTableName = "#temp_" + destTableName;
                ClsDatabase.gExecCommand(destConn, string.Format("select top 0 * into {0}  from  {1}", tempDestTableName, destTableName), false);
                List<string> mapDestColNameList = new List<string>();
                using (SqlBulkCopy sqlBulkCopy = new SqlBulkCopy(destConn))
                {
                    sqlBulkCopy.DestinationTableName = tempDestTableName;
                    foreach (var map in colMapFunc(srcTable))
                    {
                        sqlBulkCopy.ColumnMappings.Add(map.Key, map.Value);
                        mapDestColNameList.Add(map.Value);
                    }
                    sqlBulkCopy.WriteToServer(srcTable);
                }
                srcTable.ExtendedProperties.Add(MapDestColNames_String, mapDestColNameList);
                bool needUpdate = lastSaveAction(tempDestTableName, srcTable, destConn, sourceConn);

                if (needUpdate)
                {
                    if (srcTable.Columns.Contains("TranFlag"))
                    {
                        foreach (DataRow row in srcTable.Rows)
                        {
                            row["TranFlag"] = true;
                        }
                    }
                    srcAdapter.Update(srcTable);
                }

            }

            if (closeConnection)
            {
                DisposeConnections(sourceConn, destConn);
            }
        }


        /// <summary>
        /// 生成同步插入及更新目的表SQL语句
        /// </summary>
        /// <param name="destTableName"></param>
        /// <param name="tempdestTableName"></param>
        /// <param name="pkWhere"></param>
        /// <param name="mapDestColNames"></param>
        /// <returns></returns>
        public string BuildInsertOrUpdateToDestTableSql(string destTableName, string tempdestTableName, string[] pkWhereColNames, object mapDestColNames)
        {
            var mapDestColNameList = mapDestColNames as List<string>;
            string updateColNames = null;
            foreach (string col in mapDestColNameList)
            {
                if (!pkWhereColNames.Contains(col, StringComparer.OrdinalIgnoreCase))
                {
                    updateColNames += string.Format(",{0}=tnew.{0}", col);
                }
            }
            updateColNames = updateColNames.Substring(1);
            string insertColNames = string.Join(",", mapDestColNameList);

            string pkWhereSql = null;
            foreach (string col in pkWhereColNames)
            {
                pkWhereSql += string.Format("and told.{0}=tnew.{0}", col);
            }
            pkWhereSql = pkWhereSql.Substring(3);

            StringBuilder sqlBuilder = new StringBuilder();
            sqlBuilder.AppendFormat("UPDATE {0}  SET {1} FROM {0} told INNER JOIN {2} tnew ON {3}  " + Environment.NewLine,
                                    destTableName, updateColNames, tempdestTableName, pkWhereSql);
            sqlBuilder.AppendFormat("INSERT INTO {0}({1}) SELECT {1} FROM {2} tnew WHERE NOT EXISTS(SELECT 1 FROM {0} told WHERE {3}) " + Environment.NewLine,
                                    destTableName, insertColNames, tempdestTableName, pkWhereSql);

            return sqlBuilder.ToString();

        }

小结

您今后创制了2个不难的应用程序,使用 Entity Framework Core 和 SQL Server
Express LocalDB 存款和储蓄和出示数据。 在上面包车型地铁学科中,您将学习怎么样履行基本的
CRUD(创制,读取,更新和删除)操作。

 

应用示例如下:

        /// <summary>
        /// 生成同步插入及更新目的表SQL语句
        /// </summary>
        /// <param name="destTableName"></param>
        /// <param name="tempdestTableName"></param>
        /// <param name="pkWhereColNames"></param>
        /// <param name="mapDestColNames"></param>
        /// <param name="sqlType">0=生成INSERT与UPDATE;1=生成UPDATE语句;2=生成INSERT语句</param>
        /// <returns></returns>
        public string BuildInsertOrUpdateToDestTableSql(string destTableName, string tempdestTableName, string[] pkWhereColNames, object mapDestColNames, int sqlType = 0)
        {
            var mapDestColNameList = mapDestColNames as List<string>;
            string updateColNames = null;
            foreach (string col in mapDestColNameList)
            {
                if (!pkWhereColNames.Contains(col, StringComparer.OrdinalIgnoreCase))
                {
                    updateColNames += string.Format(",{0}=tnew.{0}", col);
                }
            }
            updateColNames = updateColNames.Substring(1);
            string insertColNames = string.Join(",", mapDestColNameList);

            string pkWhereSql = null;
            foreach (string col in pkWhereColNames)
            {
                pkWhereSql += string.Format(" and told.{0}=tnew.{0} ", col);
            }
            pkWhereSql = pkWhereSql.Trim().Substring(3);

            StringBuilder sqlBuilder = new StringBuilder();

            if (sqlType == 0 || sqlType == 1)
            {
                sqlBuilder.AppendFormat("UPDATE {0}  SET {1} FROM {0} told INNER JOIN {2} tnew ON {3}  " + Environment.NewLine,
                                        destTableName, updateColNames, tempdestTableName, pkWhereSql);
            }

            if (sqlType == 0 || sqlType == 2)
            {
                sqlBuilder.AppendFormat("INSERT INTO {0}({1}) SELECT {1} FROM {2} tnew WHERE NOT EXISTS(SELECT 1 FROM {0} told WHERE {3}) " + Environment.NewLine,
                                        destTableName, insertColNames, tempdestTableName, pkWhereSql);

            }

            return sqlBuilder.ToString();

        }
        public void SendData_CustomerAuthorization()
        {
            try
            {
                SqlConnection obConnLMS1 = new SqlConnection(master.connLMSStr);
                SqlConnection obConnWEB1 = new SqlConnection(master.connWEBStr);

                string selectSql = @"SELECT TOP {0} Id,Phone,Mac,IsSet,LastLoginTime,PCName,TranFlag FROM TWEB_CustomerAuthorization WHERE TranFlag=0 ORDER BY Id ";

                selectSql = string.Format(selectSql, master.batchSize);

                master.TransferBulkCopy(selectSql, obConnWEB1,
                                "TB_CustomerAuthorization", obConnLMS1,
                                 (stable) =>
                                 {
                                     var colMaps = new Dictionary<string, string>();
                                     foreach (DataColumn col in stable.Columns)
                                     {
                                         if (!col.ColumnName.Equals("TranFlag", StringComparison.OrdinalIgnoreCase))
                                         {
                                             colMaps.Add(col.ColumnName, col.ColumnName);
                                         }
                                     }
                                     return colMaps;
                                 },
                                 (tempTableName, stable, destConn, srcConn) =>
                                 {
                                     StringBuilder saveSqlBuilder = new StringBuilder("begin tran" + Environment.NewLine);

                                     string IUSql = master.BuildInsertOrUpdateToDestTableSql("TB_CustomerAuthorization", tempTableName, new[] { "Id" }, stable.ExtendedProperties[master.MapDestColNames_String]);
                                     saveSqlBuilder.Append(IUSql);

                                     saveSqlBuilder.AppendLine("commit");

                                     ClsDatabase.gExecCommand(destConn, saveSqlBuilder.ToString());

                                     master.WriteMsg(master.lstSended, string.Format("上传时间:{0:yyyy-MM-dd HH:mm}\t SendData_CustomerAuthorization \t Succeed:{1}", DateTime.Now, stable.Rows.Count));

                                     return true;

                                 });
            }
            catch (Exception ex)
            {
                master.WriteMsg(master.lstErrorInfo, DateTime.Now.ToString("yyyy-MM-dd HH:mm") + "\t" + "SendData_CustomerAuthorization" + "\t" + ex.Message.ToString());
            }
        }

  

一起原理如下:

 

一.定义好查询源服务器的急需联合的表(1般表中大家定义1个用以是不是同步的标识字段,如:TranFlag
Bit类型,0意味着新数据,未共同,一意味已1起);

运用示例如下:

二.查询源服务器的内需共同的表的笔录(1般是TranFlag=0的笔录),利用SqlDataAdapter+SqlCommandBuilder
装载Dataset,指标是持续能够利用SqlData艾达pter直接生成更新命令并实施;

        public void SendData_CustomerAuthorization()
        {
            try
            {
                SqlConnection obConnLMS1 = new SqlConnection(master.connLMSStr);
                SqlConnection obConnWEB1 = new SqlConnection(master.connWEBStr);

                string selectSql = @"SELECT TOP {0} Id,Phone,Mac,IsSet,LastLoginTime,PCName,TranFlag FROM TWEB_CustomerAuthorization WHERE TranFlag=0 ORDER BY Id ";

                selectSql = string.Format(selectSql, master.batchSize);

                master.TransferBulkCopy(selectSql, obConnWEB1,
                                "TB_CustomerAuthorization", obConnLMS1,
                                 (stable) =>
                                 {
                                     var colMaps = new Dictionary<string, string>();
                                     foreach (DataColumn col in stable.Columns)
                                     {
                                         if (!col.ColumnName.Equals("TranFlag", StringComparison.OrdinalIgnoreCase))
                                         {
                                             colMaps.Add(col.ColumnName, col.ColumnName);
                                         }
                                     }
                                     return colMaps;
                                 },
                                 (tempTableName, stable, destConn, srcConn) =>
                                 {
                                     StringBuilder saveSqlBuilder = new StringBuilder("begin tran" + Environment.NewLine);

                                     string IUSql = master.BuildInsertOrUpdateToDestTableSql("TB_CustomerAuthorization", tempTableName, new[] { "Id" }, stable.ExtendedProperties[master.MapDestColNames_String]);
                                     saveSqlBuilder.Append(IUSql);

                                     saveSqlBuilder.AppendLine("commit");

                                     ClsDatabase.gExecCommand(destConn, saveSqlBuilder.ToString());

                                     master.WriteMsg(master.lstSended, string.Format("上传时间:{0:yyyy-MM-dd HH:mm}\t SendData_CustomerAuthorization \t Succeed:{1}", DateTime.Now, stable.Rows.Count));

                                     return true;

                                 });
            }
            catch (Exception ex)
            {
                master.WriteMsg(master.lstErrorInfo, DateTime.Now.ToString("yyyy-MM-dd HH:mm") + "\t" + "SendData_CustomerAuthorization" + "\t" + ex.Message.ToString());
            }
        }

三.采取insert
into从目标服务器的将被同步的表复制结构爆发三个权且表,表名壹般是:#temp_目标服务器的将被1起表名
,那样一时半刻表与实体表的构造完全一致;

一道原理如下:

四.实例化1个SqlBulkCopy,并树立源服务器的内需1块的表字段与指标暂时表字段的照耀,然后实施跨服务器传输;

四.一.定义好查询源服务器的急需一块的表(1般表中我们定义一个用于是不是同步的标识字段,如:TranFlag
Bit类型,0表示新数据,未共同,一象征已联合);

伍.施用 BuildInsertOrUpdateToDestTableSql 方法 ,生成
 目标服务器的将被同步的表 与
一时半刻表的插入与立异SQL语句(未来在同八个库了,想怎么用SQL语句均可)  

四.二.查询源服务器的必要共同的表的记录(一般是TranFlag=0的笔录),利用SqlDataAdapter+SqlCommandBuilder
装载Dataset,指标是后续能够动用SqlDataAdapter直接扭转更新命令并推行;

陆.为有限支撑一致性,故外层还需包裹事务SQL语句,若还需到场别的处理SQL,能够加在begin
tran  …
commit代码块中即可,最终执行SQL语句:gExecCommand(ClsDatabase.gExecCommand是二个SQLDB
HELPEKuga 类的执行SQL命令的章程)

四.3.选取insert
into从指标服务器的将被1并的表复制结构发生三个近日表,表名壹般是:#temp_指标服务器的将被同台表名
,那样近年来表与实身体表面的布局完全①致;

 

四.4.实例化贰个SqlBulkCopy,并确立源服务器的内需共同的表字段与目标一时半刻表字段的映照,然后实施跨服务器传输;

四.⑤.应用 BuildInsertOrUpdateToDestTableSql 方法 ,生成
 目标服务器的将被1块的表 与
目前表的插入与立异SQL语句(今后在同二个库了,想怎么用SQL语句均可)  

四.陆.为确定保证1致性,故外层还需包裹事务SQL语句,若还需加入其余处理SQL,能够加在begin
tran  …
commit代码块中即可,最终执行SQL语句:gExecCommand(ClsDatabase.gExecCommand是一个SQLDB
HELPEPRADO 类的履行SQL命令的艺术)

 

伍.完毕同1个WINDOWS SECRUISERVICE程序
COPY多份,然后经过变更自定义的劳动ID(ServiceID)配置项来达成:同二个服务程序安装成八个不等的WINDOWS服务进程:

5.壹.创立1个WINDOWS服务项目,在ProjectInstaller设计器界面通过右键弹出菜谱接纳安装程序(serviceProcessInstaller壹、serviceInstaller1)、并安装好ServiceName、DisplayName、Description、Account等,如下图示:

美高梅开户网址 16

5.2.在ProjectInstaller构造函数中加进从CONFIG文件中读取自定义的劳务ID(ServiceID)配置项的值,然后将瑟维斯ID拼加到预设的ServiceName后边,以便实际依据瑟维斯ID能够设置成不一致瑟维斯ID后缀的服务进程,关键点在于改变ServiceName,另三个关键点是从CONFIG文件中取得ServiceID,由于设置时,守旧的艺术不也许符合规律读取到CONFIG,只可以通过Assembly.GetExecutingAssembly().Location
来获得当前履行的程序集的路子再拼成CONFIG文件路径,最终读出ServiceID的值,示例代码如下:

    public partial class ProjectInstaller : System.Configuration.Install.Installer
    {
        public ProjectInstaller()
        {
            InitializeComponent();

            string assyLocation = System.Reflection.Assembly.GetExecutingAssembly().Location;
            string assyCfgPath = assyLocation + ".config";
            string installServiceLogPath = Path.Combine(Path.GetDirectoryName(assyLocation), "InstallServiceLog.log");

            string serviceID = ConfigUtil.GetAppSettingValueForConfigPath("ServiceID", assyCfgPath);

            System.IO.File.AppendAllText(installServiceLogPath, string.Format("[{0:yyyy-MM-dd HH:mm:ss}] ServiceAssembly ConfigPath:{1};\r\n", DateTime.Now, assyCfgPath));

            if (!string.IsNullOrWhiteSpace(serviceID))
            {
                this.serviceInstaller1.DisplayName = "TestService_" + serviceID;
                this.serviceInstaller1.ServiceName = "TestService_" + serviceID;
            }

            System.IO.File.AppendAllText(installServiceLogPath, string.Format("[{0:yyyy-MM-dd HH:mm:ss}] ProjectInstaller.ProjectInstaller() ->ServiceID:{1},ServiceName:{2}; \r\n", DateTime.Now, serviceID, this.serviceInstaller1.ServiceName));
        }
    }

5.叁.在服务类的构造函数中壹样扩展从CONFIG中读取自定义的劳务ID(ServiceID)配置项的值,然后将ServiceID拼加到预设的ServiceName前边(注意应与上述ProjectInstaller中钦定的ServiceName相同),示例代码如下:  

public partial class TestService: ServiceBase
{
     public TestService()
    {
          serviceID = ConfigUtil.GetAppSettingValue("ServiceID");
            if (!string.IsNullOrWhiteSpace(serviceID))
            {
                this.ServiceName = "TestService_" + serviceID;
            }
    }

}

 上述三步就完事了同三个服务程序安装成四个不等的WINDOWS服务进程,这么些依旧相比实用的啊!上述ConfigUtil是包装的3个布局文件读写支持类,之前文章有介绍,前面也会发布二个更完整的ConfigUtil类。

发表评论

电子邮件地址不会被公开。 必填项已用*标注

网站地图xml地图