最近2018中文字幕在日韩欧美国产成人片_国产日韩精品一区二区在线_在线观看成年美女黄网色视频_国产精品一区三区五区_国产精彩刺激乱对白_看黄色黄大色黄片免费_人人超碰自拍cao_国产高清av在线_亚洲精品电影av_日韩美女尤物视频网站

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時(shí)間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
怎么理解Java中Linq4j

這篇文章給大家介紹怎么理解Java中Linq4j,內(nèi)容非常詳細(xì),感興趣的小伙伴們可以參考借鑒,希望對大家能有所幫助。

網(wǎng)站建設(shè)哪家好,找創(chuàng)新互聯(lián)!專注于網(wǎng)頁設(shè)計(jì)、網(wǎng)站建設(shè)、微信開發(fā)、微信小程序、集團(tuán)企業(yè)網(wǎng)站建設(shè)等服務(wù)項(xiàng)目。為回饋新老客戶創(chuàng)新互聯(lián)還提供了孝義免費(fèi)建站歡迎大家使用!

開發(fā)JAVA一段時(shí)間,面臨的一大問題就是集合操作,習(xí)慣了LINQ的簡潔語法,對JAVA的集合操作實(shí)在是無甚好感,只能通過C系的循環(huán)實(shí)現(xiàn)篩選等操作,由于沒有延遲執(zhí)行特性,內(nèi)存占用實(shí)在不敢恭維。因此便在網(wǎng)上找到了linq4j, 一個針對JAVA的linq移植版本。

一. 安裝

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

它并沒有部署在標(biāo)準(zhǔn)的maven庫里,因此需要手動編譯生成。使用標(biāo)準(zhǔn)命令行:

git clone git://github.com/julianhyde/linq4j.git linq4j    #git克隆到linq4j目錄下        mvn compile  #編譯        mvn test #測試        mvn jar:jar  #生成jar包

使用了maven以后,工作效率大大提升,.當(dāng)然NET下也有類似的工具nuget.

二. Linq4j的擴(kuò)展功能

由于JAVA目前還沒有匿名函數(shù)和擴(kuò)展函數(shù),而且內(nèi)置的標(biāo)準(zhǔn)迭代器接口Iterator功能偏弱。 因此Linq4j增加了一個一系列泛型接口和函數(shù):

1.  新迭代器接口: Enumerable,它擴(kuò)展了Iterator的功能

2.  一組類似“委托”性質(zhì)的函數(shù):

(1)返回R的泛型委托:public interface Function {}

(2)接收T, 返回R的泛型委托:public interface Function1 {}

(3)接收T1,T2, 返回R的泛型委托,定義如下:

/**    * Function with two parameters.    *    * @param  result type    * @param  type of parameter 1    * @param  type of parameter 2    */   public interface Function2 extends Function {     R apply(T1 v1, T2 v2);   }

當(dāng)然,內(nèi)置的函數(shù)不止這些,還有一系列非泛型的委托,包括返回bool型的Predicate函數(shù)。由于篇幅限制,此處不一一介紹。

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

三. 使用方法

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

先定義一個實(shí)體:

public class Person      {      public int Age;      public String     Name;      public boolean Sex;      }

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

//Linq4j:  public void Test(ArrayList persList)       {      java.util.List nameStrings=  Linq4j.asEnumerable(persList).where(new Predicate1()      {                    public boolean apply(Person arg0)          {                     return arg0.Sex;          }      }).select(new Function1()      {           public String apply(Person arg0)          {           return arg0.Name;          }      }).orderByDescending(new Function1()      {           public String apply(String arg0)          {          // TODO Auto-generated method stub          return arg0;          }      }).toList();       }

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

該功能利用標(biāo)準(zhǔn)Linq實(shí)現(xiàn)如下:

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

在.NET中,我們可以使用閉包,例如在篩選函數(shù)的實(shí)現(xiàn)中,訪問到外部的數(shù)據(jù)。但我們可以看如下的例子:

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

public List SelectBlackList(ArrayList persList)      {      final String[] blackList = { "zhang", "wang", "li" };      return Linq4j.asEnumerable(persList)          .where(new Predicate1()          {               public boolean apply(Person arg0)              {              return Linq4j.asEnumerable(blackList).contains(                  arg0.Name);              }           }).toList();       }

該功能使用標(biāo)準(zhǔn)Linq實(shí)現(xiàn)如下:

public  List GetBlacklist(IEnumerable persons)           {               String[] blackList = { "zhang", "wang", "li" };               var result= from d in persons where blackList.Contains(d.Name) select d;               return result.ToList();           }

***討論一下集合類型轉(zhuǎn)換,例如類Worker繼承實(shí)現(xiàn)了Person接口.

public class Worker : Person      {          public string Commpay ;      }

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

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

public  void Test3(Listworkers )          {              this.Func1(workers); //編譯器會報(bào)錯              this.Func1(workers.OfType());          }          public void Func1(IEnumerablepersons )          {              //只是演示,沒有實(shí)現(xiàn)功能        }

對于JAVA來說,一般的做法,是在外面加一個轉(zhuǎn)換,通過新建Person集合和foreach迭代器,利用強(qiáng)制類型轉(zhuǎn)換將其轉(zhuǎn)變?yōu)長ist. 這實(shí)在是太麻煩了。 利用LiNQ4J, 我們也有類似的語法:

public void Func2(List person)      {      //演示,不實(shí)現(xiàn)功能      }      public void Test3(List workers)//1.通過最簡單粗暴的循環(huán)寫法,實(shí)現(xiàn)功能,不敢恭維。      {      // Func2(workers); // 此處編譯器會報(bào)錯      List persons = new ArrayList();      for (Person person : workers)      {          persons.add(person);        }                      Func2(persons);      }      public void Test4linq(List workers)  //2.linq4j寫法      {      List persons = Linq4j.asEnumerable(workers)          .ofType(Person.class).toList();       Func2(persons);      }

linq4j除了提供了這種顯式聲明函數(shù)的寫法,還實(shí)現(xiàn)了以下的表達(dá)式寫法,看起來真是高端洋氣上檔次:

// use lambda, this time call whereN      ParameterExpression parameterE =          Expressions.parameter(Employee.class);      ParameterExpression parameterN =          Expressions.parameter(Integer.TYPE);      final Queryable nh4 =          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));

看起來很唬人,但想起來其實(shí)不難。該功能利用Expressions類的靜態(tài)方法,提供了一系列現(xiàn)成的函數(shù)供調(diào)用,一定程度上進(jìn)一步提升了可用性。具體細(xì)節(jié)可以參照linq4j的源碼,此處不打算深入討論。

四. 總結(jié)

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

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

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

關(guān)于怎么理解Java中Linq4j就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,可以學(xué)到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。


分享標(biāo)題:怎么理解Java中Linq4j
本文URL:http://fisionsoft.com.cn/article/jghojo.html