对于golang一直存有觊觎之心,但一直苦于没有下定决心去学习研究,最近开始接触golang。就我个人来说,学习golang的原动力是因为想要站在java语言之外来审视java和其它语言的区别,再就是想瞻仰一下如此NB的语言。年前就想在2019年做一件事情,希望能从各个细节处做一次java和golang的对比分析,不评判语言的优劣,只想用简单的语言和可以随时执行的代码来表达出两者的区别和底层涉及到的原理。今天是情人节,馒头妈妈在加班,送给自己一件贴心的礼物,写下第一篇对比文章:java&golang的区别之:闭包。
关于闭包到底是啥,建议参考知乎上的解释:https://www.zhihu.com/question/51402215/answer/556617311

  • java8之前的闭包
在java8之前,java其实就已经对闭包有了一定层面的支持,实现的闭包方式主要是靠匿名类来实现的,下面是java程序员经常写的一段代码:
 public class ClosureBeforeJava8 {
int y = 1; public static void main(String[] args) {
final int x = 0;
ClosureBeforeJava8 closureBeforeJava8 = new ClosureBeforeJava8();
Runnable run = closureBeforeJava8.getRunnable();
new Thread(run).start();
} public Runnable getRunnable() {
final int x = 0;
Runnable run = new Runnable() {
@Override
public void run() {
16
System.out.println("local varable x is:" + x);
//System.out.println("member varable y is:" + this.y); //error
}
};
return run;
}
}

上段代码的输出:local varable x is:0

在代码的第13行到第20行,通过匿名类的方式实现了Runnable接口的run()方法,实现了一部分操作的集合(run方法),并将这些操作映射为java的对象,在java中就可以实现将函数以变量的方式进行传递了,如果仅仅是传递函数指针,那还不能算是闭包,我们再注意第17行代码,在这段被封装可以在不同的java对象间传递的代码,引用了上层方法的局部变量,这个就有些闭包的意思在里面了。但是第18行被注释掉的代码在匿名类的情况下却无法编译通过,也就是封装的函数里面,无法引用上层方法所在对象的成员变量。总结一下,java8之前的闭包特点如下:

1.可以实现封装的函数在jvm里进行传递,可以在不同的对象里进行调用;

2.被封装的函数,可以调用上层的方法里的局部变量,但是此局部变量必须为final,也就是不可以更改的(基础类型不可以更改,引用类型不可以变更地址);

3.被封装的函数,不可以调用上层方法所在对象的成员变量;
  • java8里对闭包的支持

java8里对于闭包的支持,其实也就是lamda表达式,我们再来看一下上段代码在lamda表达式方式下的写法:

 public class ClosureInJava8 {
int y = 1; public static void main(String[] args) throws Exception{
final int x = 0;
ClosureInJava8 closureInJava8 = new ClosureInJava8();
Runnable run = closureInJava8.getRunnable();
Thread thread1 = new Thread(run);
thread1.start();
thread1.join();
new Thread(run).start();
} public Runnable getRunnable() {
final int x = 0;
Runnable run = () -> {

System.out.println("local varable x is:" + x);
System.out.println("member varable y is:" + this.y++);
};
return run;
}
}

上面对代码输出:

local varable x is:0
member varable y is:1
local varable x is:0
member varable y is:2

在代码的第16行到第20行,通过lamda表达式的方式实现了函数的封装(关于lamda表达式的用法,大家可以自行google)。通过代码的输出,大家可以发现,在lamda表达式的书写方式下,封装函数不但可以引用上层方法的effectively final类型(java8的特性之一,其实也是final类型)的局部变量,还可以引用上层方法所在对象的成员变量,并可以在其它线程和方法中对此成员变量进行修改。总结一下:java8对于闭包支持的特点如下:

1.通过lamda表达式的方式可以实现函数的封装,并可以在jvm里进行传递;

2.lamda表达式,可以调用上层的方法里的局部变量,但是此局部变量必须为final或者是effectively final,也就是不可以更改的(基础类型不可以更改,引用类型不可以变更地址);

3.lamda表达式,可以调用和修改上层方法所在对象的成员变量;
由于还没时间分析jdk和hotspot的源码,在此只能猜测推理,第2点和第3点的情况。关于第2点:上层方法的局部变量必须是final修饰的,网上的文章大部分都是说因为多线程并发的原因,无法在lamda表达式里进行修改上层方法的局部变量,这点上我是不同意这个观点的。我认为主要原因是:java在定义局部变量时,对于基础类型都是创建在stack frame上的,而一个方法执行完毕后,此方法所对应的stack frame也就没有意义了,试想一下,lamda表达式所依赖的上层方法的局部变量的存储区(stack frame)都消失了,我们还怎么能够修改这个变量,这是毫无意义的,在java里也很难实现这一点,除非像golang一下,在特定情况下,更改局部变量的存储区域(在heap里存储)。关于第3点:实现起来就比较容易,就是在lamda表达式的对象里,创建一个引用地址,地址指向原上层方法所在对象的堆存储地址即可。
  • golang里对闭包的支持

golang里对于闭包的支持,理解起来就非常容易了,就是函数可以作为变量来传递使用,代码如下:

 package main

 import "fmt"

 func main()  {
ch := make(chan int ,1)
ch2 := make(chan int ,1)
fn := closureGet()
go func() {
fn()
ch <-1
}()
go func() {
fn()
ch2 <-1
}()
<-ch
<-ch2
} func closureGet() func(){
x := 1
y := 2
fn := func(){
x = x +y
fmt.Printf("local varable x is:%d y is:%d \n", x, y)
}
return fn
}

代码输出如下:

local varable x is:3 y is:2
local varable x is:5 y is:2

代码的第24行到27行,定义了一个方法fn,此方法可以使用上层方法的局部变量,总结一下:

1.golang的闭包在表达形式上,理解起来非常容易,就是函数可以作为变量,来直接传递;

2.golang的封装函数可以没有限制的使用上层函数里的局部变量,并且在不同的goroutine里修改的值,都会有所体现。

关于第2点,大家可以参考文章:https://studygolang.com/articles/11627  中关于golang闭包的讲解部分。

  • 总结

golang的闭包从语言的简洁性、理解的难易程度、支持的力度上来说,确实还是优于java的。本文作为java和golang对比分析的第一篇文章,由于调研分析的时间有限,难免有疏忽之处,欢迎各位指正。

第一章.java&golang的区别之:闭包的更多相关文章

  1. 第一章 –– Java基础语法

    第一章 –– Java基础语法 span::selection, .CodeMirror-line > span > span::selection { background: #d7d4 ...

  2. 第一章 Java的I/O演进之路

    I/O基础入门 Java的I/O演进 第一章 Java的I/O演进之路 1.1 I/O基础入门 1.1.1 Linux网络I/O模型简介 根据UNIX网络编程对I/O模型的分类,UNIX提供了5中I/ ...

  3. Java基础知识二次学习-- 第一章 java基础

    基础知识有时候感觉时间长似乎有点生疏,正好这几天有时间有机会,就决定重新做一轮二次学习,挑重避轻 回过头来重新整理基础知识,能收获到之前不少遗漏的,所以这一次就称作查漏补缺吧!废话不多说,开始! 第一 ...

  4. javaSE习题 第一章 JAVA语言概述

    转眼就开学了,正式在学校学习SE部分,由于暑假放视频过了一遍,略感觉轻松,今天开始,博客将会记录我的课本习题,主要以文字和代码的形式展现,一是把SE基础加强一下,二是课本中有很多知识是视频中没有的,做 ...

  5. 第一章 java基本多线程技能

    第一章 java多线程技能 1 线程:进程是操作系统结构的基础,是一次程序的执行,是一个程序及其数据在处理顺序时发生的活动:是程序在一个数据集合上运行的过程,他是系统进行资源分配和调度的一个独立单位. ...

  6. 第一章 Java多线程技能

    1.初步了解"进程"."线程"."多线程" 说到多线程,大多都会联系到"进程"和"线程".那么这两者 ...

  7. 第一章Java学习(查漏补缺)

    第一章主要内容: 1.Java的地位:网络地位 语言地位 需求地位 2.Java的特点:①简单 面向对象 平台无关:软件的运行不因操作系统,处理器的变化而无法运行或出现运行错误. ②多线程 动态 3. ...

  8. 深入Java虚拟机读书笔记第一章Java体系结构介绍

    第1章 Java体系结构介绍 Java技术核心:Java虚拟机 Java:安全(先天防bug的设计.内存).健壮.平台无关.网络无关(底层结构上,对象序列化和RMI为分布式系统中各个部分共享对象提供了 ...

  9. Java核心技术(Java白皮书)卷Ⅰ 第一章 Java程序设计概述

    第1章 Java程序设计概述1.1 Java程序设计平台 具有令人赏心悦目的语法和易于理解的语言,与其他许多优秀语言一样,Java满足这些要求. 可移植性 垃圾收集 提供大型的库  如果想要有奇特的绘 ...

随机推荐

  1. 详解Java的Spring框架中的注解的用法

    转载:http://www.jb51.net/article/75460.htm 1. 使用Spring注解来注入属性 1.1. 使用注解以前我们是怎样注入属性的 类的实现: class UserMa ...

  2. maven仓库添加jar架包

    推荐几个好的 Maven 常用仓库网址:http://mvnrepository.com/http://search.maven.org/http://repository.sonatype.org/ ...

  3. hive------ Group by、join、distinct等实现原理

    1. Hive 的 distribute by Order by 能够预期产生完全排序的结果,但是它是通过只用一个reduce来做到这点的.所以对于大规模的数据集它的效率非常低.在很多情况下,并不需要 ...

  4. 利用Hive分析nginx日志

    这里用到的nginx日志是网站的访问日志,比如日志格式: 180.173.250.74 - - [08/Jan/2015:12:38:08 +0800] "GET /avatar/xxx.p ...

  5. UML类图10分钟快速入门 - From 圣杰

    虚线箭头指向依赖: 实线箭头指向关联: 虚线三角指向接口: 实线三角指向父类: 空心菱形能分离而独立存在,是聚合: 实心菱形精密关联不可分,是组合: 原文作者:圣杰 原文地址:http://www.j ...

  6. C语言下double转char*或者std::string,可以精确转换不含多余的0

    char* GetDoubleStr(double value) { char buf[32]={0};//长度可以自定义 sprintf(buf,"%.8f",value);// ...

  7. 计算机网络相关:应用层协议(一):DNS

    DNS 1.概念  DNS是:  1)  一个有分层的DNS服务器实现的分布式数据库  2)一个使得主机能够查询分布式数据库的应用协议.  它运行在UDP之上,默认使用53号端口.  主要功能 是将主 ...

  8. Netty中如何序列化数据

    JDK提供了ObjectOutputStream和ObjectInputStream,用于通过网络对POJO的基本数据类型和图进行序列化和反序列化.该API并不复杂,而且可以被应用于任何实现了java ...

  9. Java公开课-02.抽象类和接口

    在讲述抽象类和接口之前,扯点别的:封装,继承,多态,我只做个简单的涉略 一,封装 1.体现: 将变量和方法放到一个类中 私有字段封装成共有属性 2.this: 如果发现成员变量的名称和方法参数的名称相 ...

  10. 你不知道的JavaScript--Item5 全局变量

    1.尽量少用全局对象 全局变量的问题在于,你的JavaScript应用程序和web页面上的所有代码都共享了这些全局变量,他们住在同一个全局命名空间,所以当程序的两个不同部分定义同名但不同作用的全局变量 ...