custom_analysis_edb.go 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648
  1. package excel
  2. import (
  3. "encoding/json"
  4. "errors"
  5. "eta/eta_api/models/data_manage/excel"
  6. "eta/eta_api/services/data"
  7. excelServices "eta/eta_api/services/excel"
  8. "eta/eta_api/utils"
  9. "fmt"
  10. "github.com/araddon/dateparse"
  11. "github.com/shopspring/decimal"
  12. "github.com/tealeg/xlsx"
  13. "github.com/xuri/excelize/v2"
  14. "strings"
  15. )
  16. // GetCustomAnalysisExcelData 获取自定义分析的表格data数据
  17. func GetCustomAnalysisExcelData(excelInfo *excel.ExcelInfo) (luckySheet excelServices.LuckySheet, err error, errMsg string) {
  18. // 查找当前excel的sheet列表
  19. sheetList, err := excel.GetAllSheetList(excelInfo.ExcelInfoId)
  20. if err != nil {
  21. errMsg = "保存失败"
  22. err = errors.New("查找当前excel的sheet列表失败,Err:" + err.Error())
  23. return
  24. }
  25. currSheetMap := make(map[string]string)
  26. for _, sheet := range sheetList {
  27. currSheetMap[sheet.SheetName] = sheet.SheetName
  28. }
  29. sheetCellDataMapList := make(map[int][]excelServices.LuckySheetCellData)
  30. // 通过excel的id获取各个sheet的单元格数据map
  31. {
  32. dataList, tmpErr := excel.GetAllSheetDataListByExcelInfoId(excelInfo.ExcelInfoId)
  33. if tmpErr != nil {
  34. err = tmpErr
  35. return
  36. }
  37. for _, cellData := range dataList {
  38. sheetDataList, ok := sheetCellDataMapList[cellData.ExcelSheetId]
  39. if !ok {
  40. sheetDataList = make([]excelServices.LuckySheetCellData, 0)
  41. }
  42. tmpSheetDataList := make([]excelServices.LuckySheetCellData, 0)
  43. err = json.Unmarshal([]byte(cellData.Data), &tmpSheetDataList)
  44. if err != nil {
  45. err = errors.New(fmt.Sprintf("解析data的配置失败,sheetId:%d,Err:%s", cellData.ExcelDataId, err.Error()))
  46. return
  47. }
  48. sheetCellDataMapList[cellData.ExcelSheetId] = append(sheetDataList, tmpSheetDataList...)
  49. }
  50. }
  51. // 转成luckySheet的数据格式
  52. luckySheet = excelServices.LuckySheet{
  53. SheetList: make([]excelServices.LuckySheetData, 0),
  54. }
  55. for _, sheet := range sheetList {
  56. var luckySheetDataConfig excelServices.LuckySheetDataConfig
  57. err = json.Unmarshal([]byte(sheet.Config), &luckySheetDataConfig)
  58. if err != nil {
  59. err = errors.New(fmt.Sprintf("解析sheet的配置失败,sheetId:%d,Err:%s", sheet.ExcelSheetId, err.Error()))
  60. return
  61. }
  62. tmpLuckySheetDataInfo := excelServices.LuckySheetData{
  63. Name: sheet.SheetName,
  64. Index: sheet.Sort,
  65. CellData: sheetCellDataMapList[sheet.ExcelSheetId],
  66. Config: luckySheetDataConfig,
  67. }
  68. luckySheet.SheetList = append(luckySheet.SheetList, tmpLuckySheetDataInfo)
  69. }
  70. return
  71. }
  72. // GenerateExcelCustomAnalysisExcel 根据自定义分析的表格data数据生成excel
  73. func GenerateExcelCustomAnalysisExcel(excelInfo *excel.ExcelInfo) (downloadFilePath string, err error, errMsg string) {
  74. luckySheet, err, errMsg := GetCustomAnalysisExcelData(excelInfo)
  75. if err != nil {
  76. return
  77. }
  78. downloadFilePath, err = luckySheet.ToExcel(false)
  79. return
  80. }
  81. // 数据集
  82. type dataStruct struct {
  83. Value float64
  84. Ok bool
  85. }
  86. // HandleEdbSequenceVal 处理日期集和数据集(获取可用的日期、数据集)
  87. func HandleEdbSequenceVal(dateSequenceVal, dataSequenceVal []string) (newDateList []string, newDataList []float64, err error, errMsg string) {
  88. newDateList = make([]string, 0)
  89. newDataList = make([]float64, 0)
  90. dataList := make([]dataStruct, 0)
  91. {
  92. for _, v := range dataSequenceVal {
  93. // 如果没有数据集,那么就过滤
  94. if v == `` {
  95. // 如果开始插入数据了,那么就需要插入不存在值
  96. dataList = append(dataList, dataStruct{
  97. Value: 0,
  98. Ok: false,
  99. })
  100. continue
  101. }
  102. v = strings.Replace(v, ",", "", -1)
  103. v = strings.Replace(v, ",", "", -1)
  104. // 过滤空格
  105. v = strings.Replace(v, " ", "", -1)
  106. v = strings.TrimSpace(v)
  107. var tmpVal float64
  108. if strings.Contains(v, "%") {
  109. // 百分比的数
  110. isPercentage, percentageValue := utils.IsPercentage(v)
  111. if !isPercentage {
  112. dataList = append(dataList, dataStruct{
  113. Value: 0,
  114. Ok: false,
  115. })
  116. continue
  117. }
  118. tmpValDec, tmpErr := decimal.NewFromString(percentageValue)
  119. if tmpErr != nil {
  120. dataList = append(dataList, dataStruct{
  121. Value: 0,
  122. Ok: false,
  123. })
  124. continue
  125. }
  126. tmpVal, _ = tmpValDec.Div(decimal.NewFromFloat(100)).Float64()
  127. } else {
  128. tmpValDec, tmpErr := decimal.NewFromString(v)
  129. if tmpErr != nil {
  130. dataList = append(dataList, dataStruct{
  131. Value: 0,
  132. Ok: false,
  133. })
  134. continue
  135. }
  136. tmpVal, _ = tmpValDec.Float64()
  137. }
  138. dataList = append(dataList, dataStruct{
  139. Value: tmpVal,
  140. Ok: true,
  141. })
  142. }
  143. }
  144. // 日期集
  145. dateList := make([]string, 0)
  146. {
  147. for _, v := range dateSequenceVal {
  148. // 如果没有数据集,那么就过滤
  149. if v == `` {
  150. // 如果开始插入数据了,那么就需要插入不存在值
  151. dateList = append(dateList, "")
  152. continue
  153. }
  154. // 过滤空格
  155. v = strings.Replace(v, " ", "", -1)
  156. t1, tmpErr := dateparse.ParseAny(v)
  157. if tmpErr != nil {
  158. dateList = append(dateList, "")
  159. continue
  160. }
  161. dateList = append(dateList, t1.Format(utils.FormatDate))
  162. }
  163. }
  164. lenData := len(dataList)
  165. lenDate := len(dateList)
  166. // 最小个数
  167. num := lenDate
  168. if num > lenData {
  169. num = lenData
  170. }
  171. for i := 0; i < num; i++ {
  172. date := dateList[i]
  173. tmpData := dataList[i]
  174. // 日期为空、数据为空
  175. if !tmpData.Ok || date == `` {
  176. continue
  177. }
  178. newDateList = append(newDateList, date)
  179. newDataList = append(newDataList, tmpData.Value)
  180. }
  181. return
  182. }
  183. // ResetCustomAnalysisData 数据重置的结构体
  184. type ResetCustomAnalysisData struct {
  185. EdbInfoId int
  186. DateList []string
  187. DataList []float64
  188. }
  189. // Refresh 刷新表格关联的指标信息
  190. func Refresh(excelInfo *excel.ExcelInfo, lang string) (err error, errMsg string, isSendEmail bool) {
  191. isSendEmail = true
  192. list, err := excel.GetAllExcelEdbMappingItemByExcelInfoId(excelInfo.ExcelInfoId)
  193. if err != nil {
  194. errMsg = "获取失败"
  195. err = errors.New("查找所有的mapping失败" + err.Error())
  196. return
  197. }
  198. // 没有关联指标,那么就退出吧
  199. if len(list) <= 0 {
  200. return
  201. }
  202. for k, v := range list {
  203. var tmpCalculateFormula excel.CalculateFormula
  204. err = json.Unmarshal([]byte(v.CalculateFormula), &tmpCalculateFormula)
  205. if err != nil {
  206. errMsg = "获取失败"
  207. err = errors.New(fmt.Sprintf("指标id:%d,公式转换失败,Err:%s", v.EdbInfoId, err.Error()))
  208. return
  209. }
  210. v.DateSequenceStr = tmpCalculateFormula.DateSequenceStr
  211. v.DataSequenceStr = tmpCalculateFormula.DataSequenceStr
  212. list[k] = v
  213. }
  214. luckySheet, err, errMsg := GetCustomAnalysisExcelData(excelInfo)
  215. if err != nil {
  216. err = errors.New(fmt.Sprintf("获取自定义分析Excel数据失败,Err:%s", err.Error()))
  217. return
  218. }
  219. // 获取excel表格数据
  220. xlsxFile, err := luckySheet.GetExcelData(false)
  221. if err != nil {
  222. err = errors.New(fmt.Sprintf("获取excel表格数据,Err:%s", err.Error()))
  223. return
  224. }
  225. //fmt.Println(xlsxFile)
  226. edbInfoIdList := make([]int, 0)
  227. for _, v := range list {
  228. edbInfoIdList = append(edbInfoIdList, v.EdbInfoId)
  229. // 获取对应的日期和数据列表
  230. relDateList, relDataList, tmpErr, tmpErrMsg := getDateAndDataList(v, xlsxFile)
  231. if tmpErr != nil {
  232. err = errors.New(fmt.Sprintf("《%s》获取对应的日期和数据列表,Err:%s", v.EdbName, tmpErr.Error()))
  233. errMsg = fmt.Sprintf("《%s》%s", v.EdbName, tmpErrMsg)
  234. return
  235. }
  236. if len(relDateList) <= 0 {
  237. errMsg = fmt.Sprintf("《%s》的日期序列为空", v.EdbName)
  238. err = errors.New(errMsg)
  239. return
  240. }
  241. if len(relDataList) <= 0 {
  242. errMsg = fmt.Sprintf("《%s》的数据序列为空", v.EdbName)
  243. err = errors.New(errMsg)
  244. return
  245. }
  246. //fmt.Println(v)
  247. req2 := &ResetCustomAnalysisData{
  248. EdbInfoId: v.EdbInfoId,
  249. DateList: relDateList,
  250. DataList: relDataList,
  251. }
  252. // 调用指标库去更新
  253. reqJson, tmpErr := json.Marshal(req2)
  254. if tmpErr != nil {
  255. err = errors.New(fmt.Sprintf("结构体转对象失败,Err:%s", tmpErr.Error()))
  256. return
  257. }
  258. respItem, tmpErr := data.ResetCustomAnalysisData(string(reqJson), lang)
  259. if tmpErr != nil {
  260. err = errors.New(fmt.Sprintf("调用指标库去更新,Err:%s", tmpErr.Error()))
  261. return
  262. }
  263. if respItem.Ret != 200 {
  264. errMsg = respItem.Msg
  265. err = errors.New(respItem.ErrMsg)
  266. return
  267. }
  268. }
  269. if len(edbInfoIdList) > 0 {
  270. err, _ = data.EdbInfoRefreshAllFromBaseV3(edbInfoIdList, false, true, true)
  271. }
  272. //xlsxFile.Sheet[]
  273. return
  274. }
  275. // getDateAndDataList
  276. // @Description: 获取待刷新的日期和数据
  277. // @author: Roc
  278. // @datetime 2023-12-21 15:21:14
  279. // @param excelEdbMappingItem *excel.ExcelEdbMappingItem
  280. // @param xlsxFile *xlsx.File
  281. // @return newDateList []string
  282. // @return newDataList []float64
  283. // @return err error
  284. // @return errMsg string
  285. func getDateAndDataList(excelEdbMappingItem *excel.ExcelEdbMappingItem, xlsxFile *xlsx.File) (newDateList []string, newDataList []float64, err error, errMsg string) {
  286. var dateList, dataList []string
  287. // 日期序列
  288. {
  289. sheetName, startColumnName, endColumnName, startNum, endNum, isAll, isRow, isColumn, tmpErr, tmpErrMsg := GetSheetStr(excelEdbMappingItem.DateSequenceStr)
  290. if tmpErr != nil {
  291. errMsg = tmpErrMsg
  292. err = tmpErr
  293. return
  294. }
  295. // 查找sheet页
  296. sheetInfo, ok := xlsxFile.Sheet[sheetName]
  297. if !ok {
  298. errMsg = "找不到" + sheetName
  299. err = errors.New(errMsg)
  300. return
  301. }
  302. startNum = startNum - 1
  303. endNum = endNum - 1
  304. // 选择行的数据
  305. if isRow {
  306. // 因为是选择一行的数据,所以开始行和结束行时一样的
  307. //endNum = startNum - 1
  308. // 开始列名、结束列
  309. var startColumn, endColumn int
  310. if isAll {
  311. // 结束列(其实也就是整列的个数)
  312. endColumn = len(sheetInfo.Cols) - 1
  313. } else {
  314. tmpStartColumn, tmpErr := excelize.ColumnNameToNumber(startColumnName)
  315. if tmpErr != nil {
  316. errMsg = "列名异常:" + startColumnName
  317. err = errors.New(errMsg)
  318. return
  319. }
  320. tmpEndColumn, tmpErr := excelize.ColumnNameToNumber(endColumnName)
  321. if tmpErr != nil {
  322. errMsg = "列名异常:" + endColumnName
  323. err = errors.New(errMsg)
  324. return
  325. }
  326. startColumn = tmpStartColumn - 1
  327. endColumn = tmpEndColumn - 1
  328. }
  329. // 最大列数,如果设置的超过了最大列数,那么结束列就是最大列数
  330. maxCol := len(sheetInfo.Cols)
  331. if endColumn > maxCol {
  332. endColumn = maxCol - 1
  333. }
  334. // 长度固定,避免一直申请内存空间
  335. dateList = make([]string, endColumn-startColumn+1)
  336. i := 0
  337. for currColumn := startColumn; currColumn <= endColumn; currColumn++ {
  338. currCell := sheetInfo.Cell(startNum, currColumn)
  339. if currCell == nil {
  340. errMsg = fmt.Sprintf("第%d列,第%d行数据异常", startColumn, startNum)
  341. err = errors.New(errMsg)
  342. return
  343. }
  344. //dateList = append(dateList, currCell.Value)
  345. dateList[i] = currCell.Value
  346. i++
  347. }
  348. } else if isColumn { // 选择列的数据
  349. if isAll {
  350. // 选择一整列的话,结束行得根据实际情况调整(其实也就是整个sheet有多少行)
  351. endNum = len(sheetInfo.Rows) - 1
  352. }
  353. startColumn, tmpErr := excelize.ColumnNameToNumber(startColumnName)
  354. if tmpErr != nil {
  355. errMsg = "列名异常:" + startColumnName
  356. err = errors.New(errMsg)
  357. return
  358. }
  359. startColumn = startColumn - 1
  360. // 最大行数,如果设置的超过了最大行数,那么结束行就是最大行数
  361. maxRow := len(sheetInfo.Rows)
  362. if endNum > maxRow {
  363. endNum = maxRow - 1
  364. }
  365. // 长度固定,避免一直申请内存空间
  366. dateList = make([]string, endNum-startNum+1)
  367. i := 0
  368. for currRow := startNum; currRow <= endNum; currRow++ {
  369. currCell := sheetInfo.Cell(currRow, startColumn)
  370. if currCell == nil {
  371. errMsg = fmt.Sprintf("第%d列,第%d行数据异常", startColumn, startNum)
  372. err = errors.New(errMsg)
  373. return
  374. }
  375. //dateList = append(dateList, currCell.Value)
  376. dateList[i] = currCell.Value
  377. i++
  378. }
  379. }
  380. }
  381. // 数据序列
  382. {
  383. sheetName, startColumnName, endColumnName, startNum, endNum, isAll, isRow, isColumn, tmpErr, tmpErrMsg := GetSheetStr(excelEdbMappingItem.DataSequenceStr)
  384. if tmpErr != nil {
  385. errMsg = tmpErrMsg
  386. err = tmpErr
  387. return
  388. }
  389. // 查找sheet页
  390. sheetInfo, ok := xlsxFile.Sheet[sheetName]
  391. if !ok {
  392. errMsg = "找不到" + sheetName
  393. err = errors.New(errMsg)
  394. return
  395. }
  396. startNum = startNum - 1
  397. endNum = endNum - 1
  398. // 选择行的数据
  399. if isRow {
  400. // 开始列名、结束列
  401. var startColumn, endColumn int
  402. if isAll {
  403. // 结束列(其实也就是整列的个数)
  404. endColumn = len(sheetInfo.Cols) - 1
  405. } else {
  406. tmpStartColumn, tmpErr := excelize.ColumnNameToNumber(startColumnName)
  407. if tmpErr != nil {
  408. errMsg = "列名异常:" + startColumnName
  409. err = errors.New(errMsg)
  410. return
  411. }
  412. tmpEndColumn, tmpErr := excelize.ColumnNameToNumber(endColumnName)
  413. if tmpErr != nil {
  414. errMsg = "列名异常:" + endColumnName
  415. err = errors.New(errMsg)
  416. return
  417. }
  418. startColumn = tmpStartColumn - 1
  419. endColumn = tmpEndColumn - 1
  420. }
  421. // 最大列数,如果设置的超过了最大列数,那么结束列就是最大列数
  422. maxCol := len(sheetInfo.Cols)
  423. if endColumn > maxCol {
  424. endColumn = maxCol - 1
  425. }
  426. // 长度固定,避免一直申请内存空间
  427. dataList = make([]string, endColumn-startColumn+1)
  428. i := 0
  429. for currColumn := startColumn; currColumn <= endColumn; currColumn++ {
  430. currCell := sheetInfo.Cell(startNum, currColumn)
  431. if currCell == nil {
  432. errMsg = fmt.Sprintf("第%d列,第%d行数据异常", startColumn, startNum)
  433. err = errors.New(errMsg)
  434. return
  435. }
  436. //dataList = append(dataList, currCell.Value)
  437. dataList[i] = currCell.Value
  438. i++
  439. }
  440. } else if isColumn { // 选择列的数据
  441. if isAll {
  442. // 选择一整列的话,结束行得根据实际情况调整(其实也就是整个sheet有多少行)
  443. endNum = len(sheetInfo.Rows) - 1
  444. }
  445. startColumn, tmpErr := excelize.ColumnNameToNumber(startColumnName)
  446. if tmpErr != nil {
  447. errMsg = "列名异常:" + startColumnName
  448. err = errors.New(errMsg)
  449. return
  450. }
  451. startColumn = startColumn - 1
  452. // 最大行数,如果设置的超过了最大行数,那么结束行就是最大行数
  453. maxRow := len(sheetInfo.Rows)
  454. if endNum > maxRow {
  455. endNum = maxRow - 1
  456. }
  457. // 长度固定,避免一直申请内存空间
  458. dataList = make([]string, endNum-startNum+1)
  459. i := 0
  460. for currRow := startNum; currRow <= endNum; currRow++ {
  461. currCell := sheetInfo.Cell(currRow, startColumn)
  462. if currCell == nil {
  463. errMsg = fmt.Sprintf("第%d列,第%d行数据异常", startColumn, startNum)
  464. err = errors.New(errMsg)
  465. return
  466. }
  467. //dataList = append(dataList, currCell.Value)
  468. dataList[i] = currCell.Value
  469. i++
  470. }
  471. }
  472. }
  473. //fmt.Println(dateList, dataList)
  474. //fmt.Println("日期序列结束")
  475. // 将excel中的日期、数据系列处理
  476. newDateList, newDataList, err, errMsg = HandleEdbSequenceVal(dateList, dataList)
  477. return
  478. }
  479. // GetSheetStr
  480. // @return sheetName string 用户选择的sheet名称
  481. // @return startColumnName string 用户选择的开始列名称
  482. // @return endColumnName string 用户选择的结束列名称
  483. // @return startNum int 用户选择的开始列单元格位置
  484. // @return endNum int 用户选择的结束列单元格位置
  485. // @return isAll bool 是否选择整行/列数据
  486. // @return isRow bool 是否选择行数据
  487. // @return isColumn bool 是否选择列数据
  488. // @return err error 错误信息
  489. // @return errMsg string 错误信息
  490. func GetSheetStr(sequenceStr string) (sheetName, startColumnName, endColumnName string, startNum, endNum int, isAll, isRow, isColumn bool, err error, errMsg string) {
  491. // 找出sheetName
  492. tmpList := strings.Split(sequenceStr, "!")
  493. if len(tmpList) != 2 {
  494. errMsg = "错误的公式,查找sheet异常:" + sequenceStr
  495. err = errors.New(errMsg)
  496. return
  497. }
  498. sheetName = tmpList[0]
  499. // 分离开始/结束单元格
  500. tmpList = strings.Split(tmpList[1], ":")
  501. if len(tmpList) != 2 {
  502. errMsg = "错误的公式,查找开始/结束单元格异常:" + sequenceStr
  503. err = errors.New(errMsg)
  504. return
  505. }
  506. startList := strings.Split(tmpList[0], "$")
  507. endList := strings.Split(tmpList[1], "$")
  508. lenList := len(startList)
  509. if lenList != len(endList) {
  510. errMsg = "错误的公式,开始与结束单元格异常:" + sequenceStr
  511. err = errors.New(errMsg)
  512. return
  513. }
  514. if lenList != 3 && lenList != 2 {
  515. errMsg = "错误的公式:" + sequenceStr
  516. err = errors.New(errMsg)
  517. return
  518. }
  519. startColumnName = startList[1]
  520. endColumnName = endList[1]
  521. // 长度为2的话,那说明是整行或整列
  522. if lenList == 2 {
  523. isAll = true
  524. startDeci, tmpErr1 := decimal.NewFromString(startList[1])
  525. endDeci, tmpErr2 := decimal.NewFromString(endList[1])
  526. if tmpErr1 == nil && tmpErr2 == nil {
  527. isRow = true // 正常转换的话,那么就是整行
  528. startNum = int(startDeci.IntPart())
  529. endNum = int(endDeci.IntPart())
  530. startColumnName = ``
  531. endColumnName = ``
  532. return
  533. }
  534. if tmpErr1 == nil || tmpErr2 == nil {
  535. err = errors.New("错误的公式2:" + sequenceStr)
  536. return
  537. }
  538. // 如果不能转成数字,那么就是整列
  539. isColumn = true
  540. return
  541. }
  542. // 确定行
  543. startDeci, tmpErr1 := decimal.NewFromString(startList[2])
  544. endDeci, tmpErr2 := decimal.NewFromString(endList[2])
  545. if tmpErr1 != nil && errors.Is(tmpErr1, tmpErr2) {
  546. err = errors.New("错误的公式3:" + sequenceStr)
  547. return
  548. }
  549. startNum = int(startDeci.IntPart())
  550. endNum = int(endDeci.IntPart())
  551. if startColumnName != endColumnName && startNum != endNum {
  552. errMsg = `选区不允许跨行或者跨列`
  553. err = errors.New(errMsg)
  554. }
  555. if startColumnName == endColumnName {
  556. isColumn = true // 列数据
  557. } else {
  558. isRow = true // 行数据
  559. }
  560. return
  561. }