威尼斯人线上娱乐

入门教程,近年来费用项目中用到的编码小技术汇总表达

2 5月 , 2019  

一.默许EF生成的连日字符串相比较的长和奇妙,若想行使普通的总是字符串来连接EF,则足以因而创办分品种,比量齐观写1个构造函数,在构造函数中通过动态拼接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);是任重(Ren Zhong)而道远,我这里是对config中的连接字符串
也都进行了加密,故此处作者索要解密,若无那些须要能够间接:ProviderConnectionString
=connStrSetting.ConnectionString就可以。后续实例化EF上下文对象时,请使用:DataEntities(string
connName)那几个结构涵数就可以,DataEntities是有血有肉的EF上下文对象,大家的EF上下文类名均恐怕不相同等。

二.支撑贰个通用对象的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属性及德姆oDetail<T>类,德姆oDetail属性仅是为了生成detail成分节点,而子节点则由德姆oDetail<T>类来张开调换,德姆oDetail<T>是促成了IXmlSerializable接口,在XML体系化时,德姆oDetail<T>类仅将body属性对应的T类型实例内容展开种类化(WriteRaw),而反连串化时,则先反系列化body属性对应的T类型实例,然后赋值给body属性,这也是美妙之处,德姆oDetail<T>类本人并不曾真的出席到连串化中,故体系化的字符串也看不到德姆oDetail<T>类相关的因素,德姆oDetail<T>类仅仅是1个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,完结列绑定到复合属性,关键点在:EvaluateValue方法,该方法逻辑很简短,正是基于绑定的习性层级(.分隔)层层遍历获取属性的值,直到遍历完或为空时甘休,最后收获的结果就是绑定的属性的值。最终落成的职能如下图示:

威尼斯人线上娱乐 1

  

 

一.暗许EF生成的三番五次字符串相比的长和新奇,若想利用普通的连日字符串来连…

前言

Contoso 高校以身作则 Web 应用程序演示如何行使实体框架(EF)Core 二.0 和
Visual Studio 2017 成立 ASP.NET Core 二.0 MVC Web 应用程序。
演示应用程序是胡编的Contoso大学的网站。
它包罗学生入学,课程创制和老师职务等成效。
那是一雨后玉兰片教程中的第二章,介绍怎么着初步创设Contoso大学示例应用程序。
下载或查看已产生的应用程序 –
官方。

本种类文章翻译索引目录

EF Core 二.0 是EF的新颖版本,但还未有包罗具有的 EF 陆.x 功用。 有关 EF 六.x
和 EF Core 之间什么抉择的音讯,请参阅 EF Core vs.
EF6.x。
假诺你选择 EF 陆.x
,请参阅本课程种类的在此在此之前版本。

注意事项

  • 对于本学科的 ASP.NET Core 一.一 版本,请参阅PDF格式的本课程的 VS
    20一七 Update
    二版本。
  • 对此本课程的 Visual Studio 201⑤ 版本,请参阅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];
        }
    }

支出景况

设置如下工具:

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

专注上面的类是八个总局类:partial,同时BuildEntityConnectionString方法是2个静态方法,在BuildEntityConnectionString方法中ProviderConnectionString
= EncryptUtility.DesDecrypt(“XXXXX”,
connStrSetting.ConnectionString);是首要,作者这边是对config中的连接字符串
也都进展了加密,故此处小编索要解密,若无这些须求能够一向:ProviderConnectionString
=connStrSetting.ConnectionString就能够。后续实例化EF上下文对象时,请使用:DataEntities(string
connName)那么些组织涵数就能够,DataEntities是实际的EF上下文对象,大家的EF上下文类名均恐怕不平等。

故障排除

只要高出难点,您不能够消除,平时能够通过将代码与成功的花色进展比较来找到解决方案。
有关常见错误和缓和格局的列表,请参阅本类别最终叁个学科的故障排除有个别。
假若你未有找到所需的始末,您也足以在 StackOverflow.com 上发问。

小贴士

本体系涵盖10个科目,每种教程都成立在初期教程中的基础之上。
在功成名就做到各样教程之后,请思虑保存项目的别本。
然后,假如遇上标题,您能够从上二个学科重新发轫,不必要从头先导。

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

Contoso 大学网址选择

课程中创设的是3个轻易易行的大学网址。
用户能够查阅和更新学生,课程和师资消息。 上面是你就要创制的有的页面。

威尼斯人线上娱乐 2

index

威尼斯人线上娱乐 3

edit

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

1个要求被连串化成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 二.0 或越来越高版本 –
    确定保证未选用ASP.NET Core 一.1。
  • 管教认证设置为“不实行身份验证”。
  • 单击 “确定” 按钮。

威尼斯人线上娱乐 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; }
    }

安装网址样式

轻巧修改多少个职位,设置站点菜单,布局和主页。
打开 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 运营项目或从菜单中采取 调节和测试-> 初始实施(不调节和测试),
您就要浏览器中看到本学科中贯彻的首页。

威尼斯人线上娱乐 5

image.png

瞩目下面代码中,需求关切:德姆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>类仅仅是1个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高校应用程序创设实体课程。 您将从以下八个实体开首。

威尼斯人线上娱乐 6

class diagram

在 Student 与 Enrollement 实体间是一个一对多的涉及, 在 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);

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

设若导航属性能够包容三个实体(在多对多或壹对多涉及中),则其体系必须是能够加上,删除和翻新条款的列表,举例ICollection
<T>。您能够钦命ICollection <T>或项目,如List
<T>或HashSet <T>。假使钦点ICollection
<T>,EF私下认可成立3个HashSet <T>会集。

连串化的XML:

Enrollment 实体

威尼斯人线上娱乐 7

Enrollment

在 Models 文件夹中,创造三个名叫 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
实体与1个 Student 实体相关联,由此该属性只可以具有保留单个 Studnet
实体(与您事先看来的可以分包八个登记实体的 Student.Enrollments
导航属性不一致)。

CourseID 属性是外键, 对应的领航属性是 Course。 一个 Enrollment
实体与二个 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>

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 属性是导航属性。3个 Course 实体能够提到到自由四个
Enrollment 实体。

咱俩就要本类别的接轨教程中详细介绍 DatabaseGenerated 性子。
此性格允许你钦命 Course 的主键名,而不是让数据库生成它。

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

使用数据上下文 – 使用正视注入

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;
加上代码,使用测试数据早先化数据库

EF 将为你创制2个空数据库。
在本节中,您将编写制定1个创办数据库后调用的办法,以便利用测试数据实行填写。

在那边,您将选拔 EnsureCreated 方法自动创制数据库。
在后头的科目中,您将看到什么行使 Code First Migration (代码优先迁移)
来改动数据库架构而不是删除和重新创制数据库来拍卖架构改动。

Data 文件夹中,创建3个名称为 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方法,该方法逻辑相当粗略,正是基于绑定的属性层级(.分隔)层层遍历获取属性的值,直到遍历完或为空时甘休,最后收获的结果就是绑定的习性的值。最后实现的功能如下图示:

创设调控器和视图

接下去,您将使用 Visual Studio 脚手架加上 MVC 调控器和视图,并使用 EF
来查询和保留数据。

活动创制CRUD操作方法和视图称为脚手架。
脚手架与代码生成器不一样之处在于,脚手架代码只是基础代码,您能够依照自身的须要开始展览改换,而一般状态下,您不会修改生成器生成的代码。
当您须要自定义生成器生成的代码,可以选拔一些类,恐怕在状态发生改换时时重新生成代码。

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

威尼斯人线上娱乐 9

new Controller

当您单击加多时,Visual Studio 脚手架引擎成立贰个 StudentsController.cs
文件和1组与调控器一齐使用的视图(.cshtml文件)。

(脚手架引擎还是可感觉你创制数据库上下文,要是您不像在此以前在本教程中那么手动成立它。
您能够透过单击数据上下文类左边的加号在“加多调控器”框中钦点新的光景文类。
然后Visual Studio将开创您的DbContext类以及控制器和视图。)

你会小心到调整器将1个 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
链接,也有希望你必须单击右上角的领航空Logo技能收看隐藏菜单中的链接。

威尼斯人线上娱乐 10

narrow Page

威尼斯人线上娱乐 11

sdudent index

威尼斯人线上娱乐 12

查看数据库

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

1经 SSOX 窗口未有展开,在 Visual Studio 中,点击菜单 “视图” -> “SQL
Server 对象能源管理器”。
在 SSOX 中,单击(localdb)\ MSSQLLocalDB > 数据库,然后单击
ContosoUniversity一,也正是大家眼下在 appsettings.json
文件中安装的连天字符串中数据库名称。
拓展“表”节点以查看数据库中的表。

威尼斯人线上娱乐 13

ssox

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

威尼斯人线上娱乐 14

student table

.mdf 和.ldf 数据库文件位于C:\Users<你的用户名> 文件夹中。
因为您在应用程序运维时运维的初步化程序方法中调用 EnsureCreated
,所以今后得以更换 Student
类,删除数据库,再次运转应用程序,并自动重新创制数据库以协作您的更换。
比如,如若您将 EmailAddress 属性增多到 Student
类,则会在重新创制的表中看到三个新的EmailAddress 列。

四.行使BCP(sqlbulkcopy)来促成四个不等数据库之间开始展览数据差距传输(即:数据同步)

约定

据书上说约定优于配备的准绳,Entity Framework
创设叁个数据库时,你所需书写的代码很少。

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

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

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

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

预约能够被遮住。比方,你可以显式钦赐表名,如本学科前边所阅览的。
您可以设置列名称并将别的性质设置为主键或外键,那将要前边的课程中提起。

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 关键字告诉编写翻译器将该办法分为两片段。
    第2片段以异步运转的操作甘休。
    第2部分被放入回调方法,该操作在操作实现时被调用。

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

当你编写使用实体框架的异步代码时,供给留意的有个别事情:

  • 只有会吸引查询或将指令发送到数据库的语句才供给异步实施。 那包含诸如
    ToListAsyncSingleOrDefaultAsyncSaveChangesAsync
    它不应有包蕴,举例,只是改造IQueryable的语句,类似
    var students = context.Students.Where(s => s.LastName == "Davolio")
    那样的言语。

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

  • 1旦您想利用异步代码的天性优势,请确定保障您正在使用的别的库包(比如用于分页)也应用异步,假若她们调用任何导致查询发送到数据库的诀要。

有关.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);
            }
        }

小结

您现在开立了一个大约的应用程序,使用 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象征已协同);

四.二.查询源服务器的急需联合的表的记录(一般是TranFlag=0的笔录),利用SqlDataAdapter+SqlCommandBuilder
装载Dataset,目的是继续能够运用SqlDataAdapter间接扭转更新命令并实行;

四.3.选拔insert
into从目标服务器的将被联合的表复制结构爆发叁个一时半刻表,表名一般是:#temp_目标服务器的将被同台表名
,那样一时半刻表与实体表的构造完全1致;

四.肆.实例化3个SqlBulkCopy,并构建源服务器的急需一同的表字段与目标临时表字段的映射,然后实践跨过服务器务器传输;

4.5.选用 BuildInsertOrUpdateToDestTableSql 方法 ,生成
 目标服务器的将被联合的表 与
一时半刻表的插入与更新SQL语句(现在在同四个库了,想怎么用SQL语句均可)  

四.六.为确认保证一致性,故外层还需包裹事务SQL语句,若还需投入别的管理SQL,能够加在begin
tran  …
commit代码块中就能够,最终试行SQL语句:gExecCommand(ClsDatabase.gExecCommand是三个SQLDB
HELPE凯雷德 类的施行SQL命令的措施)

 

五.落成同三个WINDOWS SEMuranoVICE程序
COPY多份,然后通过改动自定义的劳务ID(ServiceID)配置项来落到实处:同3个服务程序安装成四个差异的WINDOWS服务进度:

5.一.创建二个WINDOWS服务项目,在ProjectInstaller设计器分界面通过右键弹出美食指南选拔安装程序(serviceProcessInstaller一、serviceInstaller一)、并安装好ServiceName、DisplayName、Description、Account等,如下图示:

威尼斯人线上娱乐 15

伍.贰.在ProjectInstaller构造函数中追加从CONFIG文件中读取自定义的服务ID(ServiceID)配置项的值,然后将ServiceID拼加到预设的瑟维斯Name前面,以便实际依据ServiceID能够设置成差异ServiceID后缀的劳务进程,关键点在于改造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));
        }
    }

伍.3.在服务类的构造函数中同样增添从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地图