在ASP.NET Web API中使用OData的单例模式

时间:2023-01-17 16:02:43

从OData v4开始增加了对单例模式的支持,我们不用每次根据主键等来获取某个EDM,就像在C#中使用单例模式一样。实现方式大致需要两步:

1、在需要实现单例模式的导航属性上加上[Singleton]特性
2、在EDM配置的时候使用builder.Singleton<SomeModel>("SomeModels")来创建SingletonConfiguration<SomeModel>

首先还是从模型开始。

public class Employee
{
public int ID { get; set; }
public string Name { get; set; } [Singleton]
public Company Company { get; set; }
} public enum CompanyCategory
{
IT = ,
Communication = ,
Electronics = ,
Others =
} public class Company
{
public int ID { get; set; }
public string Name { get; set; }
public Int64 Revenue { get; set; }
public CompanyCategory Category { get; set; }
public List<Employee> Employees { get; set; }
}

以上,Company和Employee存在1对多关系,我们在Employee的Compnay导航属性上加上了[Singleton]特性,也就意味着我们希望在Company上使用单例模式。

然后就在WebApiConfig中配置如下:

public static class WebApiConfig
{
public static void Register(HttpConfiguration config)
{
... config.MapODataServiceRoute("ODataRoute", "odata", GetEdmModel());
} public static IEdmModel GetEdmModel()
{
ODataConventionModelBuilder builder = new ODataConventionModelBuilder(); EntitySetConfiguration<Employee> employeesConfiguration = builder.EntitySet<Employee>("Employees");
EntityTypeConfiguration<Employee> employeeTypeConfiguration = employeesConfiguration.EntityType;
employeeTypeConfiguration.Action("ResetDataSource"); SingletonConfiguration<Company> companiesConfiguration = builder.Singleton<Company>("Umbrella");
companiesConfiguration.EntityType.Action("ResetDataSource");
companiesConfiguration.EntityType.Function("GetEmployeesCount").Returns<int>(); builder.Namespace = "Hello"; return builder.GetEdmModel();
}
}

以上,builder.Singleton<Company>("Umbrella")方法创建SingletonConfiguration<Company>类型的实例,这是EDM实现单例的方式。

Company对应的控制器UmbrellaController

再来看Company对应的控制器,大致如下:

public class UmbrellaController : ODataController
{
public static Company Umbrella; static UmbrellaController()
{
InitData();
} private static void InitData()
{
Umbrella = new Company()
{
ID = ,
Name = "Umbrella",
Revenue = ,
Category = CompanyCategory.Communication,
Employees = new List<Employee>()
};
} ... [HttpPost]
public IHttpActionResult ResetDataSourceOnCompany()
{
InitData();
return StatusCode(HttpStatusCode.NoContent);
} public IHttpActionResult GetEmployeesCount()
{
return Ok(Umbrella.Employees.Count);
}
}

以上,UmbrellaController提供的静态Company类型的Umbrella可以在全局获取。ResetDataSourceOnCompany对应配置单例EDM的companiesConfiguration.EntityType.Action("ResetDataSource")的Action,GetEmployeesCount对应配置单例EDM的companiesConfiguration.EntityType.Function("GetEmployeesCount").Returns<int>()的Function。

● 查询

[EnableQuery]
public IHttpActionResult Get()
{
return Ok(Umbrella);
} public IHttpActionResult GetRevenueFromCompany()
{
return Ok(Umbrella.Revenue);
} public IHttpActionResult GetName()
{
return Ok(Umbrella.Employees);
}

以上,GetRevenueFromCompany和GetName分别获取属性,要符合惯例,即"Get+属性名称"。

● 添加

public IHttpActionResult Put(Company newCompany)
{
Umbrella = newCompany;
return StatusCode(HttpStatusCode.NoContent);
}

● Patch

public IHttpActionResult Patch(Delta<Company> item)
{
item.Patch(Umbrella);
return StatusCode(HttpStatusCode.NoContent);
}

● 创建Company上的Employees关系

/// <summary>
/// 创建Company上Employees的关系
/// </summary>
/// <param name="navigationProperty"></param>
/// <param name="link">Empolyee的uri地址</param>
/// <returns></returns>
[AcceptVerbs("POST")]
public IHttpActionResult CreateRef(string navigationProperty, [FromBody] Uri link)
{
//获取Employee的外键
int employeeId = HelperFunction.GetKeyValue<int>(link);
Employee employee = EmployeesController.Employees.First(x => x.ID == employeeId); if(employee == null || navigationProperty!="Employees")
{
return BadRequest();
} if(Umbrella.Employees == null)
{
Umbrella.Employees = new List<Employee>() { employee};
}
else
{
Umbrella.Employees.Add(employee);
}
return StatusCode(HttpStatusCode.NoContent);
}

其实就是往Company的Employees集合导航属性中添加一个元素。其中,HelperFunction.GetKeyValue<int>()方法用来获取link中Empoyee的主键。如下:

public static class HelperFunction
{
//获取主键值
public static TKey GetKeyValue<TKey>(Uri uri)
{
if(uri ==null)
{
throw new ArgumentException("uri");
} var rootPath = uri.AbsoluteUri.Substring(, uri.AbsoluteUri.LastIndexOf('/') + );
var odataUriParser = new ODataUriParser(WebApiConfig.GetEdmModel(), new Uri(rootPath), uri);
var odataPath = odataUriParser.ParsePath();
var keySegment = odataPath.LastSegment as KeySegment;
if(keySegment==null)
{
throw new InvalidOperationException("The link does not contain a key");
}
return (TKey)keySegment.Keys.First().Value;
}
}

● 删除Company上的Employees关系

/// <summary>
/// 删除关系
/// </summary>
/// <param name="relatedKey">Employee的主键</param>
/// <param name="navigationProperty"></param>
/// <returns></returns>
public IHttpActionResult DeleteRef(string relatedKey, string navigationProperty)
{
int key = int.Parse(relatedKey);
Employee employee = Umbrella.Employees.First(x => x.ID == key); if(navigationProperty != "Employees")
{
return BadRequest();
} Umbrella.Employees.Remove(employee);
return StatusCode(HttpStatusCode.NoContent);
}

其实就是删除Company的集合属性Employees中的一个Employee元素。

● 往Company的Employees集合里添加一个Employee元素

/// <summary>
/// 从Compnay处添加某个Employee
/// </summary>
/// <param name="employee"></param>
/// <returns></returns>
[HttpPost]
public IHttpActionResult PostToEmployees([FromBody] Employee employee)
{
EmployeesController.Employees.Add(employee);
if(Umbrella.Employees == null)
{
Umbrella.Employees = new List<Employee>() { employee };
}
else
{
Umbrella.Employees.Add(employee);
}
return Created(employee);
}

EmployeesController不详诉

public class EmployeesController : ODataController
{
public static List<Employee> Employees; static EmployeesController()
{
InitData();
} private static void InitData()
{
Employees = Enumerable.Range(, ).Select(i =>
new Employee()
{
ID = i,
Name = string.Format("Name {0}", i)
}).ToList();
} [EnableQuery]
public IHttpActionResult Get()
{
return Ok(Employees.AsQueryable());
} [EnableQuery]
public IHttpActionResult Get(int key)
{
return Ok(Employees.Where(e => e.ID == key));
} public IHttpActionResult GetCompanyFromEmployee([FromODataUri] int key)
{
var company = Employees.First(e => e.ID == key).Company;
if(company==null)
{
return StatusCode(HttpStatusCode.NotFound);
}
return Ok(company);
} public IHttpActionResult Post([FromBody] Employee employee)
{
Employees.Add(employee);
return Created(employee);
} [AcceptVerbs("PUT")]
public IHttpActionResult CreateRef([FromODataUri] int key, string navigationProperty, [FromBody] Uri link)
{
if(navigationProperty!="Company")
{
return BadRequest();
}
Employees.First(e => e.ID == key).Company = UmbrellaController.Umbrella;
return StatusCode(HttpStatusCode.NoContent); } public IHttpActionResult DeleteRef([FromODataUri] int key, string navigationProperty)
{
if(navigationProperty!="Company")
{
return BadRequest();
} Employees.First(e => e.ID == key).Company = null;
return StatusCode(HttpStatusCode.NoContent);
} public IHttpActionResult PutToCompany(int key, Company company)
{
var navigateCompany = Employees.First(e => e.ID == key).Company;
Employees.First(e => e.ID == key).Company = company;
if(navigateCompany.Name == "Umbrella")
{
//体现Singleton
UmbrellaController.Umbrella = navigateCompany;
}
else
{
return BadRequest();
}
return StatusCode(HttpStatusCode.NoContent);
} public IHttpActionResult PatchToCompany(int key, Delta<Company> company)
{
var navigateCompan = Employees.First(e => e.ID == key).Company;
company.Patch(Employees.First(e => e.ID == key).Company); if(navigateCompan.Name == "Umbrella")
{
company.Patch(UmbrellaController.Umbrella);
}
else
{
return BadRequest();
}
return StatusCode(HttpStatusCode.NoContent);
} [HttpPost]
public IHttpActionResult ResetDataSourceOnCollectionOfEmployee()
{
InitData();
return Ok();
}
}