《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

 

相关文章
|
2月前
|
架构师 前端开发 Java
Java开发工程师的职业规划应该是什么样的?
Java开发工程师的职业规划涵盖多个阶段,包括初入行业(0-1年)、技能提升(1-3年)、技术专家(3-5年)及管理或专家路线选择(5年以上)。各阶段设定了明确的技能要求与职业目标,从掌握Java基础、常用框架到深入研究高级技术、微服务架构乃至担任管理职务或成为技术专家。通过持续学习与实践,结合个人兴趣,Java工程师可在技术或管理领域找到合适的发展方向,最终实现职业成功。
340 83
|
22天前
|
存储 算法 Java
java制作海报六:Graphics2D的RenderingHints方法参数详解,包括解决文字不清晰,抗锯齿问题
这篇文章是关于如何在Java中使用Graphics2D的RenderingHints方法来提高海报制作的图像质量和文字清晰度,包括抗锯齿和解决文字不清晰问题的技术详解。
29 0
java制作海报六:Graphics2D的RenderingHints方法参数详解,包括解决文字不清晰,抗锯齿问题
|
28天前
|
Java
java构造方法时对象初始化,实例化,参数赋值
java构造方法时对象初始化,实例化,参数赋值
42 1
|
2月前
|
设计模式 架构师 Java
Java开发工程师转架构师需要学习什么
Java开发工程师转型为架构师需掌握多项技能:精通Java及框架、数据库与分布式系统;熟悉设计模式与架构模式;积累项目经验;提升沟通与领导力;持续学习新技术;培养系统设计与抽象能力;了解中间件及开发工具;并注重个人特质与职业发展。具体路径应结合个人目标与实际情况制定。
63 18
|
2月前
|
人工智能 前端开发 Java
Java开发工程师转哪个行业比较好?
Java开发工程师转哪个行业比较好?
224 2
|
2月前
|
小程序 前端开发 JavaScript
Java开发工程师转小程序开发的前景如何?
Java开发工程师转小程序开发的前景如何?
45 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应用中实现直接打印文档的功能,并通过具体示例代码展示其实现过程。
249 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参数的重要性及其对解决实际问题的有效性。
93 0