java导入excel更新设备经纬度度数或者度分秒
文章目录
- 一、背景介绍
- 二、页面效果
- 三、代码
- 0.pom.xml
- 1.ImportDevice.vue
- 2.ImportDeviceError.vue
- 3.system.js
- 4.DeviceManageControl
- 5.DeviceManageUserControl
- 6.Repeater
- 7.FileUtils
- 8.ResponseModel
- 9.EnumLongitudeLatitude
- 10.词条
- 四、注意点
- 本人其他相关文章链接
一、背景介绍
项目采用:Arco Design+java+mysql+springboot+vue3
项目中转台存入有2种:
- 第一种南向上报入库;
- 第二种手动添加入库;
但是这两种方式中转台都没有经纬度和海拔值,所以需要导入excel去批量更新经纬度和海拔。
同时参数配置中展示效果有2种:
- 度数显示
- 度分秒显示
因为展示效果有2种,所以excel导入模版也有2种不同的模版,本篇文章就是记录excel导入更新入库,同时记录“度数”和“度分秒”之间的转换方式。
二、页面效果
度数显示
度分秒显示
下载excel模版和导入
数据库表字段设计
导入失败错误提示,展示所有错误的信息
导入接口会返回详细错误提示
三、代码
0.pom.xml
<dependency><groupId>org.apache.poi</groupId><artifactId>poi</artifactId><version>5.2.3</version>
</dependency>
<dependency><groupId>org.apache.poi</groupId><artifactId>poi-ooxml</artifactId><version>5.2.3</version>
</dependency>
1.ImportDevice.vue
<template><a-modal :width="570" :align-center="true"><template #title><div class="title">{{ openTitle }}</div></template><div class="frameParent"><div class="frameWrapper"><div class="frameGroup"><div class="textWrapper"><div class="text">{{$t('ImportTempletDownLoad')}}</div></div><div class="frameContainer"><div class="iconBaseParent"><div class="iconBase"> <icon-download /></div><div class="text1" @click="templateOne">{{$t('LngLatDegree')}}</div></div><div class="iconBaseParent"><div class="iconBase"> <icon-download /></div><div class="text1" @click="templateTwo">{{$t('LngLatDegreeMinuteSecond')}}</div></div></div></div></div><div class="frameDiv"><div class="groupWrapper"><div class="groupDiv"><div class="textParent"><div class="text3">{{$t('BrowseImport')}}</div><div class="parent"><a-input class="div" v-model="fileName" disabled /><a-upload accept=".xlsx, .xls" @change="onChange" :auto-upload="false" :show-file-list="false"><template #upload-button><svg-loader class="frameIcon" name="import-file"></svg-loader></template></a-upload></div></div></div></div></div></div><template #footer><a-button @click="closeDevice()">{{$t('cancelLabel')}}</a-button><a-button type="primary" @click="saveDevice()">{{$t('ExportTaskEnter')}}</a-button></template></a-modal>
</template><script setup>
import {defineEmits, inject, ref} from "vue";
import { Message } from "@arco-design/web-vue";
import {importDevive} from "@/views/pages/system/system.js";
const t = inject('t')
const openTitle = ref('')
const emits = defineEmits(['receiveImportError','refresh-flag'])const templateOne = () => {window.open(`${location.origin}/api/deviceManage/downloadImportTemplateAboutDegrees`)
}
const templateTwo = () => {window.open(`${location.origin}/api/deviceManage/downloadImportTemplateAboutDMS`)
}
const files = ref([]);
const fileName = ref('');const onChange = (fileList) => {files.value = [];if (fileList.length > 0) {const lastUploadedFile = fileList[fileList.length - 1];const lastFileName = lastUploadedFile.name;fileName.value = lastFileName;files.value.push(lastUploadedFile)}
};const closeDevice =() => {emits('refresh-flag', 1)
}const saveDevice = () => {const uploadedFile = files.value;const formData = new FormData();formData.append("file", uploadedFile[0].file);importDevive(formData).then(response =>{if (response.code === 200) {Message.success(t('HeaderSuccessfulResult'))emits('refresh-flag', 1)} else {emits('receiveImportError', response.data);}})
};</script><style scoped lang="less">
.text {position: relative;line-height: 22px;
}.textWrapper {display: flex;flex-direction: row;align-items: center;justify-content: flex-start;
}.iconBase {width: 20px;position: relative;height: 20px;overflow: hidden;flex-shrink: 0;
}.text1 {position: relative;text-decoration: underline;line-height: 22px;cursor: pointer;
}.iconBaseParent {display: flex;flex-direction: row;align-items: center;justify-content: flex-start;gap: 8px;
}.frameContainer {display: flex;flex-direction: row;align-items: center;justify-content: flex-start;gap: 32px;text-align: center;color: #3348ff;
}.frameGroup {width: 400px;display: flex;flex-direction: column;align-items: flex-start;justify-content: flex-start;gap: 16px;
}.frameWrapper {align-self: stretch;display: flex;flex-direction: row;align-items: center;justify-content: flex-start;flex-wrap: wrap;align-content: center;padding: 0px 20px 20px;
}.text3 {align-self: stretch;position: relative;line-height: 22px;
}.div1 {position: relative;
}.div {position: absolute;top: 0px;left: 0px;border-radius: 8px;background-color: #fbfcfd;border: 1px solid #dde4ed;box-sizing: border-box;width: 400px;height: 40px;display: flex;flex-direction: row;align-items: center;justify-content: flex-start;padding: 8px 10px;
}.frameIcon {position: absolute;top: 6px;left: 364px;width: 28px;height: 28px;
}.parent {width: 400px;position: relative;height: 40px;color: #202b40;font-family: 'PingFang TC';
}.textParent {position: absolute;top: 0px;left: 0px;width: 400px;display: flex;flex-direction: column;align-items: flex-start;justify-content: flex-start;gap: 4px;
}.groupDiv {width: 400px;position: relative;height: 66px;
}.groupWrapper {flex: 1;display: flex;flex-direction: column;align-items: center;justify-content: center;
}.frameDiv {align-self: stretch;display: flex;flex-direction: row;align-items: center;justify-content: center;flex-wrap: wrap;align-content: center;padding: 20px;
}.frameParent {align-self: stretch;display: flex;flex-direction: column;align-items: flex-start;justify-content: flex-start;color: #778091;
}.secondaryButton1text {box-shadow: 0px -2px 2px rgba(48, 48, 48, 0.1) inset, 0px 12px 12px rgba(255, 255, 255, 0.12) inset;border-radius: 8px;background-color: #fff;border: 1px solid #dde4ed;box-sizing: border-box;height: 36px;overflow: hidden;display: flex;flex-direction: row;align-items: center;justify-content: center;padding: 5px 16px;
}.buttontertiaryButtonnormal {height: 36px;display: flex;flex-direction: row;align-items: flex-start;justify-content: flex-start;
}.primaryButton {box-shadow: 0px -2px 2px rgba(48, 48, 48, 0.1) inset, 0px 12px 12px rgba(255, 255, 255, 0.12) inset;border-radius: 8px;background-color: #004ce5;height: 36px;display: flex;flex-direction: row;align-items: center;justify-content: flex-start;padding: 5px 16px;box-sizing: border-box;
}.buttonprimaryButtonnormal {border-radius: 8px;display: flex;flex-direction: row;align-items: flex-start;justify-content: flex-start;text-align: left;color: #fff;
}.button {align-self: stretch;border-radius: 0px 0px 12px 12px;background-color: #f6f7f9;display: flex;flex-direction: row;align-items: center;justify-content: flex-end;padding: 16px 20px;gap: 12px;text-align: center;color: #455471;
}.headerParent {width: 100%;position: relative;border-radius: 12px;background-color: #fff;display: flex;flex-direction: column;align-items: flex-start;justify-content: flex-start;gap: 20px;text-align: left;font-size: 14px;color: #202b40;font-family: 'PingFang SC';
}
</style>
2.ImportDeviceError.vue
<template><a-modal :width="1200" :align-center="true" :hide-cancel="true"><template #title><div class="title">{{ openTitle }}</div></template><div class="table-line"><a-table :data="errorTableData" :bordered="false" :pagination="false" :loading="errorTableLoading" v-model:selected-keys="errorSelectedKeys" row-key="serialNo"><template #columns><a-table-column dataIndex="alias" :title="$t('RepeaterAlias')" :tooltip="true"></a-table-column><a-table-column dataIndex="serialNo" :title="$t('SerialNo')" :tooltip="true"></a-table-column><a-table-column dataIndex="lng" :title="$t('Longitude_Degree')" :tooltip="true"><template #cell="{ record }"><div style="display: flex; align-items: center;"><a-select v-model="record.lngOr" style="margin-right: 8px;width: 100px" disabled><a-option :value=0 :label="$t('EastLongitude')"></a-option><a-option :value=1 :label="$t('WestLongitude')"></a-option></a-select>{{record.lng}}</div></template></a-table-column><a-table-column dataIndex="lat" :title="$t('Latitude_Degree')" :tooltip="true"><template #cell="{ record }"><div style="display: flex; align-items: center;"><a-select v-model="record.latOr" style="margin-right: 8px;width: 100px" disabled><a-option :value=0 :label="$t('NorthLatitude')"></a-option><a-option :value=1 :label="$t('SouthLatitude')"></a-option></a-select>{{record.lat}}</div></template></a-table-column><a-table-column dataIndex="elevation" :title="$t('Elevation')" :tooltip="true"><template #cell="{ record }">{{record.elevation}}</template></a-table-column><a-table-column dataIndex="importErrorMessage" :title="$t('CauseOfImportFailure')" :tooltip="true"></a-table-column></template></a-table></div></a-modal>
</template><script setup>
import {ref, defineExpose} from "vue";
const errorTableData = ref([])
const errorSelectedKeys = ref([])
const errorTableLoading = ref(false)
const openTitle = ref('')const setData = (record, title) => {openTitle.value = titleif (null != record) {errorTableData.value = record;}
}defineExpose({setData
})
</script><style scoped lang="less">
.table-line {box-sizing: border-box;width: 1200px;height: 500px;
}
</style>
3.system.js
import { $http, Method } from "@/http";export const importDevive = (data) => {return $http({url: `/api/deviceManage/importData`,method: Method.POST,headers: {'Content-Type': 'multipart/form-data','X-Requested-With': 'XMLHttpRequest'},data})
}
4.DeviceManageControl
@Operation(summary = "导入数据")
@PostMapping(value = "/importData")
public ResponseModel importData(@RequestParam MultipartFile file) {return deviceManageUserControl.importData(file);
}
5.DeviceManageUserControl
public ResponseModel importData(MultipartFile file) {String fileName = file.getOriginalFilename();String extension = FileUtils.getFileExtension(fileName);if (!extension.equals(".xls") && !extension.equals(".xlsx")) {return ResponseModel.ofError(languageFind.findKey("Device_Excel_ColumnIsNotExits"));}int excelColumnCount = FileUtils.getExcelColumnCount(file);if (excelColumnCount != 8 && excelColumnCount != 12) {return ResponseModel.ofError(languageFind.findKey("Device_Excel_ColumnIsNotExits"));}if (excelColumnCount > 1000) {return ResponseModel.ofError(languageFind.findKey("Device_Excel_RepeatData_1000MaximumData"));}if (excelColumnCount == 8) {return writeDegree(file);} else if (excelColumnCount == 12) {return writeDegreeMiuSec(file);}return ResponseModel.ofSuccess();}/*** 写入度的形式* @param file 文件* @return 结果*/public ResponseModel writeDegree(MultipartFile file) {List<Repeater> repeaterList = new ArrayList<>();List<Repeater> listError = new ArrayList<>();Repeater repeater = null;List<String> excelValuesList = new ArrayList<>();List<String> serialNoList = new ArrayList<>();try {Account account = (Account) SecurityUtils.getSubject().getPrincipal();List<Repeater> authorityRepeaters = commonIClient.getAuthorityRepeaters(account.getUserName());String fileName = file.getOriginalFilename();InputStream inputStream = file.getInputStream();Workbook workbook = null;if (fileName.indexOf(".xlsx") > 0) {workbook = new XSSFWorkbook(inputStream);} else if (fileName.indexOf(".xls") > 0) {workbook = new HSSFWorkbook(inputStream);}Sheet sheet = workbook.getSheetAt(0);int lastNum = sheet.getLastRowNum();Row row = sheet.getRow(0);//行int cellNum = row.getLastCellNum();Cell ce = null;for(int i = 1; i < lastNum + 1; i++){boolean isImport = true;row = sheet.getRow(i);for (int j = 0; j < cellNum; j++){ce = row.getCell(j);excelValuesList.add(getCellValueByCell(ce));}logger.info("writeDegree-excelValuesList:{}", excelValuesList);if(excelValuesList.size() > 0){repeater = new Repeater();if (excelValuesList.get(1) != null && !excelValuesList.get(1).equals("")) {repeater.setAlias(excelValuesList.get(1));}if (excelValuesList.get(2) != null && !excelValuesList.get(2).equals("")) {String serialNo = excelValuesList.get(2);repeater.setSerialNo(serialNo);List<Repeater> hasRepeater = authorityRepeaters.stream().filter(item -> item.getSerialNo().equals(serialNo)).collect(Collectors.toList());if (hasRepeater == null || hasRepeater.size() == 0) {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("SerialNo") + "," + languageFind.findKey("Device_Excel_IsNotExits_IsNotAuthority"));}if (serialNoList.contains(serialNo)) {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("SerialNo") + "," + languageFind.findKey("Device_Excel_RepeatData_ImportFailure"));}serialNoList.add(serialNo);} else {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("SerialNo") + languageFind.findKey("Device_Excel_SerialNumberIsNotEmpty"));}if (!excelValuesList.get(3).toUpperCase().equals("E") && !excelValuesList.get(3).toUpperCase().equals("W")) {repeater.setLngOr(2);isImport = false;repeater.setImportErrorMessage(languageFind.findKey("Device_ExcelHeader_EastWestLng") + languageFind.findKey("Device_Excel_NotValidLngOr"));} else {repeater.setLngOr(excelValuesList.get(3).toUpperCase().equals("E") ? 0 : 1);BigDecimal outLng = BigDecimal.ZERO;if (!FileUtils.isBigDecimal(excelValuesList.get(4))) {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("Device_ExcelHeader_Lng") + languageFind.findKey("Device_Excel_NotValidValue"));} else {outLng = new BigDecimal(excelValuesList.get(4)).abs();repeater.setLng(BigDecimal.valueOf(repeater.getLngOr() == 0 ? 1 : -1).multiply(outLng));if (outLng.compareTo(BigDecimal.ZERO) < 0 || outLng.compareTo(new BigDecimal(180)) > 0) {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("Device_ExcelHeader_Lng") + languageFind.findKey("Device_Excel_LngRange"));} else {BigDecimal[] longitudeDegreeMiuSec = getLongitudeDegreeMinuteSecond(repeater.getLng(), excelValuesList.get(3));repeater.setLngDegreeMinuteSecond(longitudeDegreeMiuSec[0].abs() + "°" + longitudeDegreeMiuSec[1].abs() + "′" + longitudeDegreeMiuSec[2].abs() + "″");repeater.setLngDegree(longitudeDegreeMiuSec[0]);repeater.setLngMinute(longitudeDegreeMiuSec[1]);repeater.setLngSecond(longitudeDegreeMiuSec[2]);}}}if (!excelValuesList.get(5).toUpperCase().equals("N") && !excelValuesList.get(5).toUpperCase().equals("S")) {repeater.setLatOr(2);isImport = false;repeater.setImportErrorMessage(languageFind.findKey("Device_ExcelHeader_SouthNorthLat") + languageFind.findKey("Device_Excel_NotValidLatOr"));} else {repeater.setLatOr(excelValuesList.get(5).toUpperCase().equals("N") ? 0 : 1);BigDecimal outLat = BigDecimal.ZERO;if (!FileUtils.isBigDecimal(excelValuesList.get(6))) {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("Device_ExcelHeader_Lat") + languageFind.findKey("Device_Excel_NotValidValue"));} else {outLat = new BigDecimal(excelValuesList.get(6)).abs();repeater.setLat(BigDecimal.valueOf(repeater.getLatOr() == 0 ? 1 : -1).multiply(outLat));if (outLat.compareTo(BigDecimal.ZERO) < 0 || outLat.compareTo(new BigDecimal(90)) > 0) {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("Device_ExcelHeader_Lat") + languageFind.findKey("Device_Excel_LatRange"));} else {BigDecimal[] latitudeDegreeMiuSec = getLatitudeDegreeMinuteSecond(repeater.getLat(), excelValuesList.get(5));repeater.setLatDegreeMinuteSecond(latitudeDegreeMiuSec[0].abs() + "°" + latitudeDegreeMiuSec[1].abs() + "′" + latitudeDegreeMiuSec[2].abs() + "″");repeater.setLatDegree(latitudeDegreeMiuSec[0]);repeater.setLatMinute(latitudeDegreeMiuSec[1]);repeater.setLatSecond(latitudeDegreeMiuSec[2]);}}}if (excelValuesList.get(7) != null && !excelValuesList.get(7).equals("")) {BigDecimal outElevation = BigDecimal.ZERO;if (!FileUtils.isBigDecimal(excelValuesList.get(7))) {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("Device_ExcelHeader_Elevation") + languageFind.findKey("Device_Excel_NotValidValue"));} else {outElevation = new BigDecimal(excelValuesList.get(7));repeater.setElevation(outElevation);}} else {repeater.setElevation(BigDecimal.ZERO);}if (isImport) {repeaterList.add(repeater);} else {listError.add(repeater);}}excelValuesList = new ArrayList<>();}inputStream.close();if (listError.size() > 0) {return ResponseModel.ofError(listError);} else {clientService.importRepeaterLngLat(repeaterList);}} catch (Exception e) {return ResponseModel.ofError(e.getMessage());}return ResponseModel.ofSuccess();}/*** 写入度分秒形式* @param file 文件* @return 结果*/public ResponseModel writeDegreeMiuSec(MultipartFile file) {List<Repeater> repeaterList = new ArrayList<>();List<Repeater> listError = new ArrayList<>();Repeater repeater = null;List<String> excelValuesList = new ArrayList<>();List<String> serialNoList = new ArrayList<>();try {Account account = (Account) SecurityUtils.getSubject().getPrincipal();List<Repeater> authorityRepeaters = commonIClient.getAuthorityRepeaters(account.getUserName());String fileName = file.getOriginalFilename();InputStream inputStream = file.getInputStream();Workbook workbook = null;if (fileName.indexOf(".xlsx") > 0) {workbook = new XSSFWorkbook(inputStream);} else if (fileName.indexOf(".xls") > 0) {workbook = new HSSFWorkbook(inputStream);}Sheet sheet = workbook.getSheetAt(0);int lastNum = sheet.getLastRowNum();Row row = sheet.getRow(0);//行int cellNum = row.getLastCellNum();Cell ce = null;for(int i = 1; i < lastNum + 1; i++){boolean isImport = true;row = sheet.getRow(i);for (int j = 0; j < cellNum; j++){ce = row.getCell(j);excelValuesList.add(getCellValueByCell(ce));}logger.info("writeDegreeMiuSec-excelValuesList:{}", excelValuesList);if(excelValuesList.size() > 0){repeater = new Repeater();if (excelValuesList.get(1) != null && !excelValuesList.get(1).equals("")) {repeater.setAlias(excelValuesList.get(1));}if (excelValuesList.get(2) != null && !excelValuesList.get(2).equals("")) {String serialNo = excelValuesList.get(2);repeater.setSerialNo(serialNo);List<Repeater> hasRepeater = authorityRepeaters.stream().filter(item -> item.getSerialNo().equals(serialNo)).collect(Collectors.toList());if (hasRepeater == null || hasRepeater.size() == 0) {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("SerialNo") + "," + languageFind.findKey("Device_Excel_IsNotExits_IsNotAuthority"));}if (serialNoList.contains(serialNo)) {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("SerialNo") + "," + languageFind.findKey("Device_Excel_RepeatData_ImportFailure"));}serialNoList.add(serialNo);} else {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("SerialNo") + languageFind.findKey("Device_Excel_SerialNumberIsNotEmpty"));}if (!excelValuesList.get(3).toUpperCase().equals("E") && !excelValuesList.get(3).toUpperCase().equals("W")) {repeater.setLngOr(2);isImport = false;repeater.setImportErrorMessage(languageFind.findKey("Device_ExcelHeader_EastWestLng") + languageFind.findKey("Device_Excel_NotValidLngOr"));} else {repeater.setLngOr(excelValuesList.get(3).toUpperCase().equals("E") ? 0 : 1);BigDecimal outLngDegree = BigDecimal.ZERO;BigDecimal outLngMinute = BigDecimal.ZERO;BigDecimal outLngSecond = BigDecimal.ZERO;if (!FileUtils.isBigDecimal(excelValuesList.get(4))) {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("Device_ExcelHeader_LngDegree") + languageFind.findKey("Device_Excel_NotValidValue"));}if (!FileUtils.isBigDecimal(excelValuesList.get(5))) {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("Device_ExcelHeader_LngMinute") + languageFind.findKey("Device_Excel_NotValidValue"));}if (!FileUtils.isBigDecimal(excelValuesList.get(6))) {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("Device_ExcelHeader_LngSecond") + languageFind.findKey("Device_Excel_NotValidValue"));}outLngDegree = new BigDecimal(excelValuesList.get(4)).abs();outLngMinute = new BigDecimal(excelValuesList.get(5)).abs();outLngSecond = new BigDecimal(excelValuesList.get(6)).abs();repeater.setLngDegreeMinuteSecond(outLngDegree + "°" + outLngMinute + "′" + outLngSecond + "″");repeater.setLngDegree(BigDecimal.valueOf(repeater.getLngOr() == 0 ? 1 : -1).multiply(outLngDegree));repeater.setLngMinute(BigDecimal.valueOf(repeater.getLngOr() == 0 ? 1 : -1).multiply(outLngMinute));repeater.setLngSecond(BigDecimal.valueOf(repeater.getLngOr() == 0 ? 1 : -1).multiply(outLngSecond));if (outLngDegree.compareTo(BigDecimal.ZERO) < 0 || outLngDegree.compareTo(new BigDecimal(180)) > 0) {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("Device_ExcelHeader_LngDegree") + languageFind.findKey("Device_Excel_LngDegreeRange"));} else if (outLngMinute.compareTo(BigDecimal.ZERO) < 0 || outLngMinute.compareTo(new BigDecimal(60)) > 0) {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("Device_ExcelHeader_LngMinute") + languageFind.findKey("Device_Excel_LngMinuteRange"));} else if (outLngSecond.compareTo(BigDecimal.ZERO) < 0 || outLngSecond.compareTo(new BigDecimal(60)) > 0) {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("Device_ExcelHeader_LngSecond") + languageFind.findKey("Device_Excel_LngSecondRange"));} else {BigDecimal lng = getLongitudeDegree(repeater.getLngDegree(), repeater.getLngMinute(), repeater.getLngSecond(), excelValuesList.get(3));repeater.setLng(lng);}}if (!excelValuesList.get(7).toUpperCase().equals("N") && !excelValuesList.get(7).toUpperCase().equals("S")) {repeater.setLatOr(2);isImport = false;repeater.setImportErrorMessage(languageFind.findKey("Device_ExcelHeader_SouthNorthLat") + languageFind.findKey("Device_Excel_NotValidLatOr"));} else {repeater.setLatOr(excelValuesList.get(7).toUpperCase().equals("N") ? 0 : 1);BigDecimal outLatDegree = BigDecimal.ZERO;BigDecimal outLatMinute = BigDecimal.ZERO;BigDecimal outLatSecond = BigDecimal.ZERO;if (!FileUtils.isBigDecimal(excelValuesList.get(8))) {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("Device_ExcelHeader_LatDegree") + languageFind.findKey("Device_Excel_NotValidValue"));}if (!FileUtils.isBigDecimal(excelValuesList.get(9))) {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("Device_ExcelHeader_LatMinute") + languageFind.findKey("Device_Excel_NotValidValue"));}if (!FileUtils.isBigDecimal(excelValuesList.get(10))) {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("Device_ExcelHeader_LatSecond") + languageFind.findKey("Device_Excel_NotValidValue"));}outLatDegree = new BigDecimal(excelValuesList.get(8)).abs();outLatMinute = new BigDecimal(excelValuesList.get(9)).abs();outLatSecond = new BigDecimal(excelValuesList.get(10)).abs();repeater.setLatDegreeMinuteSecond(outLatDegree + "°" + outLatMinute + "′" + outLatSecond + "″");repeater.setLatDegree(BigDecimal.valueOf(repeater.getLatOr() == 0 ? 1 : -1).multiply(outLatDegree));repeater.setLatMinute(BigDecimal.valueOf(repeater.getLatOr() == 0 ? 1 : -1).multiply(outLatMinute));repeater.setLatSecond(BigDecimal.valueOf(repeater.getLatOr() == 0 ? 1 : -1).multiply(outLatSecond));if (outLatDegree.compareTo(BigDecimal.ZERO) < 0 || outLatDegree.compareTo(new BigDecimal(90)) > 0) {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("Device_ExcelHeader_LatDegree") + languageFind.findKey("Device_Excel_LatDegreeRange"));}if (outLatMinute.compareTo(BigDecimal.ZERO) < 0 || outLatMinute.compareTo(new BigDecimal(90)) > 0) {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("Device_ExcelHeader_LatMinute") + languageFind.findKey("Device_Excel_LatMinuteRange"));}if (outLatSecond.compareTo(BigDecimal.ZERO) < 0 || outLatSecond.compareTo(new BigDecimal(90)) > 0) {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("Device_ExcelHeader_LatSecond") + languageFind.findKey("Device_Excel_LatSecondRange"));}BigDecimal lat = getLatitudeDegree(repeater.getLatDegree(), repeater.getLatMinute(), repeater.getLatSecond(), excelValuesList.get(7));repeater.setLat(lat);}if (excelValuesList.get(11) != null && !excelValuesList.get(11).equals("")) {BigDecimal outElevation = BigDecimal.ZERO;if (!FileUtils.isBigDecimal(excelValuesList.get(11))) {isImport = false;repeater.setImportErrorMessage(languageFind.findKey("Device_ExcelHeader_Elevation") + languageFind.findKey("Device_Excel_NotValidValue"));} else {outElevation = new BigDecimal(excelValuesList.get(11));repeater.setElevation(outElevation);}} else {repeater.setElevation(BigDecimal.ZERO);}if (isImport) {repeaterList.add(repeater);} else {listError.add(repeater);}}excelValuesList = new ArrayList<>();}inputStream.close();if (listError.size() > 0) {return ResponseModel.ofError(listError);} else {clientService.importRepeaterLngLat(repeaterList);}} catch (Exception e) {return ResponseModel.ofError(e.getMessage());}return ResponseModel.ofSuccess();}/*** 经度 -> 转度分秒* @param longitude 经度* @param enumLongitude 东西经* @return 结果*/public BigDecimal[] getLongitudeDegreeMinuteSecond(BigDecimal longitude, String enumLongitude) {try {if (longitude.compareTo(BigDecimal.ZERO) == 0) {return new BigDecimal[]{BigDecimal.ZERO, BigDecimal.ZERO, BigDecimal.ZERO};}longitude = longitude.abs();BigDecimal degree = longitude.setScale(0, RoundingMode.DOWN);BigDecimal minute = longitude.subtract(degree).multiply(BigDecimal.valueOf(60)).setScale(0, RoundingMode.DOWN);BigDecimal second = longitude.subtract(degree).multiply(BigDecimal.valueOf(60)).subtract(minute).multiply(BigDecimal.valueOf(60)).setScale(1, RoundingMode.HALF_UP);if (enumLongitude.equals(EnumLongitudeLatitude.WEST.getValue())) {degree = degree.negate();minute = minute.negate();second = second.negate();}return new BigDecimal[]{degree, minute, second};} catch (Exception ex) {logger.error("GetLongitudeDegreeMinuteSecond> Exception:{}" + ex.getMessage());return new BigDecimal[]{BigDecimal.ZERO, BigDecimal.ZERO, BigDecimal.ZERO};}}/*** 纬度 -> 转度分秒* @param latitude 纬度* @param enumLatitude 南北纬* @return 结果*/public BigDecimal[] getLatitudeDegreeMinuteSecond(BigDecimal latitude, String enumLatitude) {try {if (latitude.compareTo(BigDecimal.ZERO) == 0) {return new BigDecimal[]{BigDecimal.ZERO, BigDecimal.ZERO, BigDecimal.ZERO};}latitude = latitude.abs();BigDecimal degree = latitude.setScale(0, RoundingMode.DOWN);BigDecimal minute = latitude.subtract(degree).multiply(BigDecimal.valueOf(60)).setScale(0, RoundingMode.DOWN);BigDecimal second = latitude.subtract(degree).multiply(BigDecimal.valueOf(60)).subtract(minute).multiply(BigDecimal.valueOf(60)).setScale(1, RoundingMode.HALF_UP);if (enumLatitude.equals(EnumLongitudeLatitude.SOUTH.getValue())) {degree = degree.negate();minute = minute.negate();second = second.negate();}return new BigDecimal[]{degree, minute, second};} catch (Exception ex) {logger.error("getLatitudeDegreeMinuteSecond> Exception:{}" + ex.getMessage());return new BigDecimal[]{BigDecimal.ZERO, BigDecimal.ZERO, BigDecimal.ZERO};}}public String getCellValueByCell(Cell cell) {if (cell == null || cell.toString().trim().equals("")) {return "";}String cellValue = "";switch (cell.getCellType()) {case NUMERIC: //数字double numericValue = cell.getNumericCellValue();if (numericValue == Math.floor(numericValue)) {// 如果是整数,直接转换为整数cellValue = String.valueOf((int) numericValue);} else {// 否则,使用 DecimalFormat 保留小数部分DecimalFormat df = new DecimalFormat("#.##"); // 保留两位小数cellValue = df.format(numericValue);}break;case STRING: //字符串cellValue = String.valueOf(cell.getStringCellValue());break;case BOOLEAN: //BooleancellValue = String.valueOf(cell.getBooleanCellValue());break;case FORMULA: //公式cellValue = String.valueOf(cell.getCellFormula());break;case BLANK: //空值cellValue = "";break;case ERROR: //故障cellValue = "非法字符";break;default:cellValue = "未知类型";break;}return cellValue;}/*** 度分秒 -> 转经度* @param degree 度* @param minute 分* @param second 秒* @param enumLongitude 东西经* @return 结果*/public static BigDecimal getLongitudeDegree(BigDecimal degree, BigDecimal minute, BigDecimal second, String enumLongitude) {try {if (degree.compareTo(BigDecimal.ZERO) == 0 && minute.compareTo(BigDecimal.ZERO) == 0 && second.compareTo(BigDecimal.ZERO) == 0) {return BigDecimal.ZERO;}degree = degree.abs();minute = minute.abs();second = second.abs();BigDecimal lng = degree.add(minute.compareTo(BigDecimal.ZERO) == 0 ? BigDecimal.ZERO : minute.divide(BigDecimal.valueOf(60), 8, RoundingMode.HALF_UP)).add(second.compareTo(BigDecimal.ZERO) == 0 ? BigDecimal.ZERO : second.divide(BigDecimal.valueOf(3600), 8, RoundingMode.HALF_UP)).setScale(8, RoundingMode.HALF_UP);if (enumLongitude.equals(EnumLongitudeLatitude.WEST.getValue())) {lng = lng.negate();}return lng;} catch (Exception ex) {logger.error("getLongitudeDegree> Exception:{}" + ex.getMessage());return BigDecimal.ZERO;}}/*** 度分秒 -> 转纬度* @param degree 度* @param minute 分* @param second 秒* @param enumLongitude 南北纬* @return 结果*/public static BigDecimal getLatitudeDegree(BigDecimal degree, BigDecimal minute, BigDecimal second, String enumLongitude) {try {if (degree.compareTo(BigDecimal.ZERO) == 0 && minute.compareTo(BigDecimal.ZERO) == 0 && second.compareTo(BigDecimal.ZERO) == 0) {return BigDecimal.ZERO;}degree = degree.abs();minute = minute.abs();second = second.abs();BigDecimal lat = degree.add(minute.compareTo(BigDecimal.ZERO) == 0 ? BigDecimal.ZERO : minute.divide(BigDecimal.valueOf(60), 8, RoundingMode.HALF_UP)).add(second.compareTo(BigDecimal.ZERO) == 0 ? BigDecimal.ZERO : second.divide(BigDecimal.valueOf(3600), 8, RoundingMode.HALF_UP)).setScale(8, RoundingMode.HALF_UP);if (enumLongitude.equals(EnumLongitudeLatitude.SOUTH.getValue())) {lat = lat.negate();}return lat;} catch (Exception ex) {logger.error("getLatitudeDegree> Exception:{}" + ex.getMessage());return BigDecimal.ZERO;}}
6.Repeater
package com.xnms.data.contract.database.db;import org.apache.commons.lang3.builder.ToStringBuilder;import javax.persistence.*;
import java.math.BigDecimal;@Entity
@Table(name = "repeater")
public class Repeater {/// Repeater别名private String alias;/// 固件版本号private String firmwaveNo;/// rcdb版本号private String rcdbNo;/// RepeaterMachineNo@Transientprivate String machineNo;/// Repeater序列号信息private String serialNo;/// Repeater IP地址信息@Column(name = "IPADDRESS")private String ipAddress = "";/// Repeater snmpPort 端口:private Integer snmpPort = 0;@Transientprivate String siteAlias;/// Repeater站点所属信息/// 中转台在当前站点顺序private Integer siteIndex = 0;/// Repeater所属站点IDprivate Integer siteId = 0;/// 存储Repeater表唯一ID@Idprivate Integer repeaterId = 0;/// RadioID ID,中转台ID:private Integer radioId = 0;//在线状态【0:在线、1:离线】private Integer status = 0;/// (告警/离线/在线等状态)/// 增加中转台状态,用于拓扑地图根据状态是否显示告警图片,(告警/离线/在线等状态)@Transientprivate Integer rptState = 0;/// 用于在拓扑地图中标识,是否显示该中转台,默认全部显示,只用在勾选了check后才有选择显示@Transientprivate Integer rptChecked = 1;/// 中转台类型(语音/数据)/// 0、语音中转台/// 1、数据中转台private Integer type;/// 当前信道名称private String channelName = "";/// 当前发射频率private String currentTxFrequence = "";/// 当前接收频率private String currentRxFrequence = "";/// 系统类型/// 1:XPT/// 0:常规private Integer systemType;/// IP 互联private Integer ipConnectPort;/// 机器名称(RD980S)private String modelName;/// 机器编号(RD980-0000000S-000000-U1-0-C)private String modelNo;/// 当前工作模式/// 1、模拟模式/// 2、数字模式private String curMode;/// 中转台类型/// 0: Single Site(独立基站);/// 1: Master(主机);/// 2 : Slave(从机);/// 3 : Super Master(超级主机);/// 4: SubMaster(子网主机)private Integer repeaterSiteType;/// 中转台的主机IP地址private String masterIp;private String recursionMasterIp;/// 中转台主机端口private Integer masterPort;/// QDS IPprivate String serverIp;private String note1; //中转台备注private String note2; //中转台备注private String note3; //中转台备注private Integer deleted;/// 中转台接入码private String repeaterAccessCode;/// (真实)中转台接入码private String realAccessCode;/// 1、不同/// 0、相同@Transientprivate Integer accessCodeEqual;/// 导入文件序号@Transientprivate String importExcelXH;/// 导入时错误信息@Transientprivate String importErrorMessage;/// 经度private BigDecimal lng = BigDecimal.ZERO;/// 纬度private BigDecimal lat = BigDecimal.ZERO;/// 海拔private BigDecimal elevation = BigDecimal.ZERO;/// 0:东经/// 1:西经/// 2: 未知@Transientprivate Integer lngOr;/// 0:北纬/// 1:南纬/// 2:未知@Transientprivate Integer latOr;/// 经度(度分秒)@Transientprivate String lngDegreeMinuteSecond;/// 经度(度)private BigDecimal lngDegree;/// 经度(分)private BigDecimal lngMinute;/// 经度(秒)private BigDecimal lngSecond;/// 纬度(度分秒)@Transientprivate String latDegreeMinuteSecond;/// 纬度(度)private BigDecimal latDegree;/// 纬度(分)private BigDecimal latMinute;/// 纬度(秒)private BigDecimal latSecond;/// SNMP 版本private String repeaterSnmpVersion = "0";/// 中转台是否采用SNMPv3协议【0:否、1:是】@Column(name = "IS_SNMPV3")private String isSnmpV3 = "0";/// USM USER@Column(name = "SNMPV3_USM_USER")private String snmpV3UsmUser = "admin";/// 安全等级/// 0:no AUTH,no Priv/// 1:AUTH,no Priv/// 2:AUTH,Priv@Column(name = "SNMPV3_SECURITY_LEVEL")private String snmpV3SecurityLevel = "0";/// 鉴权/// 公钥类型,鉴权【None、MD5、SHA1】@Column(name = "SNMPV3_AUTH_ALGORITHM")private String snmpV3AuthAlgorithm = "None";/// 鉴权密码/// 公钥密码@Column(name = "SNMPV3_AUTH_PASSWORD")private String snmpV3AuthPassword;/// 加密/// 私钥类型,加密【None、DES、AES】@Column(name = "SNMPV3_PRIVACY_ALGORITHM")private String snmpV3PrivacyAlgorithm = "None";/// 加密密码/// 私钥密码@Column(name = "SNMPV3_PRIVACY_PASSWORD")private String snmpV3PrivacyPassword;/// 手填@Column(name = "SNMPV3_CONTEXT_NAME")private String snmpV3ContextName;@Transientprivate Integer Index;public Integer getIndex() {return Index;}public void setIndex(Integer index) {Index = index;}public String getAlias() {return alias;}public void setAlias(String alias) {this.alias = alias;}public String getFirmwaveNo() {return firmwaveNo;}public void setFirmwaveNo(String firmwaveNo) {this.firmwaveNo = firmwaveNo;}public String getRcdbNo() {return rcdbNo;}public void setRcdbNo(String rcdbNo) {this.rcdbNo = rcdbNo;}public String getMachineNo() {return machineNo;}public void setMachineNo(String machineNo) {this.machineNo = machineNo;}public String getSerialNo() {return serialNo;}public void setSerialNo(String serialNo) {this.serialNo = serialNo;}public String getIpAddress() {return ipAddress;}public void setIpAddress(String ipAddress) {this.ipAddress = ipAddress;}public Integer getSnmpPort() {return snmpPort;}public void setSnmpPort(Integer snmpPort) {this.snmpPort = snmpPort;}public String getSiteAlias() {return siteAlias;}public void setSiteAlias(String siteAlias) {this.siteAlias = siteAlias;}public Integer getSiteIndex() {return siteIndex;}public void setSiteIndex(Integer siteIndex) {this.siteIndex = siteIndex;}public Integer getSiteId() {return siteId;}public void setSiteId(Integer siteId) {this.siteId = siteId;}public Integer getRepeaterId() {return repeaterId;}public void setRepeaterId(Integer repeaterId) {this.repeaterId = repeaterId;}public Integer getRadioId() {return radioId;}public void setRadioId(Integer radioId) {this.radioId = radioId;}public Integer getStatus() {return status;}public void setStatus(Integer status) {this.status = status;}public Integer getRptState() {return rptState;}public void setRptState(Integer rptState) {this.rptState = rptState;}public Integer getRptChecked() {return rptChecked;}public void setRptChecked(Integer rptChecked) {this.rptChecked = rptChecked;}public Integer getType() {return type;}public void setType(Integer type) {this.type = type;}public String getChannelName() {return channelName;}public void setChannelName(String channelName) {this.channelName = channelName;}public String getCurrentTxFrequence() {return currentTxFrequence;}public void setCurrentTxFrequence(String currentTxFrequence) {this.currentTxFrequence = currentTxFrequence;}public String getCurrentRxFrequence() {return currentRxFrequence;}public void setCurrentRxFrequence(String currentRxFrequence) {this.currentRxFrequence = currentRxFrequence;}public Integer getSystemType() {return systemType;}public void setSystemType(Integer systemType) {this.systemType = systemType;}public Integer getIpConnectPort() {return ipConnectPort;}public void setIpConnectPort(Integer ipConnectPort) {this.ipConnectPort = ipConnectPort;}public String getModelName() {return modelName;}public void setModelName(String modelName) {this.modelName = modelName;}public String getModelNo() {return modelNo;}public void setModelNo(String modelNo) {this.modelNo = modelNo;}public String getCurMode() {return curMode;}public void setCurMode(String curMode) {this.curMode = curMode;}public Integer getRepeaterSiteType() {return repeaterSiteType;}public void setRepeaterSiteType(Integer repeaterSiteType) {this.repeaterSiteType = repeaterSiteType;}public String getMasterIp() {return masterIp;}public void setMasterIp(String masterIp) {this.masterIp = masterIp;}public Integer getMasterPort() {return masterPort;}public void setMasterPort(Integer masterPort) {this.masterPort = masterPort;}public String getServerIp() {return serverIp;}public void setServerIp(String serverIp) {this.serverIp = serverIp;}public String getNote1() {return note1;}public void setNote1(String note1) {this.note1 = note1;}public String getNote2() {return note2;}public void setNote2(String note2) {this.note2 = note2;}public String getNote3() {return note3;}public void setNote3(String note3) {this.note3 = note3;}public Integer getDeleted() {return deleted;}public void setDeleted(Integer deleted) {this.deleted = deleted;}public String getRepeaterAccessCode() {return repeaterAccessCode;}public void setRepeaterAccessCode(String repeaterAccessCode) {this.repeaterAccessCode = repeaterAccessCode;}public String getRealAccessCode() {return realAccessCode;}public void setRealAccessCode(String realAccessCode) {this.realAccessCode = realAccessCode;}public Integer getAccessCodeEqual() {return accessCodeEqual;}public void setAccessCodeEqual(Integer accessCodeEqual) {this.accessCodeEqual = accessCodeEqual;}public String getImportExcelXH() {return importExcelXH;}public void setImportExcelXH(String importExcelXH) {this.importExcelXH = importExcelXH;}public String getImportErrorMessage() {return importErrorMessage;}public void setImportErrorMessage(String importErrorMessage) {this.importErrorMessage = importErrorMessage;}public BigDecimal getLng() {return lng;}public void setLng(BigDecimal lng) {this.lng = lng;}public BigDecimal getLat() {return lat;}public void setLat(BigDecimal lat) {this.lat = lat;}public BigDecimal getElevation() {return elevation;}public void setElevation(BigDecimal elevation) {this.elevation = elevation;}public Integer getLngOr() {return lngOr;}public void setLngOr(Integer lngOr) {this.lngOr = lngOr;}public Integer getLatOr() {return latOr;}public void setLatOr(Integer latOr) {this.latOr = latOr;}public String getLngDegreeMinuteSecond() {return lngDegreeMinuteSecond;}public void setLngDegreeMinuteSecond(String lngDegreeMinuteSecond) {this.lngDegreeMinuteSecond = lngDegreeMinuteSecond;}public BigDecimal getLngDegree() {return lngDegree;}public void setLngDegree(BigDecimal lngDegree) {this.lngDegree = lngDegree;}public BigDecimal getLngMinute() {return lngMinute;}public void setLngMinute(BigDecimal lngMinute) {this.lngMinute = lngMinute;}public BigDecimal getLngSecond() {return lngSecond;}public void setLngSecond(BigDecimal lngSecond) {this.lngSecond = lngSecond;}public String getLatDegreeMinuteSecond() {return latDegreeMinuteSecond;}public void setLatDegreeMinuteSecond(String latDegreeMinuteSecond) {this.latDegreeMinuteSecond = latDegreeMinuteSecond;}public BigDecimal getLatDegree() {return latDegree;}public void setLatDegree(BigDecimal latDegree) {this.latDegree = latDegree;}public BigDecimal getLatMinute() {return latMinute;}public void setLatMinute(BigDecimal latMinute) {this.latMinute = latMinute;}public BigDecimal getLatSecond() {return latSecond;}public void setLatSecond(BigDecimal latSecond) {this.latSecond = latSecond;}public String getRepeaterSnmpVersion() {return repeaterSnmpVersion;}public void setRepeaterSnmpVersion(String repeaterSnmpVersion) {this.repeaterSnmpVersion = repeaterSnmpVersion;}public String getIsSnmpV3() {return isSnmpV3;}public void setIsSnmpV3(String isSnmpV3) {this.isSnmpV3 = isSnmpV3;}public String getSnmpV3UsmUser() {return snmpV3UsmUser;}public void setSnmpV3UsmUser(String snmpV3UsmUser) {this.snmpV3UsmUser = snmpV3UsmUser;}public String getSnmpV3SecurityLevel() {return snmpV3SecurityLevel;}public void setSnmpV3SecurityLevel(String snmpV3SecurityLevel) {this.snmpV3SecurityLevel = snmpV3SecurityLevel;}public String getSnmpV3AuthAlgorithm() {return snmpV3AuthAlgorithm;}public void setSnmpV3AuthAlgorithm(String snmpV3AuthAlgorithm) {this.snmpV3AuthAlgorithm = snmpV3AuthAlgorithm;}public String getSnmpV3AuthPassword() {return snmpV3AuthPassword;}public void setSnmpV3AuthPassword(String snmpV3AuthPassword) {this.snmpV3AuthPassword = snmpV3AuthPassword;}public String getSnmpV3PrivacyAlgorithm() {return snmpV3PrivacyAlgorithm;}public void setSnmpV3PrivacyAlgorithm(String snmpV3PrivacyAlgorithm) {this.snmpV3PrivacyAlgorithm = snmpV3PrivacyAlgorithm;}public String getSnmpV3PrivacyPassword() {return snmpV3PrivacyPassword;}public void setSnmpV3PrivacyPassword(String snmpV3PrivacyPassword) {this.snmpV3PrivacyPassword = snmpV3PrivacyPassword;}public String getSnmpV3ContextName() {return snmpV3ContextName;}public void setSnmpV3ContextName(String snmpV3ContextName) {this.snmpV3ContextName = snmpV3ContextName;}public String getRecursionMasterIp() {return recursionMasterIp;}public void setRecursionMasterIp(String recursionMasterIp) {this.recursionMasterIp = recursionMasterIp;}@Overridepublic String toString() {return ToStringBuilder.reflectionToString(this);}}
7.FileUtils
package com.xnms.client.service.utils;import org.apache.poi.hssf.usermodel.HSSFWorkbook;
import org.apache.poi.ss.usermodel.Row;
import org.apache.poi.ss.usermodel.Sheet;
import org.apache.poi.ss.usermodel.Workbook;
import org.apache.poi.xssf.usermodel.XSSFWorkbook;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.web.multipart.MultipartFile;import java.io.InputStream;
import java.math.BigDecimal;public class FileUtils {private static final Logger logger = LoggerFactory.getLogger(FileUtils.class);/*** 获取文件后缀扩展名* @param fileName 文件名* @return 结果*/public static String getFileExtension(String fileName) {int index = fileName.lastIndexOf(".");if (index > 0 && index < fileName.length() - 1) {return fileName.substring(index);}return "";}/*** 获取excel的列数* @param file 文件* @return 结果*/public static int getExcelColumnCount(MultipartFile file) {int columnCount = 0;String fileName = file.getOriginalFilename();Workbook workbook = null;try {InputStream inputStream = file.getInputStream();if (fileName.indexOf(".xlsx") > 0) {workbook = new XSSFWorkbook(inputStream);} else if (fileName.indexOf(".xls") > 0) {workbook = new HSSFWorkbook(inputStream);}Sheet sheet = workbook.getSheetAt(0);Row firstRow = sheet.getRow(0);if (firstRow != null) {columnCount = firstRow.getPhysicalNumberOfCells();}inputStream.close();} catch (Exception e) {e.printStackTrace();logger.error("getExcelColumnCount-Exception:{}", e.getMessage());}return columnCount;}/*** 校验字符串数值是否是BigDecimal类型* @param str 字符串数值* @return 结果*/public static boolean isBigDecimal(String str) {try {new BigDecimal(str);return true;} catch (Exception e) {return false;}}
}
8.ResponseModel
package com.xnms.client.service.controller.common;import com.xnms.data.contract.database.db.Pagination;public class ResponseModel<T> {private int code; // 状态码,表示请求是否成功private String msg; // 信息,通常用于描述错误或操作成功的提示private T data; // 数据,通常是请求返回的具体内容private Pagination pagination; // 分页参数,封装分页相关的数据// 私有构造函数private ResponseModel(int code, String msg, T data, Pagination pagination) {this.code = code;this.msg = msg;this.data = data;this.pagination = pagination;}// 默认构造函数private ResponseModel() {this.code = ConstantsCode.HTTP_OK; // 默认 code 为 200this.msg = "成功"; // 默认 msg 为 "成功"this.pagination = null; // 默认 pagination 为 null}// 静态方法:只返回数据和分页信息public static <T> ResponseModel<T> of(T data, Pagination pagination) {ResponseModel<T> response = new ResponseModel<>();response.setData(data);response.setPagination(pagination);return response;}// 静态方法:返回数据和分页信息,默认成功信息public static <T> ResponseModel<T> ofSuccess(T data, Pagination pagination) {return new ResponseModel<>(ConstantsCode.HTTP_OK, "成功", data, pagination);}public static <T> ResponseModel<T> ofSuccess(T data, String msg, Pagination pagination) {return new ResponseModel<>(ConstantsCode.HTTP_OK, msg, data, pagination);}// 静态方法:返回数据和成功信息,分页信息为空public static <T> ResponseModel<T> ofSuccess(T data) {return new ResponseModel<>(ConstantsCode.HTTP_OK, "成功", data, null);}public static <T> ResponseModel<T> ofSuccess() {return new ResponseModel<>(ConstantsCode.HTTP_OK, "成功", null, null);}public static <T> ResponseModel<T> ofError(T data) {return new ResponseModel<>(ConstantsCode.HTTP_INTERNAL_SERVER_ERROR, "失败", data, null);}// 静态方法:返回错误信息public static <T> ResponseModel<T> ofError(String msg) {return new ResponseModel<>(ConstantsCode.HTTP_INTERNAL_SERVER_ERROR, msg, null, null);}public static <T> ResponseModel<T> ofError(int code, String msg) {return new ResponseModel<>(code, msg, null, null);}// 静态方法:只返回分页信息,数据为空public static <T> ResponseModel<T> ofPagination(Pagination pagination) {return new ResponseModel<>(ConstantsCode.HTTP_OK, "成功", null, pagination);}// 静态方法:无数据,无分页,仅返回状态码和信息public static <T> ResponseModel<T> ofStatus(int code, String msg) {return new ResponseModel<>(code, msg, null, null);}// Getter 和 Setter 方法public int getCode() {return code;}public void setCode(int code) {this.code = code;}public String getMsg() {return msg;}public void setMsg(String msg) {this.msg = msg;}public T getData() {return data;}public void setData(T data) {this.data = data;}public Pagination getPagination() {return pagination;}public void setPagination(Pagination pagination) {this.pagination = pagination;}}
9.EnumLongitudeLatitude
package com.xnms.client.service.Utility;//经纬度枚举
public enum EnumLongitudeLatitude {NORTH("N"),SOUTH("S"),EAST("E"),WEST("W"),UnKnown("");private String value;public String getValue() {return value;}EnumLongitudeLatitude(String value) {this.value = value;}
}
10.词条
CauseOfImportFailure=导入失败原因
ImportFailedData=导入数据失败
Device_ExcelHeader_No=序号
Device_ExcelHeader_Alias=别名
Device_ExcelHeader_SerialNumber=序列号
Device_ExcelHeader_EastWestLng=东西经
Device_ExcelHeader_Lng=经度
Device_ExcelHeader_SouthNorthLat=南北纬
Device_ExcelHeader_Lat=纬度
Device_ExcelHeader_Elevation=海拔
Device_ExcelHeader_LngDegree=经度(度)
Device_ExcelHeader_LngMinute=经度(分)
Device_ExcelHeader_LngSecond=经度(秒)
Device_ExcelHeader_LatDegree=纬度(度)
Device_ExcelHeader_LatMinute=纬度(分)
Device_ExcelHeader_LatSecond=纬度(秒)
Device_Excel_ColumnIsInUse=文件被占用
Device_Excel_ColumnIsNotExits=模板格式错误
Device_Excel_SerialNumberIsNotEmpty=不能为空
Device_Excel_IsNotExits_IsNotAuthority=不存在或没有权限
Device_Excel_NotValidValue=不是有效数值
Device_Excel_NotValidLngOr=值为E或W
Device_Excel_NotValidLatOr=值为S或N
Device_Excel_LngRange=超出经度范围0-180
Device_Excel_LngDegreeRange=超出经度(度)范围 0-180
Device_Excel_LngMinuteRange=超出经度(分)范围 0-60
Device_Excel_LngSecondRange=超出经度(秒)范围 0-60
Device_Excel_LatRange=超出纬度范围0-90
Device_Excel_LatDegreeRange=超出纬度(度)范围 0-90
Device_Excel_LatMinuteRange=超出纬度(分)范围 0-60
Device_Excel_LatSecondRange=超出纬度(秒)范围 0-60
Device_Excel_RepeatData_ImportFailure=重复数据,导入失败.
Device_Excel_RepeatData_1000MaximumData=最大数据1000条
Device_Excel_DataNotAllowedToImport=未找到可以导入的数据
四、注意点
注意点1:
页面设备管理哪里可以直接手动修改经纬度和高海拔,该设计是考虑当你想单一修改个别设备时可以直接修改保存,而无需导入excel去修改。
注意点2:
点击“导入”按钮时可以看到,允许下载2套不同模版。
注意点3:
导入excel如果校验成功,则直接弹窗提示“导入成功”,或者其他词条;而有部分校验失败的地方则展示所有错误信息,方便精确知道哪行信息未通过。
注意点4:
上传经纬度(度数)excel时同步更新度、分、秒字段;同理上传经纬度(度分秒)excel时同步更新经纬度字段;因为实体有字段会动态拼接“度数字符串”和“度分秒字符串”返回前端进行展示。
注意点5:
- 经度 -> 转度分秒 => 对应方法getLongitudeDegreeMinuteSecond()
- 纬度 -> 转度分秒 => 对应方法getLatitudeDegreeMinuteSecond()
- 度分秒 -> 转经度 => 对应方法getLongitudeDegree()
- 度分秒 -> 转纬度 => 对应方法getLatitudeDegree()
本人其他相关文章链接
1.vue3 开发电子地图功能
2.vue java 实现大地图切片上传
3.java导入excel更新设备经纬度度数或者度分秒
4.快速上手Vue3国际化 (i18n)
相关文章:
java导入excel更新设备经纬度度数或者度分秒
文章目录 一、背景介绍二、页面效果三、代码0.pom.xml1.ImportDevice.vue2.ImportDeviceError.vue3.system.js4.DeviceManageControl5.DeviceManageUserControl6.Repeater7.FileUtils8.ResponseModel9.EnumLongitudeLatitude10.词条 四、注意点本人其他相关文章链接 一、背景介…...
使用python访问mindie部署的vl多模态模型
说明 今天使用mindie1.0部署了qwen2_7b_vl模型,测试过程出现一些问题,这里总结下。 问题1:transformers版本太低 报错信息: [ERROR] [model_deploy_config.cpp:159] Failed to get vocab size from tokenizer wrapper with ex…...
github发布个人中英文简历网站CaoYongshengcys.github.io
在GitHub上创建个人主页(也称为GitHub Pages)是一个展示个人项目、技能和经历的好方法。以下是详细步骤: 1. 创建GitHub账号 • 如果你还没有GitHub账号,先访问GitHub官网注册一个账号。 • 选择一个专业的用户名,因…...
动态规划算法深度解析:0-1背包问题(含完整流程)
简介: 0-1背包问题是经典的组合优化问题:给定一组物品(每个物品有重量和价值),在背包容量限制下选择物品装入背包,要求总价值最大化且每个物品不可重复选取。 动态规划核心思想 通过构建二维状态表dp[i]…...
QML面试笔记--UI设计篇04交互控件
1. QML中常用交互控件 1.1. Button1.2. Slider1.3. ProgressBar1.4. TextField1.5. TextArea1.6. ComboBox1.7. CheckBox1.8. RadioButton1.9. Menu1.10. Dialog 1. QML中常用交互控件 在万物互联的智能时代,QML凭借其声明式语法和跨平台能力,…...
【数学】线性代数(Python)
参考:https://aibydoing.com/notebooks/appendix01-01-linear-algebra-with-python 目录 矩阵的定义矩阵的运算矩阵的属性矩阵的分解矩阵的本质遗留问题 矩阵的定义 通过数组的维度来区分向量(1 维数组)、矩阵(2 维数组࿰…...
ragflow开启https访问:添加证书后,使用浏览器还是有警告,如何解决?
如果在 Windows 系统中安装了 PEM 证书(使用方法一通过证书管理器 MMC 导入),但浏览器仍然提示安全警告,可能有以下几个原因及解决方法: 1. 证书未正确安装到受信任的存储位置 问题:如果证书被导入到错误的存储位置(如“个人”而非“受信任的根证书颁发机构”),浏览器…...
vue.config.js配置代理(输出代理前后的地址)
devServer: {host: 0.0.0.0,port: port,open: true,before(app) {app.use((req, res, next) > {// console.log(原始地址:, req.originalUrl) // 原始地址,如 /api/some-api/xxxxxnext()})},proxy: {[process.env.VUE_APP_BASE_API]: {target: http://192.168.50…...
【八股文】http1.0和1.1的区别
http1.0默认使用短连接,每次请求都需要建立TCP连接(三次握手),响应完成后立即关闭连接,导致资源浪费和延迟增加。 支持通过Connection:Keep-alive 手动开启长连接,但需客户端和服务端显式协商 …...
【Prompt实战】邮件分类专家
本文原创作者:姚瑞南 AI-agent 大模型运营专家,先后任职于美团、猎聘等中大厂AI训练专家和智能运营专家岗;多年人工智能行业智能产品运营及大模型落地经验,拥有AI外呼方向国家专利与PMP项目管理证书。(转载需经授权&am…...
K8S核心技术点
Pod,Service和Deployment的关系 Pod:Kubernetes 中最小的部署单元,用于运行容器化应用。 Service:提供服务发现和负载均衡,为 Pod 提供稳定的网络端点,ClusterIP,NodePort,LoadBala…...
Python手写“随机森林”解决鸢尾花数据集分类问题
Python使用“随机森林”解决鸢尾花数据集分类问题 任务描述解题1. 导入必要的库2. 数据采样函数 sample3. 设置随机种子和超参数4. 定义随机森林类 random_forest5. 加载数据集并划分训练集和测试集6. 创建并训练随机森林模型7. 进行预测并计算准确率 代码 任务描述 您的任务是…...
Python 字典和集合(泛映射类型)
本章内容的大纲如下: 常见的字典方法 如何处理查找不到的键 标准库中 dict 类型的变种set 和 frozenset 类型 散列表的工作原理 散列表带来的潜在影响(什么样的数据类型可作为键、不可预知的 顺序,等等) 泛映射类型 collections.…...
CrystalDiskInfo电脑硬盘监控工具 v9.6.0中文绿色便携版
前言 CrystalDiskInfo是一个不用花钱的硬盘小帮手软件,它可以帮你看看你的电脑硬盘工作得怎么样,健不健康。这个软件能显示硬盘的温度高不高、还有多少地方没用、传输东西快不快等等好多信息。用了它,你就能很容易地知道硬盘现在是什么情况&…...
rqlite:一个基于SQLite构建的分布式数据库
今天给大家介绍一个基于 SQLite 构建的轻量级分布式关系型数据库:rqlite。 rqlite 基于 Raft 协议,结合了 SQLite 的简洁性以及高可用分布式系统的稳健性,对开发者友好,操作极其简便,其核心设计理念是以最低的复杂度实…...
网络1 网络设备
计算机网络设备 集线器: 易发生阻塞:所有端口共享一条带宽,两个端口发生传输时,其他端口若想传输数据给这两个端口,需等待这两个端口传输数据完毕。 端口数量限制:10M带宽下可用15口。15口共享10Md带宽 集线…...
mybatis 某些特殊的 ORA-00979:not a GROUP BY expression
打印的日志sql执行都是正常的 但是 就是报ORA-00979: not a GROUP BY expression 可能是 GROUP BY中不能使用动态参数 或特殊方法 使用 硬编码可以解决问题 <if test"statisticsInVo.timeTypeSql!null and statisticsInVo.timeTypeSql yyyy">TO_CHAR(CARD_T…...
基于OpenCV的图像处理程序设计实践
一.安装OpenCV3.x # 安装依赖 sudo apt update sudo apt install build-essential cmake git libgtk2.0-dev pkg-config libavcodec-dev libavformat-dev libswscale-dev# 下载OpenCV源码 git clone https://github.com/opencv/opencv.git -b 3.4 cd opencv mkdir build &…...
DeepSeek 全套汇总资料pdf免费下载(最新更新8篇)
DeepSeek 全套汇总资料pdf目前仍然在持续更新中,今天更新了8篇,合计的汇总都在这里了,有需要的朋友可以直接去下载了。 后续更新请关注文章:DeepSeek 全套汇总资料pdf免费下载(持续更新) _ 潘子夜个人博客…...
前端面试题(六):HTTP和HTTPS的区别以及他们如何保障数据安全
HTTP(HyperText Transfer Protocol)和HTTPS(HyperText Transfer Protocol Secure)都是用于在互联网上传输数据的协议,但它们之间有一个重要的区别:安全性。 1. HTTP(超文本传输协议)…...
Buffer Pool 的核心作用与工作机制
Buffer Pool 的核心作用与工作机制 1. Buffer Pool 是什么? Buffer Pool 是 InnoDB 存储引擎的核心内存区域,用于 缓存磁盘中的数据页。 作用:通过内存缓存减少直接磁盘 I/O,加速数据库的读写操作。默认大小:通常设…...
使用uglifyjs对静态引入的js文件进行压缩
前言 因为有时候js文件没有npm包,或者需要修改,只能引入静态的js,那么这个时候就可以对js进行压缩了。我其实想通过vite、webpack等插件进行压缩的,可是他都不能定位到public目录下面的文件,所以我只能自己压缩了。编…...
Vue 3 的<Teleport>功能与用法
Vue 3 的 <Teleport> 功能与用法 1. 基本用法 <Teleport> 是 Vue 3 的一个内置组件,允许将组件的内容渲染到 DOM 中的任意位置,而不改变其逻辑结构。以下是基本用法: 定义目标 DOM 元素:<div id"teleport-…...
2025 年江苏交安安全员考试:借助本地培训资源提升能力
江苏拥有丰富的教育和培训资源,为交安安全员备考提供了有力支持。考生可关注本地专业培训机构开设的交安安全员培训课程,这些课程往往由经验丰富的讲师授课,他们熟悉本地考试特点和行业实际需求。课程内容不仅涵盖考试大纲的知识点࿰…...
Umi Max 和 Ant Design Pro 的区别
1、前言: Ant Design Pro Umi Max Umi Max 和 Ant Design Pro 其实关系很紧密,但用途不同、定位不同。 我们一起来搞清楚它们的区别、联系、使用场景👇 2、一句话总结 名称作用Umi Max是现代前端框架,用来构建中后台项目&#x…...
《 Scikit-learn与MySQL的深度协同:构建智能数据生态系统的架构哲学》
在机器学习工程实践中,数据存储与模型训练的割裂始终是制约算法效能的关键瓶颈。Scikit-learn作为经典机器学习库,其与MySQL的深度协同并非简单的数据管道连接,而是构建了一个具备自组织能力的智能数据生态系统。这种集成突破了传统ETL流程的…...
无公网实体服务器加装多个操作系统供多个用户互不打扰使用_part1
背景介绍 因笔者业务需求,入手了一个实体服务器,但为了避免出现在一个操作系统中搭建编程环境后有许多相关的进程和服务,拖慢日常的使用,也能让其他人短期使用,更好的利用服务器的性能,让服务器专注于“什…...
C#调用Lua方法1+C#调用Lua方法2,3
xLua中Lua调用C#代码 原因:C#实现的系统,因为Lua可以调用,所以完全可以换成Lua实现,因为Lua可以即时更改,即时运行,所以游戏的代码逻辑就可以随时更改。 实现和C#相同效果的系统,如何实现&#…...
浅层神经网络:从数学原理到实战应用的全面解析
浅层神经网络:从数学原理到实战应用的全面解析 前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,可以分享一下给大家。点击跳转到网站。 https://www.captainbed.cn/ccc 一、神经网络演进简史:浅层网络的奠…...
【深度学习:理论篇】--Pytorch基础入门
目录 1.Pytorch--安装 2.Pytorch--张量 3.Pytorch--定义 4.Pytorch--运算 4.1.Tensor数据类型 4.2.Tensor创建 4.3.Tensor运算 4.4.Tensor--Numpy转换 4.5.Tensor--CUDA(GPU) 5.Pytorch--自动微分 (autograd) 5.1.back…...
C++中数组的概念
文章目录 一、数组的定义二、什么是一维数组?2.1 一维数组的声明2.2 一维数组的初始化2.3 一维数组的使用 三、什么是一维数组的数组名?四、一维数组与指针的关系五、数组指针和指针数组的区别5.1 指针数组(array of pointers)5.2…...
996引擎-源码学习:Cocos2d-Lua 的 class(classname, ...)
996引擎-源码学习:Cocos2d-Lua 的 class(classname, ...) 一、核心方法调用顺序用户调用入口完整调用链二、__create 工厂方法的三种情形情形1:父类为函数(自定义工厂)情形2:父类为Cocos原生类情形3:父类为普通Lua表三、方法职责与内存管理对照表四、正确使用示例示例1…...
@linux系统SSL证书转换(Openssl转换PFX)
在Linux中,你可以使用OpenSSL工具将PFX/P12格式的证书转换为单独的CRT(证书)、KEY(私钥)文件以及提取证书链 1. 提取私钥文件(.key) openssl pkcs12 -in your_certificate.pfx -nocerts -out private.key -nodes系统会…...
flask返回json或者中文字符串不要编码
在 Flask 中返回中文字符串时,如果希望浏览器直接显示中文(而非编码后的 Unicode 转义字符如 \uXXXX),需确保以下两点: 正确设置 HTTP 响应的字符集(如 utf-8)。 避免 Flask 默认的 JSON 序列化转义中文字符。 以下是具体实现方法: 方法 1:直接返回纯文本(非 JSON) …...
打造船岸“5G+AI”智能慧眼 智驱力赋能客船数智管理
项目介绍 船舶在航行、作业过程中有着严格的规范要求,但在实际航行与作业中往往会因为人为的疏忽,发生事故,导致人员重大伤亡和财产损失; 为推动安全治理模式向事前预防转型,实现不安全状态和行为智能预警,…...
【Proteus仿真】【32单片机-A007】PT100热敏温度检测系统设计
目录 一、主要功能 二、使用步骤 三、硬件资源 四、软件设计 五、实验现象 联系作者 一、主要功能 1、LCD1602显示当前检测的温度值以及温度阈值 2、超过上限温度,降温模块启动 3、PT100热敏电阻测量-60C-135C 4、按键设置温度阈值 5、超过阈值࿰…...
MPDrive:利用基于标记的提示学习提高自动驾驶的空间理解能力
25年4月来自南方科技大学、百度、英国 KCL和琶洲实验室(广东 AI 和数字经济实验室)的论文“MPDrive: Improving Spatial Understanding with Marker-Based Prompt Learning for Autonomous Driving”。 自动驾驶视觉问答(AD-VQA)…...
PhotoShop学习08
1.应用滤镜 PhotoShop提供了很多滤镜,借助滤镜可以打造很多有趣的效果。滤镜可以通过点击菜单栏的滤镜,并选择滤镜库进入滤镜调整界面。 进入到滤镜库后,左侧是实时进行预览的图片,右侧可以选择滤镜效果,最右边可以调…...
Photoshop2025最新版v26超详细图文安装教程(附安装包)
前言 Photoshop是一款基于位图的图像处理软件,专注于对已有图像的编辑、修复、合成及特效制作。其核心功能包括图层管理、色彩校正、选区工具、滤镜效果等,支持多种颜色模型(如RGB、CMYK、CIELAB)和文件格式(如.PSD、…...
Plusar集群搭建-Ubuntu20.04-Winterm
1 背景 已经部署了Pulsar集群在生产上,新项目需要用到Pulsar。对Pulsar不熟,故搭建练手。 环境:Windows10vmwareUbuntu20.04,ssh工具使用的Winterm。 使用的是root账户,ubuntu防火墙都ufw disable了。 2 参考文档 集…...
Qt与C++数据类型转换
本文深入探讨Qt与C中相似但不同的数据类型处理技巧。 一、QString与std::string的相互转换 1. QString → std::string 方法1:使用toStdString()(推荐) QString qstr "你好,Qt世界"; std::string str qstr.toStdS…...
Excel处理控件Aspose.Cells指南:如何查看、编辑和删除 Excel 元数据
本文是如何使用Aspose.Cells的在线工具和编码解决方案查看、编辑和删除 Excel 元数据的综合指南。无论您是寻找快速Excel 元数据查看器的普通用户,还是寻求强大的Excel 元数据编辑器的开发人员,本指南都能满足您的需求。您可以选择使用简单的在线转换器来…...
Rust 在汽车 MCU 编程中的进展及安全特性剖析
在当今汽车行业,软件定义汽车的趋势正深刻改变着汽车的设计与用户体验。随着汽车电子系统复杂性的不断提升,对汽车微控制器(MCU)编程的安全性、可靠性和效率提出了更高要求。Rust 作为一种新兴的编程语言,凭借其独特的…...
Pytorch 第十四回:神经网络编码器——变分自动编解码器
Pytorch 第十四回:神经网络编码器——变分自动编解码器 本次开启深度学习第十四回,基于Pytorch的神经网络编码器。本回分享VAE变分自动编码器。在本回中,通过minist数据集来分享如何建立一个变分自动编码器。接下来给大家分享具体思路。 本次…...
hive排序函数
在 Hive 中,排序可以通过几种不同的方法来实现,通常依赖于 ORDER BY 或 SORT BY 等函数。这里简要介绍这几种排序方法: 1. ORDER BY ORDER BY 用于对结果集进行全局排序。它会将所有数据加载到一个节点进行排序,因此可能会导致性能问题,尤其是在数据量很大的时候。 语法…...
Android测试王炸:Appium + UI Automator2
Android平台主流开源框架简介 在Android平台上,有多个开源且好用的自动化测试框架。以下是几个被广泛使用和认可的框架: 1.1 Appium Appium是一个跨平台的移动测试工具,支持iOS和Android上的原生、混合及移动Web应用。 它使用了供应商提供的…...
用Python打造增强现实的魔法:实时对象叠加系统全解析
友友们好! 我是Echo_Wish,我的的新专栏《Python进阶》以及《Python!实战!》正式启动啦!这是专为那些渴望提升Python技能的朋友们量身打造的专栏,无论你是已经有一定基础的开发者,还是希望深入挖掘Python潜力的爱好者,这里都将是你不可错过的宝藏。 在这个专栏中,你将会…...
const let var 在react jsx中的使用方法 。
在 JavaScript 里,const 和 let 都是 ES6(ES2015)引入的用于声明变量的关键字,它们和之前的 var 关键字有所不同。下面为你详细介绍 const 和 let 的区别: 1. 块级作用域 const 和 let 都具备块级作用域,…...
C++隐式转换的机制、风险与消除方法
引言 C作为一门强类型语言,类型安全是其核心特性之一。 然而,隐式转换(Implicit Conversion)的存在既为开发者提供了便利,也可能成为程序中的“隐藏炸弹”。 一、隐式转换的定义与分类 1.1 什么是隐式转换…...
Python 为什么要保留显式的 self ?
当你在类中定义方法时,Python要求第一个参数必须表示当前对象实例。当你调用obj.method(),Python 本质上会将它转换为ClassName.method(obj)。 所以你需要通过self参数显式接收这个实例,才能访问该对象的属性和其他方法。如果不加self&#…...