java的LINQ :Linq4j简明介绍
开发JAVA一段时间,面临的一大问题就是集合操作,习惯了LINQ的简洁语法,对JAVA的集合操作实在是无甚好感,只能通过C系的循环实现筛选等操作,由于没有延迟执行特性,内存占用实在不敢恭维。因此便在网上找到了linq4j, 一个针对JAVA的linq移植版本。下面的文章,就会对这一工具进行简要的介绍。
一. 安装
该项目的Github地址是:https://github.com/julianhyde/linq4j. 显然是一个个人项目,向作者致敬。
它并没有部署在标准的maven库里,因此需要手动编译生成。使用标准命令行:
git clone git://github.com/julianhyde/linq4j.git linq4j #git克隆到linq4j目录下
mvn compile #编译
mvn test #测试
mvn jar:jar #生成jar包
使用了maven以后,工作效率大大提升,.当然NET下也有类似的工具nuget.
二. Linq4j的扩展功能
由于JAVA目前还没有匿名函数和扩展函数,而且内置的标准迭代器接口Iterator功能偏弱。 因此Linq4j增加了一个一系列泛型接口和函数:
1. 新迭代器接口: Enumerable<T>,它扩展了Iterator的功能
2. 一组类似“委托”性质的函数:
(1)返回R的泛型委托:public interface Function<R> {}
(2)接收T, 返回R的泛型委托:public interface Function1<T,R> {}
(3) 接收T1,T2, 返回R的泛型委托,定义如下:
/**
* Function with two parameters.
*
* @param <R> result type
* @param <T1> type of parameter 1
* @param <T2> type of parameter 2
*/
public interface Function2<T1, T2, R> extends Function<R> {
R apply(T1 v1, T2 v2);
}
当然,内置的函数不止这些,还有一系列非泛型的委托,包括返回bool型的Predicate函数。由于篇幅限制,此处不一一介绍。
3. 一系列Expressions,具体使用下面有介绍。
三. 使用方法
该库实现了大部分LINQ的功能,其中包括了筛选器,排序器,分组器,类型转换等功能。下面我们以一个实例来介绍它。
先定义一个实体:
public class Person
{
public int Age;
public String Name;
public boolean Sex;
}
我们的基本任务,是将一个Person集合中,所有性别为男(true)的名字取出来,并按照string的默认降序排列。最后得到的应该是List<String>类型。
//Linq4j:
public void Test(ArrayList<Person> persList)
{
java.util.List<String> nameStrings= Linq4j.asEnumerable(persList).where(new Predicate1<Linq4jTest.Person>()
{ public boolean apply(Person arg0)
{ return arg0.Sex;
}
}).select(new Function1<Linq4jTest.Person, String>()
{ public String apply(Person arg0)
{
return arg0.Name;
}
}).orderByDescending(new Function1<String, String>()
{ public String apply(String arg0)
{
// TODO Auto-generated method stub
return arg0;
}
}).toList();
}
这段代码的风格和C#的很像,由于接口Enumerable可以拼接,因此通过简单的Where,Select和orderByDescending即可实现。但由于LINQ没有匿名函数,不得不在函数中加入函数,看起来实在是让人头疼。另外,由于没有扩展函数,需要在方法前使用Linq4j的静态方法。
该功能利用标准Linq实现如下:
var userNames = from d in persons where d.Sex orderby d.Name descending select d.Name;
在.NET中,我们可以使用闭包,例如在筛选函数的实现中,访问到外部的数据。但我们可以看如下的例子:
该函数的基本逻辑是找到personList中名字在黑名单里的人。套了两个Linq4j, 但是,注意blacklist数组的final关键字, 如果没有该关键字会报错,JAVA没有闭包,因此blacklist数组就不应该修改,这个语法糖到底是不是利大于弊,还需要读者讨论。
public List<Person> SelectBlackList(ArrayList<Person> persList)
{
final String[] blackList = { "zhang", "wang", "li" };
return Linq4j.asEnumerable(persList)
.where(new Predicate1<Linq4jTest.Person>()
{ public boolean apply(Person arg0)
{
return Linq4j.asEnumerable(blackList).contains(
arg0.Name);
} }).toList(); }
该功能使用标准Linq实现如下:
public List<Person> GetBlacklist(IEnumerable<Person> persons)
{
String[] blackList = { "zhang", "wang", "li" };
var result= from d in persons where blackList.Contains(d.Name) select d;
return result.ToList();
}
最后讨论一下集合类型转换,例如类Worker继承实现了Person接口.
public class Worker : Person
{
public string Commpay ;
}
那么,一个函数的定义是 void Func(List<Person> nodes). 而我要传入的参数类型是List<Worker>,编译器肯定是要报错的!怎么办?
对于.NET来说,有逆变和协变特性,或者我可以这么做:
public void Test3(List<Worker>workers )
{
this.Func1(workers); //编译器会报错
this.Func1(workers.OfType<Person>());
}
public void Func1(IEnumerable<Person>persons )
{
//只是演示,没有实现功能
}
对于JAVA来说,一般的做法,是在外面加一个转换,通过新建Person集合和foreach迭代器,利用强制类型转换将其转变为List<Person>. 这实在是太麻烦了。 利用LiNQ4J, 我们也有类似的语法:
public void Func2(List<Person> person)
{
//演示,不实现功能
}
public void Test3(List<Worker> workers)//1.通过最简单粗暴的循环写法,实现功能,不敢恭维。
{
// Func2(workers); // 此处编译器会报错
List<Person> persons = new ArrayList<Linq4jTest.Person>();
for (Person person : workers)
{
persons.add(person);
}
Func2(persons);
}
public void Test4linq(List<Worker> workers) //2.linq4j写法
{
List<Person> persons = Linq4j.asEnumerable(workers)
.ofType(Person.class).toList();
Func2(persons);
}
linq4j除了提供了这种显式声明函数的写法,还实现了以下的表达式写法,看起来真是高端洋气上档次:
// use lambda, this time call whereN
ParameterExpression parameterE =
Expressions.parameter(Employee.class);
ParameterExpression parameterN =
Expressions.parameter(Integer.TYPE);
final Queryable<Employee> nh3 =
Linq4j.asEnumerable(emps)
.asQueryable()
.whereN(
Expressions.lambda(
Predicate2.class,
Expressions.andAlso(
Expressions.equal(
Expressions.field(
parameterE,
Employee.class,
"deptno"),
Expressions.constant(10)),
Expressions.lessThan(
parameterN,
Expressions.constant(3))),
parameterE,
parameterN));
看起来很唬人,但想起来其实不难。该功能利用Expressions类的静态方法,提供了一系列现成的函数供调用,一定程度上进一步提升了可用性。具体细节可以参照linq4j的源码,此处不打算深入讨论。
四. 总结
Linq4j实现了标准Linq的绝大多数功能,同时利用Expression类简化了很多简单函数的实现。使用起来还是很方便的,但我没有时间做具体的性能测试,因此在性能上没有发言权。但不论如何,膜拜一下作者的技术水平。如果大家有空,可以看看linq4j的源码,一定会有很多收获。
集合操作是应用开发中最普遍的开发情形,可惜JAVA本身在该处并无太大建树,linq4j能不能用在大型项目上很难说,如果能在语言本身享受这种便利,那是最好不过的了,.NET系同学应该感到幸福。我们只能期待JAVA8带来的lamda表达式新特性,能更好的解决这个问题,当然这只能在2014年了。
为了方便那些不用maven的同学,附件加上linq4j的jar包下载。 注意下载后改后缀名为jar.
java的LINQ :Linq4j简明介绍的更多相关文章
- Linq4j简明介绍
Linq4j简明介绍 开发JAVA一段时间,面临的一大问题就是集合操作,习惯了LINQ的简洁语法,对JAVA的集合操作实在是无甚好感,只能通过C系的循环实现筛选等操作,由于没有延迟执行特性,内存占用实 ...
- Spark如何使用Akka实现进程、节点通信的简明介绍
<深入理解Spark:核心思想与源码分析>一书前言的内容请看链接<深入理解SPARK:核心思想与源码分析>一书正式出版上市 <深入理解Spark:核心思想与源码分析> ...
- Atitit 实现java的linq 以及与stream api的比较
Atitit 实现java的linq 以及与stream api的比较 1.1. Linq 和stream api的关系,以及主要优缺点1 1.2. Linq 与stream api的适用场景1 1. ...
- Java XML解析工具 dom4j介绍及使用实例
Java XML解析工具 dom4j介绍及使用实例 dom4j介绍 dom4j的项目地址:http://sourceforge.net/projects/dom4j/?source=directory ...
- Java 并发和多线程(一) Java并发性和多线程介绍[转]
作者:Jakob Jenkov 译者:Simon-SZ 校对:方腾飞 http://tutorials.jenkov.com/java-concurrency/index.html 在过去单CPU时 ...
- [原创]Java静态代码检查工具介绍
[原创]Java静态代码检查工具介绍 一 什么是静态代码检查? 静态代码分析是指无需运行被测代码,仅通过分析或检查源程序的语法.结构.过程.接口等来检查程序的正确性,找出代码隐藏的错误和缺陷,如参数 ...
- Java并发性和多线程介绍
java并发性和多线程介绍: 单个程序内运行多个线程,多任务并发运行 多线程优点: 高效运行,多组件并行.读->操作->写: 程序设计的简单性,遇到多问题,多开线程就好: 快速响应,异步式 ...
- java基础-Eclipse开发工具介绍
java基础-Eclipse开发工具介绍 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 所谓工欲善其事必先利其器,即将身为一名Java开发工程师怎么能没有一款好使的IDE呢?今天就 ...
- java基础-Idea开发工具介绍
java基础-Idea开发工具介绍 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 之前给大家介绍过一款Java的IDE叫eclipse,有些功能用起来不是很得心应手,尤其是在导报的 ...
随机推荐
- LintCode 463 Sort Integer
这个是O(n2)的排序的总结 /* bubble sort */public static void sortIntegers(int[] A) { // Write your code here i ...
- 《理解 ES6》阅读整理:函数(Functions)(八)Tail Call Optimization
尾调用优化(Tail Call Optimization) 尾调用是指函数的最后一条语句是函数调用,比如下面的代码: function doSomething() { return doSomethi ...
- iOS常用的忽略警告
在iOS开发过程中,偶尔会碰到一些编译器警告,如果能够确定该警告不会影响到程序的正常运行,则可以手动告诉编译器忽略掉这个警告 iOS常用的忽略警告类型: 1.方法弃用警告 #pragma clang ...
- 如何在ASP.NET 5上搭建基于TypeScript的Angular2项目
一.前言 就在上月,公司的一个同事建议当前的前端全面改用AngularJs进行开发,而我们采用的就是ASP.NET 5项目,原本我的计划是采用TypeScript直接进行Angular2开发.所以借用 ...
- CSS Font知识整理总结
1.什么是字体 字体是文字的外在形式,就是文字的风格,是文字的外衣.比如行书.楷书.草书,都是一种字体.同样一个字每个人写起来都会有差异,可以说每个人都有一套潜在的字体库.对于web页面来说,字体就是 ...
- ASP.NET MVC请求处理管道生命周期的19个关键环节(1-6)
ASP.NET和ASP.NET MVC的HttpApplication请求处理管道有共同的部分和不同之处,本系列将体验ASP.NET MVC请求处理管道生命周期的19个关键环节. ①以IIS6.0为例 ...
- 译文---C#堆VS栈(Part Three)
前言 在本系列的第一篇文章<C#堆栈对比(Part Two)>中,介绍了值类型和引用类型在参数传递时的不同,本文将讨论如何应用ICloneable接口实现去修复引在堆上的用变量所带来的问题 ...
- 使用Location对象查询字符串参数
location是BOM中最有用的对象之一: 1.它提供了与当前窗口中加载的文档有关的信息: 2.他还提供了一些导航功能. location对象的属性有: hash, host, hostname, ...
- RabbitMQ(六)远程连接
RabbitMQ(六)远程连接 默认情况下,rabbitmq使用`guest`来连接本地(localhost)的server,当需要远程连接时,就会失效. "guest" user ...
- [异常解决] ubuntu上安装虚拟机遇到的问题(vmware坑了,virtual-box简单安装,在virtual-box中安装精简版win7)
利用周末时间将整个电脑格式化,换成了ubuntu系统- 所谓:扫清屋子再请客! 但是有些软件只在win上有,于是还是考虑装个虚拟机来个——逐步过度策略,一点点地从win上转移到linux上 我的系统是 ...