本文适合对 Java 或 C 有一些了解的用户阅读,推荐阅读时间15分钟。

导言

写这个系列的原因?

我曾经听过一种说法,如果不了解Liunx的网络通讯,就很难理解理解Java的IO;如果不知道Java的IO也很难理解之后的NIO,Netty。(理解是建立在对底层的理解之上的

一门语言的设计是一项工程,是需要向很多其他编程语言、操作系统学习,站在巨人的肩膀上进行设计的;好的设计,好的语法也不是一蹴而就的,需要对之前版本的优点和缺点进行评估,再进行新语法的设想。
所以,本系列想采用Java从设计之初一直发展到Java8相关的新特性来讲述语言为何如此设计,以及语言发展各个重大版本的变化这个角度进行解析如何编写好“风格”的代码。

注:Java已经发展了很多年,目前为止最新的LTS版本为Java11。由于Java8商用已经较为普及,所以采用此版本。

这一节,先从布尔,常量一些特性开始讲起。

一、布尔类型

布尔类型在语法上类似于语言内置的常量,所以也做一些简单的介绍。

在过去编写C语言的代码时,由于缺少布尔类型(C99加入了布尔类型!),默认任何 非零 的值(不为null)假定为 true,把 零 和 null 假定为 false

#include<stdio.h>
int main() {
/* 输出 if(null) is false */
if (null)
print("if(null) is true");
else
print("if(null) is false"); /* 输出 if(0) is false */
if (0)
print("if(0) is true");
else
print("if(0) is false"); /* 输出 if(1) is true */
if (1)
print("if(1) is true");
else
print("if(1) is false");
}

这在当时没有问题,但这种写法可读性差,对程序员不太友好。

在这点上,C++往前走了一步,添加了布尔类型bool,但由于要兼容C语言,也可以同时使用C的语法(条件表达式接收int值),打印bool值真假实际为1和0。

#include <iostream>
using namespace std; int main(){
bool flag = true;
if(flag)
cout<<"true"<<endl;
else
cout<<"false"<<endl; cout<<flag<<endl; flag = false;
if(flag)
cout<<"true"<<endl;
else
cout<<"false"<<endl; cout<<flag<<endl; return 0;
} /* 输出
true
1
false
0
*/

Java也沿袭下来,增加了boolean类型,但条件表达式不接受int值,在语法上int值和boolean彻底无关。

package com.xx;

public class Demo {

    public static void main(String[] args) {
boolean b1 = true;
System.out.println(b1);
boolean b2 = false;
System.out.println(b2); }
} /*输出
true
false
*/

从C、C++一直到Java,可以看到语言设计者在一步步让程序编写者编出更直观的、人类可读的代码。

二、魔术数字(magic number)

编程初学者碰到一些特殊变量最开始可能会直接采用硬编码的方式编写。这种编码被人们称之为魔术数字(原因在于作者在若干月后,甚至自己也不知道这个数字是什么意义)

//100表示成功,其他值表示失败
if (status == 100) {
//成功处理
}
else {
//失败处理
}
其中100就是一个典型的魔术数字

这种编码习惯有以下两个明显的缺点

1.数值的意义难以了解

2.数值涉及变动时,可能要改不只一处

后来的编程人员使用了常量来代替魔术数字。常量名力求见名知意(某人说,好的代码是自解释(self-explanatory)的)。

常量的加入给程序带来了这两个好处。

1.增加可读性。

2.方便修改。

//1.这样的常量通常用相同的前缀表示同一类型 USER_SEX,且定义位置在最开头
//2.从C沿袭至今的全大写,下划线分隔
public static final int USER_SEX_MALE = 1;
public static final int USER_SEX_FEMALE = 0;

但是,在越来越多的常量使用中,人们又发现了自定义的常量也有一些难以解决的问题

1.需编译后才能生效
2.存在类型转换的风险

p.s. 第二点正是和C++的布尔值bool有着同样的问题。

再后来,常量的编写方式越来越多,比如注解;

通过注解使用的常量值


//application.properties
# 默认使用dev的配置文件
spring.profiles.active=dev

//application-dev.properties
# foo在开发环境的默认次数
default.foo.numbers = 10
//application-test.properties
#foo在测试环境的默认次数
default.foo.numbers = 1000


//application-prod.properties
# foo在生产环境的默认次数
default.foo.numbers = 100
import org.springframework.beans.factory.annotation.Value;

@Service
public class FooService {
@Value("${default.foo.numbers}")
private Integer fooNumbers;
}

使用注解的好处在于用户可以定义不同的application-xxx.properties等多个配置文件。每个文件可以定义同名不同值的常量, 在启动项目时通过启动参数 -Dspring.profiles.active=XXX 来选择使用的配置文件,实现改变配置时改变常量值

比如 dev配置文件的default.foo.numbers设置为10, test配置文件值设置为1000,根据 启动项目调用 -Dspring.profiles.active=test 调用到的值就是1000。

Java设计之初并没有考虑枚举类型,而是沿用从C一直以来的传统使用全局int/string值表示常量。为了让用户更好地使用常量,从1.5之后引入了枚举类型

/**
* 1.自动私有构造函数
* 2.强类型,不会因为类型转换带来问题
* 3.可以直接使用 == 比较,和String常量需要equals()对比,效率高
* 4.无需编译生效
*/
public enum UserSex {
male, //男
female, //女
;
//error
// UserSex();
public static UserSex get(String str) {
for (UserSex t : UserSex.values()) {
if (t.name().equals(str))
return t;
}
return null;
}
}

目前为止博主的最佳实践

对于只有真假的值,在Java中推荐使用布尔类型表示,比如是否为会员(isVip: true, false)。

对于超过两种以上或今后可能会增加新的类型的推荐用枚举表示。比如会员类型(VipType: week, month, year),考虑之后可能会新加入quarter等时间。

对于不同服之间同一个常量值由于各种需要测试等原因不一致,可以在各个服的配置文件定义。

最新文章

  1. Codeforces Round #379 (Div. 2)
  2. 【转】ubuntu vpn自动切换路由
  3. 线程优先级抢占实验【RT-Thread学习笔记 3】
  4. hive数据库的一些应用
  5. JavaScript的一些认识
  6. [翻译]你不会想知道的kobject,kset,和ktypes
  7. V9自定义分页函数
  8. angular中的promise
  9. Android 手机上安装并运行 Ubuntu 12.04
  10. BZOJ 1738: [Usaco2005 mar]Ombrophobic Bovines 发抖的牛( floyd + 二分答案 + 最大流 )
  11. Middleware开发入门
  12. myeclipse中打开java文件中文乱码
  13. datatables 学习笔记1 基础篇
  14. Java 第八周总结
  15. Kafka 详解(三)------Producer生产者
  16. 使用python写的一个代码统计程序
  17. python3-基础2
  18. 在 Intellij IDEA 中部署 Java 应用到 阿里云 ECS
  19. SDUT 1269-走迷宫(DFS打印路径)
  20. java 方向术语

热门文章

  1. 介绍 golang json数据的处理
  2. 面试28k职位,老乡面试官从HashCode到HashMap给我讲了一下午!「回家赶忙整理出1.6万字的面试材料」
  3. 你可能不了解的java枚举
  4. 【Android】listview 嵌套gridview报错,代码:”during second layout pass: posting in next frame
  5. P2414 [NOI2011]阿狸的打字机 AC自动机
  6. 利用负margin-bottom去除列表最后一个li元素的border-bottom
  7. 【问题】【SpringBoot】记一次springboot框架下用jackson解析RequestBody失败的问题
  8. jQuery提供的Ajax方法
  9. centos7安装jdk11
  10. 关于Vue-Router的那些事儿