《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

 

相关文章
|
8月前
|
搜索推荐 算法 Java
2025 年互联网大厂校园招聘 JAVA 工程师笔试题及备考要点解析
本文针对互联网大厂校招Java工程师笔试题进行解析,涵盖基础知识、面向对象编程、数据结构与算法、异常处理及集合框架等核心内容。从数据类型、运算符到流程控制语句,从类与对象、继承多态到数组链表、排序算法,再到异常捕获与集合框架应用,结合实际案例深入剖析,助你系统掌握考点,提升应试能力。资源链接:[点此获取](https://pan.quark.cn/s/14fcf913bae6)。
310 9
|
11月前
|
Java Linux 定位技术
Minecraft配置文件参数说明(JAVA服务器篇)
Minecraft JAVA版服务器启动后会生成server.properties配置文件,位于minecraft_server/根目录下。该文件包含多项关键设置,如游戏模式(gamemode)、最大玩家数(max-players)、难度(difficulty)等。此文档详细说明了各配置项的功能与默认值,帮助用户高效管理服务器环境。
2589 60
|
8月前
|
Java 数据库连接 API
互联网大厂校招 JAVA 工程师笔试题解析及常见考点分析
本文深入解析互联网大厂校招Java工程师笔试题,涵盖基础知识(数据类型、流程控制)、面向对象编程(类与对象、继承与多态)、数据结构与算法(数组、链表、排序算法)、异常处理、集合框架、Java 8+新特性(Lambda表达式、Stream API)、多线程与并发、IO与NIO、数据库操作(JDBC、ORM框架MyBatis)及Spring框架基础(IoC、DI、AOP)。通过技术方案讲解与实例演示,助你掌握核心考点,提升解题能力。
327 2
|
10月前
|
Java
java中一个接口A,以及一个实现它的类B,一个A类型的引用对象作为一个方法的参数,这个参数的类型可以是B的类型吗?
本文探讨了面向对象编程中接口与实现类的关系,以及里氏替换原则(LSP)的应用。通过示例代码展示了如何利用多态性将实现类的对象传递给接口类型的参数,满足LSP的要求。LSP确保子类能无缝替换父类或接口,不改变程序行为。接口定义了行为规范,实现类遵循此规范,从而保证了多态性和代码的可维护性。总结来说,接口与实现类的关系天然符合LSP,体现了多态性的核心思想。
264 0
|
12月前
|
机器学习/深度学习 Java PyTorch
Java工程师如何理解张量?
刚接触AI和PyTorch,理解“张量(Tensor)”是入门关键。张量可类比为Java中的多维数组,但更强大,尤其在AI领域支持GPU加速、自动求导等特性。它不仅能高效存储数据,还能进行复杂运算,是深度学习的核心数据结构。掌握张量的维度、数据类型及GPU加速特性,对学习PyTorch至关重要。
331 3
|
Java
实现java执行kettle并传参数
实现java执行kettle并传参数
282 1
|
存储 算法 Java
java制作海报六:Graphics2D的RenderingHints方法参数详解,包括解决文字不清晰,抗锯齿问题
这篇文章是关于如何在Java中使用Graphics2D的RenderingHints方法来提高海报制作的图像质量和文字清晰度,包括抗锯齿和解决文字不清晰问题的技术详解。
601 0
java制作海报六:Graphics2D的RenderingHints方法参数详解,包括解决文字不清晰,抗锯齿问题
|
Java
java构造方法时对象初始化,实例化,参数赋值
java构造方法时对象初始化,实例化,参数赋值
396 1
在Java中定义一个不做事且没有参数的构造方法的作用
Java程序在执行子类的构造方法之前,如果没有用super()来调用父类特定的构造方法,则会调用父类中“没有参数的构造方法”。因此,如果父类中只定义了有参数的构造方法,而在子类的构造方法中又没有用super()来调用父类中特定的构造方法,则编译时将发生错误,因为Java程序在父类中找不到没有参数的构造方法可供执行。解决办法是在父类里加上一个不做事且没有参数的构造方法。
|
人工智能 前端开发 Java
Java开发工程师转哪个行业比较好?
Java开发工程师转哪个行业比较好?
1029 2