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

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

接上篇:https://developer.aliyun.com/article/1228187?spm=a2c6h.13148508.setting.16.358c4f0eIHLsiZ


二、 一次失败的重构经历

 

1. 理论依据

 

当看到一个方法定义过长时或者这段方法需要很多注释才能让人理解的时候,这时候就要考虑是不是把这个方法的部分代码提取出来,形成一个新的方法,方便调用和理解,同时也减小方法的粒度。我们把这种方法叫做提炼函数(Extract Function),在Java语言中可叫做提炼方法(Extract Method)。

 

2. 重构步骤

 

创建一个新方法,并根据这个方法的意图来命名;

将待提炼的代码段从原方法中拷贝到新方法中;

检查提炼的代码段,把缺少的变量添加到方法的参数中;

如果部分参数成对出现,可以把这些参数合并为一个参数;

如果方法需要有返回值,确定返回值的类型,并在合适的位置返回;

在原方法中,删除被提炼的代码段,替换为新方法的调用。

 

3. 代码实现

 

/**

* 几何辅助类

*/

public final class GeometryHelper {

 

   /** 原有静态常量 */

   ......

 

   /**

    * 划分线串

    *

    * @param lineString 原始线串

    * @param segmentLengthes 分段长度数组

    * @return 线串数组

    */

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

       // 原有计算逻辑

       ......

 

       // 初始化参数值

       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++) {

           lineStrings[i] = combineLineString(coordinates, index, coordinate, targetLengthes[i]);

       }

 

       // 添加最后一段

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

 

       // 返回线串数组

       return lineStrings;

   }

 

   /**

    * 组装线串

    *

    * @param coordinates 坐标数组

    * @param index 当前序号

    * @param coordinate 当前坐标

    * @param targetLength 目标长度

    * @return 线串

    */

   private static LineString combineLineString(Coordinate[] coordinates, int index, Coordinate coordinate, long targetLength) {

       // 添加线串坐标

       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, targetLength);

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

           if (compareResult < 0) {

               addupLength += segmentLength;

               coordinate = coordinates[index];

               coordinateList.add(coordinate);

               isBreak = false;

           }

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

           else if (compareResult > 0) {

               long deltaLength = targetLength - addupLength;

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

           }

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

           else {

               index++;

               coordinate = coordinates[index];

           }

 

           // 是否跳出循环

           if (isBreak) {

               break;

           }

       }

       coordinateList.add(coordinate);

 

       // 返回线串对象

       return buildLineString(coordinateList);

   }

 

   /** 原有静态方法 */

   ......

 

}

 

4. 存在问题

 

粗看这段代码,似乎没有什么问题。但是,通过测试发现,并没有得到正确的结果。

 

5. 分析原因

 

在《Thinking in Java》中有这样一段话:

 

| When you’re passing primitives into a method,you get a distinct copy of the primitive. When you’re passing a reference into a method, you get a copy of the reference.

 

当您将基本类型传递到方法中时,您将得到该基本类型的副本。当您将对象引用传递到方法中时,您将得到该对象引用的副本。

 

原来参数index(当前序号)和coordinate(当前坐标)在方法combineLineString(组装线串)中的修改,只是对该方法中的参数副本进行修改,并没有体现到调用方法splitLineString(划分线串)中,从而导致每次调用都在使用参数的初始化值。其实,这是在提取方法的过程中,没有考虑到参数的作用域。

 

6. 检查技巧

 

这里给出一个作者累试不爽的检查技巧——把提取方法的所有参数添加上final关键字,编译后观察到哪个参数出现编译错误,就说明这个参数是一个输入输出参数(Inout Parameter)。

 

 

7. 解决方案

 

在Java语言中,没有直接的输入输出参数机制,无法简单地实现参数的输入输出功能。所以,需要借助其它解决方案,来实现参数的输入输出功能。在这里,作者通过实践总结,给出了以下几种解决方案。


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

 

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