Wie Sie vergleichen mit .NET-Typen in einer NHibernate ICriteria Abfrage für eine ICompositeUserType?
-
27-09-2019 - |
Frage
Ich habe eine beantwortet Stackoverflow Frage, wie auf Legacy CHAR
Datenbank Datums- und Zeitfelder in eine .NET DateTime
Eigenschaft in meiner POCO kombinieren
hier (dank viel Berryl !). Jetzt versuche ich, eine benutzerdefinierte ICritera Abfrage Arbeit gegen diese sehr DateTime
Eigenschaft ohne Erfolg zu erhalten. hier ist meine Frage:
ICriteria criteria =
Session.CreateCriteria<InputFileLog>()
.Add(Expression.Gt(MembersOf<InputFileLog>.GetName(x => x.FileCreationDateTime), DateTime.Now.AddDays(-14)))
.AddOrder(Order.Desc(Projections.Id()))
.CreateCriteria(typeof(InputFile).Name)
.Add(Expression.Eq(MembersOf<InputFile>.GetName(x => x.Id), inputFileName));
IList<InputFileLog> list = criteria.List<InputFileLog>();
Und hier ist die Abfrage es der Erzeugungs:
SELECT this_.input_file_token as input1_9_2_,
this_.file_creation_date as file2_9_2_,
this_.file_creation_time as file3_9_2_,
this_.approval_ind as approval4_9_2_,
this_.file_id as file5_9_2_,
this_.process_name as process6_9_2_,
this_.process_status as process7_9_2_,
this_.input_file_name as input8_9_2_,
gonogo3_.input_file_token as input1_6_0_,
gonogo3_.go_nogo_ind as go2_6_0_,
inputfile1_.input_file_name as input1_3_1_,
inputfile1_.src_code as src2_3_1_,
inputfile1_.process_cat_code as process3_3_1_
FROM input_file_log this_
left outer join go_nogo gonogo3_ on this_.input_file_token=gonogo3_.input_file_token
inner join input_file inputfile1_ on this_.input_file_name=inputfile1_.input_file_name
WHERE this_.file_creation_date > :p0 and
this_.file_creation_time > :p1 and
inputfile1_.input_file_name = :p2
ORDER BY this_.input_file_token desc;
:p0 = '20100401',
:p1 = '15:15:27',
:p2 = 'LMCONV_JR'
Die Abfrage ist genau das, was ich erwarten würde, tatsächlich, es sei denn es tatsächlich geben Sie mir nicht, was ich will (alle Zeilen in den letzten 2 Wochen), weil in der DB es statt CHAR
s eine mehr als Vergleich mit DATE
s tut . Ich habe keine Ahnung, wie die Abfrage erhalten die CHAR
Werte in eine DATE
in der Abfrage, ohne einen CreateSQLQuery zu tun () zu konvertieren, , die ich möchte vermeiden . Jedermann weiß, wie dies zu tun;
UPDATE:
Ich habe geschaut, in versuchen Projections.SqlFunction()
oder Formeln zu verwenden, um dies zu erreichen, aber ohne bisher vergeblich. Hier ist der Code i SqlFunction()
haben, aber ich erhalte eine NHibernate.QueryException : property does not map to a single column: FileCreationDateTime
Fehler:
DateTime twoWeeksAgo = DateTime.Now.AddDays(-14);
ICriteria criteria =
Session.CreateCriteria<InputFileLog>()
.Add(Restrictions.Gt(Projections.SqlFunction("to_date", NHibernateUtil.DateTime, Projections.Property(MembersOf<InputFileLog>.GetName(x => x.FileCreationDateTime))), twoWeeksAgo))
//.Add(Expression.Gt(MembersOf<InputFileLog>.GetName(x => x.FileCreationDateTime), DateTime.Now.AddDays(-14)))
.AddOrder(Order.Desc(Projections.Id()))
.CreateCriteria(typeof(InputFile).Name)
.Add(Expression.Eq(MembersOf<InputFile>.GetName(x => x.Id), inputFileName));
Ich bin sicher, dass ich hier etwas falsch mache und es funktioniert, wie es immer noch nicht trotzdem, weil FileCreationDateTime
eine benutzerdefinierte ICompositeUserType
verwendet, die die .NET DateTime
Eigenschaft in zwei Oracle SQL CHAR
Spalten aufteilt (siehe diese Stackoverflow question für weitere Details).
Lösung
Ich stelle dar, endlich diese aus! hier ist der Code (aus irgendeinem Grunde ist Stackoverflow einige der Methoden, Namen in der diesem ersten Code macht die Syntax Farbe eines Typ-Snippet):
IList<InputFileLog> list = null;
DateTime twoWeeksAgo = DateTime.Now.AddDays(-14);
IProjection datePropProj =
DefaultStringFileCreationDateTimeType.GetFileCreationDateToDateSQLProjection();
IProjection timePropProj =
DefaultStringFileCreationDateTimeType.GetFileCreationTimeToDateSQLProjection();
IProjection dateConstProj =
DefaultStringFileCreationDateTimeType.GetFileCreationDateToDateSQLFunction(twoWeeksAgo);
IProjection timeConstProj =
DefaultStringFileCreationDateTimeType.GetFileCreationTimeToDateSQLFunction(twoWeeksAgo);
ICriteria criteria =
Session.CreateCriteria<InputFileLog>()
.Add(Restrictions.Or(Restrictions.GtProperty(datePropProj, dateConstProj),
Restrictions.And(Restrictions.EqProperty(datePropProj, dateConstProj),
Restrictions.GeProperty(timePropProj, timeConstProj))))
.AddOrder(Order.Desc(Projections.Id()))
.CreateCriteria(typeof(InputFile).Name)
.Add(Expression.Eq(MembersOf<InputFile>.GetName(x => x.Id), inputFileName));
list = criteria.List<InputFileLog>();
Und hier sind die Methoden i verwendet, um die SQLProjections
und SQLFunctions
zu erstellen. Ich habe sie in meinem ICompositeUserType
(DefaultStringFileCreationDateTime
), dass ich für die benutzerdefinierte Typenzuordnung auf der FileCreationDateTime
Eigenschaft verwendet.
public class DefaultStringFileCreationDateTime : ICompositeUserType
{
.
.
.
public const string DotNetDateFormat = "yyyyMMdd";
public const string DotNetTimeFormat = "HH:mm:ss";
public const string DbDateFormat = "YYYYMMDD";
public const string DbTimeFormat = "HH24:MI:SS";
private const string _nullDateRepresentationInDb = "00000000";
public struct DatabaseFieldNames
{
/// <summary>
/// File creation date column name.
/// </summary>
public const string FileCreationDate = "file_creation_date";
/// <summary>
/// File creation time column name.
/// </summary>
public const string FileCreationTime = "file_creation_time";
}
public static IProjection GetFileCreationDateToDateSQLProjection()
{
return ProjectionUtil.GetToDateSQLProjection(DatabaseFieldNames.FileCreationDate, DbDateFormat, NHibernateUtil.DateTime);
}
public static IProjection GetFileCreationTimeToDateSQLProjection()
{
return ProjectionUtil.GetToDateSQLProjection(DatabaseFieldNames.FileCreationTime, DbTimeFormat, NHibernateUtil.DateTime);
}
public static IProjection GetFileCreationDateToDateSQLFunction(DateTime dt)
{
return ProjectionUtil.GetToDateSQLFunction(dt, DotNetDateFormat, DbDateFormat);
}
public static IProjection GetFileCreationTimeToDateSQLFunction(DateTime dt)
{
return ProjectionUtil.GetToDateSQLFunction(dt, DotNetTimeFormat, DbTimeFormat);
}
}
Ich war schon mit dem consts
DatabaseFieldNames
struct
für die PropertyNames
Mitglied Umsetzung, so konnte ich diese hartcodierte Spaltennamen für die Projections
wiederzuverwenden i auch benötigt wird.
Hier ist die Projection
Utility-Klasse, wo die generischen to_date
Methoden leben:
public class ProjectionUtil
{
public static IProjection GetToDateSQLProjection(
string columnName, string dbToDateFormat, IType returnType)
{
return Projections.SqlProjection(
string.Format("to_date({0}, '{1}') as {0}", columnName, dbToDateFormat),
new string[] { columnName },
new IType[] { returnType });
}
public static IProjection GetToDateSQLFunction(
DateTime dt, string dotNetFormatString, string dbFormatString)
{
return Projections.SqlFunction(
"to_date",
NHibernateUtil.DateTime,
Projections.Constant(dt.ToString(dotNetFormatString)),
Projections.Constant(dbFormatString));
}
}
Schließlich ist hier die Oracle SQL, dass NHibernate generiert:
SELECT
this_.input_file_token as input1_9_2_,
this_.file_creation_date as file2_9_2_,
this_.file_creation_time as file3_9_2_,
this_.approval_ind as approval4_9_2_,
this_.file_id as file5_9_2_,
this_.process_name as process6_9_2_,
this_.process_status as process7_9_2_,
this_.input_file_name as input8_9_2_,
gonogo3_.input_file_token as input1_6_0_,
gonogo3_.go_nogo_ind as go2_6_0_,
inputfile1_.input_file_name as input1_3_1_,
inputfile1_.src_code as src2_3_1_,
inputfile1_.process_cat_code as process3_3_1_
FROM
input_file_log this_
left outer join go_nogo gonogo3_ on this_.input_file_token=gonogo3_.input_file_token
inner join input_file inputfile1_ on this_.input_file_name=inputfile1_.input_file_name
WHERE
(
to_date(file_creation_date, 'YYYYMMDD') > to_date(:p0, :p1) or
(
to_date(file_creation_date, 'YYYYMMDD') = to_date(:p2, :p3) and
to_date(file_creation_time, 'HH24:MI:SS') >= to_date(:p4, :p5)
)
) and
inputfile1_.input_file_name = :p6
ORDER BY this_.input_file_token desc;
:p0 = '20100415',
:p1 = 'YYYYMMDD',
:p2 = '20100415',
:p3 = 'YYYYMMDD',
:p4 = '18:48:48',
:p5 = 'HH24:MI:SS',
:p6 = 'LMCONV_JR'
kann nicht glauben, dass ich dieses bekam! Ich dachte, ich gehen zu müssen, zu einem ISQLQuery
sicher greifen!