| | |
| | | import com.se.nsl.domain.po.SimuData; |
| | | import com.se.nsl.domain.vo.ConfigVo; |
| | | import com.se.nsl.helper.ComHelper; |
| | | import com.se.nsl.helper.GdalHelper; |
| | | import com.se.nsl.helper.StringHelper; |
| | | import com.se.nsl.helper.WebHelper; |
| | | import com.se.nsl.utils.AreaType; |
| | | import lombok.SneakyThrows; |
| | | import lombok.extern.slf4j.Slf4j; |
| | | import org.gdal.gdal.Band; |
| | | import org.gdal.gdal.Dataset; |
| | | import org.gdal.gdal.gdal; |
| | | import org.gdal.gdalconst.gdalconstConstants; |
| | | import org.gdal.ogr.Geometry; |
| | | import org.gdal.ogr.ogr; |
| | | import org.gdal.osr.CoordinateTransformation; |
| | | import org.gdal.osr.SpatialReference; |
| | | import org.springframework.stereotype.Service; |
| | | import org.springframework.util.StringUtils; |
| | | |
| | |
| | | import java.nio.file.StandardCopyOption; |
| | | import java.sql.Timestamp; |
| | | import java.text.SimpleDateFormat; |
| | | import java.time.Instant; |
| | | import java.time.temporal.ChronoUnit; |
| | | import java.util.*; |
| | | import java.util.concurrent.ExecutorService; |
| | | import java.util.concurrent.Executors; |
| | |
| | | SimuData data = JSONUtil.toBean(simu.getData(), SimuData.class); |
| | | |
| | | update(simu, 1, "初始化参数"); |
| | | initArgs(data); |
| | | initArgs(simu, data); |
| | | createRainfallFile(simu, data); |
| | | |
| | | update(simu, 2, "调用求解器"); |
| | |
| | | createNsl(data); |
| | | |
| | | update(simu, 10, "完成"); |
| | | log.info("模拟完成"); |
| | | } catch (Exception ex) { |
| | | log.error(ex.getMessage(), ex); |
| | | update(simu, 20, ex.getMessage()); |
| | |
| | | /** |
| | | * 初始化参数 |
| | | */ |
| | | public void initArgs(SimuData data) throws IOException { |
| | | public void initArgs(Simu simu, SimuData data) throws IOException { |
| | | String inPath = config.getInPath() + File.separator + data.getInPath(); |
| | | createDir(inPath); |
| | | createDir(inPath + File.separator + "depth"); |
| | | createDir(inPath + File.separator + "velocity"); |
| | | createDir(config.getOutPath() + File.separator + data.getOutPath()); |
| | | |
| | | // 临时复制高程tif,以后需要自行切割 |
| | | File uwBat = new File(config.getUwSolverBat()); |
| | | String sourceTif = uwBat.getParent() + File.separator + "case1" + File.separator + "LiuLiMiaoZhen_5m_f32.tif"; |
| | | String targetTif = inPath + File.separator + config.getDemFile(); |
| | | Files.copy(Paths.get(sourceTif), Paths.get(targetTif), StandardCopyOption.REPLACE_EXISTING); |
| | | Short areaType = simu.getAreaType(); |
| | | AreaType at = AreaType.of(areaType); |
| | | String terrainTif = config.getSourceDem(); |
| | | String landuseTif = config.getSourceLanduse(); |
| | | if (at == AreaType.KEY_DITCH) { |
| | | File keyDitchDir = new File(config.getKeyDitch()); |
| | | String areaName = simu.getAreaName(); |
| | | Optional<File> first = Arrays.stream(keyDitchDir.listFiles()).filter(f -> f.getName().equals(areaName)).findFirst(); |
| | | if (first.isPresent()) { |
| | | File targetTifDir = first.get(); |
| | | File[] files = targetTifDir.listFiles(); |
| | | for (File file : files) { |
| | | String name = file.getName(); |
| | | if (name.toLowerCase().contains("dem")) { |
| | | terrainTif = file.getAbsolutePath(); |
| | | } |
| | | if (name.toLowerCase().contains("landuse")) { |
| | | landuseTif = file.getAbsolutePath(); |
| | | } |
| | | } |
| | | } |
| | | } |
| | | |
| | | Geometry geom = Geometry.CreateFromWkt(simu.getGeom()); |
| | | if (geom.GetGeometryType() == ogr.wkbMultiPolygon) geom = geom.GetGeometryRef(0); |
| | | SpatialReference dstSR = GdalHelper.createSpatialReference(config.getEpsg()); |
| | | |
| | | CoordinateTransformation ct = CoordinateTransformation.CreateCoordinateTransformation(GdalHelper.SR4326, dstSR); |
| | | geom.Transform(ct); |
| | | String wkt = geom.ExportToWkt(); |
| | | |
| | | String terrainFile = inPath + File.separator + config.getTerrainFile(); |
| | | |
| | | Dataset dsDem = gdal.Open(terrainTif, gdalconstConstants.GA_ReadOnly); |
| | | ComHelper.Resample(dsDem, null, terrainFile, null, wkt, null, null); |
| | | dsDem.delete(); |
| | | |
| | | String landuseFile = inPath + File.separator + config.getLanduseFile(); |
| | | |
| | | Dataset dsLanduse = gdal.Open(landuseTif, gdalconstConstants.GA_ReadOnly); |
| | | ComHelper.Resample(dsLanduse, null, landuseFile, null, wkt, null, null); |
| | | dsLanduse.delete(); |
| | | } |
| | | |
| | | public void updateTif(Simu simu, SimuData data2) throws IOException { |
| | | Dataset ds = gdal.Open(config.getSourceLanduse(), gdalconstConstants.GA_Update); // 以读写模式打开TIFF文件 |
| | | |
| | | Band band = ds.GetRasterBand(1); |
| | | if (band.GetRasterDataType() != gdalconstConstants.GDT_Byte) { |
| | | System.err.println("错误:非Byte类型数据"); |
| | | ds.delete(); |
| | | return; |
| | | } |
| | | |
| | | int width = band.getXSize(); |
| | | int height = band.getYSize(); |
| | | |
| | | // 读取Byte类型数据 |
| | | byte[] data = new byte[width * height]; |
| | | band.ReadRaster(0, 0, width, height, data); |
| | | |
| | | // 替换0值为8(注意Java的byte是有符号的,需处理0-255范围) |
| | | for (int i = 0; i < data.length; i++) { |
| | | if ((data[i] & 0xFF) == 0) { // 无符号比较 |
| | | data[i] = (byte) 8; |
| | | } |
| | | } |
| | | |
| | | band.WriteRaster(0, 0, width, height, data); // 写回数据并保存 |
| | | band.SetNoDataValue(8.0); // 设置新Nodata值 |
| | | band.FlushCache(); // 强制写入更改 |
| | | ds.delete(); |
| | | } |
| | | |
| | | private void createDir(String path) { |
| | |
| | | FileUtil.del(f); |
| | | } |
| | | f.mkdirs(); |
| | | } |
| | | |
| | | enum RainFallUnit { |
| | | MM_H("mm/h", 60), |
| | | MM_15MIN("mm/15min", 15), |
| | | MM_5MIN("mm/5min", 5), |
| | | MM_MIN("mm/min", 1); |
| | | private String unit; |
| | | private int c; //系数 |
| | | RainFallUnit(String unit, int c) { |
| | | this.unit = unit; |
| | | this.c = c; |
| | | } |
| | | |
| | | public static RainFallUnit of(String unit) { |
| | | RainFallUnit[] values = values(); |
| | | for (RainFallUnit v : values) { |
| | | if (v.unit.equals(unit)) { |
| | | return v; |
| | | } |
| | | } |
| | | return MM_H; //默认按照mm/h计算 |
| | | } |
| | | |
| | | } |
| | | |
| | | public void createRainfallFile(Simu simu, SimuData data) throws Exception { |
| | |
| | | double centerX = ComHelper.getMinVal((data.getMinx() + data.getMaxx()) / 2, DIGIT); |
| | | double centerY = ComHelper.getMinVal((data.getMiny() + data.getMaxy()) / 2, DIGIT); |
| | | String prefix = config.getRainfallSite() + " " + centerX + " " + centerY + " "; |
| | | int unit = StringUtils.isEmpty(data.getIntensityUnit()) || "mm/h".equals(data.getIntensityUnit()) ? 60 : 5; |
| | | |
| | | //如果没有单位或者单位为mm/h则按照 mm/h计算,否则按照mm/5min计算 |
| | | // int unit = StringUtils.isEmpty(data.getIntensityUnit()) || "mm/h".equals(data.getIntensityUnit()) ? 60 : 5; |
| | | String iu = data.getIntensityUnit(); |
| | | RainFallUnit rfUnit = RainFallUnit.of(iu); |
| | | data.setIntensityUnit(rfUnit.unit); |
| | | int unit = rfUnit.c; |
| | | int c = rainfalls.size() - 1; |
| | | for (int i = 0; i < c; i++) { |
| | | Rainfall r1 = rainfalls.get(i); |
| | |
| | | } |
| | | //list.add(prefix + YYYYMDHM.format(rainfalls.get(c).getTime()) + getMinVal(rainfalls.get(c).getIntensity() / unit, DIGIT)); |
| | | list.add(String.format("%s%s%f", prefix, YYYYMDHM.format(rainfalls.get(c).getTime()), getMinVal(rainfalls.get(c).getIntensity() / unit, DIGIT))); |
| | | list.add(0, "1 " + (list.size() - 1)); |
| | | // list.add(0, "1 " + (list.size() - 1)); |
| | | |
| | | Files.write(Paths.get(rainfallFile), list, StandardCharsets.UTF_8); |
| | | } |
| | |
| | | |
| | | int duration = 3600 * data.getDuration(); // 秒数 |
| | | if (null != data.getRainfalls() && data.getRainfalls().size() > 1) { |
| | | duration = (int) (Math.abs(data.getRainfalls().get(data.getRainfalls().size() - 1).getTime().getTime() - data.getRainfalls().get(0).getTime().getTime()) / 60); |
| | | List<Rainfall> rainfalls = data.getRainfalls(); |
| | | int size = rainfalls.size(); |
| | | Rainfall last = rainfalls.get(size - 1); |
| | | Rainfall first = rainfalls.get(0); |
| | | Instant end = last.getTime().toInstant(); |
| | | Instant start = first.getTime().toInstant(); |
| | | long diff = ChronoUnit.SECONDS.between(end, start); |
| | | duration = (int) (Math.abs(diff)); |
| | | } |
| | | |
| | | String inPath = config.getInPath() + File.separator + data.getInPath(); |
| | | String terrainFile = (inPath + File.separator + config.getDemFile()); |
| | | String terrainFile = inPath + File.separator + config.getTerrainFile(); |
| | | String landuseFile = inPath + File.separator + config.getLanduseFile(); |
| | | String rainfallFile = (inPath + File.separator + "rainfall.dat"); |
| | | String saveName = inPath + File.separator + "result.zarr"; |
| | | ConfigVo vo = new ConfigVo(terrainFile, terrainFile, terrainFile, rainfallFile, saveName, duration, config.getSaveFrames()); |
| | | // ConfigVo vo = new ConfigVo(terrainFile, landuseFile, terrainFile, rainfallFile, saveName, duration, config.getSaveFrames()); |
| | | Integer saveFrameInterval = config.getSaveFrameInterval(); |
| | | int saveFrames = duration / 60 / saveFrameInterval; |
| | | ConfigVo vo = new ConfigVo(terrainFile, landuseFile, terrainFile, rainfallFile, saveName, duration, saveFrames); |
| | | |
| | | String configFile = config.getInPath() + File.separator + data.getInPath() + File.separator + data.getInPath() + ".json"; |
| | | ComHelper.writeJson(configFile, JSON.toJSONString(vo)); |
| | |
| | | String inPath = config.getInPath() + File.separator + data.getInPath(); |
| | | String zarrFile = inPath + File.separator + "result.zarr"; |
| | | String geotiffDir = inPath + File.separator + "depth"; |
| | | String terrainFile = inPath + File.separator + config.getDemFile(); |
| | | String terrainFile = inPath + File.separator + config.getTerrainFile(); |
| | | String jsonPath = inPath + File.separator + "zarr2tif.json"; |
| | | |
| | | Zarr2Tif zarr2Tif = new Zarr2Tif(zarrFile, geotiffDir, terrainFile, data.getStartTime()); |
| | |
| | | try (BufferedReader reader = new BufferedReader(new InputStreamReader(process.getInputStream(), "GBK"))) { |
| | | String line; |
| | | while ((line = reader.readLine()) != null) { |
| | | System.out.println(line); |
| | | // System.out.println(line); |
| | | //sb.append(line); |
| | | log.info(line); |
| | | } |
| | | } |
| | | |
| | |
| | | File uwBat = new File(config.getUwSolverBat()); |
| | | String zarrFile = uwBat.getParent() + File.separator + "result.zarr"; |
| | | String inPath = config.getInPath() + File.separator + data.getInPath(); |
| | | String terrainFile = inPath + File.separator + config.getDemFile(); |
| | | String terrainFile = inPath + File.separator + config.getTerrainFile(); |
| | | String waterPath = inPath + File.separator + "depth"; |
| | | |
| | | String cmd = String.format("%s \"%s\" \"%s\" \"%s\" \"%s\"", config.getZarr2tifBat(), "depth", zarrFile, terrainFile, waterPath); |
| | |
| | | |
| | | cal.add(Calendar.MINUTE, 1); |
| | | } |
| | | list.add(0, "1 " + (list.size() - 1)); |
| | | // list.add(0, "1 " + (list.size() - 1)); |
| | | |
| | | Files.write(Paths.get(dat), list, StandardCharsets.UTF_8); |
| | | } |