aqiUtil.go 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292
  1. package common
  2. import (
  3. "math"
  4. "strconv"
  5. )
  6. //计算空气质量转换等级
  7. func CalculateAirQuality(d float64) string {
  8. //d, err := strconv.ParseFloat(airQuality, 64)
  9. //if err != nil {
  10. // panic(err.Error())
  11. //}
  12. val := "-"
  13. if d <= 50 {
  14. val = "优"
  15. } else if d > 50 && d <= 100 {
  16. val = "良"
  17. } else if d > 100 && d <= 150 {
  18. val = "轻度污染"
  19. } else if d > 150 && d <= 200 {
  20. val = "中度污染"
  21. } else if d > 200 && d < 300 {
  22. val = "重度污染"
  23. } else if d > 300 {
  24. val = "严重污染"
  25. }
  26. return val
  27. }
  28. /**
  29. * 特别注意此方法 不能在这里使用
  30. * 根据提供污染物的各项指标,对AQI进行计算
  31. *
  32. * @param pmtw PM2.5
  33. * @param pmte PM10
  34. * @param co 一氧化碳浓度
  35. * @param no2 二氧化氮浓度
  36. * @param o3 臭氧浓度
  37. * @param so2 二氧化硫浓度
  38. * @return
  39. */
  40. //func CountAqi(pmtw float64, pmte float64, co float64, no2 float64, o3 float64, so2 float64) *dao.AqiData {
  41. // var pmtwIaqi float64 = getPm25IAQI(pmtw)
  42. // var pmteIaqi float64 = getPm10IAQI(pmte)
  43. // var coIaqi float64 = getCoIAQI(co)
  44. // var no2Iaqi float64 = getNo2IAQI(no2)
  45. // var o3Iaqi float64 = getO3OneHourIAQI(o3)
  46. // var so2Iaqi float64 = getSo2IAQI(so2)
  47. // var aList []dao.AqiData
  48. // //// 初始化对象数组
  49. // if pmtwIaqi != 0 {
  50. // aList = append(aList, dao.AqiData{Name: "PM2.5", Aqi: pmtwIaqi})
  51. // }
  52. // if pmteIaqi != 0 {
  53. // aList = append(aList, dao.AqiData{Name: "PM10", Aqi: pmteIaqi})
  54. // }
  55. // if coIaqi != 0 {
  56. // aList = append(aList, dao.AqiData{Name: "CO", Aqi: coIaqi})
  57. // }
  58. // if no2Iaqi != 0 {
  59. // aList = append(aList, dao.AqiData{Name: "NO2", Aqi: no2Iaqi})
  60. // }
  61. // if o3Iaqi != 0 {
  62. // aList = append(aList, dao.AqiData{Name: "O3", Aqi: o3Iaqi})
  63. // }
  64. // if so2Iaqi != 0 {
  65. // aList = append(aList, dao.AqiData{Name: "SO2", Aqi: so2Iaqi})
  66. // }
  67. // sort.Slice(aList, func(i, j int) bool {
  68. // f := aList[i].Aqi - aList[j].Aqi
  69. // if f > 0 {
  70. // return true
  71. // }
  72. // return false
  73. // })
  74. // var aqi dao.AqiData
  75. // if len(aList) > 0 {
  76. // aqi = aList[len(aList)-1]
  77. // } else {
  78. // aqi = dao.AqiData{"PM10", 0.0}
  79. // }
  80. // return &aqi
  81. //}
  82. func getCoIAQI(co float64) float64 {
  83. if co > 0 {
  84. return countPerIaqi(co, 3)
  85. }
  86. return 0
  87. }
  88. func getNo2IAQI(no2 float64) float64 {
  89. if no2 > 0 {
  90. return countPerIaqi(no2, 4)
  91. }
  92. return 0
  93. }
  94. func getO3OneHourIAQI(o3One float64) float64 {
  95. if o3One > 0 {
  96. return countPerIaqi(o3One, 5)
  97. }
  98. return 0
  99. }
  100. func getSo2IAQI(so2 float64) float64 {
  101. if so2 > 0 {
  102. return countPerIaqi(so2, 6)
  103. }
  104. return 0
  105. }
  106. /**
  107. * 计算风向角度,转换成中文方向
  108. *
  109. * @param windDirection
  110. * @return
  111. */
  112. func CalculateDirection(windDirection string) string {
  113. d, _ := strconv.ParseFloat(windDirection, 64)
  114. val := "-"
  115. if d > 337.5 || d < 22.5 {
  116. //Namezh: "北", Name: "N", Center: 0, DirectionA: 337.5, DirectionB: 22.5
  117. val = "北"
  118. } else if d > 22.5 && d < 67.5 {
  119. //Direction{Namezh: "东北", Name: "NE", Center: 45, DirectionA: 22.5, DirectionB: 67.5}
  120. val = "东北"
  121. } else if d > 67.5 && d < 112.5 {
  122. //Namezh:"东", Name:"E", Center:90, DirectionA:67.5, DirectionB:112.5
  123. val = "东"
  124. } else if d > 112.5 && d < 157.5 {
  125. //Namezh: "东南", Name: "SN", Center: 135, DirectionA: 112.5, DirectionB: 157.5
  126. val = "东南"
  127. } else if d > 157.5 && d < 202.5 {
  128. //Namezh: "南", Name: "S", Center: 180, DirectionA: 157.5, DirectionB: 202.5
  129. val = "南"
  130. } else if d > 202.5 && d < 247.5 {
  131. //Namezh: "西南", Name: "SW", Center: 225, DirectionA: 202.5, DirectionB: 247.5
  132. val = "西南"
  133. } else if d > 247.5 && d < 292.5 {
  134. //Namezh: "西", Name: "W", Center: 270, DirectionA: 247.5, DirectionB: 292.5
  135. val = "西"
  136. } else if d > 292.5 && d < 337.5 {
  137. //Namezh: "西北", Name: "NW", Center: 315, DirectionA: 292.5, DirectionB: 337.5
  138. val = "西北"
  139. }
  140. return val
  141. }
  142. /**
  143. * 计算风速等级
  144. *
  145. * @param speed
  146. * @return
  147. */
  148. func CalculateSpeed(speed string) string {
  149. //风力等级范围
  150. d, _ := strconv.ParseFloat(speed, 64)
  151. val := "-"
  152. if d >= 0.0 && d <= 0.2 {
  153. val = "无风"
  154. } else if d >= 0.3 && d <= 1.5 {
  155. val = "软风(一级)"
  156. } else if d >= 1.6 && d <= 3.3 {
  157. val = "轻风(二级)"
  158. } else if d >= 3.4 && d <= 5.4 {
  159. val = "微风(三级)"
  160. } else if d >= 5.5 && d <= 7.9 {
  161. val = "和风(四级)"
  162. } else if d >= 8.0 && d <= 10.7 {
  163. val = "清劲风(五级)"
  164. } else if d >= 10.8 && d <= 13.8 {
  165. val = "强风(六级)"
  166. } else if d >= 13.9 && d <= 17.1 {
  167. val = "疾风(七级)"
  168. } else if d >= 17.2 && d <= 20.7 {
  169. val = "大风(八级)"
  170. } else if d >= 20.8 && d <= 24.4 {
  171. val = "烈风(九级)"
  172. } else if d >= 24.5 && d <= 28.4 {
  173. val = "狂风(十级)"
  174. } else if d >= 28.5 && d <= 32.6 {
  175. val = "暴风(11级)"
  176. } else if d > 32.6 {
  177. val = "台风(12级)"
  178. }
  179. return val
  180. }
  181. func getPollutionDegree(aqi float64) int {
  182. var pollutionDegree int = 1
  183. if aqi <= 50 {
  184. pollutionDegree = 1
  185. } else if aqi > 50 && aqi <= 100 {
  186. pollutionDegree = 2
  187. } else if aqi > 100 && aqi <= 150 {
  188. pollutionDegree = 3
  189. } else if aqi > 150 && aqi <= 200 {
  190. pollutionDegree = 4
  191. } else if aqi > 200 && aqi <= 250 {
  192. pollutionDegree = 5
  193. } else if aqi > 250 && aqi <= 300 {
  194. pollutionDegree = 6
  195. } else if aqi > 300 {
  196. pollutionDegree = 7
  197. }
  198. return pollutionDegree
  199. }
  200. func getDegree(pollutionDegree int) string {
  201. if pollutionDegree == 1 {
  202. return "优"
  203. } else if pollutionDegree == 2 {
  204. return "良"
  205. } else if pollutionDegree == 3 {
  206. return "轻微污染"
  207. } else if pollutionDegree == 4 {
  208. return "轻度污染"
  209. } else if pollutionDegree == 5 {
  210. return "中度污染"
  211. } else if pollutionDegree == 6 {
  212. return "中度重污染"
  213. } else if pollutionDegree == 7 {
  214. return "重度污染"
  215. }
  216. return "良"
  217. }
  218. func countPerIaqi(cp float64, r int) float64 {
  219. var bph float64 = 0 // 与 cp相近的污染物浓度限值的高位值
  220. var bpl float64 = 0 // 与 cp相近的污染物浓度限值的低位值
  221. var iaqih float64 = 0 // 与 bph对应的空气质量分指数
  222. var iaqil float64 = 0 // 与 bpl对应的空气质量分指数
  223. var iaqip float64 = 0 // 当前污染物项目P的空气质量分指数
  224. // 空气质量分指数及对应的污染物项目浓度限值
  225. var aqiArr [3][8]float64 = [3][8]float64{{0, 50, 100, 150, 200, 300, 400, 500}, {0, 50, 150, 250, 350, 420, 500, 600}, {0, 35, 75, 115, 150, 250, 350, 500}}
  226. var min float64 = aqiArr[r][0]
  227. var index int = len(aqiArr[r]) - 1
  228. var max float64 = aqiArr[r][index]
  229. if cp <= min || cp >= max {
  230. return 0.0
  231. } else {
  232. // 对每种污染物的bph、bpl、iaqih、iaqil进行赋值
  233. for i := r; i < (r + 1); i++ {
  234. for j := 0; j < len(aqiArr[0]); j++ {
  235. if cp < aqiArr[i][j] {
  236. bph = aqiArr[i][j]
  237. bpl = aqiArr[i][j-1]
  238. iaqih = aqiArr[0][j]
  239. iaqil = aqiArr[0][j-1]
  240. break
  241. }
  242. }
  243. }
  244. // 计算污染物项目P的空气质量分指数
  245. iaqip = (iaqih-iaqil)/(bph-bpl)*(cp-bpl) + iaqil
  246. return iaqip
  247. }
  248. }
  249. func getPm10IAQI(pmte float64) float64 {
  250. if pmte > 0 {
  251. return countPerIaqi(pmte, 1)
  252. }
  253. return 0
  254. }
  255. func getPm25IAQI(pmtw float64) float64 {
  256. if pmtw > 0 {
  257. return countPerIaqi(pmtw, 2)
  258. }
  259. return 0
  260. }
  261. func CountAqi(pmtw, pmte float64) float64 {
  262. var pmtwIaqi float64 = getPm25IAQI(pmtw)
  263. var pmteIaqi float64 = getPm10IAQI(pmte)
  264. return math.Max(pmteIaqi, pmtwIaqi)
  265. }
  266. func GetDegree(pmtw, pmte float64) string {
  267. return getDegree(getPollutionDegree(CountAqi(pmtw, pmte)))
  268. }
  269. func GetAqiAndDegree(pmtw, pmte float64) (float64, string) {
  270. aqi := CountAqi(pmtw, pmte)
  271. degree := getDegree(getPollutionDegree(aqi))
  272. return aqi, degree
  273. }