源码学习之ASP.NET MVC Application Using Entity Framework

时间:2021-01-26 04:12:11

源码学习的重要性,再一次让人信服。
ASP.NET MVC Application Using Entity Framework Code First

做MVC已经有段时间了,但看了一些CodePlex上的代码,仍然觉得需要学习的东西还很多,基本上功能都可以实现,但如果放弃去利用框架本身已经聚合与提供的便利性功能,那往往会事倍功半,即使实现,也不漂亮不专业,不舒服。

很多专业的东西都在那里,没事瞅瞅,不要动不动一上来就老三样,或者放弃好东西,用自己的老办法去经验性的解决,既然可以实现,那就要去找最漂亮的解决办法,否则何来提升。

  • 虚拟属性

    //虚拟自动属性。子类可以使用overrides重写该属性,
    //子类可以实现get和set访问器,以实现自定义的操作。
    public virtual string Name { get; set; }

    
    // 带支持字段的虚拟属性
         private int num;
         public virtual int Number
         {
             get { return num; }
             set { num = value; }
         }
  • DataType

    Custom,
    DateTime,
    Date,
    Time,
    Duration,
    PhoneNumber,
    Currency,
    Text,
    Html,
    MultilineText,
    EmailAddress,
    Password,
    Url,
    ImageUrl,
    CreditCard,
    PostalCode,
    Upload,
    很多常用的验证都已经被封装好了,只需要用就可以了,不要再费劲的写正则了,亲。

MetadataTypeAttribute

[MetadataType(typeof(CustomerMetaData))]
public partial class Customer
{   }
public class CustomerMetaData
{
    // Apply RequiredAttribute
    [Required(ErrorMessage = "Title is required.")]
    public object Title;
}

通过使用 MetadataTypeAttribute 属性,您可以将类与数据模型分部类关联。在此关联的类中,您可以提供数据模型中所没有的附加元数据信息。这样较好的区分了,DataBase中类的基本定义与Coding时类的附加性验证,使两张能够结构清晰,分工明确。

[MetadataType(typeof(CompanyMetadata))]
public partial class Company
{
    public class CompanyMetadata
    {
        [JsonIgnore]//防止API调用中的嵌套输出
        public virtual ICollection<Advertise> Advertise { get; set; }
    }
}
  • rowversion

使用rowversion数据类型实现乐观锁
SQL Server 2008 提供了一个特殊数据类型rowversion,它可以用于在应用程序中实现乐观锁。rowversion数据类型在乐观锁模式下充当版本号。无论何时包含rowversion类型数据列的行被插入或更新时,SQL Server 自动为该列生成一个值。rowversion数据类型是8字节的二进制数据类型,除了保证值的唯一性和单向增长外,它的值不具有意义。你不能够查看它的每个字节来搞懂它是什么意思。

客户端从表中读取数据,确保返回的结果集中包含了主键和rowversion列,以及其他想要的数据列。由于查询并不运行在事务中,一旦数据被读取,SELECT查询获取的锁即被释放。当一段时间过后用户想要更新某行时,必须确保在此期间该数据没有被其他客户端修改过。Update语句必须包含WHERE子句用以比较取回的rowversion值与数据库中该列的当前值。如果两个值匹配(即相同),说明该行记录在此期间没有被修改过。因此可以放心提交更改。如果不匹配,则说明该行记录已经被修改过。为了避免Lost Update问题发生,不应提交本次更新。

  • Linq to SQl 与传统SQl语句的整合

```

ViewBag.RowsAffected = db.Database.ExecuteSqlCommand("UPDATE Course SET Credits = Credits * {0}", multiplier);

string query = "SELECT EnrollmentDate, COUNT(*) AS StudentCount "
            + "FROM Person "
            + "WHERE Discriminator = 'Student' "
            + "GROUP BY EnrollmentDate";
        IEnumerable<EnrollmentDateGroup> data = db.Database.SqlQuery<EnrollmentDateGroup>(query);

 IQueryable<Course> courses = db.Courses;
 var sql = courses.ToString();
  • 延迟,饿汉,以及显式加载关联数据

    var departments = context.Departments;
    foreach(Department d in departments)
    {foreach get course.} 

    //Lazy Loading当第一次访问关联属性时,关联数据才会被自动读取。

    var departments = context.Departments.Include(x=x.Course){foreach,foreach get course}

    //Eager Loading 当实体加载时,相关联的数据也一起被加载。性能最好,因为只给Db发送一次查询。

    var departments = context.Departments.ToList();
    Context.Entry(d).Collection(x=>x.Courses).Load(); 

    //Explict Loading 类似于延迟加载,如果不常访问关联实体属性,或仅访问一小部分,则延迟加载更有效。

数据库上下文默认支持延迟加载,有两种方法可以关闭延迟加载:
对于特定的导航属性,在定义属性的时候取消 virtual
对于所有的导航属性,设置 LazyLoadingEnabled 为假。