2019Java依然免费,你应该知道的事实,不要被误导—技术领导必读

2019年关于Java是否免费,一直存在误解,国内新闻断章取义,搞错概念,误导开发者。 作为企业的技术领导应该弄清楚关键的概念,避免被误导。

国内一直谣传Java2019年1月1号收费,今天都2号了,使用Java的企业运转一切正常。各种新闻标题耸人听闻,但是如果你也被吓到,那不妨看看本文调节一下心情。

相信许多java开发者也会被这种新闻或者文章困扰,老外也遇到过,自2001年担任美国加州圣地亚哥Java UG主席的Paul Webber也看过这种文章:

“在过去的六个月里阅读很多博客,看不下去,因为那里有许多关于Oracle许可和JDK发布周期的错误信息,”

“本次大会Mark Reinhold的演讲,这是一个可信的消息来源- 他可能是真相的来源 - 不要被误导。“

本文会讲解Oracle JDK、Open JDK、Oracle首席架构师声明、Amazon JDK Corretto、大批公司转Java、后期应对策略,给大家做个参考,尤其是公司的技术领导。

1、Java依然免费,国内新闻断章取义误导大众,混淆概念

首先搞清楚几个基本概念,国内很多文章作者是完全没有搞清楚概念的前提下,乱下结论,发文章误导开发者。

Oracle JDK 8 is undergoing the “End of Public Updates” process, which means there are no longer free updates for commercial use after January 2019.

Oracle plans to provide full paid support for Oracle JDK 8 until at least 2025 and Oracle JDK 11 until at least 2026 (details). There is a wide range of paid support options for Java SE / OpenJDK 8 and 11 binaries from Azul, IBM, Red Hat, and others.

声明的意思:Oracle 不在对JDK8的商业用途提供免费的公开的更新服务。看清楚第一行英文,断章取义的国内文章基本就翻译为Java收费。。。

国内新闻偷换概念的地方在于,做个类比,Redhat自己研发的企业版Linux收费,然后新闻标题偷换概念:Linux收费!偷换了概念。误导大家。

当然 Oracle投入大量资金,维护 Oracle JDK,收费无可非议,但是这些基本概念完全不同,不要被误导。

关于Java的几个概念,范围要界定一下。

Oracle收费!=Java收费,这是偷换概念。

Oracle JDK收取技术支持费用!=Java收费。

Oracle JDK != Java JDK。

2019Java依然免费,你应该知道的事实,不要被误导—技术领导必读
Java的JDK版本很多,包括Open JDK 和 Oracle JDK。

Open JDK又存在许多的分支。

Oracle JDK 8正在进行“公共更新结束”流程,这意味着2019年1月之后不再有商业用途的免费更新。

Amazon Corretto 是Java之父James Gosling 加盟了亚马逊之后,今年11月15号永久开源了自己的JDK,其他公司可以免费使用,无需付费。

当然还有其他的免费的JDK,以及对应的支持,我们下面详细说明一下。

但是,自Java SE 9以来,Oracle还提供Oracle的OpenJDK版本,可免费用于商业用途,其他公司提供免费的OpenJDK版本,如Amazon、谷歌、阿里巴巴、AdoptOpenJDK,Azul,IBM,Red Hat,Linux distros等。

Java规范由Java Community Process(JCP)委员会管理制定,并通过Java规范(JSR)体现。国内阿里巴巴是唯一的JCP委员公司,参与Java规范制定。

2、Oracle首席Java架构师Mark Reinhold:Java永远开源、免费

2019Java依然免费,你应该知道的事实,不要被误导—技术领导必读
2018年10月22日,Oracle首席Java架构师Mark Reinhold在Oracle Code One技术大会上演讲表示:Java仍然是开放和免费的。

Paul Webber ,美国加州圣地亚哥 Java Users Group 主席,2001年至今,说:

“I stopped reading a lot of blogs in the last six months because there was so much misinformation out there about Oracle licensing and the JDK cycle,” said Paul Webber, who has been president of the San Diego Java Users Group since 2001.

“This was a keynote from a responsible source—he is probably the source of truth—to get rid of the misconceptions. He knows his stuff, so that adds to the credibility of the presentation.”

3、Open JDK永久免费,包含太多开源、免费的分支

Open JDK是最大的开源、免费 JDK项目。官方网站: http://openjdk.java.net/

众所周知,如果只有Java语言是完全没有用的,企业开发需要许多底层的库,以及周边的框架,架构方案。这样才能完整地开发系统。

2019Java依然免费,你应该知道的事实,不要被误导—技术领导必读
需要掌握的一个关键点是,世界上大多数JDK构建都基于开源、免费的OpenJDK项目。

Oracle JDK只是基于OpenJDK代码库的众多分支之一。

虽然过去几年Oracle在JDK中有额外开发的扩展功能,但从Java 11开始,情况就不再如此。

使用Open JDK扩展的公司很多,国内著名的阿里巴巴,严格来说,BAT都有自己内部维护的Linux和Open JDK分支,为了防止可能的版权问题。

许多公司也提供基于OpenJDK代码库的构建分支。这些构建可能具有额外的品牌和/或其他非核心功能。这些公司中的大多数也开源回馈贡献给之前OpenJDK项目,包括安全补丁和更新包。

所以不要被 误导,绝大部分公司,都已经做好了准备。

此外,获得JDK有两种主要方式。第一个是更新机制buit到操作系统(例如linux redhat)。第二种自己下载二进制文件安装。

为了进一步研究这一点,让我们分别看一下Java 8和Java 11。

Java SE / OpenJDK / Oracle的OpenJDK构建/ Oracle JDK

OpenJDK社区创建并维护Java SE规范的(GPLv2 + CE),

来自各种大厂(例如Amazon Azul,Eclipse,IBM,Red Hat,Oracle,SAP等)的Java SE实现,其中最常见的是Oracle JDK。

Oracle JDK 8正在进行“公共更新结束”流程,这意味着2019年1月之后不再有商业用途的免费更新。

红色字体部分,也是国内新闻断章取义的关键点。

4、亚马逊JDK Corretto 永久免费,并且长期支持Java 11

2019Java依然免费,你应该知道的事实,不要被误导—技术领导必读
Java之父 James Gosling 加盟了亚马逊

Amazon Corretto是Open Java Development Kit(OpenJDK)的免费、开源的、多平台,生产就绪版本。 基于GPLv2 开源协议。

源码已经开放 https://github.com/corretto/corretto-8。

Corretto提供长期支持,包括性能增强和安全修复。 亚马逊在内部运行Corretto数千种生产服务,Corretto被认证为与Java SE标准兼容。 使用Corretto,您可以在流行的操作系统上开发和运行Java应用程序,包括Amazon Linux 2,Windows和macOS。 亚马逊Corretto 8是预览版,但是可以用于生产环境。

2019Java依然免费,你应该知道的事实,不要被误导—技术领导必读
Amazon Corretto 官方网站 https://aws.amazon.com/cn/corretto/

Amazon Corretto根据开源许可证GNU公共许可证版本2的条款获得许可,具有类路径异常(带有CPE的GPLv2)。 亚马逊不对其使用或分发收费。

5、Amazon Corretto的长期支持包括哪些内容?

Amazon在2023年6月之前为Corretto 8提供安全更新”,

至少在2024年8月之前支持Corretto 11进行季度更新”。

亚马逊计划在2019年上半年发布亚马逊Corretto 11.

Amazon Corretto 8和11的当前更新计划每季度进行一次,直到截止支持(2023年6月) 对于Corretto 8和824年8月为Corretto 11)。

6、使用JDK8或者9、10、11的选择方案?

2019年1月1日之后如果继续使用Java 8,但是注意更新自己的版本,所以即时跟进Open JDK更新:永远留在最后一个免费发布。但是存在安全隐患,所以请关注Open JDK。

1)依靠Linux操作系统获取更新包。

在Linux平台上,可以通过操作系统获得最新JDK(例如Red Hat,Debian,Fedora,Arch等)。因此,JDK的更新通过操作系统提供商获取。

Red Hat承诺在Red Hat Enterprise Linux中直到2023年6月更新Java 8 - 他们更愿意将修复程序即时推回到“上游”OpenJDK项目。

所以使用Centos版本可以即时得JDK的最新安全补丁。

2)自己构建OpenJDK。

安全补丁流*在GPL许可下发布到公共Mercurial存储库。因此,完全可以通过跟踪对该存储库的提交来自己构建OpenJDK。国内的BAT,阿里、腾讯、百度应该都有自己的分支维护团队。小公司技术难度和成本太高,不如追随大公司,风险最低。

3)使用AdoptOpenJDK的构建。

AdoptOpenJDK的社区团队在过去几年一直忙于创建一个构建开源JDK和测试平台。因此,他们现在能够获取安全补丁并将其转换为版本,就像商业版获得的那样。他们还运行Java TCK(测试兼容性工具包),以允许这些构建完全认证为与Java SE规范兼容。计划支持Java 8版本,直到2023年9月或之后(Java 17发布两年后)。对于大多数想要在不付费的情况下使用Java 8的用户来说,这可能是最佳选择。

7、不要被误导,技术领导必读

Oracle首席Java架构师Mark Reinhold在Oracle Code One技术大会上演讲表示:Java仍然是开放和免费的。Oracle收取的是技术支持更新服务费用,这是可以理解的。 比如汽车免费送你开,但是收取修车保养费合情合理,你可以选择自己保养修车,也可以选择找到免费的修车店保养车。

对于绝大部分企业来说,担心可能的版权或者收费问题,注意一下几点:

不要轻易下载Oracle JDK8后续更新,避免许可证陷阱,后续更新注意方式,当然如果你打算付费购买技术更新支持服务也可以。
目前Java 8版本已经足够使用,大部分企业选择稳定版本,后续语法糖特性应该影响不大。无需升级版本,除非万不得已。
直接使用 Open JDK 8发行版本,并且接受社区的免费的更新支持,Redhat红帽已经表示会领导更新工作。
使用Amazon Corretto 8,永久免费长期支持。
Java 11的所有部分都将作为长期支持版本进行维护。 但是,与Java 6,7和8不同,Oracle不会长期支持。
Redhat红帽将接管这项任务 - 他们已经公开承诺。
在Java 11的前6个月中,Oracle将在jdk.java.net上提供带有安全补丁的GPL + CE许可免费下载。
Java依然免费,Open JDK依然免费,Amazon JDK Corretto 永久免费!所以请放心使用Java。太多开源免费的方案。

8、Java长期第一,大批公司转Java

根据国外权威网站programming languages ranking的统计排名,Java长期排名第一。请看下图

Java长期排名第一,市场份额超过10%的只有Java和c语言,超过5%的只有python、c++、vb。其他语言长期不到5%

2019Java依然免费,你应该知道的事实,不要被误导—技术领导必读
阿里巴巴 腾讯 百度 蚂蚁金服 今日头条 滴滴 美团 小米 京东 360 携程 拼多多 中国平安 等名 互联网公司 都用Java。超过90%的中国互联网公司是Java架构。

京东、携程、拍拍贷、沪江网 VIPABC 汽车之家、易车网、同程旅游、易果网、途虎养车、中通、申通、汉庭集团 都转Java了。名单还在继续。。。

点击链接加入群聊【java/spring/boot/kafka】:https://jq.qq.com/?_wv=1027&k=5R11ZzY

欢迎工作一到五年的Java工程师朋友们加入Java架构工程师:611481448

本群提供免费的学习指导 架构资料 以及免费的解答

不懂得问题都可以在本群提出来 之后还会有职业生涯规划以及面试指导

1JAVA SE 1.1深入JAVA API 1.1.1Lang包 1.1.1.1String类和StringBuffer类 位于java.lang包中,这个包中的类使用时不用导入 String类一旦初始化就不可以改变,而stringbuffer则可以。它用于封装内容可变的字符串。它可以使用tostring()转换成string字符串。 String x=”a”+4+”c”编译时等效于String x=new StringBuffer().append(“a”).append(4).append(“c”).toString(); 字符串常量是一种特殊的匿名对象,String s1=”hello”;String s2=”hello”;则s1==s2;因为他们指向同一个匿名对象。 如果String s1=new String(“hello”);String s2=new String(“hello”);则s1!=s2; /*逐行读取键盘输入,直到输入为“bye”时,结束程序 注:对于回车换行,在windows下面,有'\r'和'\n'两个,而unix下面只有'\n',但是写程序的时候都要把他区分开*/ public class readline { public static void main(String args[]) { String strInfo=null; int pos=0; byte[] buf=new byte[1024];//定义一个数组,存放换行前的各个字符 int ch=0; //存放读入的字符 system.out.println(“Please input a string:”); while(true) { try { ch=System.in.read(); //该方法每次读入一个字节的内容到ch变量中。 } catch(Exception e) { } switch(ch) { case '\r': //回车时,不进行处理 break; case '\n': //换行时,将数组总的内容放进字符串中 strInfo=new String(buf,0,pos); //该方法将数组中从第0个开始,到第pos个结束存入字符串。 if(strInfo.equals("bye")) //如果该字符串内容为bye,则退出程序。 { return; } else //如果不为bye,则输出,并且竟pos置为0,准备下次存入。 { System.out.println(strInfo); pos=0; break; } default: buf[pos++]=(byte)ch; //如果不是回车,换行,则将读取的数据存入数组中。 } } } } String类的常用成员方法 1、构造方法: String(byte[] byte,int offset,int length);这个在上面已经用到。 2、equalsIgnoreCase:忽略大小写的比较,上例中如果您输入的是BYE,则不会退出,因为大小写不同,但是如果使用这个方法,则会退出。 3、indexOf(int ch);返回字符ch在字符串中首次出现的位置 4、substring(int benginIndex); 5、substring(int beginIndex,int endIndex); 返回字符串的子字符串,4返回从benginindex位置开始到结束的子字符串,5返回beginindex和endindex-1之间的子字符串。 基本数据类型包装类的作用是:将基本的数据类型包装成对象。因为有些方法不可以直接处理基本数据类型,只能处理对象,例如vector的add方法,参数就只能是对象。这时就需要使用他们的包装类将他们包装成对象。 例:在屏幕上打印出一个*组成的矩形,矩形的宽度和高度通过启动程序时传递给main()方法的参数指定。 public class testInteger { public static void main(String[] args) //main()的参数是string类型的数组,用来做为长,宽时,要转换成整型。 { int w=new Integer(args[0]).intValue(); int h=Integer.parseInt(args[1]); //int h=Integer.valueOf(args[1]).intValue(); //以上为三种将字符串转换成整形的方法。 for(int i=0;i<h;i++) { StringBuffer sb=new StringBuffer(); //使用stringbuffer,是因为它是可追加的。 for(int j=0;j<w;j++) { sb.append('*'); } System.out.println(sb.toString()); //在打印之前,要将stringbuffer转化为string类型。 } } } 比较下面两段代码的执行效率: (1)String sb=new String(); For(int j=0;j<w;j++) { Sb=sb+’*’; } (2) StringBuffer sb=new StringBuffer(); For(int j=0;j<w;j++) { Sb.append(‘*’); } (1)和(2)在运行结果上相同,但效率相差很多。 (1)在每一次循环中,都要先将string类型转换为stringbuffer类型,然后将‘*’追加进去,然后再调用tostring()方法,转换为string类型,效率很低。 (2)在没次循环中,都只是调用原来的那个stringbuffer对象,没有创建新的对象,所以效率比较高。 1.1.1.2System类与Runtime类 由于java不支持全局函数和全局变量,所以java设计者将一些与系统相关的重要函数和变量放在system类中。 我们不能直接创建runtime的实例,只能通过runtime.getruntime()静态方法来获得。 编程实例:在java程序中启动一个windows记事本程序的运行实例,并在该运行实例中打开该运行程序的源文件,启动的记事本程序5秒后关闭。 public class Property { public static void main(String[] args) { Process p=null; //java虚拟机启动的进程。 try { p=Runtime.getRuntime().exec("notepad.exe Property.java"); //启动记事本并且打开源文件。 Thread.sleep(5000); //持续5秒 p.destroy(); //关闭该进程 } catch(Exception ex) { ex.printStackTrace(); } } } 1.1.1.3Java语言中两种异常的差别 Java提供了两类主要的异常:runtime exception和checked exception。所有的checked exception是从java.lang.Exception类衍生出来的,而runtime exception则是从java.lang.RuntimeException或java.lang.Error类衍生出来的。    它们的不同之处表现在两方面:机制上和逻辑上。    一、机制上    它们在机制上的不同表现在两点:1.如何定义方法;2. 如何处理抛出的异常。请看下面CheckedException的定义:    public class CheckedException extends Exception    {    public CheckedException() {}    public CheckedException( String message )    {    super( message );    }    }    以及一个使用exception的例子:    public class ExceptionalClass    {    public void method1()    throws CheckedException    {     // ... throw new CheckedException( “...出错了“ );    }    public void method2( String arg )    {     if( arg == null )     {      throw new NullPointerException( “method2的参数arg是null!” );     }    }    public void method3() throws CheckedException    {     method1();    }    }    你可能已经注意到了,两个方法method1()和method2()都会抛出exception,可是只有method1()做了声明。另外,method3()本身并不会抛出exception,可是它却声明会抛出CheckedException。在向你解释之前,让我们先来看看这个类的main()方法:    public static void main( String[] args )    {    ExceptionalClass example = new ExceptionalClass();    try    {    example.method1();    example.method3();    }    catch( CheckedException ex ) { } example.method2( null );    }    在main()方法中,如果要调用method1(),你必须把这个调用放在try/catch程序块当中,因为它会抛出Checked exception。    相比之下,当你调用method2()时,则不需要把它放在try/catch程序块当中,因为它会抛出的exception不是checked exception,而是runtime exception。会抛出runtime exception的方法在定义时不必声明它会抛出exception。    现在,让我们再来看看method3()。它调用了method1()却没有把这个调用放在try/catch程序块当中。它是通过声明它会抛出method1()会抛出的exception来避免这样做的。它没有捕获这个exception,而是把它传递下去。实际上main()方法也可以这样做,通过声明它会抛出Checked exception来避免使用try/catch程序块(当然我们反对这种做法)。    小结一下:    * Runtime exceptions:    在定义方法时不需要声明会抛出runtime exception;    在调用这个方法时不需要捕获这个runtime exception;    runtime exception是从java.lang.RuntimeException或java.lang.Error类衍生出来的。    * Checked exceptions:    定义方法时必须声明所有可能会抛出的checked exception;    在调用这个方法时,必须捕获它的checked exception,不然就得把它的exception传递下去;    checked exception是从java.lang.Exception类衍生出来的。    二、逻辑上    从逻辑的角度来说,checked exceptions和runtime exception是有不同的使用目的的。checked exception用来指示一种调用方能够直接处理的异常情况。而runtime exception则用来指示一种调用方本身无法处理或恢复的程序错误。    checked exception迫使你捕获它并处理这种异常情况。以java.net.URL类的构建器(constructor)为例,它的每一个构建器都会抛出MalformedURLException。MalformedURLException就是一种checked exception。设想一下,你有一个简单的程序,用来提示用户输入一个URL,然后通过这个URL去下载一个网页。如果用户输入的URL有错误,构建器就会抛出一个exception。既然这个exception是checked exception,你的程序就可以捕获它并正确处理:比如说提示用户重新输入。 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值