《Java工程师必读手册》——Java编程技巧系列——Java编程技巧之输入输出参数(1)

简介: 《Java工程师必读手册》——Java编程技巧系列——Java编程技巧之输入输出参数(1)

前言

 

软件开发方法学的泰斗肯特·贝克(Kent Beck)曾说过:

 

| 我不是一个伟大的程序员,我只是一个具有良好习惯的优秀程序员。

 

养成良好的习惯,尤其是不断重构的习惯,是每一个优秀程序员都应该具备的素质。重构(Refactoring)就是在不改变软件现有功能的基础上,通过调整程序的结构、提高程序的质量、优化程序的性能……使其程序的设计模式和架构更趋合理,从而提高软件的稳定性、扩展性和维护性。


一、 一个需要重构的方法

 

1. 需求描述

 

需要把一个线串(一组经纬度坐标串),按照指定分段长度数组进行按比例划分。(由于指定线串的长度较小,可以近似地认为在几何平面上,无需进行球面距离换算。)

 

2. 代码实现

 

/**

* 几何辅助类

*/

public final class GeometryHelper {

 

   /** 常量相关 */

   /** 小数位数 */

   private static final int DIGIT_SCALE = 8;

   /** 放大比例 */

   private static final double ZOOM_SCALE = 10000000000L;

   /** 几何工厂 */

   private static final GeometryFactory GEOMETRY_FACTORY = new GeometryFactory(new PrecisionModel(PrecisionModel.FLOATING));

 

   /**

    * 构造方法

    */

   private GeometryHelper() {

       throw new UnsupportedOperationException();

   }

 

   /**

    * 划分线串

    *

    * @param lineString 原始线串

    * @param segmentLengthes 分段长度数组

    * @return 线串数组

    */

   public static LineString[] splitLineString(LineString lineString, double[] segmentLengthes) {

       // 检查分段数量

       if (Objects.isNull(segmentLengthes) || segmentLengthes.length < 1) {

           return new LineString[] {lineString};

       }

 

       // 计算总共长度

       double totalLength = Arrays.stream(segmentLengthes)

           .map(segmentLength -> Math.max(segmentLength, 0.0D))

           .sum();

 

       // 计算目标长度

       double lineLength = lineString.getLength();

       long[] targetLengthes = Arrays.stream(segmentLengthes)

           .mapToLong(segmentLength -> getTargetLength(lineLength, totalLength, segmentLength))

           .toArray();

 

       // 初始化参数值

       int index = 1;

       Coordinate[] coordinates = lineString.getCoordinates();

       Coordinate coordinate = coordinates[0];

       int length = targetLengthes.length;

       LineString[] lineStrings = new LineString[length];

 

       // 添加前面N段

       for (int i = 0; i < length - 1; i++) {

           // 添加线串坐标

           long addupLength = 0L;

           List coordinateList = new ArrayList<>();

           coordinateList.add(coordinate);

           for (; index < coordinates.length; index++) {

               // 计算分段长度

               long segmentLength = Math.round(coordinate.distance(coordinates[index]) * ZOOM_SCALE);

 

               // 根据长度处理

               boolean isBreak = true;

               int compareResult = Long.compare(addupLength + segmentLength, targetLengthes[i]);

               // 根据长度处理: 未达目标长度

               if (compareResult < 0) {

                   addupLength += segmentLength;

                   coordinate = coordinates[index];

                   coordinateList.add(coordinate);

                   isBreak = false;

               }

               // 根据长度处理: 超过目标长度

               else if (compareResult > 0) {

                   long deltaLength = targetLengthes[i] - addupLength;

                   coordinate = buildMiddleCoordinate(coordinate, coordinates[index], segmentLength, deltaLength);

               }

               // 根据长度处理: 等于目标长度

               else {

                   index++;

                   coordinate = coordinates[index];

               }

 

               // 是否跳出循环

               if (isBreak) {

                   break;

               }

           }

           coordinateList.add(coordinate);

 

           // 设置线串对象

           lineStrings[i] = buildLineString(coordinateList);

       }

 

       // 添加最后一段

       lineStrings[length - 1] = buildLineString(coordinates, index, coordinate);

 

       // 返回线串数组

       return lineStrings;

   }

 

   /**

    * 构建线串

    *

    * @param coordinates 坐标数组

    * @param index 当前序号

    * @param coordinate 当前坐标

    * @return 线串

    */

   private static LineString buildLineString(Coordinate[] coordinates, int index, Coordinate coordinate) {

       List coordinateList = new ArrayList<>();

       coordinateList.add(coordinate);

       coordinateList.addAll(Arrays.asList(ArrayUtils.subarray(coordinates, index, coordinates.length)));

       return buildLineString(coordinateList);

   }

 

   /**

    * 构建线串

    *

    * @param coordinateList 坐标列表

    * @return 线串

    */

   private static LineString buildLineString(List coordinateList) {

       return GEOMETRY_FACTORY.createLineString(coordinateList.toArray(new Coordinate[0]));

   }

 

   /**

    * 构建中间坐标

    *

    * @param coordinate1 坐标1

    * @param coordinate2 坐标2

    * @param segmentLength 分段长度

    * @param deltaLength 增量长度

    * @return 中间坐标

    */

   private static Coordinate buildMiddleCoordinate(Coordinate coordinate1, Coordinate coordinate2,

       long segmentLength, long deltaLength) {

       double deltaScale = deltaLength * 1.0D / segmentLength;

       double middleX = round(coordinate1.x + (coordinate2.x - coordinate1.x) * deltaScale, DIGIT_SCALE);

       double middleY = round(coordinate1.y + (coordinate2.y - coordinate1.y) * deltaScale, DIGIT_SCALE);

       return new Coordinate(middleX, middleY);

   }

 

   /**

    * 获取目标长度

    *

    * @param lineLength 线路长度

    * @param totalLength 总共长度

    * @param segmentLength 段长度

    * @return 目标长度

    */

   private static long getTargetLength(double lineLength, double totalLength, double segmentLength) {

       return Math.round(Math.max(segmentLength, 0.0D) * ZOOM_SCALE * lineLength / totalLength);

   }

 

   /**

    * 四舍五入

    *

    * @param value 双精度浮点值

    * @param scale 保留小数位数

    * @return 四舍五入值

    */

   private static double round(double value, int scale) {

       return BigDecimal.valueOf(value).setScale(scale, BigDecimal.ROUND_HALF_UP).doubleValue();

   }  

   

}

 

 

3. 备注说明

 

在超过目标长度时,获取了一个中间坐标,由于数据精度的关系,这个坐标可能跟上一坐标或下一坐标重合。这里为了降低这块逻辑的复杂度,没有进行前后坐标的去重处理。

 

4. 存在问题

 

在方法splitLineString(划分线串)中,存在一个两层循环,导致了方法逻辑复杂、层级较深、代码量大。如果把外层循环体提炼为一个方法,就能够使代码更简洁、更清晰、更容易维护。


 接下篇:https://developer.aliyun.com/article/1228185?groupCode=java

 

相关文章
|
2月前
|
架构师 前端开发 Java
Java开发工程师的职业规划应该是什么样的?
Java开发工程师的职业规划涵盖多个阶段,包括初入行业(0-1年)、技能提升(1-3年)、技术专家(3-5年)及管理或专家路线选择(5年以上)。各阶段设定了明确的技能要求与职业目标,从掌握Java基础、常用框架到深入研究高级技术、微服务架构乃至担任管理职务或成为技术专家。通过持续学习与实践,结合个人兴趣,Java工程师可在技术或管理领域找到合适的发展方向,最终实现职业成功。
320 83
|
17天前
|
存储 算法 Java
java制作海报六:Graphics2D的RenderingHints方法参数详解,包括解决文字不清晰,抗锯齿问题
这篇文章是关于如何在Java中使用Graphics2D的RenderingHints方法来提高海报制作的图像质量和文字清晰度,包括抗锯齿和解决文字不清晰问题的技术详解。
25 0
java制作海报六:Graphics2D的RenderingHints方法参数详解,包括解决文字不清晰,抗锯齿问题
|
23天前
|
Java
java构造方法时对象初始化,实例化,参数赋值
java构造方法时对象初始化,实例化,参数赋值
36 1
|
2月前
|
设计模式 架构师 Java
Java开发工程师转架构师需要学习什么
Java开发工程师转型为架构师需掌握多项技能:精通Java及框架、数据库与分布式系统;熟悉设计模式与架构模式;积累项目经验;提升沟通与领导力;持续学习新技术;培养系统设计与抽象能力;了解中间件及开发工具;并注重个人特质与职业发展。具体路径应结合个人目标与实际情况制定。
62 18
|
2月前
|
人工智能 前端开发 Java
Java开发工程师转哪个行业比较好?
Java开发工程师转哪个行业比较好?
207 2
|
2月前
|
小程序 前端开发 JavaScript
Java开发工程师转小程序开发的前景如何?
Java开发工程师转小程序开发的前景如何?
43 0
|
3月前
|
Java
【Azure 应用服务】如何查看App Service Java堆栈JVM相关的参数默认配置值?
【Azure 应用服务】如何查看App Service Java堆栈JVM相关的参数默认配置值?
【Azure 应用服务】如何查看App Service Java堆栈JVM相关的参数默认配置值?
|
3月前
|
C# 开发者 Windows
震撼发布:全面解析WPF中的打印功能——从基础设置到高级定制,带你一步步实现直接打印文档的完整流程,让你的WPF应用程序瞬间升级,掌握这一技能,轻松应对各种打印需求,彻底告别打印难题!
【8月更文挑战第31天】打印功能在许多WPF应用中不可或缺,尤其在需要生成纸质文档时。WPF提供了强大的打印支持,通过`PrintDialog`等类简化了打印集成。本文将详细介绍如何在WPF应用中实现直接打印文档的功能,并通过具体示例代码展示其实现过程。
224 0
|
3月前
|
缓存 前端开发 Java
【Azure 应用服务】App Service 使用Tomcat运行Java应用,如何设置前端网页缓存的相应参数呢(-Xms512m -Xmx1204m)?
【Azure 应用服务】App Service 使用Tomcat运行Java应用,如何设置前端网页缓存的相应参数呢(-Xms512m -Xmx1204m)?
|
3月前
|
运维 监控 Java
【JVM 调优秘籍】实战指南:JVM 调优参数全解析,让 Java 应用程序性能飙升!
【8月更文挑战第24天】本文通过一个大型在线零售平台的例子,深入探讨了Java虚拟机(JVM)性能调优的关键技术。面对应用响应延迟的问题,文章详细介绍了几种常用的JVM参数调整策略,包括堆内存大小、年轻代配置、垃圾回收器的选择及日志记录等。通过具体实践(如设置`-Xms`, `-Xmx`, `-XX:NewRatio`, `-XX:+UseParallelGC`等),成功降低了高峰期的响应时间,提高了系统的整体性能与稳定性。案例展示了合理配置JVM参数的重要性及其对解决实际问题的有效性。
88 0