在Java8下更好地利用枚举

在我们的云使用分析API中,返回了格式化过的分析数据(这里指生成分析图)。最近,我们添加了一个特性,允许用户选择时间段(最开始只可以按天选择)。问题是,代码中每天中的时间段部分高度耦合了……

创新互联是一家专业提供德化企业网站建设,专注与成都网站制作、成都网站设计、html5、小程序制作等业务。10年已为德化众多企业、政府机构等服务。创新互联专业网站设计公司优惠进行中。

例如,下面这段代码:

 
 
  1. private static List createListWithZerosForTimeInterval(DateTime from,
  2.     DateTime to,
  3.     ImmutableSet> metrics) {
  4.     List points = new ArrayList<>();
  5.     for (int i = 0; i <= Days.daysBetween(from, to).getDays(); i++) {
  6.         points.add(new DataPoint().withDatas(createDatasWithZeroValues(metrics))
  7.             .withDayOfYear(from.withZone(DateTimeZone.UTC)
  8.                 .plusDays(i)
  9.                 .withTimeAtStartOfDay()));
  10.     }
  11.     return points;
  12. }

注意:Days、Minutes、Hours、Weeks 和Months一样出现在代码的后面部分。这些代码来自Joda-Time Java时间和日期API。甚至方法的名字都没有反应出(各自的功能)。这些名字牢牢的绑定到了days的概念上。

我也尝试过使用不同时间段方式(比如月、周、小时)。但我看到了糟糕的switch/case鬼鬼祟祟地隐藏在代码里。

你需要知道,switch/case=罪恶 已经深入我心了。在我大学期间的两段实习经历中就已经这么认为了。因此,我会不惜任何代价避免使用switch/case。这主要是因为它们违反了开放闭合原则。我深深地相信,遵循这个原则是写出面向对象代码的***实践。我不是唯一一个这样想的,Robert C. Martin曾经说:

在很多方面,开放闭合原则是面向对象设计的核心。遵循这个原则会从面向对象技术中收获巨大的好处,比如可重用性和可维护性。(http://www.objectmentor.com/resources/articles/ocp.pdf )

我告诉自己:“我们使用Java8或许可以发现一些新的特性来避免swtich/case的危险场面出现”。使用Java8的新 functions(不是那么新,不过你知道我的意思)。我决定使用枚举代表不同的可得到时间段。

  
 
  1. public enum TimePeriod
  2. {
  3.     MINUTE(Dimension.MINUTE, 
  4.            (from,
  5.             to) -> Minutes.minutesBetween(from, to).getMinutes() + 1,
  6.            Minutes::minutes, 
  7.            from -> from.withZone(DateTimeZone.UTC)
  8.                        .withSecondOfMinute(0)
  9.                        .withMillisOfSecond(0)),
  10.     HOUR(Dimension.HOUR,
  11.          (from,
  12.           to) -> Hours.hoursBetween(from, to).getHours() + 1,
  13.          Hours::hours,
  14.          from -> from.withZone(DateTimeZone.UTC)
  15.                      .withMinuteOfHour(0)
  16.                      .withSecondOfMinute(0)
  17.                      .withMillisOfSecond(0)),
  18.     DAY(Dimension.DAY,
  19.         (from,
  20.          to) -> Days.daysBetween(from, to).getDays() + 1,
  21.         Days::days,
  22.         from -> from.withZone(DateTimeZone.UTC)
  23.                     .withTimeAtStartOfDay()),
  24.     WEEK(Dimension.WEEK,
  25.          (from,
  26.           to) -> Weeks.weeksBetween(from, to).getWeeks() + 1,
  27.          Weeks::weeks,
  28.          from -> from.withZone(DateTimeZone.UTC)
  29.                      .withDayOfWeek(1)
  30.                      .withTimeAtStartOfDay()),
  31.     MONTH(Dimension.MONTH,
  32.           (from,
  33.            to) -> Months.monthsBetween(from, to).getMonths() + 1,
  34.           Months::months,
  35.           from -> from.withZone(DateTimeZone.UTC)
  36.                       .withDayOfMonth(1)
  37.                       .withTimeAtStartOfDay());
  38.  
  39.     private Dimension dimension;
  40.     private BiFunction getNumberOfPoints;
  41.     private Function getPeriodFromNbOfInterval;
  42.     private Function getStartOfInterval;
  43.  
  44.     private TimePeriod(Dimension dimension,
  45.                        BiFunction getNumberOfPoints,
  46.                        Function getPeriodFromNbOfInterval,
  47.                        Function getStartOfInterval)
  48.     {
  49.         this.dimension = dimension;
  50.         this.getNumberOfPoints = getNumberOfPoints;
  51.         this.getPeriodFromNbOfInterval = getPeriodFromNbOfInterval;
  52.         this.getStartOfInterval = getStartOfInterval;
  53.     }
  54.  
  55.     public Dimension getDimension()
  56.     {
  57.         return dimension;
  58.     }
  59.  
  60.     public int getNumberOfPoints(DateTime from,
  61.                                  DateTime to)
  62.     {
  63.         return getNumberOfPoints.apply(from, to);
  64.     }
  65.  
  66.     public ReadablePeriod getPeriodFromNbOfInterval(int nbOfInterval)
  67.     {
  68.         return getPeriodFromNbOfInterval.apply(nbOfInterval);
  69.     }
  70.  
  71.     public DateTime getStartOfInterval(DateTime from)
  72.     {
  73.         return getStartOfInterval.apply(from);
  74.     }
  75. }

通过枚举,我就能够很容易地修改代码,允许用户给图表数据点指定时间段。

原来是这样调用:

  
 
  1. for (int i = 0; i <= Days.daysBetween(from, to).getDays(); i++)

变成这样调用:

 
 
  1. for (int i = 0; i < timePeriod.getNumberOfPoints(from, to); i++)

支持getGraphDataPoints调用的Usage Analytics服务代码已经完成了,并且支持时间段。值得一提的是,它考虑了我之前说过的开放闭合原则。

当前题目:在Java8下更好地利用枚举
文章URL:http://www.csdahua.cn/qtweb/news31/271881.html

网站建设、网络推广公司-快上网,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等

广告

声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 快上网