[英]Storing blobs in windows azure store additional data such as content type in DB
[英]Storing decimal data type in Azure Tables
覆盖基类中的ReadEntity
和WriteEntity
对此有好处。 每次检索实体时都没有必要编写EntityResolver
。
public class CustomTableEntity : TableEntity
{
public override void ReadEntity(IDictionary<string, EntityProperty> properties, OperationContext operationContext)
{
base.ReadEntity(properties, operationContext);
foreach (var thisProperty in
GetType().GetProperties().Where(thisProperty =>
thisProperty.GetType() != typeof(string) &&
properties.ContainsKey(thisProperty.Name) &&
properties[thisProperty.Name].PropertyType == EdmType.String))
{
var parse = thisProperty.PropertyType.GetMethods().SingleOrDefault(m =>
m.Name == "Parse" &&
m.GetParameters().Length == 1 &&
m.GetParameters()[0].ParameterType == typeof(string));
var value = parse != null ?
parse.Invoke(thisProperty, new object[] { properties[thisProperty.Name].StringValue }) :
Convert.ChangeType(properties[thisProperty.Name].PropertyAsObject, thisProperty.PropertyType);
thisProperty.SetValue(this, value);
}
}
public override IDictionary<string, EntityProperty> WriteEntity(OperationContext operationContext)
{
var properties = base.WriteEntity(operationContext);
foreach (var thisProperty in
GetType().GetProperties().Where(thisProperty =>
!properties.ContainsKey(thisProperty.Name) &&
typeof(TableEntity).GetProperties().All(p => p.Name != thisProperty.Name)))
{
var value = thisProperty.GetValue(this);
if (value != null)
{
properties.Add(thisProperty.Name, new EntityProperty(value.ToString()));
}
}
return properties;
}
}
使用时,只需从CustomTableEntity
扩展实体,插入或检索实体时它将是透明的。 它支持DateTime
, TimeSpan
, decimal
和那些具有Parse
方法或实现IConvertible
接口的类型。
您可以覆盖TableEntity中的WriteEntity方法并使用EntityResolver
public class CustomTableEntity : TableEntity
{
private const string DecimalPrefix = "D_";
public override IDictionary<string, EntityProperty> WriteEntity(OperationContext operationContext)
{
var entityProperties = base.WriteEntity(operationContext);
var objectProperties = GetType().GetProperties();
foreach (var item in objectProperties.Where(f => f.PropertyType == typeof (decimal)))
{
entityProperties.Add(DecimalPrefix + item.Name, new EntityProperty(item.GetValue(this, null).ToString()));
}
return entityProperties;
}
}
我们将使用的实体
public class MyEntity : CustomTableEntity
{
public string MyProperty { get; set; }
public decimal MyDecimalProperty1 { get; set; }
public decimal MyDecimalProperty2 { get; set; }
}
用法包括Create Table / Insert / Retreive
#region connection
CloudStorageAccount account = CloudStorageAccount.DevelopmentStorageAccount;
CloudTableClient client = account.CreateCloudTableClient();
CloudTable table = client.GetTableReference("mytable");
table.CreateIfNotExists();
#endregion
const string decimalPrefix = "D_";
const string partitionKey = "BlaBlaBla";
string rowKey = DateTime.Now.ToString("yyyyMMddHHmmss");
#region Insert
var entity = new MyEntity
{
PartitionKey = partitionKey,
RowKey = rowKey,
MyProperty = "Test",
MyDecimalProperty1 = (decimal) 1.2,
MyDecimalProperty2 = (decimal) 3.45
};
TableOperation insertOperation = TableOperation.Insert(entity);
table.Execute(insertOperation);
#endregion
#region Retrieve
EntityResolver<MyEntity> myEntityResolver = (pk, rk, ts, props, etag) =>
{
var resolvedEntity = new MyEntity {PartitionKey = pk, RowKey = rk, Timestamp = ts, ETag = etag};
foreach (var item in props.Where(p => p.Key.StartsWith(decimalPrefix)))
{
string realPropertyName = item.Key.Substring(decimalPrefix.Length);
System.Reflection.PropertyInfo propertyInfo = resolvedEntity.GetType().GetProperty(realPropertyName);
propertyInfo.SetValue(resolvedEntity, Convert.ChangeType(item.Value.StringValue, propertyInfo.PropertyType), null);
}
resolvedEntity.ReadEntity(props, null);
return resolvedEntity;
};
TableOperation retrieveOperation = TableOperation.Retrieve(partitionKey, rowKey, myEntityResolver);
TableResult retrievedResult = table.Execute(retrieveOperation);
var myRetrievedEntity = retrievedResult.Result as MyEntity;
// myRetrievedEntity.Dump();
#endregion
您是否尝试过使用Lokad.Cloud FatEntities产品?
我认为他们只是使用二进制序列化器来处理要存储在表中的整个对象。 看看“对象到云映射器”项目也是值得的:
@EUYUIL已经提出了一个很好的通用解决方案,我已经习惯了很好的效果,但是当他的回答表明它在使用Nullable类型时会失败。
// Get the underlying types 'Parse' method
if (curType.IsGenericType && curType.GetGenericTypeDefinition() == typeof(Nullable<>))
{
curType = Nullable.GetUnderlyingType(curType);
}
如果它帮助任何人,readEntity的内容覆盖foreach内部的方法。 可能有更好的方法来写这个,但为了说明的目的,这将做。
var curType = thisProperty.PropertyType;
// Get the underlying types 'Parse' method
if (curType.IsGenericType && curType.GetGenericTypeDefinition() == typeof(Nullable<>))
{
curType = Nullable.GetUnderlyingType(curType);
}
var parse = curType.GetMethods().SingleOrDefault(m =>
m.Name == "Parse" &&
m.GetParameters().Length == 1 &&
m.GetParameters()[0].ParameterType == typeof(string));
var value = parse != null ?
parse.Invoke(thisProperty, new object[] { properties[thisProperty.Name].StringValue }) :
Convert.ChangeType(properties[thisProperty.Name].PropertyAsObject, thisProperty.PropertyType);
thisProperty.SetValue(this, value);
您可以将属性的类型更改为double
。 然后,您必须通过将表实体映射到您自己的域类型来在decimal
和double
之间进行转换。 另一种选择是在由单个decimal
字段支持的实体上具有两个属性。 但是,您可能希望继续使用decimal
属性的Quantity
名称,因为它是存储在表中的double
属性,您必须通过重写ReadEntity
和WriteEntity
将此属性重命名为Quantity
。 那么你不妨使用这里提出的其他一些解决方案。
现在,您可能认为将decimal
存储为double
会导致某些值无法正确地进行往返。 虽然肯定存在不会往返的值,因为两种类型的范围和精度是非常不同的,大多数“正常”值,例如货币值不是天文数字大且具有人类可读的精度,将会没有任何问题地往返。 原因是Convert.ToDouble
执行的从double
到decimal
转换具有一个特殊属性:
此方法返回的Decimal值最多包含15位有效数字。
下面是一个例子,说明如何解决其他有问题的数字,因为这样:
var originalValue = 2.24M;
var doubleValue = (double) originalValue;
问题是使用浮点没有精确表示十进制数2.24,因为没有使用十进制数(2.24是有理数224/100)的有理数1/3的精确表示。 0.3333333333333333与1/3不同。 您可以通过以足够的精度打印doubleValue
来验证这一点。 Console.WriteLine($"{doubleValue:G17}")
产生
2.2400000000000002
但是,绕过该值仍然有效:
var roundTripValue = (decimal) doubleValue;
现在Console.WriteLine(roundTripValue)
产生
2.24
因此,只要您不对double
值进行任何计算,就可以使用它们来存储decimal
值,前提是double
和decimal
之间的转换符合上面引用的.NET规则。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.