中文字幕在线观看,亚洲а∨天堂久久精品9966,亚洲成a人片在线观看你懂的,亚洲av成人片无码网站,亚洲国产精品无码久久久五月天

在Java 8下更好地利用枚舉

2018-07-20    來(lái)源:編程學(xué)習(xí)網(wǎng)

容器云強(qiáng)勢(shì)上線!快速搭建集群,上萬(wàn)Linux鏡像隨意使用

在我們的云使用分析API中,返回了格式化過(guò)的分析數(shù)據(jù)(這里指生成分析圖)。最近,我們添加了一個(gè)特性,允許用戶選擇時(shí)間段(最開(kāi)始只可以按天選擇)。問(wèn)題是,代碼中每天中的時(shí)間段部分高度耦合了……

例如,下面這段代碼:

private static List<DataPoint> createListWithZerosForTimeInterval(DateTime from,
	DateTime to,
	ImmutableSet<Metric<? extends Number>> metrics) {
	List<DataPoint> points = new ArrayList<>();
	for (int i = 0; i <= Days.daysBetween(from, to).getDays(); i++) {
		points.add(new DataPoint().withDatas(createDatasWithZeroValues(metrics))
			.withDayOfYear(from.withZone(DateTimeZone.UTC)
				.plusDays(i)
				.withTimeAtStartOfDay()));
	}
	return points;
}

注意:Days、Minutes、Hours、Weeks 和Months一樣出現(xiàn)在代碼的后面部分。這些代碼來(lái)自Joda-Time Java時(shí)間和日期API。甚至方法的名字都沒(méi)有反應(yīng)出(各自的功能)。這些名字牢牢的綁定到了days的概念上。

我也嘗試過(guò)使用不同時(shí)間段方式(比如月、周、小時(shí))。但我看到了糟糕的switch/case鬼鬼祟祟地隱藏在代碼里。

你需要知道,switch/case=罪惡 已經(jīng)深入我心了。在我大學(xué)期間的兩段實(shí)習(xí)經(jīng)歷中就已經(jīng)這么認(rèn)為了。因此,我會(huì)不惜任何代價(jià)避免使用switch/case。這主要是因?yàn)樗鼈冞`反了開(kāi)放閉合原則。我深深地相信,遵循這個(gè)原則是寫出面向?qū)ο蟠a的最好實(shí)踐。我不是唯一一個(gè)這樣想的,Robert C. Martin曾經(jīng)說(shuō):

在很多方面,開(kāi)放閉合原則是面向?qū)ο笤O(shè)計(jì)的核心。遵循這個(gè)原則會(huì)從面向?qū)ο蠹夹g(shù)中收獲巨大的好處,比如可重用性和可維護(hù)性1

我告訴自己:“我們使用Java8或許可以發(fā)現(xiàn)一些新的特性來(lái)避免swtich/case的危險(xiǎn)場(chǎng)面出現(xiàn)”。使用Java8的新 functions(不是那么新,不過(guò)你知道我的意思)。我決定使用枚舉代表不同的可得到時(shí)間段。

public enum TimePeriod
{
    MINUTE(Dimension.MINUTE, 
           (from,
            to) -> Minutes.minutesBetween(from, to).getMinutes() + 1,
           Minutes::minutes, 
           from -> from.withZone(DateTimeZone.UTC)
                       .withSecondOfMinute(0)
                       .withMillisOfSecond(0)),
    HOUR(Dimension.HOUR,
         (from,
          to) -> Hours.hoursBetween(from, to).getHours() + 1,
         Hours::hours,
         from -> from.withZone(DateTimeZone.UTC)
                     .withMinuteOfHour(0)
                     .withSecondOfMinute(0)
                     .withMillisOfSecond(0)),
    DAY(Dimension.DAY,
        (from,
         to) -> Days.daysBetween(from, to).getDays() + 1,
        Days::days,
        from -> from.withZone(DateTimeZone.UTC)
                    .withTimeAtStartOfDay()),
    WEEK(Dimension.WEEK,
         (from,
          to) -> Weeks.weeksBetween(from, to).getWeeks() + 1,
         Weeks::weeks,
         from -> from.withZone(DateTimeZone.UTC)
                     .withDayOfWeek(1)
                     .withTimeAtStartOfDay()),
    MONTH(Dimension.MONTH,
          (from,
           to) -> Months.monthsBetween(from, to).getMonths() + 1,
          Months::months,
          from -> from.withZone(DateTimeZone.UTC)
                      .withDayOfMonth(1)
                      .withTimeAtStartOfDay());

    private Dimension<Timestamp> dimension;
    private BiFunction<DateTime, DateTime, Integer> getNumberOfPoints;
    private Function<Integer, ReadablePeriod> getPeriodFromNbOfInterval;
    private Function<DateTime, DateTime> getStartOfInterval;

    private TimePeriod(Dimension<Timestamp> dimension,
                       BiFunction<DateTime, DateTime, Integer> getNumberOfPoints,
                       Function<Integer, ReadablePeriod> getPeriodFromNbOfInterval,
                       Function<DateTime, DateTime> getStartOfInterval)
    {
        this.dimension = dimension;
        this.getNumberOfPoints = getNumberOfPoints;
        this.getPeriodFromNbOfInterval = getPeriodFromNbOfInterval;
        this.getStartOfInterval = getStartOfInterval;
    }

    public Dimension<Timestamp> getDimension()
    {
        return dimension;
    }

    public int getNumberOfPoints(DateTime from,
                                 DateTime to)
    {
        return getNumberOfPoints.apply(from, to);
    }

    public ReadablePeriod getPeriodFromNbOfInterval(int nbOfInterval)
    {
        return getPeriodFromNbOfInterval.apply(nbOfInterval);
    }

    public DateTime getStartOfInterval(DateTime from)
    {
        return getStartOfInterval.apply(from);
    }
}

通過(guò)枚舉,我就能夠很容易地修改代碼,允許用戶給圖表數(shù)據(jù)點(diǎn)指定時(shí)間段。

原來(lái)是這樣調(diào)用:

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

變成這樣調(diào)用:

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

支持getGraphDataPoints調(diào)用的Usage Analytics服務(wù)代碼已經(jīng)完成了,并且支持時(shí)間段。值得一提的是,它考慮了我之前說(shuō)過(guò)的開(kāi)放閉合原則。

標(biāo)簽: 代碼

版權(quán)申明:本站文章部分自網(wǎng)絡(luò),如有侵權(quán),請(qǐng)聯(lián)系:west999com@outlook.com
特別注意:本站所有轉(zhuǎn)載文章言論不代表本站觀點(diǎn)!
本站所提供的圖片等素材,版權(quán)歸原作者所有,如需使用,請(qǐng)與原作者聯(lián)系。

上一篇:EhCache緩存系統(tǒng)在集成環(huán)境中的使用詳解

下一篇:Java中有關(guān)Null的9件事