我有一个我正面临的设计问题,我把它带到这里,所以也许我可以得到更多的想法。
目前在我的项目中,我有一个场景,我需要代表孩子的年龄。
年龄可以从0到15岁从1岁到15岁很容易,我可以将它存储为整数, 但问题是如果年龄介于0和1之间。那么我想将它存储在几个月内。
将年龄存储为java.util.Date
将无效,因为我不想与当前日期进行比较(我正在处理一个非常大的数据库)。关于浮点,你知道一年有12个月,所以我需要翻译0.5到6个月和0.7到我不知道...
我正在寻找一种将年龄存储在一种类型中的好方法。它可以是Enum
等。
它应该尽可能小。并且应该可以在没有任何问题的情况下将类型保留在数据库中......
答案 0 :(得分:7)
您可以随时将年龄存储在适当的小单位(例如几个月),并在需要时将其转换为年份。
答案 1 :(得分:4)
我建议两件事:
在数据库中,将birthdate存储为数据库的相应日期类型中的日期。如果您将年龄存储在数据库中,那么它将不断变得过时,并且需要更新。你说你的数据库很大,但这不应该是存储像'age'这样的瞬态值的理由。在birthdate上放一个好的索引,让数据库在需要时为你计算年龄。
如果需要,使用上述建议之一在java类中适当地表示(计算的)年龄(尽管这是有问题的)。一种方法是拥有单独的“月”和“年”成员,这些月只在年值为零时才显着。
答案 2 :(得分:2)
我会用下列两种方法中的任何一种来做:
int
,并且boolean
标记表示age
的值是年还是数月。答案 3 :(得分:2)
当您存储相对于当前日期的值时,您需要每天更新值。您仍然可以将生日值存储在自19.1.100以来的数字表示中。
答案 4 :(得分:1)
... SQL
SELECT * FROM person WHERE birthdate <= ? ;
多年......
LocalDate.now( ZoneId.of( "Africa/Tunis" ) )
.minus( Period.ofYears( 15 ) )
几个月......
LocalDate.now( ZoneId.of( "Africa/Tunis" ) )
.minus( Period.ofMonths( 6 ) )
接受的Answer by GreyBeardedGeek是正确的。您应该将日期值存储在日期类型的列中。将查询框架化为查找其出生日期在特定年龄的特定日期之前/之后的行。
以下是一些Java代码示例。现代方法使用Java 8及更高版本中内置的java.time类。
避免使用麻烦的旧日期时间类,例如Date
和Calendar
。它们设计糟糕且令人困惑。从JSR 310开始,它们现在已经遗留下来了。
LocalDate
LocalDate
类表示没有时间且没有时区的仅限日期的值。
时区对于确定日期至关重要。对于任何给定的时刻,日期在全球范围内因地区而异。例如,在Paris France午夜后的几分钟是新的一天,而Montréal Québec中仍然是“昨天”。
以continent/region
的格式指定proper time zone name,例如America/Montreal
,Africa/Casablanca
或Pacific/Auckland
。切勿使用诸如EST
或IST
之类的3-4字母缩写,因为它们不是真正的时区,不是标准化的,甚至不是唯一的(!)。
ZoneId z = ZoneId.of( "America/Montreal" );
LocalDate today = LocalDate.now( z );
Period
将未附加到时间轴的时间段表示为Period
或Duration
。
Period fifteenYears = Period.ofYears( 15 ) ;
计算SQL查询的目标日期。
LocalDate ld = today.minus( fifteenYears ) ; // Get the date fifteen years ago.
您的SQL将运行类似此示例的内容。 ?
是上面确定的LocalDate
值的占位符。是使用<=
还是<
进行比较取决于您的业务规则对年龄的定义。
SELECT *
FROM person
WHERE birthdate <= ?
;
关于几个月与几年的问题完全没问题。使用Period
个月而不是几年。
Period sixMonths = Period.ofMonths( 6 ) ;
您可能希望将该范围的可能Period
个对象设为常量或Enum
。
从JDBC 4.2开始,我们可以直接用数据库交换java.time类型。与日期时间相关的java.sql类现在是遗留的,例如java.sql.Timestamp
和java.sql.Date
。很好地解决了这些问题。
myPreparedStatement.setObject( 1 , ld ) ;
...和...
LocalDate birthdate = myResultSet.getObject( … , LocalDate.class ) ;
如果您只想要15岁的孩子,而不是16岁的孩子,请使用两个标准,一对早期和晚期的出生日期。
LocalDate earlierBirthDate = today.minus( Period.ofYears( 16 ) ) ;
LocalDate laterBirthDate = today.minus( Period.ofYears( 15 ) ) ;
SQL将是这样的:
SELECT *
FROM person
WHERE birthdate > ?
AND birthdate <= ?
;
java.time框架内置于Java 8及更高版本中。这些类取代了麻烦的旧legacy日期时间类,例如java.util.Date
,Calendar
和&amp; SimpleDateFormat
现在位于Joda-Time的maintenance mode项目建议迁移到java.time类。
要了解详情,请参阅Oracle Tutorial。并搜索Stack Overflow以获取许多示例和解释。规范是JSR 310。
从哪里获取java.time类?
ThreeTen-Extra项目使用其他类扩展java.time。该项目是未来可能添加到java.time的试验场。您可以在此处找到一些有用的课程,例如Interval
,YearWeek
,YearQuarter
和more。
答案 5 :(得分:0)
我愿意
enum Age
{
ZERO_MONTHS ,
ONE_MONTH ,
TWO_MONTHS ,
THREE_MONTHS ,
... ,
ELEVEN_MONTHS ,
ONE_YEAR ,
TWO_YEARS ,
THREE_YEARS ,
... ,
FIFTEEN_YEARS
}