1. 程式人生 > >『自己構建節假日API』

『自己構建節假日API』

RickAndMorty.png

大家好,我叫謝偉,是一名程式設計師。

之前梳理了一些內建庫的學習,收到了一些評論,絕大多數評論都在直指一個問題:為什麼梳理這些無關痛癢的內建庫?

好吧,看上去確實都是些簡單的內建庫的梳理。主要原因是為了:《後端工程師的攻略》這個系列,從零起步,教程到這個地步,看上去是需要提高難度了。後續加以改善。另外一個原因,其實是想告訴初學者,內建庫的很多程式碼組織方式,程式碼的編寫方式指的學習、借鑑、參考。

這期是我之前準備的,趁著這期還是放送出來吧。

核心很簡單:懂 Go 的基本語法,會使用內建庫的 time, 基本能搞到這些。

大綱:

前段日子專案中需要使用的國家規定的節假日,所以需要獲取這些資料。其實獲取這些資料的方式也很多:

  • 比如比較笨的方式:搜尋引擎,手動整理
  • 使用一些免費開放的第三方節假日 API: 不穩定,雖然這些資料也不需要頻繁的使用
  • 使用一些付費的第三方節假日 API:付費

於是本著簡潔的方式,編寫這麼一個節假日的庫。

要求:

  • 簡單的功能
  • 簡單的API

1. 資料獲取

資料來源需要可靠,所以需要尋找官方的通知來源。

比如:國務院辦公廳關於2018年部分節假日安排的通知

一般的方式呢,就是網頁資料抓取,解析出得到的資料。

這是第一步,獲取資料;當然,很多網站都可以找到這些資訊,這裡僅僅舉例。

2. 定義結構體

關於節假日,我們最需要知道的是什麼資訊?

  • 名稱
  • 時間安排

基於此,可以這麼設計結構體:

type OneCollection struct {
	Start  string `json:"start"`
	End    string `json:"end"`
	ChName string `json:"ch_name"`
	EnName string `json:"en_name"`
}
複製程式碼

包括:

  • 中文名稱
  • 英文名稱
  • 開始時間
  • 結束時間

關於節假日名稱呢,國家法定的節日是這麼幾個:元旦、春節、清明、端午、勞動、中秋、國慶

借鑑許多內建庫的處理方式:這種固定的資料的處理,可以使用列舉型別:

const
( NewYearDay = iota SpringFestivalDay TombSweepingDay LaborDay DragonBoatFestivalDay NationalDay MidAutumnFestivalDay ) var ChHolidays = [...]string{ "元旦", "春節", "清明節", "勞動節", "端午節", "中秋節", "國慶節", } var EnHolidays = [...]string{ "New Year\\'s Day", "Spring Festival", "Tomb-sweeping Day", "Labour Day", "Dragon Boat Festival", "Mid-autumn Festival", "National Day", } 複製程式碼

中英文,獲取指定偏移量上的資料即可,這種處理方式在內建庫很常見:比如時間型別的時間基本單位月:一月、二月、三月等

3. 歷史資料

基於上文的分析,要構建這個簡單的庫,要組織歷史節假日,這邊選取 2010年到 2019 年的資料。

// 一年
type YearCollection struct {
	Data []OneCollection `json:"data"`
}
// n 年
type CollectionYearHistory struct {
	Data [][]OneCollection `json:"data"`
}

// 2010 年到 2019年曆史資料
func FetchCollectionYearHistory() CollectionYearHistory {
	return CollectionYearHistory{
		Data: [][]OneCollection{
			holiday2019,
			holiday2018,
			holiday2017,
			holiday2016,
			holiday2015,
			holiday2014,
			holiday2013,
			holiday2012,
			holiday2011,
			holiday2010,
		},
	}
}

複製程式碼

4. 構建API

  • 選擇好的命名方式
  • 選擇好的資料返回格式
- FetchAll
- FetchByChName(year int, name string)
- FetchByEnName(year int, name string)
- FetchByMonth(year int, month int)
- FetchByYear(year int)
- FetchMonthHolidayCount(year int, month int)
- FetchYearHolidayCount(year int)
- IsHoliday
- IsWeekDay
- IsWorkDay

複製程式碼

之所以這樣設計, 是因為專案中經常會是這樣的操作:

  • 獲取所有的歷史資料
  • 獲取某年的歷史資料
  • 獲取某月的歷史資料
  • 統計某年某月的放假天數
  • 統計某年的放假天數
  • 判斷一個日期是否是節假日
  • 判斷一個日期是否是週末
  • 判斷一個日期是否是工作日

基於這些需求,構建了上文的API

以幾個API 為例,詳細的操作如何實現?

  • FetchByYear(year int): 從歷史資料中獲取
// FetchByYear get holidays by year in china
func FetchByYear(year int) []history.OneCollection {
	var index int
	nowYear, _, _ := time.Now().Date()
	if year > nowYear+1 {
		return nil
	}
	index = nowYear + 1 - year
	return history.FetchCollectionYearHistory().Data[index]

}
複製程式碼
  • FetchByMonth: 從某年的歷史資料中獲取
func FetchByMonth(year int, month int) []history.OneCollection {
	if month < 1 || month > 12 {
		return nil

	}
	collections := FetchByYear(year)
	var data []history.OneCollection
	for _, collection := range collections {
		collectionTime, _ := time.Parse("2006/01/02", collection.End)
		if int(collectionTime.Month()) == month {
			data = append(data, collection)
		}
	}
	return data
}
複製程式碼
  • IsHoliday: 歷史資料中是否能擊中目標
func IsHoliday(value string) bool {
	collectionTime, err := time.Parse("2006/01/02", value)
	if err != nil {
		return false
	}
	nowYear, _, _ := time.Now().Date()
	if collectionTime.Year() > nowYear+1 {
		return false
	}
	collections := FetchByYear(collectionTime.Year())
	for _, collection := range collections {
		startDate, _ := getDate(collection.Start)
		endDate, _ := getDate(collection.End)
		if collectionTime.Unix() >= startDate.Unix() && collectionTime.Unix() <= endDate.Unix() {
			return true
		}
	}
	return false

}
複製程式碼
  • IsWeekDay: 不是節假日、也不是工作日的
// IsWeekDay: judge date is week day or not
func IsWeekDay(value string) bool {
	return !IsWorkDay(value) && !IsHoliday(value)
}
複製程式碼

<後記>

基於上文的理念,其實可以寫很多小工具:

比如:

  • 獲取古詩詞
  • 給定一個日期,判斷星座
  • 給定一個身份證,判斷是否有效,屬於哪個地區等
  • 給定一個ip, 給出地理位置
  • 給定一箇中文,給出英文或者拼音
  • 給定文字,解釋其中文含義 ...

<完>