一.暗中认可EF生成的连日字符串相比较的长和奇怪,若想使用普通的总是字符串来连接EF,则足以经过创办分项目,同仁一视写四个构造函数,在构造函数中经过动态拼接EntityConnectionString获得EF所需的接二连三字符串,具代完毕代码如下:

壹.默许EF生成的几次三番字符串相比的长和奇怪,若想使用普通的连日字符串来连接EF,则足以经过创办分档次,一碗水端平写二个构造函数,在构造函数中经过动态拼接EntityConnectionString获得EF所需的接连字符串,具代完成代码如下:

方今支出品种中用到的编码小技巧汇总表达,开发项目小技巧

壹.暗中同意EF生成的连日字符串比较的长和奇怪,若想使用普通的总是字符串来连接EF,则足以经过成立分种类,仁同一视写2个构造函数,在构造函数中经过动态拼接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方法是1个静态方法,在BuildEntityConnectionString方法中ProviderConnectionString
= EncryptUtility.DesDecrypt(“XXXXX”,
connStrSetting.ConnectionString);是第一,小编那边是对config中的连接字符串
也都进展了加密,故此处笔者急需解密,若无那些必要能够直接:ProviderConnectionString
=connStrSetting.ConnectionString即可。后续实例化EF上下文对象时,请使用:DataEntities(string
connName)那些布局涵数即可,DataEntities是有血有肉的EF上下文对象,大家的EF上下文类名均大概不壹样。

2.支撑一个通用对象的XML种类化(即:二个类中有可变类型属性成员,供给分歧的队列结果及变化不相同的系列成分名称),具体达成代码如下:

一个亟需被种类化成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属性及DemoDetail<T>类,德姆oDetail属性仅是为着生成detail成分节点,而子节点则由德姆oDetail<T>类来开始展览转变,德姆oDetail<T>是落到实处了IXmlSerializable接口,在XML体系化时,德姆oDetail<T>类仅将body属性对应的T类型实例内容进行连串化(WriteRaw),而反系列化时,则先反连串化body属性对应的T类型实例,然后赋值给body属性,这也是抢眼之处,德姆oDetail<T>类本身并未当真加入到类别化中,故类别化的字符串也看不到DemoDetail<T>类相关的成分,德姆oDetail<T>类仅仅是3个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>

3.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方法,该措施逻辑很简单,正是依据绑定的品质层级(.分隔)层层遍历获取属性的值,直到遍历完或为空时截止,最终获得的结果正是绑定的本性的值。最后落到实处的功用如下图示:

金沙注册送58 1

  

 

1.暗中同意EF生成的接连字符串相比较的长和奇怪,若想利用普通的连接字符串来连…

前言

Contoso 大学示范 Web 应用程序演示怎么样选择实体框架(EF)Core 二.0 和
Visual Studio 20一7 创造 ASP.NET Core 二.0 MVC Web 应用程序。
演示应用程序是胡编的Contoso高校的网址。
它包涵学生入学,课程创造和教育者职分等功能。
那是一文山会海教程中的第2章,介绍怎么样起头营造Contoso高校示例应用程序。
下载或查看已到位的应用程序 –
官方。

本连串小说翻译索引目录

EF Core 2.0 是EF的风行版本,但还并未有包含富有的 EF 陆.x 功能。 有关 EF 六.x
和 EF Core 之间如何抉择的音讯,请参阅 EF Core vs.
EF6.x。
如若您选拔 EF 6.x
,请参阅本学科类别的先前版本。

注意事项

  • 对此本学科的 ASP.NET Core 一.一 版本,请参阅PDF格式的本课程的 VS
    20一7 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
    V一伍.三 或更高版本,安装 ASP.NET 和 Web 开发工具。

只顾上面包车型大巴类是二个分部类:partial,同时BuildEntityConnectionString方法是二个静态方法,在BuildEntityConnectionString方法中ProviderConnectionString
= EncryptUtility.DesDecrypt(“XXXXX”,
connStrSetting.ConnectionString);是珍视,作者那边是对config中的连接字符串
也都举行了加密,故此处笔者要求解密,若无那个必要可以平昔:ProviderConnectionString
=connStrSetting.ConnectionString即可。后续实例化EF上下文对象时,请使用:DataEntities(string
connName)这么些结构涵数即可,DataEntities是切实的EF上下文对象,我们的EF上下文类名均也许不雷同。

小心下面包车型客车类是五个分部类:partial,同时BuildEntityConnectionString方法是贰个静态方法,在BuildEntityConnectionString方法中ProviderConnectionString
= EncryptUtility.DesDecrypt(“XXXXX”,
connStrSetting.ConnectionString);是重中之重,笔者那边是对config中的连接字符串
也都开始展览了加密,故此处作者急需解密,若无那一个须求能够一直:ProviderConnectionString
=connStrSetting.ConnectionString即可。后续实例化EF上下文对象时,请使用:DataEntities(string
connName)那几个布局涵数即可,DataEntities是现实的EF上下文对象,我们的EF上下文类名均恐怕不一样等。

故障排除

假设遇上难题,您无法缓解,平时能够通过将代码与完结的花色进展比较来找到消除方案。
有关常见错误和缓解格局的列表,请参阅本类别最终1个科指标故障排除有些。
假诺您未有找到所需的内容,您也得以在 StackOverflow.com 上发问。

小贴士

本类别涵盖拾二个科目,每个教程都建立在早期教程中的基础之上。
在成功达成各样教程之后,请思虑保存项目标副本。
然后,假设遇上题目,您能够从上一个科目重新起首,无需从头初叶。

2.援救1个通用对象的XML系列化(即:贰个类中有可变类型属性成员,须要差异的系列结果及变化区别的行列成分名称),具体完毕代码如下:

二.支撑3个通用对象的XML系列化(即:3个类中有可变类型属性成员,须要差异的连串结果及变化分裂的行列成分名称),具体贯彻代码如下:

Contoso 大学网址选拔

课程中塑造的是1个简便的高等学校网址。
用户能够查阅和创新学生,课程和教育者新闻。 上边是您将要创造的部分页面。

金沙注册送58 2

index

金沙注册送58 3

edit

网址的 UI 风格与内置模板生成的 UI 风格保持一致,本学科主要关切怎么样使用
Entity Framework。

一个急需被体系化成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 二.0 和 EF Core 2.0 或更高版本 –
    确认保证未选用ASP.NET Core 一.1。
  • 担保认证设置为“不开始展览身份验证”。
  • 单击 “确定” 按钮。

金沙注册送58 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>

按 CTLX570L+F5 运维项目或从菜单中采用 调节和测试-> 开首履行(不调节和测试),
您将在浏览器中看看本课程中落到实处的首页。

金沙注册送58 5

image.png

留神下边代码中,需求关心:德姆oDetail属性及DemoDetail<T>类,德姆oDetail属性仅是为着生成detail成分节点,而子节点则由DemoDetail<T>类来举办转变,德姆oDetail<T>是兑现了IXmlSerializable接口,在XML体系化时,德姆oDetail<T>类仅将body属性对应的T类型实例内容展开连串化(WriteRaw),而反连串化时,则先反系列化body属性对应的T类型实例,然后赋值给body属性,这也是巧妙之处,DemoDetail<T>类本人并未当真参加到系列化中,故连串化的字符串也看不到德姆oDetail<T>类相关的因素,DemoDetail<T>类仅仅是三个XML种类化格式生成的中介。种类化的XML结果如下:

专注上边代码中,必要关爱:DemoDetail属性及德姆oDetail<T>类,德姆oDetail属性仅是为着生成detail成分节点,而子节点则由DemoDetail<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 包中,因而我们今后如何都无须做。

其1顺序包 (Microsoft.EntityFrameworkCore) 及其正视项
(Microsoft.EntityFrameworkCore.Relational) 提供了EF运维时支持。在稍后的
”数据库迁移“教程中,你将会学习添加叁个工具包。

关于可用于 Entity Framework Core 的其余数据库帮忙程序的新闻,请参阅
Data
Providers。

类别化代码:

连串化代码:

创造数据模型

接下去,您将为Contoso高校应用程序创立实体课程。 您将从以下多少个实体伊始。

金沙注册送58 6

class diagram

在 Student 与 Enrollement 实体间是一个一对多的涉及, 在 Course 与
Enrollment 间也设有一些多关乎。
换句话说,学生能够参与任意数量的教程,课程能够有专擅数量的学生登记。

在偏下1些中,您将为各样实体创制三个类。

            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
属性是导航属性。导航属性用于关联其余实体。对于1个学员实体(数据)来说,个中的
Enrollments 属性包蕴全体与该学生相关联的 Enrollment
实体(数据)。也正是说,如若数据库中的3个学员行数据涉嫌多个注册行数据(壹对多,在
Enrollment 表中外键关联StudentID 值为该学员的主键值),则 Student
实体中的 Enrollments 导航属性将富含那三个 Enrollment 实体。

如果导航属性能够兼容多个实体(在多对多或一对多关系中),则其品种必须是足以拉长,删除和更新条目标列表,例如ICollection
<T>。您能够钦点ICollection <T>或项目,如List
<T>或HashSet <T>。假使钦命ICollection
<T>,EF暗许创造二个HashSet <T>集合。

系列化的XML:

连串化的XML:

Enrollment 实体

金沙注册送58 7

Enrollment

在 Models 文件夹中,创制2个名称叫 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 形式。 平常你会只选用一种情势,并在方方面面数据模型中选取。
在此处,分歧的情势是为着演示,表达你可以使用任一方式。
在后面包车型大巴科目中,您将见到哪些运用未有类名的 ID
能够更便于地在数据模型中达成一连。

Grade (等级) 属性是一个枚举类型。 Grade 类型注明后的 ? 表示
可为空类型。 三个空的等级和1个值为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 实体

金沙注册送58 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 的主键名,而不是让数据库生成它。

3.winform DataGridView
实现钦点列接纳密码框方式展现与编辑,以及列绑定到复合属性(即:绑定到多层次属性)
,具体落到实处代码如下:

三.winform DataGridView
达成钦点列选用密码框形式突显与编写制定,以及列绑定到复合属性(即:绑定到多层次属性),具体贯彻代码如下:

创制数据库上下文 Database Context

将数据模型与 Entity Framework 功用协同工作的重点类是数据库上下文类。
通过从 Microsoft.EntityFrameworkCore.DbContext 类派生来创立此类。
在代码中,能够内定数据模型中隐含哪些实体。 还能自定义有些 Entity
Framework 行为。 在那一个体系中,该类被命名称叫 SchoolContext

在档次文件夹中,创立3个名称为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数据库上下文)在应用程序运转时期通过重视注入注册实例。
那2个急需运用劳务的组件通过构造函数参数获得劳动的实例。
稍后大家得以看出控制器构造函数获取上下文实例的代码。

要将 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文件并累加2个再而三字符串,如下例所示。

{
  "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 将为您成立二个空数据库。
在本节中,您将编辑1个创立数据库后调用的办法,以便利用测试数据实行填写。

在那边,您将运用 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及EditingControlShowing事件,并在其间写转换当前Cell的Value,落成列绑定到复合属性,关键点在:伊娃luateValue方法,该措施逻辑非常的粗略,就是基于绑定的性质层级(.分隔)层层遍历获取属性的值,直到遍历完或为空时截止,最终获得的结果便是绑定的属性的值。最后落到实处的功效如下图示:

兑现钦点列采纳密码框形式展现与编辑,以及列绑定到复合属性均要求订阅DataGridView的CellFormatting及EditingControlShowing事件,并在里边写转换当前Cell的Value,达成列绑定到复合属性,关键点在:伊娃luateValue方法,该措施逻辑非常粗大略,就是依据绑定的习性层级(.分隔)层层遍历获取属性的值,直到遍历完或为空时结束,最终获得的结果就是绑定的品质的值。最后落实的成效如下图示:

创建控制器和视图

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

电动创立CRUD操作方法和视图称为脚手架。
脚手架与代码生成器差异之处在于,脚手架代码只是基础代码,您能够遵照自个儿的内需展开改动,而普通景况下,您不会修改生成器生成的代码。
当您供给自定义生成器生成的代码,能够动用部分类,只怕在景况时有发生转移时时重新生成代码。

  • 右键单击化解方案能源管理器中的 Controllers 文件夹,然后选择 添加
    -> 控制器。
  • 在“添加基架”对话框中,接纳“视图使用 Entity Framework 的 MVC
    控制器”,点击“添加”
  • 在“添加控制器”对话框中:
    • 模型类采用 Student
    • 数量上下文类选拔 SchoolContext
    • 点击 “添加”。

金沙注册送58 9

new Controller

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

(脚手架引擎还是能为你创造数据库上下文,借使您不像以前在本教程中那么手动制造它。
您能够通过单击数据上下文类左边的加号在“添加控制器”框中钦点新的上下文类。
然后Visual Studio将创造您的DbContext类以及控制器和视图。)

你会小心到控制器将3个 SchoolContext 作为3个构造函数参数。

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

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

ASP.NET 注重注入负责将 SchoolContext 的三个实例传递到控制器中。
前文中,已经
在 Startup.cs 文件中安顿 SchoolContext 的注重注入。

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

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

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

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

按 CT安德拉L + F五 运转品种或从菜单中选取 调节和测试 -> 开始履行(不调试)。

单击 Student 链接,可以见见 DbInitializer.Initialize
方法中插入的测试数据。 依照浏览器窗口的狭窄程度,您会师到页面顶部的
Student
链接,也有希望你必须单击右上角的领航空图标才能旁观隐藏菜单中的链接。

金沙注册送58 10

narrow Page

金沙注册送58 11

sdudent index

金沙注册送58 12

金沙注册送58 13

查阅数据库

当你运行应用程序时,DbInitializer.Initialize 方法调用 EnsureCreated 。
EF 看到未有数据库,所以它创设了三个,然后 Initialize
方法代码的其他部分用数码填充数据库。 在 Visual Studio 中,您可以应用 SQL
Server 对象能源管理器(SSOX)查看数据库。

若果 SSOX 窗口未有打开,在 Visual Studio 中,点击菜单 “视图” -> “SQL
Server 对象能源管理器”。
在 SSOX 中,单击(localdb)\ MSSQLLocalDB > 数据库,然后单击
ContosoUniversity一,约等于我们眼下在 appsettings.json
文件中设置的连日字符串中数据库名称。
举行“表”节点以查看数据库中的表。

金沙注册送58 14

ssox

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

金沙注册送58 15

student table

.mdf 和.ldf 数据库文件位于C:\Users<你的用户名> 文件夹中。
因为你在应用程序运转时运营的伊始化程序方法中调用 EnsureCreated
,所今后后得以更改 Student
类,删除数据库,再一次运转应用程序,并机关重新成立数据库以分外您的更动。
例如,假诺你将 EmailAddress 属性添加到 Student
类,则会在重复创制的表中看到三个新的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();

        }

小结

你今后成立了二个简练的应用程序,使用 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());
            }
        }

  

联合原理如下:

金沙注册送58, 

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

运用示例如下:

二.查询源服务器的急需共同的表的笔录(一般是TranFlag=0的笔录),利用SqlDataAdapter+SqlCommandBuilder
装载Dataset,目标是继续能够采取SqlDataAdapter直接生成更新命令并推行;

        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从指标服务器的将被一块的表复制结构发生1个近期表,表名1般是:#temp_指标服务器的将被联合表名
,那样一时半刻表与实体表的构造完全一致;

联合原理如下:

四.实例化3个SqlBulkCopy,并确立源服务器的内需1块的表字段与目标权且表字段的照射,然后实施跨服务器传输;

4.1.概念好查询源服务器的须求联合的表(壹般表中大家定义3个用来是否同步的标识字段,如:TranFlag
Bit类型,0表示新数据,未共同,1象征已联手);

5.选拔 BuildInsertOrUpdateToDestTableSql 方法 ,生成
 目标服务器的将被1块的表 与
权且表的插入与立异SQL语句(以后在同3个库了,想怎么用SQL语句均可)  

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

6.为保障一致性,故外层还需包裹事务SQL语句,若还需出席别的处理SQL,能够加在begin
tran  …
commit代码块中即可,最终执行SQL语句:gExecCommand(ClsDatabase.gExecCommand是八个SQLDB
HELPEEvoque 类的施行SQL命令的艺术)

四.三.利用insert
into从目的服务器的将被联合的表复制结构产生多个一时表,表名1般是:#temp_目标服务器的将被一起表名
,那样近期表与实体表的结构完全壹致;

 

4.四.实例化3个SqlBulkCopy,并建立源服务器的内需1起的表字段与目标暂时表字段的照射,然后实施跨服务器传输;

四.5.选用 BuildInsertOrUpdateToDestTableSql 方法 ,生成
 指标服务器的将被一起的表 与
最近表的插入与更新SQL语句(未来在同一个库了,想怎么用SQL语句均可)  

四.陆.为保障壹致性,故外层还需包裹事务SQL语句,若还需参与其余处理SQL,能够加在begin
tran  …
commit代码块中即可,最终执行SQL语句:gExecCommand(ClsDatabase.gExecCommand是二个SQLDB
HELPEKuga 类的实践SQL命令的法子)

 

5.达成同3个WINDOWS SE途乐VICE程序
COPY多份,然后经过转移自定义的劳务ID(ServiceID)配置项来落到实处:同一个服务程序安装成多少个不等的WINDOWS服务进程:

伍.一.开立二个WINDOWS服务项目,在ProjectInstaller设计器界面通过右键弹出菜谱选用安装程序(serviceProcessInstaller一、serviceInstaller壹)、并设置好ServiceName、DisplayName、Description、Account等,如下图示:

金沙注册送58 16

5.二.在ProjectInstaller构造函数中追加从CONFIG文件中读取自定义的服务ID(ServiceID)配置项的值,然后将ServiceID拼加到预设的ServiceName前边,以便实际遵照ServiceID可以设置成不一样ServiceID后缀的劳务进程,关键点在于改变瑟维斯Name,另三个关键点是从CONFIG文件中拿走ServiceID,由于设置时,守旧的主意不能寻常读取到CONFIG,只可以通过Assembly.GetExecutingAssembly().Location
来获得当前履行的顺序集的门道再拼成CONFIG文件路径,最后读出瑟维斯ID的值,示例代码如下:

    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;
            }
    }

}

 上述三步就完了了同3个服务程序安装成多少个不等的WINDOWS服务进程,那几个照旧相比实用的哦!上述ConfigUtil是包裹的一个配备文件读写扶助类,从前小说有介绍,后边也会发布一个更完整的ConfigUtil类。

相关文章

网站地图xml地图