当前位置:首页 > 科技  > 软件

99%的Java程序员都会写这样的垃圾代码

来源: 责编: 时间:2023-11-13 09:06:41 379观看
导读在软件开发领域,编程经验往往与设计、编码、重构和测试的能力相辅相成。随着时间的推移,这些技能的提升使您能够在日常工作中脱颖而出。然而,有时候我们可能陷入固定的编程模式,导致编码习惯停滞不前。在这篇文章中,我们将

Giz28资讯网——每日最新资讯28at.com

在软件开发领域,编程经验往往与设计、编码、重构和测试的能力相辅相成。随着时间的推移,这些技能的提升使您能够在日常工作中脱颖而出。然而,有时候我们可能陷入固定的编程模式,导致编码习惯停滞不前。在这篇文章中,我们将介绍10个Java编程习惯,它们可以帮助您提高编码技能,写出更加干净、健壮的Java代码。Giz28资讯网——每日最新资讯28at.com

1、调用equals()方法时使用String字面值或已知对象

这条肯定中!很多人以为由于equals()方法是对称的,因此调用 a.equals(b) 与调用b.equals(a)相同的,所以习惯性的这样写:Giz28资讯网——每日最新资讯28at.com

if (givenString.equals("YES")){    // 执行一些操作}

尽管这种写法在可读性上有优势,但它并不安全。如果givenString为null,那么这段代码将抛出NullPointerException。为了避免这种情况,我们应该将equals方法调用放在已知对象的一侧,如下所示:Giz28资讯网——每日最新资讯28at.com

"YES".equals(givenString)

这样,如果givenString为null,它将返回false,而不会抛出异常。这是一种更加安全和健壮的编码习惯,同时也是避免NullPointerException的一种流行方式。Giz28资讯网——每日最新资讯28at.com

2、使用entrySet遍历HashMap

在遍历HashMap时,我们通常会使用键集合(key set)来获取键,并通过键获取对应的值。例如:Giz28资讯网——每日最新资讯28at.com

Set<Key> keySet = map.keySet();for (Key k : keySet){    Value v = map.get(k);    System.out.println(k + ": " + v);}

然而,这种方法需要进行两次查找操作,可能会导致性能下降。如果我们需要同时访问键和值,更好的方式是使用entrySet,如下:Giz28资讯网——每日最新资讯28at.com

Set<Map.Entry<Key, Value>> entrySet = map.entrySet();for (Map.Entry<Key, Value> entry : entrySet){    Key k = entry.getKey();    Value v = entry.getValue();    System.out.println(k + ": " + v);}

这种方式效率更高,因为它直接从entry对象中获取值,而不需要再次查找。Giz28资讯网——每日最新资讯28at.com

3、使用枚举作为单例

想象一下,您需要创建一个线程安全的单例模式。以前,这可能需要大量的代码和同步操作。但现在,您可以使用Java的枚举类型,仅需一行代码即可创建一个线程安全的单例:Giz28资讯网——每日最新资讯28at.com

public enum Singleton{  INSTANCE;}

这个枚举实例在多线程环境下也能保证只有一个实例存在,即使在序列化和反序列化过程中也是如此。这是一种简洁而强大的单例模式的实现方式。Giz28资讯网——每日最新资讯28at.com

4、使用Arrays.asList()或List.of()初始化集合

在Java中,初始化集合时,我们通常会逐个添加元素。例如:Giz28资讯网——每日最新资讯28at.com

List<String> listOfCurrencies = new ArrayList<>();listOfCurrencies.add("USD/AUD");listOfCurrencies.add("USD/JPY");listOfCurrencies.add("USD/INR");

这种方法虽然有效,但相对繁琐。您可以使用Arrays.asList()方法以更简洁的方式初始化集合,如下:Giz28资讯网——每日最新资讯28at.com

List<String> listOfPairs = new ArrayList<>(Arrays.asList("USD/AUD", "USD/JPY", "USD/INR"));

此外,从Java 9开始,您还可以使用List.of()和Set.of()方法来创建不可变的列表和集合。这些方法提供了更好的不可变性保证。Giz28资讯网——每日最新资讯28at.com

List<String> newList = List.of("One", "Two", "Infinity");

5、在循环中检查wait()条件

当我们使用wait()、notify()和notifyAll()方法进行多线程通信时,通常会在if语句中检查等待条件,然后调用wait()。例如:Giz28资讯网——每日最新资讯28at.com

synchronized(queue) {  if(queue.isFull()){    queue.wait();  }}

然而,这种写法存在一个问题,即可能会发生虚假通知(spurious notification)。为了解决这个问题,我们应该将检查等待条件的操作放在一个while循环内,如下:Giz28资讯网——每日最新资讯28at.com

synchronized(queue) {  while(queue.isFull()){    queue.wait();  }}

这样,即使在通知之前等待条件再次被满足,我们的代码也可以正确地工作。Giz28资讯网——每日最新资讯28at.com

6、捕获CloneNotSupportedException并返回子类实例

在Java中,对象克隆的实现机制常常受到批评,因为它的性能不佳。如果您需要实现clone()方法,可以使用以下习惯来减轻这种痛苦:Giz28资讯网——每日最新资讯28at.com

public Course clone() {   Course c = null;   try {     c = (Course)super.clone();   } catch (CloneNotSupportedException e) {} // 不会发生   return c;}

这个习惯利用了clone()方法实际上不会抛出CloneNotSupportedException的事实,只要类实现了Cloneable接口。这种方式返回了子类的实例,被称为协变方法覆盖(covariant method overriding),从Java 5开始支持。这可以减少客户端代码中的类型强制转换,使代码更加清晰。Giz28资讯网——每日最新资讯28at.com

7、在可能的情况下使用接口

在定义方法的返回类型、变量类型或方法参数类型时,应尽量使用接口而不是具体Giz28资讯网——每日最新资讯28at.com

的类。例如,不要这样写:Giz28资讯网——每日最新资讯28at.com

ArrayList<Integer> listOfNumbers = new ArrayList<>();public ArrayList<Integer> getNumbers(){    return listOfNumbers;}public void setNumbers(ArrayList<Integer> numbers){    listOfNumbers = numbers;}

而应该这样写:Giz28资讯网——每日最新资讯28at.com

List<Integer> listOfNumbers;public List<Integer> getNumbers(){    return listOfNumbers;}public void setNumbers(List<Integer> numbers){    listOfNumbers = numbers;}

这种方式提供了更大的灵活性,可以传递不同的集合实现。您还可以使用泛型中的通配符扩展功能,进一步提高灵活性。Giz28资讯网——每日最新资讯28at.com

public void processList(List<? extends Number> numbers){    // 执行操作}

8、使用迭代器遍历List

在Java中,有多种遍历List的方法,包括使用索引的for循环、增强的for循环和迭代器。最佳实践是使用迭代器,因为它是一种安全且能够防止不可预测行为的方法:Giz28资讯网——每日最新资讯28at.com

Iterator<String> iterator = list.iterator();while(iterator.hasNext()){    String name = iterator.next();    // 执行操作}

使用迭代器的好处包括能够遍历不同实现的List,例如ArrayList和LinkedList,同时避免了多线程环境下的问题。Giz28资讯网——每日最新资讯28at.com

9、在编写代码时考虑依赖注入

在以前的编程实践中,我们常常会硬编码依赖关系,例如:Giz28资讯网——每日最新资讯28at.com

public Game {    private HighScoreService service = HighScoreService.getInstance();    public void showLeaderBoard(){        List<Player> listOfTopPlayers = service.getLeaderBoard();        System.out.println(listOfTopPlayers);    }}

这种方式使Game类与HighScoreService类紧密耦合,不容易进行单元测试,因为必须使用HighScoreService的实际实现。为了避免这个问题,我们应该使用依赖注入,将依赖作为构造函数参数传递:Giz28资讯网——每日最新资讯28at.com

public Game {    private HighScoreService service;    public Game(HighScoreService service){        this.service = service;    }    public void showLeaderBoard(){        List<Player> listOfTopPlayers = service.getLeaderBoard();        System.out.println(listOfTopPlayers);    }}

这种方式使代码更加可测试,可以轻松地使用模拟对象进行测试。Giz28资讯网——每日最新资讯28at.com

10、在它们自己的try块中关闭流

在处理输入流和输出流时,我们经常需要进行异常处理和关闭操作。以前,我们可能会这样写:Giz28资讯网——每日最新资讯28at.com

InputStream is = null;OutputStream os = null;try {     is = new FileInputStream("application.json");    os = new FileOutputStream("application.log");} catch (IOException io) {} finally {    is.close();     os.close();}

然而,这种方式存在一个问题,如果第一个流的操作抛出异常,第二个流将永远不会被关闭。为了解决这个问题,我们可以使用Java 7引入的try-with-resources语法,更加简洁地处理流的关闭操作:Giz28资讯网——每日最新资讯28at.com

try (InputStream is = new FileInputStream("application.json");     OutputStream os = new FileOutputStream("application.log")) {    // 读取输入流并写入输出流的操作} catch (IOException e) {    // 异常处理代码}

使用try-with-resources后,不需要再手动关闭流,它们会在try块结束时自动关闭。这样的写法更加简洁和安全。Giz28资讯网——每日最新资讯28at.com

结语

这些Java编程习惯可以帮助您写出更加高效、健壮的Java代码,提高编码技能。如果您是刚刚开始学习Java或已经有一到两年的经验,这些习惯将为您打开Java编程的新视角。随着Java版本的不断更新,一些习惯可能会被更好的API方法所取代,但掌握它们仍然比不掌握更好。如果您还知道或遵循其他Java编程习惯,欢迎在评论中分享,我们期待从有经验的读者那里学习。希望这些习惯对您的Java编程之旅有所帮助。Giz28资讯网——每日最新资讯28at.com

本文链接:http://www.28at.com/showinfo-26-22475-0.html99%的Java程序员都会写这样的垃圾代码

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: 五个提高Java代码安全性的VS Code插件

下一篇: Python开发者的宝典:CSV和JSON数据处理技巧大公开!

标签:
  • 热门焦点
  • 直屏旗舰来了 iQOO 12和K70 Pro同台竞技

    旗舰机基本上使用的都是双曲面屏幕,这就让很多喜欢直屏的爱好者在苦等一款直屏旗舰,这次,你们等到了。据博主数码闲聊站带来的最新爆料称,Redmi下代旗舰K70 Pro和iQOO 12两款手
  • 天猫精灵Sound Pro体验:智能音箱没有音质?来听听我的

    这几年除了手机作为智能生活终端最主要的核心之外,第二个可以成为中心点的产品是什么?——是智能音箱。 手机在执行命令的时候有两种操作方式,手和智能语音助手,而智能音箱只
  • 6月安卓手机性价比榜:Note 12 Turbo断层式碾压

    6月份有一个618,虽然这是京东周年庆的日子,但别的电商也都不约而同的跟进了,反正促销没坏处,厂商和用户都能满意。618期间一些产品也出现了历史低价,那么各个价位段的产品性价比
  • 线程通讯的三种方法!通俗易懂

    线程通信是指多个线程之间通过某种机制进行协调和交互,例如,线程等待和通知机制就是线程通讯的主要手段之一。 在 Java 中,线程等待和通知的实现手段有以下几种方式:Object 类下
  • 摸鱼心法第一章——和配置文件说拜拜

    为了能摸鱼我们团队做了容器化,但是带来的问题是服务配置文件很麻烦,然后大家在群里进行了“亲切友好”的沟通图片图片图片图片对比就对比,简单对比下独立配置中心和k8s作为配
  • 三言两语说透设计模式的艺术-单例模式

    写在前面单例模式是一种常用的软件设计模式,它所创建的对象只有一个实例,且该实例易于被外界访问。单例对象由于只有一个实例,所以它可以方便地被系统中的其他对象共享,从而减少
  • 如何正确使用:Has和:Nth-Last-Child

    我们可以用CSS检查,以了解一组元素的数量是否小于或等于一个数字。例如,一个拥有三个或更多子项的grid。你可能会想,为什么需要这样做呢?在某些情况下,一个组件或一个布局可能会
  • 如何通过Python线程池实现异步编程?

    线程池的概念和基本原理线程池是一种并发处理机制,它可以在程序启动时创建一组线程,并将它们置于等待任务的状态。当任务到达时,线程池中的某个线程会被唤醒并执行任务,执行完任
  • 电博会与软博会实现"线下+云端"的双线融合

    在本次“电博会”与“软博会”双展会利好条件的加持下,既可以发挥展会拉动人流、信息流、资金流实现快速交互流动的作用,继而推动区域经济良性发展;又可以聚
Top