【手撸一个ORM】第二步、封装实体描述和实体属性描述
一、实体属性描述 [MyProperty.cs]
- Name,属性名称
- PropertyInfo,反射获取的属性信息,后面很多地方需要通过该属性获取对应的实体类型,或调用SetValue进行赋值
- FieldName,对应的数据表列名
- IsKey,是否主键
- IsMap,查询时是否映射该属性,若属性非值类型或string,则默认为false,其他默认为true;若需手动设置为false,如计算属性,需要在MyColumnAttribute中配置 IsMap=false
- InsertIgnore,插入时忽略
- UpdateIgnore,更新时忽略
- JoinAble,是否可以通过Join进行查询,等同于导航属性
- ForignKey,如果可以通过Join查询,对应的外键名
- MasterKey,如果可以通过Join查询,对应的主表Id,默认为Id,若主表主键列名不是Id,需手动在MyForeignKey中配置 MasterKey="StudentId"
using System.Reflection; namespace MyOrm.Reflections
{
public class MyProperty
{
public string Name { get; set; } public PropertyInfo PropertyInfo { get; set; }public string FieldName { get; set; } public bool IsKey { get; set; } public bool IsMap { get; set; } = true; public bool InsertIgnore { get; set; } public bool UpdateIgnore { get; set; } public bool JoinAble { get; set; } public string ForeignKey { get; set; } public string MasterKey { get; set; } public MyProperty(PropertyInfo property)
{
Name = property.Name;
TypeName = property.PropertyType.Name;
PropertyInfo = property; if (property.IsMapAble())
{
// 判断是否主键
var keyAttribute = property.GetKeyAttribute();
if (keyAttribute != null)
{
// 有
IsKey = true;
FieldName = string.IsNullOrWhiteSpace(keyAttribute.FieldName) ? Name : keyAttribute.FieldName;
if (keyAttribute.IsIncrement)
{
// 如果是自增列,不能插入和修改
InsertIgnore = true;
UpdateIgnore = true;
}
else
{
// 如果不是自增列,可插入但不能修改
InsertIgnore = true;
}
}
else if (Name == "Id")
{
FieldName = "Id";
IsKey = true;
InsertIgnore = true;
UpdateIgnore = true;
}
else
{
// 可映射的属性
var columnAttribute = property.GetMyColumnAttribute(); if (columnAttribute != null)
{
FieldName = string.IsNullOrWhiteSpace(columnAttribute.ColumnName)
? Name
: columnAttribute.ColumnName;
InsertIgnore = columnAttribute.Ignore || columnAttribute.InsertIgnore;
UpdateIgnore = columnAttribute.Ignore || columnAttribute.UpdateIgnore;
}
else
{
FieldName = Name;
}
}
}
else if (property.IsJoinAble())
{
// 可关联查询的属性
IsMap = false;
JoinAble = true;
UpdateIgnore = true;
InsertIgnore = true;
var foreignAttribute = property.GetForeignKeyAttribute();
if (foreignAttribute == null)
{
ForeignKey = Name + "Id";
MasterKey = "Id";
}
else
{
ForeignKey = string.IsNullOrWhiteSpace(foreignAttribute.ForeignKey)
? Name + "Id"
: foreignAttribute.ForeignKey;
MasterKey = string.IsNullOrWhiteSpace(foreignAttribute.MasterKey)
? "Id"
: foreignAttribute.MasterKey;
}
}
else
{
// 其他属性
IsMap = false;
UpdateIgnore = true;
InsertIgnore = true;
}
}
}
}
二、数据实体描述 [MyEntity.cs]
- KeyColumn,数据表中主键列名称
- Name,实体名称
- TableName,实体对应的数据表名称
- IsSoftDelete,是否软删除(继承ISoftDelete),后面会有说明
- IsCreateAudit,是否创建审计(保存创建人、创建时间)
- IsUpdateAudit,是否更新审计(保存修改人、修改时间)
- Properties,封装过的属性信息列表-考虑过这里用字典保存,但是因为后面需要大量的遍历操作,个人感觉还是List用起来方便,所以最终选择了List类型
using MyOrm.Attributes;
using MyOrm.Commons;
using System;
using System.Collections.Generic; namespace MyOrm.Reflections
{
public class MyEntity
{
public string KeyColumn { get; set; } public string Name { get; set; } public string TableName { get; set; } public bool IsSoftDelete { get; set; } public bool IsCreateAudit { get; set; } public bool IsUpdateAudit { get; set; } public List<MyProperty> Properties { get; set; } public MyEntity(Type type)
{
Name = type.Name;
IsSoftDelete = type.IsInstanceOfType(typeof(ISoftDelete));
IsCreateAudit = type.IsInstanceOfType(typeof(ICreateAudit));
IsUpdateAudit = type.IsInstanceOfType(typeof(IUpdateAudit)); var tableAttr = type.GetCustomAttributes(typeof(MyTableAttribute), false);
if (tableAttr.Length > )
{
var tableName = ((MyTableAttribute)tableAttr[]).TableName;
TableName = string.IsNullOrWhiteSpace(tableName) ? type.Name.Replace("Entity", "") : tableName;
}
else
{
TableName = Name;
} Properties = new List<MyProperty>(); foreach (var propertyInfo in type.GetProperties())
{
var property = new MyProperty(propertyInfo);
if (property.IsKey)
{
KeyColumn = property.FieldName;
}
Properties.Add(property);
}
}
}
}
三、实体容器
上面对实体及其属性进行了封装,但是如果每次都需要反射获取,那性能损耗会非常厉害,因此将生成的内容缓存在起来便十分必要了。这里使用的是线程安全的静态字典作为缓存容器。
using System;
using System.Collections.Concurrent; namespace MyOrm.Reflections
{
public class MyEntityContainer
{
private static readonly ConcurrentDictionary<string, MyEntity> Dict =
new ConcurrentDictionary<string, MyEntity>(); public static MyEntity Get(Type type)
{
if (type == null) throw new ArgumentNullException(nameof(type));
if (Dict.TryGetValue(type.FullName ?? throw new InvalidOperationException(), out var result))
{
return result;
}
else
{
var entity = new MyEntity(type);
Dict.TryAdd(type.FullName, entity);
return entity;
}
}
}
}
【手撸一个ORM】第二步、封装实体描述和实体属性描述的更多相关文章
- 【手撸一个ORM】MyOrm的使用说明
[手撸一个ORM]第一步.约定和实体描述 [手撸一个ORM]第二步.封装实体描述和实体属性描述 [手撸一个ORM]第三步.SQL语句构造器和SqlParameter封装 [手撸一个ORM]第四步.Ex ...
- 【手撸一个ORM】第一步、实体约定和描述
一.约定 数据实体必须实现 IEntity 接口,该接口定义了一个int类型的Id属性,既每个实体必须有一个名称为Id的自增主键. 若数据表的主键列名称不是Id,可以通过 [MyKey("主 ...
- 【手撸一个ORM】第六步、对象表达式解析和Select表达式解析
说明 一个Orm自然不仅仅包含条件表达式,还会有如下的场景: OrderBy(s => s.StudentName) Select<StudentDto>(s => new S ...
- 【手撸一个ORM】第七步、SqlDataReader转实体
说明 使用Expression(表达式目录树)转Entity的文章在园子里有很多,思路也大致也一样,我在前面有篇文章对解决思路有些说明,有兴趣的小伙伴可以看下 (传送门),刚接触表达式目录树时写的,不 ...
- 【手撸一个ORM】第三步、SQL语句构造器和SqlParameter封装
既然是数据库工具,自然少不了增删改查的sql语句,在这里将这些常用SQL拼接操作集成到 [SqlServerBuilder.cs] 当中,方便后面调用. 近几年在项目中一直使用Dapper操作数据库, ...
- 【手撸一个ORM】第十步、数据操作工具类 MyDb
说明 其实就是数据库操作的一些封装,很久不用SqlCommand操作数据库了,看了点园子里的文章就直接上手写了,功能上没问题,但写法上是否完美高效无法保证,建议有需要的朋友自己重写,当然如果能把最佳实 ...
- 【手撸一个ORM】第五步、Expression(表达式目录树)转换为Where子句
说明 在SQL中,查询.修改比较常用到WHERE子句,在这里根据使用场景不同,定义了两个类,一个用于查询,一个用于修改(插入)操作.原因是: 查询操作支持一级导航属性查询,如student.Schoo ...
- 【手撸一个ORM】第四步、Expression(表达式目录树)扩展
到这里,Orm的基架已经搭起来了,接下来就是激动人心的部分,表达式目录树转Sql语句,SqlDataReader转数据实体等等,但是在这之前,我们需要扩展下表达式目录树的方法,以方便后面的相关操作. ...
- 【手撸一个ORM】第八步、查询工具类
一.实体查询 using MyOrm.Commons; using MyOrm.DbParameters; using MyOrm.Expressions; using MyOrm.Mappers; ...
随机推荐
- 搭建LoadRunner中的场景(四)控制器的全局设置
选择“Tools”菜单-“Options”选项打开设置窗口 1.超时设置 2.运行时设置 3.运行时文件存储设置 4.路径翻译表 路径翻译表是一种映射,将控制器上的文件路径转换为远程主机上的文件路径. ...
- 使用谷歌浏览器进行Web开发技巧
1.为了避免缓存影响开发,使用使用那个“Ctrl+Shift+N”进入浏览器的隐身模式
- EmbarassedBirds全体开发人员落泪
Github (李昆乘,赖展飞) 现阶段还在开发后期,API调试过程中. 本周无法上线. 全体开发人员留下眼泪. 贴上几个功能图, 给大家尝尝鲜吧! 现阶段仍在API调试 因为队员李昆乘经常出去玩没有 ...
- AtCoder Grand Contest #026 C - String Coloring
Time Limit: 3 sec / Memory Limit: 1024 MB Score : 600600 points Problem Statement You are given a st ...
- P2024 [NOI2001]食物链[扩展域并查集]
大水题一道啊,几分钟切掉. 还是扩展域,每个点拆3个点,之间连边表示有关系(即捕食关系).然后随便判定一下就好了,不难,毕竟NOI上古题目. #include<iostream> #inc ...
- DDP入门
DDP,即动态动态规划,可以用于解决一类带修改的DP问题. 我们从一个比较简单的东西入手,最大子段和. 带修改的最大子段和其实是常规问题了,经典的解决方法是用线段树维护从左,右开始的最大子段和和区间最 ...
- 【Lintcode】159.Find Minimum in Rotated Sorted Array
题目: Suppose a sorted array is rotated at some pivot unknown to you beforehand. (i.e., 0 1 2 4 5 6 7 ...
- poj2228Naptime——环形DP
题目:http://poj.org/problem?id=2228 dp[i][j][0/1]表示前i小时中第j小时睡(1)或不睡(0)的最优值: 注意第一个小时,若睡则对最终取结果有要求,即第n个小 ...
- Linq 支持动态字查询集合, 也就是说根据传入的值进行查询。
Linq 支持动态字查询集合, 也就是说根据传入的值进行查询. 比如我们有个类Patient, 其中有个字段PatientName, 现在有Patient集合, 想要查询PatientName为&qu ...
- JSON 的正确用法:Python、MongoDB、JavaScript与AjaxJSON 的正确用法:Python、MongoDB、JavaScript与Ajax
本文主要总结网站编写以来在传递 JSON 数据方面遇到的一些问题以及目前采用的解决方案.网站数据库采用 MongoDB,后端是 Python,前端采用“半分离”形式的 Riot.js,所谓半分离,是说 ...