JAVA JDK1.5-1.9新特性
1.自动装箱与拆箱:
2.枚举(常用来设计单例模式)
3.静态导入
4.可变参数
5.内省
1.6
1.Web服务元数据
2.脚本语言支持
3.JTable的排序和过滤
4.更简单,更强大的JAX-WS
5.轻量级Http Server
6.嵌入式数据库 Derby
1.7
1,switch中可以使用字串了
2.运用List tempList = new ArrayList<>(); 即泛型实例化类型自动推断
3.语法上支持集合,而不一定是数组
4.新增一些取环境信息的工具方法
5.Boolean类型反转,空指针安全,参与位运算
6.两个char间的equals
7.安全的加减乘除
8.map集合支持并发请求,且可以写成 Map map = {name:"xxx",age:18};
1.8
1. 允许在接口中有默认方法实现
2. Lambda表达式
3. 函数式接口
4. 方法和构造函数引用
5. Lambda的范围
6. 内置函数式接口
7. Streams
8. Parallel Streams
9. Map
10. 时间日期API
11. Annotations
1.9
1. Jigsaw 项目;模块化源码
2. 简化进程API
3. 轻量级 JSON API
4. 钱和货币的API
5. 改善锁争用机制
6. 代码分段缓存
7. 智能Java编译, 第二阶段
8. HTTP 2.0客户端
9. Kulla计划: Java的REPL实现
---------------------------------------------------------------------------
JDK1.5新特性:
1自动装箱与拆箱
Integer
System.out.println(iObj
枚举
public
public
}
public
}
public
}
}
3 静态导入
import
public
public
}
}
可变参数
public
public
}
public
}
}
内省
ReflectPoint
BeanInfo
--------------------------------------
jdk1.6新特性:
1.Web服务元数据
Java
package
import
import
import
import
import
import
import
import
@WebService(targetNamespace="http://blog.csdn.net/chinajash",serviceName="HelloService")
public
}
如果想看到Web
1.将/bin加入path环境变量
2.在命令行下切换当前目录到WSProvider的class文件所在的目录,运行下面命令
wsgen
在这个例子中会生成以下3个类的源代码文件及class文件
SayHi
SayHiResponse
PrintTime
3.执行如下代码发布WSProvider到http://localhost:8888/chinajash/WSProvider,在这里可以执行WSProvider类的main方法就可以
Endpoint.publish("http://localhost:8888/chinajash/WSProvider",new
4.在浏览器输入http://localhost:8888/chinajash/WSProvider?wsdl就可以看到生成的WSDL文件,为了节省篇幅,这里就不把生成的WSDL文件贴上了,大家可以自己动手试试.
2.脚本语言支持
JDK6增加了对脚本语言的支持(JSR
1、许多脚本语言都有动态特性,比如,你不需要用一个变量之前先声明它,你可以用一个变量存放完全不同类型的对象,你不需要做强制类型转换,因为转换都是自动的。现在Java语言也可以通过对脚本语言的支持间接获得这种灵活性。
2、
3、通过引入脚本语言可以轻松实现Java应用程序的扩展和自定义,我们可以把原来分布在在Java应用程序中的配置逻辑,数学表达式和业务规则提取出来,转用JavaScript来处理。
Sun的JDK6实现包含了一个基于Mozilla
Scripting
Scripting
1、创建一个ScriptEngineManager对象
2、通过ScriptEngineManager获得ScriptEngine对象
3、用ScriptEngine的eval方法执行脚本
下面是一个Hello
public
package
import
import
import
import
public
--------------------------------------------------------------------------------
SUN 提供的JDK6中有一个命令行工具??jrunscript,你可以在/bin下面找到这个工 具,jrunscript是一个脚本语言的解释程序,它独立于脚本语言,但默认是用JavaScript,我们可以用jrunscript来测试自己写的 脚本语言是否正确,下面是一个在命令行运行jrunscript的简单例子
jrunscript
js>println("Hello,JrunScript");
Hello,JrunScript
js>9*8
72.0
js>
3.JTable的排序和过滤
原来的JTable基本上是只能显示数据,在JDK6新增了对JTable的排序和过滤功能,下面代码演示了这两个功能
public
}
运行上面程序,单击JTable的某一个title,这个title对应的列就会按照升序/降序重新排列;在下面的Criteria文本框中输入"ese",点击"Do
4.更简单,更强大的JAX-WS
JAX-WS2.0的来历
--------------------------------------------------------------------------------
JAX-WS(JSR-224)
JAX-WS2.0的架构
--------------------------------------------------------------------------------
JAX-WS不是一个孤立的框架,它依赖于众多其他的规范,本质上它由以下几部分组成
1.用来开发Web
2.用来处理Marshal/Unmarshal的XML
3.众多元数据(Annotations)会被JAX-WS用来描述Web
4.Annotation
5.JAX-WS还包括JAX-WS
JAX-WS2.0的编程模型
--------------------------------------------------------------------------------
现在用JAX-WS2.0来编写Web
1.编写服务类,并用Web
@WebService(targetNamespace="http://blog.csdn.net/chinajash",serviceName="HelloService")
public
}
2.用wsgen生成上面服务类的必要的帮助类,然后调用用EndPoint类的静态方法publish发布服务类(步骤请参考我的另一篇Blog
3.用wsimport为服务消费者(也就是服务的客户端)生成必要的帮助类,命令如下:
wsimport
这会在<当前目录>\net\csdn\blog\chinajash下生成客户端的帮助类,在这个例子中会生成7个类
HelloService.class
ObjectFactory.class
package-info.class
PrintSystemTime.class
SayHi.class
SayHiResponse.class
WSProvider.class
4.在客户端用下面代码即可调用步骤1定义的Web
HelloService
WSProvider
System.out.println(ws.sayHi("chinajash"));
ws.printSystemTime();
调用上述代码后客户端控制台输出
hi,chinajash
服务端控制台输出服务器当前系统时间
5.轻量级Http
JDK6的新特性之五:轻量级Http
JDK6提供了一个简单的Http
public
}
class
Happy New Year 2007!--Chinajash
";
}
运行程序后,在浏览器内输入 http://localhost:8888/xx ,浏览器输出
6. 嵌入式数据库
Derby是IBM送给开源社区的又一个礼物,是一个pure
不知道对于大数据量的性能如何,但传说中启动derby只会给JVM添加2M的内存,对那些小数据库应用,比如像用access那种应该是挺有诱惑力的。
另外,麻雀虽小,五脏俱全,功能要比access多得多咯,包括事务处理,并发,触发器都有,管理又简单,因此自己用来做点工具正好合适。
废话少说,介绍一下我折腾了半天的经验吧。
我的Derby配置过程:
1,下载db-derby-10.1.3.1-bin.tar.gz,derby_core_plugin_10.1.3.zip和derby_ui_plugin_1.1.0.zip,把两个插件安装到eclipse上
2,打开ecllipse,新建一个project
3,右键这个project,选择Apache
4,右键Project,在apache
5,右键Project,在apache
6,输入connect
7,用标准的SQL语句来建一个数据库试试:
create
居然可以用,太神奇了,呵呵
8,再插入一条语句试试呢,insert
9,再插一下:insert
10,好了,现在可以像你控制的其他数据库一样来控制Derby了。
如果上述方法不行,或者你习惯了在eclipse之外使用和管理数据库,那么可以很方便的把Derby“装”在系统里。下面我说一下步骤:
1,把db-derby-10.1.3.1-bin.tar.gz解压到c:\derby,使lib和framework两个目录在c:\derby下边即可
2,设置环境变量
设置一个c:\derby\framework\embeded\bin或c:\derby\framework\NetworkServe\bin到Path中,这样我们就可以直接执行上边介绍的connect这样的命令而不用每次钻到那个目录下去执行了
设置c:\derby\lib\derby.jar;c:\derby\lib\derbytoos.jar到CLASSPATH中,以便让这些java编成的命令能够正确执行;
3,打开cmd
4,敲入startNetworkServer,可以看到像在eclisp中提示的那样启动了server
5,再打开一个cmd,敲入sysinfo,可以看到derby的环境信息了,注意在java
6,敲入ij,好了,进入到上边的交互界面,可以建一个数据库看看了。
7,最后在另外一个cmd中敲入stopNetworkServer就可以关闭数据库了。
如果你两种方法都试过了,那么需要注意的,还是上边步骤5的问题,这个问题是你可能随时会启动一个数据库或新建一个数据库,但如果你刚刚使用derby,你可能还没有察觉。
derby实际上有两种启动方式,一种是嵌入式的,一种是网络服务器的启动。
1,我们在eclipse中右键start
connect
的方式进行链接。
2,第二种启动方式是在ij里边就直接
connect
这实际是在连当前配置环境下java
看到这里可能有点糊涂了,这么就会出问题了那?
实际上derby的访问更像是一种使用derby
另外一个问题是,在eclipse中和在系统中连接服务器,在connect的时候这个当前配置环境是不一样的,eclipse默认工程所在路径是数据库的所在路径,而在系统中“装”derby则会认为
jdk1.7新特性:
1,switch中可以使用字串了
String
switch
case
case
default
}
2.运用List
3.语法上支持集合,而不一定是数组
final
4.新增一些取环境信息的工具方法
File
File
File
File
5.Boolean类型反转,空指针安全,参与位运算
Boolean
True
boolean
boolean
boolean
boolean
boolean
boolean
6.两个char间的equals
boolean
7.安全的加减乘除
int
int
long
long
int
long
long
long
int
long
long
int
8.map集合支持并发请求,且可以写成
--------------------------------------------------------------------------
java1.8新特性
允许在接口中有默认方法实现
- interface
Formula { -
double calculate(int a); -
-
default double sqrt(int a) { -
return Math.sqrt(a); -
} - }
在接口Formula中,除了抽象方法caculate以外,还定义了一个默认方法sqrt。Formula的实现类只需要实现抽象方法caculate就可以了。默认方法sqrt可以直接使用。
- Formula
formula = new Formula() { -
@Override -
public double calculate(int a) { -
return sqrt(a * 100); -
} - };
-
- formula.calculate(100);
// 100.0 - formula.sqrt(16);
// 4.0
formula对象以匿名对象的形式实现了Formula接口。代码很啰嗦:用了6行代码才实现了一个简单的计算功能:a*100开平方根。我们在下一节会看到,Java 8 还有一种更加优美的方法,能够实现包含单个函数的对象。
Lambda表达式
- List
names = Arrays.asList("peter", "anna", "mike", "xenia"); -
- Collections.sort(names,
new Comparator() { -
@Override -
public int compare(String a, String b) { -
return b.compareTo(a); -
} - });
静态工具方法Collections.sort接受一个list,和一个Comparator接口作为输入参数,Comparator的实现类可 以对输入的list中的元素进行比较。通常情况下,你可以直接用创建匿名Comparator对象,并把它作为参数传递给sort方法。
除了创建匿名对象以外,Java 8 还提供了一种更简洁的方式,Lambda表达式。
- Collections.sort(names,
(String a, String b) -> { -
return b.compareTo(a); - });
你可以看到,这段代码就比之前的更加简短和易读。但是,它还可以更加简短:
- Collections.sort(names,
(String a, String b) -> b.compareTo(a));
只要一行代码,包含了方法体。你甚至可以连大括号对{}和return关键字都省略不要。不过这还不是最短的写法:
- Collections.sort(names,
(a, b) -> b.compareTo(a));
Java编译器能够自动识别参数的类型,所以你就可以省略掉类型不写。让我们再深入地研究一下lambda表达式的威力吧。
函数式接口
任意只包含一个抽象方法的接口,我们都可以用来做成lambda表达式。为了让你定义的接口满足要求,你应当在接口前加上@FunctionalInterface 标注。编译器会注意到这个标注,如果你的接口中定义了第二个抽象方法的话,编译器会抛出异常。
举例:
- @FunctionalInterface
- interface
Converter { -
T convert(F from); - }
-
- Converter
converter = (from) -> Integer.valueOf(from); - Integer
converted = converter.convert("123"); - System.out.println(converted);
// 123
注意,如果你不写@FunctionalInterface 标注,程序也是正确的。
方法和构造函数引用
- Converter
converter = Integer::valueOf; - Integer
converted = converter.convert("123"); - System.out.println(converted);
// 123
Java 8 允许你通过::关键字获取方法或者构造函数的的引用。上面的例子就演示了如何引用一个静态方法。而且,我们还可以对一个对象的方法进行引用:
- class
Something { -
String startsWith(String s) { -
return String.valueOf(s.charAt(0)); -
} - }
-
- Something
something = new Something(); - Converter
converter = something::startsWith; - String
converted = converter.convert("Java"); - System.out.println(converted);
// "J"
让我们看看如何使用::关键字引用构造函数。首先我们定义一个示例bean,包含不同的构造方法:
- class
Person { -
String firstName; -
String lastName; -
-
Person() {} -
-
Person(String firstName, String lastName) { -
this.firstName = firstName; -
this.lastName = lastName; -
} - }
接下来,我们定义一个person工厂接口,用来创建新的person对象:
- interface
PersonFactoryextends Person> { -
P create(String firstName, String lastName); - }
然后我们通过构造函数引用来把所有东西拼到一起,而不是像以前一样,通过手动实现一个工厂来这么做。
- PersonFactory
personFactory = Person::new; - Person
person = personFactory.create("Peter", "Parker");
我们通过Person::new来创建一个Person类构造函数的引用。Java编译器会自动地选择合适的构造函数来匹配PersonFactory.create函数的签名,并选择正确的构造函数形式。
Lambda的范围
访问局部变量
我们可以访问lambda表达式外部的final局部变量:
- final
int num = 1; - Converter
stringConverter = -
(from) -> String.valueOf(from + num); -
- stringConverter.convert(2);
// 3
但是与匿名对象不同的是,变量num并不需要一定是final。下面的代码依然是合法的:
- int
num = 1; - Converter
stringConverter = -
(from) -> String.valueOf(from + num); -
- stringConverter.convert(2);
// 3
然而,num在编译的时候被隐式地当做final变量来处理。下面的代码就不合法:
- int
num = 1; - Converter
stringConverter = -
(from) -> String.valueOf(from + num); - num
= 3;
在lambda表达式内部企图改变num的值也是不允许的。
访问成员变量和静态变量
与局部变量不同,我们在lambda表达式的内部能获取到对成员变量或静态变量的读写权。这种访问行为在匿名对象里是非常典型的。
- class
Lambda4 { -
static int outerStaticNum; -
int outerNum; -
-
void testScopes() { -
Converter stringConverter1 = (from) -> { -
outerNum = 23; -
return String.valueOf(from); -
}; -
-
Converter stringConverter2 = (from) -> { -
outerStaticNum = 72; -
return String.valueOf(from); -
}; -
} - }
访问默认接口方法
还记得第一节里面formula的那个例子么? 接口Formula定义了一个默认的方法sqrt,该方法能够访问formula所有的对象实例,包括匿名对象。这个对lambda表达式来讲则无效。
默认方法无法在lambda表达式内部被访问。因此下面的代码是无法通过编译的:
- Formula
formula = (a) -> sqrt( a * 100);
内置函数式接口
此外,Java 8 API 还提供了很多新的函数式接口,来降低程序员的工作负担。有些新的接口已经在 Google Guava库中很有名了。如果你对这些库很熟的话,你甚至闭上眼睛都能够想到,这些接口在类库的实现过程中起了多么大的作用。
Predicates
Predicate是一个布尔类型的函数,该函数只有一个输入参数。Predicate接口包含了多种默认方法,用于处理复杂的逻辑动词(and, or,negate):
- Predicate
predicate = (s) -> s.length() > 0; -
- predicate.test("foo");
// true - predicate.negate().test("foo");
// false -
- Predicate
nonNull = Objects::nonNull; - Predicate
isNull = Objects::isNull; -
- Predicate
isEmpty = String::isEmpty; - Predicate
isNotEmpty = isEmpty.negate();
Functions
Function接口接收一个参数,并返回单一的结果。默认方法可以将多个函数串在一起(compse, andThen):
- Function
toInteger = Integer::valueOf; - Function
backToString = toInteger.andThen(String::valueOf); -
- backToString.apply("123");
// "123"
Suppliers
Supplier接口产生一个给定类型的结果。与Function不同的是,Supplier没有输入参数。
- Supplier
personSupplier = Person::new; - personSupplier.get();
// new Person
Consumers
Consumer代表了在一个输入参数上需要进行的操作。
- Consumer
greeter = (p) -> System.out.println("Hello, " + p.firstName); - greeter.accept(new
Person("Luke", "Skywalker"));
Comparators
Comparator接口在早期的Java版本中非常著名。Java 8 为这个接口添加了不同的默认方法。
- Comparator
comparator = (p1, p2) -> p1.firstName.compareTo(p2.firstName); -
- Person
p1 = new Person("John", "Doe"); - Person
p2 = new Person("Alice", "Wonderland"); -
- comparator.compare(p1,
p2); // > 0 - comparator.reversed().compare(p1,
p2); // < 0
Optionals
Optional不是一个函数式接口,而是一个精巧的工具接口,用来防止NullPointerEception产生。这个概念在下一节会显得很重要,所以我们在这里快速地浏览一下Optional的工作原理。
Optional是一个简单的值容器,这个值可以是null,也可以是non-null。考虑到一个方法可能会返回一个non-null的值,也可能返回一个空值。为了不直接返回null,我们在Java 8中就返回一个Optional。
- Optional
optional = Optional.of("bam"); -
- optional.isPresent();
// true - optional.get();
// "bam" - optional.orElse("fallback");
// "bam" -
- optional.ifPresent((s)
-> System.out.println(s.charAt(0))); // "b"
Streams
我们先了解一下序列流。首先,我们通过string类型的list的形式创建示例数据:
- List
stringCollection = new ArrayList<>(); - stringCollection.add("ddd2");
- stringCollection.add("aaa2");
- stringCollection.add("bbb1");
- stringCollection.add("aaa1");
- stringCollection.add("bbb3");
- stringCollection.add("ccc");
- stringCollection.add("bbb2");
- stringCollection.add("ddd1");
Java 8中的Collections类的功能已经有所增强,你可以之直接通过调用Collections.stream()或者Collection.parallelStream()方法来创建一个流对象。下面的章节会解释这个最常用的操作。
Filter
Filter接受一个predicate接口类型的变量,并将所有流对象中的元素进行过滤。该操作是一个中间操作,因此它允许我们在返回结果的基 础上再进行其他的流操作(forEach)。ForEach接受一个function接口类型的变量,用来执行对每一个元素的操作。ForEach是一个 中止操作。它不返回流,所以我们不能再调用其他的流操作。
- stringCollection
-
.stream() -
.filter((s) -> s.startsWith("a")) -
.forEach(System.out::println); -
- //
"aaa2", "aaa1"
Sorted
Sorted是一个中间操作,能够返回一个排过序的流对象的视图。流对象中的元素会默认按照自然顺序进行排序,除非你自己指定一个Comparator接口来改变排序规则。
- stringCollection
-
.stream() -
.sorted() -
.filter((s) -> s.startsWith("a")) -
.forEach(System.out::println); -
- //
"aaa1", "aaa2"
一定要记住,sorted只是创建一个流对象排序的视图,而不会改变原来集合中元素的顺序。原来string集合中的元素顺序是没有改变的。
- System.out.println(stringCollection);
- //
ddd2, aaa2, bbb1, aaa1, bbb3, ccc, bbb2, ddd1
Map
map是一个对于流对象的中间操作,通过给定的方法,它能够把流对象中的每一个元素对应到另外一个对象上。下面的例子就演示了如何把每个 string都转换成大写的string. 不但如此,你还可以把每一种对象映射成为其他类型。对于带泛型结果的流对象,具体的类型还要由传递给map的泛型方法来决定。
- stringCollection
-
.stream() -
.map(String::toUpperCase) -
.sorted((a, b) -> b.compareTo(a)) -
.forEach(System.out::println); -
- //
"DDD2", "DDD1", "CCC", "BBB3", "BBB2", "AAA2", "AAA1"
Match
匹配操作有多种不同的类型,都是用来判断某一种规则是否与流对象相互吻合的。所有的匹配操作都是终结操作,只返回一个boolean类型的结果。
- boolean
anyStartsWithA = -
stringCollection -
.stream() -
.anyMatch((s) -> s.startsWith("a")); -
- System.out.println(anyStartsWithA);
// true -
- boolean
allStartsWithA = -
stringCollection -
.stream() -
.allMatch((s) -> s.startsWith("a")); -
- System.out.println(allStartsWithA);
// false -
- boolean
noneStartsWithZ = -
stringCollection -
.stream() -
.noneMatch((s) -> s.startsWith("z")); -
- System.out.println(noneStartsWithZ);
// true
Count
Count是一个终结操作,它的作用是返回一个数值,用来标识当前流对象中包含的元素数量。
- long
startsWithB = -
stringCollection -
.stream() -
.filter((s) -> s.startsWith("b")) -
.count(); -
- System.out.println(startsWithB);
// 3
Reduce
该操作是一个终结操作,它能够通过某一个方法,对元素进行削减操作。该操作的结果会放在一个Optional变量里返回。
- Optional
reduced = -
stringCollection -
.stream() -
.sorted() -
.reduce((s1, s2) -> s1 + "#" + s2); -
- reduced.ifPresent(System.out::println);
- //
"aaa1#aaa2#bbb1#bbb2#bbb3#ccc#ddd1#ddd2"
Parallel Streams
下面的例子就演示了如何使用并行流进行操作来提高运行效率,代码非常简单。
首先我们创建一个大的list,里面的元素都是唯一的:
- int
max = 1000000; - List
values = new ArrayList<>(max); - for
(int i = 0; i < max; i++) { -
UUID uuid = UUID.randomUUID(); -
values.add(uuid.toString()); - }
现在,我们测量一下对这个集合进行排序所使用的时间。
顺序排序
- long
t0 = System.nanoTime(); -
- long
count = values.stream().sorted().count(); - System.out.println(count);
-
- long
t1 = System.nanoTime(); -
- long
millis = TimeUnit.NANOSECONDS.toMillis(t1 - t0); - System.out.println(String.format("sequential
sort took: %d ms", millis)); -
- //
sequential sort took: 899 ms
并行排序
- long
t0 = System.nanoTime(); -
- long
count = values.parallelStream().sorted().count(); - System.out.println(count);
-
- long
t1 = System.nanoTime(); -
- long
millis = TimeUnit.NANOSECONDS.toMillis(t1 - t0); - System.out.println(String.format("parallel
sort took: %d ms", millis)); -
- //
parallel sort took: 472 ms
如你所见,所有的代码段几乎都相同,唯一的不同就是把stream()改成了parallelStream(), 结果并行排序快了50%。
Map
- Map
map = new HashMap<>(); -
- for
(int i = 0; i < 10; i++) { -
map.putIfAbsent(i, "val" + i); - }
-
- map.forEach((id,
val) -> System.out.println(val));
上面的代码风格是完全自解释的:putIfAbsent避免我们将null写入;forEach接受一个消费者对象,从而将操作实施到每一个map中的值上。
下面的这个例子展示了如何使用函数来计算map的编码:
- map.computeIfPresent(3,
(num, val) -> val + num); - map.get(3);
// val33 -
- map.computeIfPresent(9,
(num, val) -> null); - map.containsKey(9);
// false -
- map.computeIfAbsent(23,
num -> "val" + num); - map.containsKey(23);
// true -
- map.computeIfAbsent(3,
num -> "bam"); - map.get(3);
// val33
接下来,我们将学习,当给定一个key值时,如何把一个实例从对应的key中移除:
- map.remove(3,
"val3"); - map.get(3);
// val33 -
- map.remove(3,
"val33"); - map.get(3);
// null
另一个有用的方法:
- map.getOrDefault(42,
"not found"); // not found
将map中的实例合并也是非常容易的:
- map.merge(9,
"val9", (value, newValue) -> value.concat(newValue)); - map.get(9);
// val9 -
- map.merge(9,
"concat", (value, newValue) -> value.concat(newValue)); - map.get(9);
// val9concat
合并操作先看map中是否没有特定的key/value存在,如果是,则把key/value存入map,否则merging函数就会被调用,对现有的数值进行修改。
时间日期API
Clock
Clock提供了对当前时间和日期的访问功能。Clock是对当前时区敏感的,并可用于替代 System.currentTimeMillis()方法来获取当前的毫秒时间。当前时间线上的时刻可以用Instance类来表示。Instance 也能够用于创建原先的java.util.Date对象。
- Clock
clock = Clock.systemDefaultZone(); - long
millis = clock.millis(); -
- Instant
instant = clock.instant(); - Date
legacyDate = Date.from(instant); // legacy java.util.Date
Timezones
时区类可以用一个ZoneId来表示。时区类的对象可以通过静态工厂方法方便地获取。时区类还定义了一个偏移量,用来在当前时刻或某时间与目标时区时间之间进行转换。
- System.out.println(ZoneId.getAvailableZoneIds());
- //
prints all available timezone ids -
- ZoneId
zone1 = ZoneId.of("Europe/Berlin"); - ZoneId
zone2 = ZoneId.of("Brazil/East"); - System.out.println(zone1.getRules());
- System.out.println(zone2.getRules());
-
- //
ZoneRules[currentStandardOffset=+01:00] - //
ZoneRules[currentStandardOffset=-03:00]
LocalTime
本地时间类表示一个没有指定时区的时间,例如,10 p.m.或者17:30:15,下面的例子会用上面的例子定义的时区创建两个本地时间对象。然后我们会比较两个时间,并计算它们之间的小时和分钟的不同。
- LocalTime
now1 = LocalTime.now(zone1); - LocalTime
now2 = LocalTime.now(zone2); -
- System.out.println(now1.isBefore(now2));
// false -
- long
hoursBetween = ChronoUnit.HOURS.between(now1, now2); - long
minutesBetween = ChronoUnit.MINUTES.between(now1, now2); -
- System.out.println(hoursBetween);
// -3 - System.out.println(minutesBetween);
// -239
LocalTime是由多个工厂方法组成,其目的是为了简化对时间对象实例的创建和操作,包括对时间字符串进行解析的操作。
- LocalTime
late = LocalTime.of(23, 59, 59); - System.out.println(late);
// 23:59:59 -
- DateTimeFormatter
germanFormatter = -
DateTimeFormatter -
.ofLocalizedTime(FormatStyle.SHORT) -
.withLocale(Locale.GERMAN); -
- LocalTime
leetTime = LocalTime.parse("13:37", germanFormatter); - System.out.println(leetTime);
// 13:37
LocalDate
本地时间表示了一个独一无二的时间,例如:2014-03-11。这个时间是不可变的,与LocalTime是同源的。下面的例子演示了如何通过加减日,月,年等指标来计算新的日期。记住,每一次操作都会返回一个新的时间对象。
- LocalDate
today = LocalDate.now(); - LocalDate
tomorrow = today.plus(1, ChronoUnit.DAYS); - LocalDate
yesterday = tomorrow.minusDays(2); -
- LocalDate
independenceDay = LocalDate.of(2014, Month.JULY, 4); - DayOfWeek
dayOfWeek = independenceDay.getDayOfWeek(); - System.out.println(dayOfWeek);
// FRIDAYParsing a LocalDate from a string is just as simple as parsing a LocalTime:
解析字符串并形成LocalDate对象,这个操作和解析LocalTime一样简单。
- DateTimeFormatter
germanFormatter = -
DateTimeFormatter -
.ofLocalizedDate(FormatStyle.MEDIUM) -
.withLocale(Locale.GERMAN); -
- LocalDate
xmas = LocalDate.parse("24.12.2014", germanFormatter); - System.out.println(xmas);
// 2014-12-24
LocalDateTime
LocalDateTime表示的是日期-时间。它将刚才介绍的日期对象和时间对象结合起来,形成了一个对象实例。LocalDateTime是不可变的,与LocalTime和LocalDate的工作原理相同。我们可以通过调用方法来获取日期时间对象中特定的数据域。
- LocalDateTime
sylvester = LocalDateTime.of(2014, Month.DECEMBER, 31, 23, 59, 59); -
- DayOfWeek
dayOfWeek = sylvester.getDayOfWeek(); - System.out.println(dayOfWeek);
// WEDNESDAY -
- Month
month = sylvester.getMonth(); - System.out.println(month);
// DECEMBER -
- long
minuteOfDay = sylvester.getLong(ChronoField.MINUTE_OF_DAY); - System.out.println(minuteOfDay);
// 1439
如果再加上的时区信息,LocalDateTime能够被转换成Instance实例。Instance能够被转换成以前的java.util.Date对象。
- Instant
instant = sylvester -
.atZone(ZoneId.systemDefault()) -
.toInstant(); -
- Date
legacyDate = Date.from(instant); - System.out.println(legacyDate);
// Wed Dec 31 23:59:59 CET 2014
格式化日期-时间对象就和格式化日期对象或者时间对象一样。除了使用预定义的格式以外,我们还可以创建自定义的格式化对象,然后匹配我们自定义的格式。
- DateTimeFormatter
formatter = -
DateTimeFormatter -
.ofPattern("MMM dd, yyyy - HH:mm"); -
- LocalDateTime
parsed = LocalDateTime.parse("Nov 03, 2014 - 07:13", formatter); - String
string = formatter.format(parsed); - System.out.println(string);
// Nov 03, 2014 - 07:13
不同于java.text.NumberFormat,新的DateTimeFormatter类是不可变的,也是线程安全的。
更多的细节,请看 这里
Annotations
首先,我们定义一个包装 注解,它包括了一个实际 注解的数组
- @interface
Hints { -
Hint[] value(); - }
-
- @Repeatable(Hints.class)
- @interface
Hint { -
String value(); - }
只要在前面加上 注解名:@Repeatable,Java 8 允许我们对同一类型使用多重 注解:
变体1:使用 注解容器(老方法):
- @Hints({@Hint("hint1"),
@Hint("hint2")}) - class
Person {}
变体2:使用可重复 注解(新方法):
- @Hint("hint1")
- @Hint("hint2")
- class
Person {}
使用变体2,Java编译器能够在内部自动对@Hint进行设置。这对于通过反射来读取 注解信息来说,是非常重要的。
- Hint
hint = Person.class.getAnnotation(Hint.class); - System.out.println(hint);
// null -
- Hints
hints1 = Person.class.getAnnotation(Hints.class); - System.out.println(hints1.value().length);
// 2 -
- Hint[]
hints2 = Person.class.getAnnotationsByType(Hint.class); - System.out.println(hints2.length);
// 2
尽管我们绝对不会在Person类上声明@Hints 注解,但是它的信息仍然可以通过getAnnotation(Hints.class)来读 取。并且,getAnnotationsByType方法会更方便,因为它赋予了所有@Hints 注解标注的方法直接的访问权限。
- @Target({ElementType.TYPE_PARAMETER,
ElementType.TYPE_USE}) - @interface
MyAnnotation {}
总结
我希望这个博文能够对您有所帮助,也希望您阅读愉快。完整的教程源代码放在了 GitHub上。您可以尽情地 fork,并请通过 Twitter告诉我您的反馈。
原文链接: winterbe 翻译: ImportNew.com - 黄小非
译文链接: http://www.importnew.com/10360.html
这些重要的特性都包括在Jigsaw项目中。显著的性能改善和期待已久的API包括:进程API更新,JSON将成为java.util的一部分,货币处理API对于想处在技术最前沿的你,可从这里获得Java 9的初期版本。
被接受的特性
1. Jigsaw 项目;模块化源码
Jigsaw项目是为了模块化Java代码、将JRE分成可相互协作的组件,这也是Java 9 众多特色种的一个。JEP是迈向Jigsaw四步中的第一步,它不会改变JRE和JDK的真实结构。JEP是为了模块化JDK源代码,让编译系统能够模块 编译并在构建时检查模块边界。这个项目原本是随Java 8发布的,但由于推迟,所以将把它加到Java 9.
一旦它完成,它可能允许根据一个项目需求自定义组件从而减少rt.jar的大小。在JDK 7 和JDK 8的rt.jar包中有大约20,000个类,但有很多类在一些特定的环境里面并没有被用到(即使在Java 8的紧凑分布特性中已经包含了一部分解决方法也存在着类冗余)。这么做是为了能让Java能够容易应用到小型计算设备(比如网络设备)中,提高它的安全和 性能,同时也能让开发者更容易构建和维护这些类库。
2. 简化进程API
截止到目前,Java控制与管理系统进程的能力是有限的。举个例子,现在为了简便获取你程序的进程PID,你要么调用本地程序要么要自己使用一些变通方案。更多的是,每个(系统)平台需要有一个不同实现来确保你能获得正确的结果。
期望代码能获取Linux PIDS,现在是如下方式:
在Java 9中,可以变换成如下方式(支持所有的操作系统):
这次更新将会扩展Java与操作系统的交互能力:新增一些新的直接明了的方法去处理PIDs,进程名字和状态以及枚举多个JVM和进程以及更多事情。
3. 轻量级 JSON API
目前有多种处理JSON的Java工具,但JSON API 独到之处在于JSON API将作为Java语言的一部分,轻量并且运用Java 8的新特性。它将放在java.util包里一起发布(但在JSR 353里面的JSON是用第三方包或者其他的方法处理的).
4. 钱和货币的API
在Java 8引进了日期和时间的API之后, Java 9引入了新的货币API, 用以表示货币, 支持币种之间的转换和各种复杂运算. 关于这个项目的具体情况, 请访问https://github.com/JavaMoney,里面已经给出了使用说明和示例, 以下是几个重要的例子:
5. 改善锁争用机制
锁争用是限制许多Java多线程应用性能的瓶颈. 新的机制在改善Java对象监视器的性能方面已经得到了多种基准(benchmark)的验证, 其中包括Volano. 测试中通讯服务器开放了海量的进程来连接客户端, 其中有很多连接都申请同一个资源, 以此模拟重负荷日常应用.
通过诸如此类的压力测试我们可以估算JVM的极限吞吐量(每秒的消息数量). JEP在22种不同的测试中都得到了出色的成绩, 新的机制如果能在Java 9中得到应用的话, 应用程序的性能将会大大提升.
6. 代码分段缓存
Java 9的另一个性能提升来自于JIT(Just-in-time)编译器. 当某段代码被大量重复执行的时候, 虚拟机会把这段代码编译成机器码(native code)并储存在代码缓存里面, 进而通过访问缓存中不同分段的代码来提升编译器的效率.
和原来的单一缓存区域不同的是, 新的代码缓存根据代码自身的生命周期而分为三种:
- 永驻代码(JVM 内置 / 非方法代码)
- 短期代码(仅在某些条件下适用的配置性(profiled)代码)
- 长期代码(非配置性代码)
缓存分段会在各个方面提升程序的性能, 比如做垃圾回收扫描的时候可以直接跳过非方法代码(永驻代码), 从而提升效率.
7. 智能Java编译, 第二阶段
智能Java编译工具sjavac的第一阶段开始于JEP 139这个项目, 用于在多核处理器上提升JDK的编译速度. 现在这个项目已经进入第二阶段(JEP 199), 目的是改进sjavac并让其成为取代目前JDK编译工具javac的Java默认的通用编译工具.
其他值得期待的内容:
8. HTTP 2.0客户端
HTTP 2.0标准虽然还没正式发布, 但是已经进入了最终审查阶段, 预计可以在Java 9发布之前审查完毕. JEP 110将会重新定义并实现一个全新的Java HTTP客户端, 用来取代现在的HttpURLConnection, 同时也会实现HTTP 2.0和网络接口(原文websockets). 它现在还没被JEP正式认可但我们希望在Java 9中包含这一项目的内容.
官方的HTTP 2.0 RFC(Request for Comments, 官方技术讨论/会议记录等等的一系列文档记录)预订于2015年2月发布, 它是基于Google发布的SPDY(Speedy, 快速的)协议. 基于SPDY协议的网络相对于基于HTTP 1.1协议的网络有11.81%到47.7%之间的显著提速, 现在已经有浏览器实现了这个协议.
9. Kulla计划: Java的REPL实现
这个取名为Kulla的项目最近宣布将于2015年4月整合测试, 虽然已经不太有希望能赶上Java 9的发布, 但如果进度快的话或许刚好能赶上. 现在Java并没有来自官方的REPL(Read-Eval-Print-Loop)方式, 也就是说现在如果你想要跑几行Java代码做一个快速的测试, 你仍然需要把这几行代码封装在项目或者方法里面. 虽然在一些流行的IDE里面有Java REPL工具, 但它们并没有官方支持, 而Kulla项目或许就能成为Java官方发布的REPL解决方案.
这些新功能出自何处?
JEP和JSR并不是无中生有,下面就介绍一下Java发展的生态环境:
小组 - 对特定技术内容, 比如安全、网络、Swing、HotSpot、有共同兴趣的组织和个人。
项目 - 编写代码, 文档以及其他工作,至少由一个小组赞助和支持,比如最近的Lambda计划,Jigsaw计划和Sumatra计划。
JDK改进提案(JEP) - 每当需要有新的尝试的时候, JEP可以在JCP(Java Community Process)之前或者同时提出非正式的规范(specification),被正是认可的JEP正式写进JDK的发展路线图并分配版本号。
Java规范提案(JSR) - 新特性的规范出现在这一个阶段,可以来自于小组 / 项目、JEP、 JCP成员或者Java社区(community)成员的提案,每个Java版本都由相应的JSR支持, Java 9暂时还没有。