0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

怎么样才能让Java代码编写更规范化

5jek_harmonyos 来源:博客园 作者:涛姐涛哥 2021-08-27 09:31 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

作者 | 涛姐涛哥

链接 | cnblogs.com/taojietaoge/p/11575376.html

如何更规范化编写Java 代码

Many of the happiest people are those who own the least. But are we really so happy with our IPhones, our big houses, our fancy cars

忘川如斯,拥有一切的人才更怕失去。

背景:如何更规范化编写Java 代码的重要性想必毋需多言,其中最重要的几点当属提高代码性能、使代码远离Bug、令代码更优雅。

一、MyBatis 不要为了多个查询条件而写 1 = 1

当遇到多个查询条件,使用where 1=1 可以很方便的解决我们的问题,但是这样很可能会造成非常大的性能损失,因为添加了 “where 1=1 ”的过滤条件之后,数据库系统就无法使用索引等查询优化策略,数据库系统将会被迫对每行数据进行扫描(即全表扫描) 以比较此行是否满足过滤条件,当表中的数据量较大时查询速度会非常慢;此外,还会存在SQL 注入的风险。

反例:

《select id=“queryBookInfo” parameterType=“com.tjt.platform.entity.BookInfo” resultType=“java.lang.Integer”》 select count(*) from t_rule_BookInfo t where 1=1《if test=“title !=null and title !=‘’ ”》 AND title = #{title} 《/if》 《if test=“author !=null and author !=‘’ ”》 AND author = #{author}《/if》 《/select》

正例:

《select id=“queryBookInfo” parameterType=“com.tjt.platform.entity.BookInfo” resultType=“java.lang.Integer”》 select count(*) from t_rule_BookInfo t《where》《if test=“title !=null and title !=‘’ ”》 title = #{title} 《/if》《if test=“author !=null and author !=‘’ ”》 AND author = #{author}《/if》《/where》 《/select》

UPDATE 操作也一样,可以用《set》 标记代替 1=1。

二、 迭代entrySet() 获取Map 的key 和value

当循环中只需要获取Map 的主键key时,迭代keySet() 是正确的;但是,当需要主键key 和取值value 时,迭代entrySet() 才是更高效的做法,其比先迭代keySet() 后再去通过get 取值性能更佳。

反例:

//Map 获取value 反例:HashMap《String, String》 map = new HashMap《》();for (String key : map.keySet()){String value = map.get(key);}

正例:

//Map 获取key & value 正例:HashMap《String, String》 map = new HashMap《》(); for (Map.Entry《String,String》 entry : map.entrySet()){ String key = entry.getKey(); String value = entry.getValue();}

三、使用Collection.isEmpty() 检测空

使用Collection.size() 来检测是否为空在逻辑上没有问题,但是使用Collection.isEmpty() 使得代码更易读,并且可以获得更好的性能;除此之外,任何Collection.isEmpty() 实现的时间复杂度都是O(1) ,不需要多次循环遍历,但是某些通过Collection.size() 方法实现的时间复杂度可能是O(n)。O(1)纬度减少循环次数 例子

反例:

LinkedList《Object》 collection = new LinkedList《》();if (collection.size() == 0){ System.out.println(“collection is empty.”); }

正例:

LinkedList《Object》 collection = new LinkedList《》(); if (collection.isEmpty()){ System.out.println(“collection is empty.”); }

//检测是否为null 可以使用CollectionUtils.isEmpty() if (CollectionUtils.isEmpty(collection)){ System.out.println(“collection is null.”);

}

四、初始化集合时尽量指定其大小

尽量在初始化时指定集合的大小,能有效减少集合的扩容次数,因为集合每次扩容的时间复杂度很可能时O(n),耗费时间和性能。

反例:

//初始化list,往list 中添加元素反例:int[] arr = new int[]{1,2,3,4};List《Integer》 list = new ArrayList《》();for (int i : arr){ list.add(i);}

正例:

//初始化list,往list 中添加元素正例: int[] arr = new int[]{1,2,3,4}; //指定集合list 的容量大小 List《Integer》 list = new ArrayList《》(arr.length); for (int i : arr){ list.add(i); }

五、使用StringBuilder 拼接字符串

一般的字符串拼接在编译期Java 会对其进行优化,但是在循环中字符串的拼接Java 编译期无法执行优化,所以需要使用StringBuilder 进行替换。

反例:

//在循环中拼接字符串反例String str = “”;for (int i = 0; i 《 10; i++){ //在循环中字符串拼接Java 不会对其进行优化 str += i;}

正例:

//在循环中拼接字符串正例 String str1 = “Love”; String str2 = “Courage”; String strConcat = str1 + str2; //Java 编译器会对该普通模式的字符串拼接进行优化 StringBuilder sb = new StringBuilder(); for (int i = 0; i 《 10; i++){ //在循环中,Java 编译器无法进行优化,所以要手动使用StringBuilder sb.append(i); }

六、若需频繁调用Collection.contains 方法则使用Set

在Java 集合类库中,List的contains 方法普遍时间复杂度为O(n),若代码中需要频繁调用contains 方法查找数据则先将集合list 转换成HashSet 实现,将O(n) 的时间复杂度将为O(1)。

反例:

//频繁调用Collection.contains() 反例List《Object》 list = new ArrayList《》();for (int i = 0; i 《= Integer.MAX_VALUE; i++){ //时间复杂度为O(n) if (list.contains(i)) System.out.println(“list contains ”+ i); }

正例:

//频繁调用Collection.contains() 正例 List《Object》 list = new ArrayList《》(); Set《Object》 set = new HashSet《》(); for (int i = 0; i 《= Integer.MAX_VALUE; i++){ //时间复杂度为O(1) if (set.contains(i)){ System.out.println(“list contains ”+ i); } }

七、使用静态代码块实现赋值静态成员变量

对于集合类型的静态成员变量,应该使用静态代码块赋值,而不是使用集合实现来赋值。

反例:

//赋值静态成员变量反例 private static Map《String, Integer》 map = new HashMap《String, Integer》(){ { map.put(“Leo”,1); map.put(“Family-loving”,2); map.put(“Cold on the out side passionate on the inside”,3); } }; private static List《String》 list = new ArrayList《》(){ { list.add(“Sagittarius”); list.add(“Charming”); list.add(“Perfectionist”); } };

正例:

//赋值静态成员变量正例private static Map《String, Integer》 map = new HashMap《String, Integer》(); static { map.put(“Leo”,1); map.put(“Family-loving”,2); map.put(“Cold on the out side passionate on the inside”,3); }

private static List《String》 list = new ArrayList《》(); static { list.add(“Sagittarius”); list.add(“Charming”); list.add(“Perfectionist”); }

八、删除未使用的局部变量、方法参数、私有方法、字段和多余的括号。

九、工具类中屏蔽构造函数

工具类是一堆静态字段和函数的集合,其不应该被实例化;但是,Java 为每个没有明确定义构造函数的类添加了一个隐式公有构造函数,为了避免不必要的实例化,应该显式定义私有构造函数来屏蔽这个隐式公有构造函数。

反例:

public class PasswordUtils { //工具类构造函数反例 private static final Logger LOG = LoggerFactory.getLogger(PasswordUtils.class);

public static final String DEFAULT_CRYPT_ALGO = “PBEWithMD5AndDES”;

public static String encryptPassword(String aPassword) throws IOException { return new PasswordUtils(aPassword).encrypt(); }

正例:

public class PasswordUtils { //工具类构造函数正例 private static final Logger LOG = LoggerFactory.getLogger(PasswordUtils.class);

//定义私有构造函数来屏蔽这个隐式公有构造函数 private PasswordUtils(){}

public static final String DEFAULT_CRYPT_ALGO = “PBEWithMD5AndDES”;

public static String encryptPassword(String aPassword) throws IOException { return new PasswordUtils(aPassword).encrypt(); }

十、删除多余的异常捕获并抛出

用catch 语句捕获异常后,若什么也不进行处理,就只是让异常重新抛出,这跟不捕获异常的效果一样,可以删除这块代码或添加别的处理。

反例:

//多余异常反例private static String fileReader(String fileName)throws IOException{

try (BufferedReader reader = new BufferedReader(new FileReader(fileName))) { String line; StringBuilder builder = new StringBuilder(); while ((line = reader.readLine()) != null) { builder.append(line); } return builder.toString(); } catch (Exception e) { //仅仅是重复抛异常 未作任何处理 throw e; }}

正例:

//多余异常正例private static String fileReader(String fileName)throws IOException{

try (BufferedReader reader = new BufferedReader(new FileReader(fileName))) { String line; StringBuilder builder = new StringBuilder(); while ((line = reader.readLine()) != null) { builder.append(line); } return builder.toString(); //删除多余的抛异常,或增加其他处理: /*catch (Exception e) { return “fileReader exception”; }*/ }}

十一、字符串转化使用String.valueOf(value) 代替 “ ” + value

把其它对象或类型转化为字符串时,使用String.valueOf(value) 比 “”+value 的效率更高。

反例:

//把其它对象或类型转化为字符串反例:int num = 520;// “” + valueString strLove = “” + num;

正例:

//把其它对象或类型转化为字符串正例:int num = 520;// String.valueOf() 效率更高String strLove = String.valueOf(num);

十二、避免使用BigDecimal(double)

BigDecimal(double) 存在精度损失风险,在精确计算或值比较的场景中可能会导致业务逻辑异常。

反例:

// BigDecimal 反例BigDecimal bigDecimal = new BigDecimal(0.11D);

正例:

// BigDecimal 正例BigDecimal bigDecimal1 = bigDecimal.valueOf(0.11D);

图1. 失去精度

十三、返回空数组和集合而非 null

若程序运行返回null,需要调用方强制检测null,否则就会抛出空指针异常;返回空数组或空集合,有效地避免了调用方因为未检测null 而抛出空指针异常的情况,还可以删除调用方检测null 的语句使代码更简洁。

反例:

//返回null 反例public static Result[] getResults() { return null;}

public static List《Result》 getResultList() { return null;}

public static Map《String, Result》 getResultMap() { return null;}

正例:

//返回空数组和空集正例public static Result[] getResults() { return new Result[0];}

public static List《Result》 getResultList() { return Collections.emptyList();}

public static Map《String, Result》 getResultMap() { return Collections.emptyMap();}

十四、优先使用常量或确定值调用equals 方法

对象的equals 方法容易抛空指针异常,应使用常量或确定有值的对象来调用equals 方法。

反例:

//调用 equals 方法反例private static boolean fileReader(String fileName)throws IOException{ // 可能抛空指针异常 return fileName.equals(“Charming”); }

正例:

//调用 equals 方法正例private static boolean fileReader(String fileName)throws IOException{

// 使用常量或确定有值的对象来调用 equals 方法 return “Charming”.equals(fileName);

//或使用:java.util.Objects.equals() 方法 return Objects.equals(“Charming”,fileName); }

十五、枚举的属性字段必须是私有且不可变

枚举通常被当做常量使用,如果枚举中存在公共属性字段或设置字段方法,那么这些枚举常量的属性很容易被修改;理想情况下,枚举中的属性字段是私有的,并在私有构造函数中赋值,没有对应的Setter 方法,最好加上final 修饰符。

反例:

public enum SwitchStatus { // 枚举的属性字段反例 DISABLED(0, “禁用”), ENABLED(1, “启用”);

public int value; private String description;

private SwitchStatus(int value, String description) { this.value = value; this.description = description; }

public String getDescription() { return description; }

public void setDescription(String description) { this.description = description; }}

正例:

public enum SwitchStatus { // 枚举的属性字段正例 DISABLED(0, “禁用”), ENABLED(1, “启用”);

// final 修饰 private final int value; private final String description;

private SwitchStatus(int value, String description) { this.value = value; this.description = description; }

// 没有Setter 方法 public int getValue() { return value; }

public String getDescription() { return description; }}

十六、tring.split(String regex)部分关键字需要转译

使用字符串String 的plit 方法时,传入的分隔字符串是正则表达式,则部分关键字(比如 。[]()| 等)需要转义。

反例:

// String.split(String regex) 反例String[] split = “a.ab.abc”.split(“。”);System.out.println(Arrays.toString(split)); // 结果为[]String[] split1 = “a|ab|abc”.split(“|”);System.out.println(Arrays.toString(split1)); // 结果为[“a”, “|”, “a”, “b”, “|”, “a”, “b”, “c”]

正例:

// String.split(String regex) 正例// 。 需要转译String[] split2 = “a.ab.abc”.split(“\。”);System.out.println(Arrays.toString(split2)); // 结果为[“a”, “ab”, “abc”]

// | 需要转译String[] split3 = “a|ab|abc”.split(“\|”);System.out.println(Arrays.toString(split3)); // 结果为[“a”, “ab”, “abc”]

责任编辑:haq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • JAVA
    +关注

    关注

    20

    文章

    2997

    浏览量

    115671
  • 编程
    +关注

    关注

    90

    文章

    3707

    浏览量

    96764
  • 代码
    +关注

    关注

    30

    文章

    4941

    浏览量

    73145

原文标题:16 条 yyds 的代码规范

文章出处:【微信号:harmonyos_developer,微信公众号:harmonyos_developer】欢迎添加关注!文章转载请注明出处。

收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    编写简单高效单片机汇编程序

    ,避免重复编写。 宏指令:编写重复代码块时,把它封装为宏指令,并用传参简化,比如显示数字(数码管号, 数值)。这样能让代码
    发表于 11-24 07:00

    Arm Neoverse CPU上大代码Java应用的性能测试

    Java 是互联网领域广泛使用的编程语言。Java 应用的一些特性使其性能表现与提前编译的原生应用(例如 C 程序)大相径庭。由于 Java 字节码无法直接在 CPU 上执行,因此通常运行时在
    的头像 发表于 11-05 11:25 371次阅读
    Arm Neoverse CPU上大<b class='flag-5'>代码</b>量<b class='flag-5'>Java</b>应用的性能测试

    Java 25正式发布,重要特性详解(附代码示例):灵活构造函数体、模块导入声明、AOT方法分析等

    Java 25现已发布,更多新特性来了!配合Perforce JRebel,代码修改即时生效,无需重启服务,即可实现“改完就看效果”。新特性+快工具,让你的Java开发体验双倍提升!
    的头像 发表于 10-29 13:16 851次阅读
    <b class='flag-5'>Java</b> 25正式发布,重要特性详解(附<b class='flag-5'>代码</b>示例):灵活构造函数体、模块导入声明、AOT方法分析等

    代码格式化工具Clang-Format提升你的CW32工程质量

    它能自动统一团队代码风格,让不同开发者写出的代码如出一辙。就像 CW32 官方库函数遵循统一规范,Clang-Format 能让团队所有
    的头像 发表于 10-09 17:43 903次阅读
    <b class='flag-5'>代码</b>格式化工具Clang-Format提升你的CW32工程质量

    嵌入式达到什么水平才能就业?

    工具定位问题具备嵌入式软件模块设计能力:能按功能划分代码模块,编写规范的头文件与源文件,实现模块间低耦合调用会使用Git 进行版本管理:能提交代码
    发表于 09-15 10:20

    Perforce JRebel 简介:即时加载代码变更,加速Java应用开发

    Perforce JRebel 专为Java开发提速而生!支持跳过构建与重新部署,实时加载代码变更,支持100+框架,无缝集成主流IDE与应用服务器。
    的头像 发表于 08-14 14:35 487次阅读
    Perforce JRebel 简介:即时加载<b class='flag-5'>代码</b>变更,加速<b class='flag-5'>Java</b>应用开发

    镭神智能深度参与两项激光雷达国家标准制定 引领行业规范化发展新征程

    ,深圳市镭神智能系统有限公司在董事长胡小波先生的带领下,深度参与标准制定工作,为推动激光雷达技术规范化、产业发展贡献了核心力量。国家标准《车载激光雷达性能要求及
    的头像 发表于 05-09 18:57 870次阅读
    镭神智能深度参与两项激光雷达国家标准制定 引领行业<b class='flag-5'>规范化</b>发展新征程

    众合云科林枚参编的人力资源AI领域团体标准正式发布,助推行业数字才能规范化建设

    专家组成员,参与该标准的起草、研讨、修订及技术审核工作,为人工智能与人力资源数字应用技能领域的人才能规范化建设提供专业支持。 (众合云科人力资源副总裁林枚受邀担任《标准》编委) 本次《标准》共有9个章节,规定AI+人力资源数
    的头像 发表于 05-09 15:05 320次阅读
    众合云科林枚参编的人力资源AI领域团体标准正式发布,助推行业数字<b class='flag-5'>化</b>人<b class='flag-5'>才能</b>力<b class='flag-5'>规范化</b>建设

    Java开发者必备的效率工具——Perforce JRebel是什么?为什么很多Java开发者在用?

    Perforce JRebel是一款Java开发效率工具,旨在帮助java开发人员更快地编写更好的应用程序。JRebel可即时重新加载对代码的修改,无需重启或重新部署应用程序,就
    的头像 发表于 04-27 13:44 648次阅读
    <b class='flag-5'>Java</b>开发者必备的效率工具——Perforce JRebel是什么?为什么很多<b class='flag-5'>Java</b>开发者在用?

    RAKsmart企业服务器上部署DeepSeek编写运行代码

    在RAKsmart企业服务器上部署并运行DeepSeek模型的代码示例和详细步骤。假设使用 Python + Transformers库 + FastAPI实现一个基础的AI服务。主机推荐小编为您整理发布RAKsmart企业服务器上部署DeepSeek编写运行
    的头像 发表于 03-25 10:39 539次阅读

    零基础入门:如何在树莓派上编写和运行Python程序?

    是一种非常有用的编程语言,其语法易于阅读,允许程序员使用比汇编、C或Java等语言更少的代码行。Python编程语言最初实际上是作为Linux的脚本语言而开发的。Py
    的头像 发表于 03-25 09:27 1523次阅读
    零基础入门:如何在树莓派上<b class='flag-5'>编写</b>和运行Python程序?

    Java的SPI机制详解

    接口规范以及可以发现接口实现的机制,而不需要实现接口。 SPI机制在Java中应用广泛。例如:JDBC中的数据库连接驱动使用SPI机制,只定义了数据库连接接口的规范,而具体实现由各大数据库厂商实现,不同数据库的实现不同,我们常用
    的头像 发表于 03-05 11:35 1111次阅读
    <b class='flag-5'>Java</b>的SPI机制详解

    如何给ADS1247发送指令,才能让它开始转换温度?

    最新在使用ADS1247测量PT100,MCU51模拟SPI访问ADS1247,可以读写所有的寄存器。但是无法测量温度,DOUT交永远是低电平,而且差分电压也总是0.000V。想请问,应当如何给ADS1247发送指令,才能让它开始转换温度。
    发表于 01-22 06:25

    EE-134:为SHARC系列编写C兼容汇编代码中断处理程序

    电子发烧友网站提供《EE-134:为SHARC系列编写C兼容汇编代码中断处理程序.pdf》资料免费下载
    发表于 01-07 13:58 0次下载
    EE-134:为SHARC系列<b class='flag-5'>编写</b>C兼容汇编<b class='flag-5'>代码</b>中断处理程序

    STM32CubeMX生成的代码,是怎样的HAL架构?

    需要注意,CubeMX只是一个初始化工具,它只是帮你完成了HAL库外设和时钟的初始,具体程序的编写还需要查阅HAL库的说明,使用相关函数实现你想实现的功能。CubeMX生成的代码包含了丰富的注释
    的头像 发表于 01-01 21:01 1233次阅读
    STM32CubeMX生成的<b class='flag-5'>代码</b>,是怎样的HAL架构?