मान लें कि आप EF6 का उपयोग कर रहे हैं और आप Kind
. सेट करना चाहते हैं किसी भी DateTime
. की संपत्ति डेटाबेस से Utc
. पर पुनः प्राप्त मान ।
इसी तरह के प्रश्न पूछे गए हैं, और उत्तर ObjectContext.ObjectMaterialized
घटना, लेकिन यह प्रक्षेपण का उपयोग करने वाले प्रश्नों के लिए सक्रिय नहीं हुआ।
मैं जिस समाधान का प्रस्ताव करने जा रहा हूं, वह DbDataReader
पर रूपांतरण करके, इकाई और प्रक्षेपण प्रश्नों दोनों के लिए काम करता है। स्तर (जिसका उपयोग इस प्रकार के प्रश्नों द्वारा किया जाता है)।
ऐसा करने के लिए, हमें एक कस्टम DbDataReader
. की आवश्यकता है कार्यान्वयन जो GetDateTime
तरीका। दुर्भाग्य से DbDataReader
को लागू कर रहा है व्युत्पन्न वर्ग को बहुत सारे बॉयलरप्लेट कोड की आवश्यकता होती है। सौभाग्य से मैंने C# सिंटैक्स त्रुटियों से बचने के लिए डायनामिक अनुवाद
जो बस प्रत्येक विधि को अंतर्निहित DbDataReader
. को सौंपता है उदाहरण के लिए, इसलिए मैं इसे वहीं से लूंगा:
abstract class DelegatingDbDataReader : DbDataReader
{
readonly DbDataReader source;
public DelegatingDbDataReader(DbDataReader source)
{
this.source = source;
}
public override object this[string name] { get { return source[name]; } }
public override object this[int ordinal] { get { return source[ordinal]; } }
public override int Depth { get { return source.Depth; } }
public override int FieldCount { get { return source.FieldCount; } }
public override bool HasRows { get { return source.HasRows; } }
public override bool IsClosed { get { return source.IsClosed; } }
public override int RecordsAffected { get { return source.RecordsAffected; } }
public override bool GetBoolean(int ordinal) { return source.GetBoolean(ordinal); }
public override byte GetByte(int ordinal) { return source.GetByte(ordinal); }
public override long GetBytes(int ordinal, long dataOffset, byte[] buffer, int bufferOffset, int length) { return source.GetBytes(ordinal, dataOffset, buffer, bufferOffset, length); }
public override char GetChar(int ordinal) { return source.GetChar(ordinal); }
public override long GetChars(int ordinal, long dataOffset, char[] buffer, int bufferOffset, int length) { return source.GetChars(ordinal, dataOffset, buffer, bufferOffset, length); }
public override string GetDataTypeName(int ordinal) { return source.GetDataTypeName(ordinal); }
public override DateTime GetDateTime(int ordinal) { return source.GetDateTime(ordinal); }
public override decimal GetDecimal(int ordinal) { return source.GetDecimal(ordinal); }
public override double GetDouble(int ordinal) { return source.GetDouble(ordinal); }
public override IEnumerator GetEnumerator() { return source.GetEnumerator(); }
public override Type GetFieldType(int ordinal) { return source.GetFieldType(ordinal); }
public override float GetFloat(int ordinal) { return source.GetFloat(ordinal); }
public override Guid GetGuid(int ordinal) { return source.GetGuid(ordinal); }
public override short GetInt16(int ordinal) { return source.GetInt16(ordinal); }
public override int GetInt32(int ordinal) { return source.GetInt32(ordinal); }
public override long GetInt64(int ordinal) { return source.GetInt64(ordinal); }
public override string GetName(int ordinal) { return source.GetName(ordinal); }
public override int GetOrdinal(string name) { return source.GetOrdinal(name); }
public override string GetString(int ordinal) { return source.GetString(ordinal); }
public override object GetValue(int ordinal) { return source.GetValue(ordinal); }
public override int GetValues(object[] values) { return source.GetValues(values); }
public override bool IsDBNull(int ordinal) { return source.IsDBNull(ordinal); }
public override bool NextResult() { return source.NextResult(); }
public override bool Read() { return source.Read(); }
public override void Close() { source.Close(); }
public override T GetFieldValue<T>(int ordinal) { return source.GetFieldValue<T>(ordinal); }
public override Task<T> GetFieldValueAsync<T>(int ordinal, CancellationToken cancellationToken) { return source.GetFieldValueAsync<T>(ordinal, cancellationToken); }
public override Type GetProviderSpecificFieldType(int ordinal) { return source.GetProviderSpecificFieldType(ordinal); }
public override object GetProviderSpecificValue(int ordinal) { return source.GetProviderSpecificValue(ordinal); }
public override int GetProviderSpecificValues(object[] values) { return source.GetProviderSpecificValues(values); }
public override DataTable GetSchemaTable() { return source.GetSchemaTable(); }
public override Stream GetStream(int ordinal) { return source.GetStream(ordinal); }
public override TextReader GetTextReader(int ordinal) { return source.GetTextReader(ordinal); }
public override Task<bool> IsDBNullAsync(int ordinal, CancellationToken cancellationToken) { return source.IsDBNullAsync(ordinal, cancellationToken); }
public override Task<bool> ReadAsync(CancellationToken cancellationToken) { return source.ReadAsync(cancellationToken); }
public override int VisibleFieldCount { get { return source.VisibleFieldCount; } }
}
और उस वास्तविक वर्ग का निर्माण करें जिसकी हमें इसके ऊपर आवश्यकता है:
class UtcDateTimeConvertingDbDataReader : DelegatingDbDataReader
{
public UtcDateTimeConvertingDbDataReader(DbDataReader source) : base(source) { }
public override DateTime GetDateTime(int ordinal)
{
return DateTime.SpecifyKind(base.GetDateTime(ordinal), DateTimeKind.Utc);
}
}
एक बार हमारे पास यह हो जाने के बाद, हमें इसे EF इंटरसेप्शन<का उपयोग करके EF इंफ्रास्ट्रक्चर में प्लग करना होगा। /ए> ।
हम एक कस्टम DbCommandInterceptor
. बनाकर शुरुआत करेंगे व्युत्पन्न वर्ग:
class UtcDateTimeConvertingDbCommandInterceptor : DbCommandInterceptor
{
public override void ReaderExecuted(DbCommand command, DbCommandInterceptionContext<DbDataReader> interceptionContext)
{
base.ReaderExecuted(command, interceptionContext);
if (!(interceptionContext.Result is UtcDateTimeConvertingDbDataReader)
&& interceptionContext.Result != null
&& interceptionContext.Exception == null)
interceptionContext.Result = new UtcDateTimeConvertingDbDataReader(interceptionContext.Result);
}
}
इसे पंजीकृत करें (उदाहरण के लिए आपके DbContext
. से व्युत्पन्न वर्ग स्थिर निर्माता):
public class YourDbContext : DbContext
{
static YourDbContext()
{
DbInterception.Add(new UtcDateTimeConvertingDbCommandInterceptor());
}
// ...
}
और हम कर रहे हैं।
अब हर DateTime
डेटाबेस से आने वाले मान में Kind
होगा संपत्ति Utc
. पर सेट है ।