数据表示和存储 - 人的年龄

时间:2011-12-10 08:03:52

标签: java database types

我有一个我正面临的设计问题,我把它带到这里,所以也许我可以得到更多的想法。

目前在我的项目中,我有一个场景,我需要代表孩子的年龄。

年龄可以从0到15岁从1岁到15岁很容易,我可以将它存储为整数, 但问题是如果年龄介于0和1之间。那么我想将它存储在几个月内。

将年龄存储为java.util.Date将无效,因为我不想与当前日期进行比较(我正在处理一个非常大的数据库)。关于浮点,你知道一年有12个月,所以我需要翻译0.5到6个月和0.7到我不知道...

我正在寻找一种将年龄存储在一种类型中的好方法。它可以是Enum等。 它应该尽可能小。并且应该可以在没有任何问题的情况下将类型保留在数据库中......

6 个答案:

答案 0 :(得分:7)

您可以随时将年龄存储在适当的小单位(例如几个月),并在需要时将其转换为年份。

答案 1 :(得分:4)

我建议两件事:

  1. 在数据库中,将birthdate存储为数据库的相应日期类型中的日期。如果您将年龄存储在数据库中,那么它将不断变得过时,并且需要更新。你说你的数据库很大,但这不应该是存储像'age'这样的瞬态值的理由。在birthdate上放一个好的索引,让数据库在需要时为你计算年龄。

  2. 如果需要,使用上述建议之一在java类中适当地表示(计算的)年龄(尽管这是有问题的)。一种方法是拥有单独的“月”和“年”成员,这些月只在年值为零时才显着。

答案 2 :(得分:2)

我会用下列两种方法中的任何一种来做:

  • 以几个月的时间存储每个人的年龄,并在必要时将其转换为年份。
  • 年龄为int,并且boolean标记表示age的值是年还是数月。

答案 3 :(得分:2)

当您存储相对于当前日期的值时,您需要每天更新值。您仍然可以将生日值存储在自19.1.100以来的数字表示中。

答案 4 :(得分:1)

TL;博士

... 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.time

以下是一些Java代码示例。现代方法使用Java 8及更高版本中内置的java.time类。

避免使用麻烦的旧日期时间类,例如DateCalendar。它们设计糟糕且令人困惑。从JSR 310开始,它们现在已经遗留下来了。

LocalDate

LocalDate类表示没有时间且没有时区的仅限日期的值。

时区对于确定日期至关重要。对于任何给定的时刻,日期在全球范围内因地区而异。例如,在Paris France午夜后的几分钟是新的一天,而Montréal Québec中仍然是“昨天”。

continent/region的格式指定proper time zone name,例如America/MontrealAfrica/CasablancaPacific/Auckland。切勿使用诸如ESTIST之类的3-4字母缩写,因为它们不是真正的时区,不是标准化的,甚至不是唯一的(!)。

ZoneId z = ZoneId.of( "America/Montreal" );
LocalDate today = LocalDate.now( z );

Period

将未附加到时间轴的时间段表示为PeriodDuration

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

从JDBC 4.2开始,我们可以直接用数据库交换java.time类型。与日期时间相关的java.sql类现在是遗留的,例如java.sql.Timestampjava.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.time框架内置于Java 8及更高版本中。这些类取代了麻烦的旧legacy日期时间类,例如java.util.DateCalendar和&amp; SimpleDateFormat

现在位于Joda-Timemaintenance mode项目建议迁移到java.time类。

要了解详情,请参阅Oracle Tutorial。并搜索Stack Overflow以获取许多示例和解释。规范是JSR 310

从哪里获取java.time类?

ThreeTen-Extra项目使用其他类扩展java.time。该项目是未来可能添加到java.time的试验场。您可以在此处找到一些有用的课程,例如IntervalYearWeekYearQuartermore

答案 5 :(得分:0)

我愿意

enum Age
{
     ZERO_MONTHS ,
     ONE_MONTH ,
     TWO_MONTHS ,
     THREE_MONTHS ,
     ... ,
     ELEVEN_MONTHS ,
     ONE_YEAR ,
     TWO_YEARS ,
     THREE_YEARS ,
     ... ,
     FIFTEEN_YEARS
}
相关问题