成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Java的LINQ:Linq4j簡明介紹

開發 后端
開發JAVA一段時間,面臨的一大問題就是集合操作,習慣了LINQ的簡潔語法,對JAVA的集合操作實在是無甚好感,只能通過C系的循環實現篩選等操作,由于沒有延遲執行特性,內存占用實在不敢恭維。因此便在網上找到了linq4j, 一個針對JAVA的linq移植版本。下面的文章,就會對這一工具進行簡要的介紹。

開發JAVA一段時間,面臨的一大問題就是集合操作,習慣了LINQ的簡潔語法,對JAVA的集合操作實在是無甚好感,只能通過C系的循環實現篩選等操作,由于沒有延遲執行特性,內存占用實在不敢恭維。因此便在網上找到了linq4j, 一個針對JAVA的linq移植版本。下面的文章,就會對這一工具進行簡要的介紹。

一. 安裝

該項目的Github地址是:https://github.com/julianhyde/linq4j. 顯然是一個個人項目,向作者致敬。

它并沒有部署在標準的maven庫里,因此需要手動編譯生成。使用標準命令行:

  1. git clone git://github.com/julianhyde/linq4j.git linq4j    #git克隆到linq4j目錄下  
  2.  
  3.      mvn compile  #編譯  
  4.  
  5.      mvn test #測試  
  6.  
  7.      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的泛型委托,定義如下:

  1. /**   
  2.  * Function with two parameters.   
  3.  *   
  4.  * @param <R> result type   
  5.  * @param <T1> type of parameter 1   
  6.  * @param <T2> type of parameter 2   
  7.  */   
  8. public interface Function2<T1, T2, R> extends Function<R> {   
  9.   R apply(T1 v1, T2 v2);   

當然,內置的函數不止這些,還有一系列非泛型的委托,包括返回bool型的Predicate函數。由于篇幅限制,此處不一一介紹。

3. 一系列Expressions,具體使用下面有介紹。

三. 使用方法

該庫實現了大部分LINQ的功能,其中包括了篩選器,排序器,分組器,類型轉換等功能。下面我們以一個實例來介紹它。

先定義一個實體:

  1. public class Person  
  2.     {  
  3.     public int Age;  
  4.     public String     Name;  
  5.     public boolean Sex;  
  6.     } 

我們的基本任務,是將一個Person集合中,所有性別為男(true)的名字取出來,并按照string的默認降序排列。***得到的應該是List<String>類型。

  1. //Linq4j:  
  2. public void Test(ArrayList<Person> persList)  
  3.      {  
  4.     java.util.List<String> nameStrings=  Linq4j.asEnumerable(persList).where(new Predicate1<Linq4jTest.Person>()  
  5.     {  
  6.           
  7.         public boolean apply(Person arg0)  
  8.         {  
  9.           
  10.          return arg0.Sex;  
  11.         }  
  12.     }).select(new Function1<Linq4jTest.Person, String>()  
  13.     {  
  14.  
  15.         public String apply(Person arg0)  
  16.         {  
  17.          return arg0.Name;  
  18.         }  
  19.     }).orderByDescending(new Function1<String, String>()  
  20.     {  
  21.  
  22.         public String apply(String arg0)  
  23.         {  
  24.         // TODO Auto-generated method stub  
  25.         return arg0;  
  26.         }  
  27.     }).toList();  
  28.      } 

這段代碼的風格和C#的很像,由于接口Enumerable可以拼接,因此通過簡單的Where,Select和 orderByDescending即可實現。但由于LINQ沒有匿名函數,不得不在函數中加入函數,看起來實在是讓人頭疼。另外,由于沒有擴展函數,需要在方法前使用Linq4j的靜態方法。

該功能利用標準Linq實現如下:

  1. var userNames = from d in persons where d.Sex orderby d.Name descending select d.Name; 

在.NET中,我們可以使用閉包,例如在篩選函數的實現中,訪問到外部的數據。但我們可以看如下的例子:

該函數的基本邏輯是找到personList中名字在黑名單里的人。套了兩個Linq4j, 但是,注意blacklist數組的final關鍵字, 如果沒有該關鍵字會報錯,JAVA沒有閉包,因此blacklist數組就不應該修改,這個語法糖到底是不是利大于弊,還需要讀者討論。

  1. public List<Person> SelectBlackList(ArrayList<Person> persList)  
  2.     {  
  3.     final String[] blackList = { "zhang""wang""li" };  
  4.     return Linq4j.asEnumerable(persList)  
  5.         .where(new Predicate1<Linq4jTest.Person>()  
  6.         {  
  7.  
  8.             public boolean apply(Person arg0)  
  9.             {  
  10.             return Linq4j.asEnumerable(blackList).contains(  
  11.                 arg0.Name);  
  12.             }  
  13.  
  14.         }).toList();  
  15.  
  16.     } 

該功能使用標準Linq實現如下:

  1. public  List<Person> GetBlacklist(IEnumerable<Person> persons)  
  2.          {  
  3.              String[] blackList = { "zhang""wang""li" };  
  4.              var result= from d in persons where blackList.Contains(d.Name) select d;  
  5.              return result.ToList();  
  6.          } 

***討論一下集合類型轉換,例如類Worker繼承實現了Person接口.

  1. public class Worker : Person  
  2.     {  
  3.         public string Commpay ;  
  4.     }  
  5.  

那么,一個函數的定義是  void Func(List<Person> nodes). 而我要傳入的參數類型是List<Worker>,編譯器肯定是要報錯的!怎么辦?

對于.NET來說,有逆變和協變特性,或者我可以這么做:

  1. public  void Test3(List<Worker>workers )  
  2.         {  
  3.             this.Func1(workers); //編譯器會報錯  
  4.             this.Func1(workers.OfType<Person>());  
  5.         }  
  6.         public void Func1(IEnumerable<Person>persons )  
  7.         {  
  8.             //只是演示,沒有實現功能  
  9.       } 

對于JAVA來說,一般的做法,是在外面加一個轉換,通過新建Person集合和foreach迭代器,利用強制類型轉換將其轉變為List<Person>. 這實在是太麻煩了。 利用LiNQ4J, 我們也有類似的語法:

  1. public void Func2(List<Person> person)  
  2.     {  
  3.     //演示,不實現功能  
  4.     }  
  5.     public void Test3(List<Worker> workers)//1.通過最簡單粗暴的循環寫法,實現功能,不敢恭維。  
  6.     {  
  7.     // Func2(workers); // 此處編譯器會報錯  
  8.     List<Person> persons = new ArrayList<Linq4jTest.Person>();  
  9.     for (Person person : workers)  
  10.     {  
  11.         persons.add(person);    
  12.     }  
  13.                     Func2(persons);  
  14.     }  
  15.     public void Test4linq(List<Worker> workers)  //2.linq4j寫法  
  16.     {  
  17.     List<Person> persons = Linq4j.asEnumerable(workers)  
  18.         .ofType(Person.class).toList();   
  19.     Func2(persons);  
  20.     } 

linq4j除了提供了這種顯式聲明函數的寫法,還實現了以下的表達式寫法,看起來真是高端洋氣上檔次:

  1. // use lambda, this time call whereN  
  2.     ParameterExpression parameterE =  
  3.         Expressions.parameter(Employee.class);  
  4.     ParameterExpression parameterN =  
  5.         Expressions.parameter(Integer.TYPE);  
  6.     final Queryable<Employee> nh3 =  
  7.         Linq4j.asEnumerable(emps)  
  8.             .asQueryable()  
  9.             .whereN(  
  10.                 Expressions.lambda(  
  11.                     Predicate2.class,  
  12.                     Expressions.andAlso(  
  13.                         Expressions.equal(  
  14.                             Expressions.field(  
  15.                                 parameterE,  
  16.                                 Employee.class,  
  17.                                 "deptno"),  
  18.                             Expressions.constant(10)),  
  19.                         Expressions.lessThan(  
  20.                             parameterN,  
  21.                             Expressions.constant(3))),  
  22.                     parameterE,  
  23.                     parameterN)); 

看起來很唬人,但想起來其實不難。該功能利用Expressions類的靜態方法,提供了一系列現成的函數供調用,一定程度上進一步提升了可用性。具體細節可以參照linq4j的源碼,此處不打算深入討論。

四. 總結

Linq4j實現了標準Linq的絕大多數功能,同時利用Expression類簡化了很多簡單函數的實現。使用起來還是很方便的,但我沒有時間做具體的性能測試,因此在性能上沒有發言權。但不論如何,膜拜一下作者的技術水平。如果大家有空,可以看看linq4j的源碼,一定會有很多收獲。

集合操作是應用開發中最普遍的開發情形,可惜JAVA本身在該處并無太大建樹,linq4j能不能用在大型項目上很難說,如果能在語言本身享受這種便利,那是***不過的了,.NET系同學應該感到幸福。我們只能期待JAVA8帶來的lamda表達式新特性,能更好的解決這個問題,當然這只能在2014年了。

為了方便那些不用maven的同學,附件加上linq4j的jar包下載。 注意下載后改后綴名為jar.

原文鏈接:http://www.cnblogs.com/buptzym/p/3282020.html

責任編輯:林師授 來源: 博客園
相關推薦

2009-09-08 10:50:20

2009-09-08 13:07:15

介紹Linq to S

2009-09-16 10:38:43

LINQ查詢

2009-09-11 09:41:19

LINQ to SQL

2009-09-10 17:30:15

LINQ Where子

2009-08-13 11:01:32

LINQPadLINQ工具

2009-09-09 14:20:49

LINQ To Luc

2009-09-18 09:25:06

LINQ Framew

2009-09-07 17:05:10

LINQ進行查詢

2009-09-07 17:46:18

LINQ高級特性

2009-09-10 14:02:08

LINQ ASP.NE

2009-09-09 09:36:25

Linq對象引用

2009-09-15 09:39:38

LINQ查詢架構

2009-09-14 14:17:11

Linq異常處理

2009-09-15 13:20:34

LINQ DataCo

2009-09-07 16:13:14

LINQ to SQL

2009-09-16 09:09:23

Linq Contai

2009-09-18 16:56:22

System.LINQ

2009-09-17 13:54:26

LINQ to XML

2009-09-17 13:30:32

LINQ to XML
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 在线不卡| h在线播放| 免费一级片| 九九色综合 | 欧美一级免费看 | 欧美久久免费观看 | 成人国产精品久久 | 欧美激情视频一区二区三区免费 | 亚洲欧美综合网 | 亚洲精品视频一区 | 污污的网站在线观看 | 欧美高清成人 | 一级片免费视频 | 天天射视频 | 91在线观看视频 | 日本欧美视频 | 美女在线一区二区 | 国产乱码精品1区2区3区 | 亚洲91精品 | 国产小视频在线 | 亚洲在线高清 | 麻豆一区 | www精品美女久久久tv | 国产欧美日韩一区 | 亚洲免费精品 | 欧美精品综合在线 | 精品99久久 | www.性色 | 国产精品视频导航 | 亚洲国产二区 | 国产九九精品 | 精品国产视频 | 台湾av在线 | 亚洲成年在线 | 国产精品久久久久久吹潮 | 涩涩视频在线观看免费 | 成人av一区二区三区 | 日韩精品在线免费 | 久久久青草| 国产精品久久久久永久免费观看 | 国产精品1区2区3区 一区中文字幕 |