麻豆小视频在线观看_中文黄色一级片_久久久成人精品_成片免费观看视频大全_午夜精品久久久久久久99热浪潮_成人一区二区三区四区

首頁 > 學院 > 開發設計 > 正文

克服J2SE 1.3 ~ 1.4不兼容問題 HK2000c

2019-11-18 13:12:57
字體:
來源:轉載
供稿:網友

  概要
  假如你要實現javaAPI中的一個,那么可能是件比較痛苦的事情。你經常會需要實現許多交叉依靠的接口。對新特性的需求促成了升級現有的JavaAPI,這就造成了提供這些API的供給商對他們的相關實現不斷的升級以維持相關功能。隨著這些API的升級更改越來越頻繁,API代碼的不兼容使你不得不分別維護新舊版本的代碼庫。這直接到導致了你維護成本和難度的增加。本文演示了解決此問題的技術,揭示了如何僅使用一個代碼庫編譯不同JavaAPI版本的代碼。
  
  現在非常多的API被加入到到Java的標準庫中,比如JDBC。這樣做的好處是,Java可選包在部署時不必被綁定到相關的部署應用中去。這些API由專門的專業開發小組實現,在實際的使用當中這些API變得越來越受歡迎,使用的深度及廣度也在不斷的增加。但是有時候對一些API升級會變得使一些類及方法不可用。開發小組寧愿讓這些API包成為可選組件而不是作為Java標準支持庫的形式來發布。但是一旦加入標準庫中的API包,就像是和用戶簽定了終生契約,想再成為可選包是不可能的。所以作為用戶的你,可能會忽然發現你一下子自己的代碼庫變成了不兼容的2個代碼庫,一個是使用新API的代碼庫,另一個是使用舊API的代碼庫。你可能會以為情況不像你想象的那樣糟糕。我這里舉一個簡單的例子。J2SE1.4中由于對JDBC中的一些API的升級使的java.sql.Connection 不能同時被1.3 及 1.4 版本編譯通過。你可能會碰到我這樣的困境:我可能需要實現java.sql.Connection這個接口,但是我的代碼需要同時通過1.3 及1.4 得編譯。但是我不想同時維護2個版本的代碼庫。所以我開始尋找更好的解決方法。
  
  假如你依靠于javac來編譯你的應用的話,那么很不幸,Java聞名的一次編寫,到處運行(WORA)并不包括WOCA(一次編寫,到處編譯^_^;)。
  
  不過別太沮喪,編碼的反射技巧以及編譯的Ant技巧是你能夠安然過關。我能夠僅僅使用一組Java文件以及Ant工具,就能使一個版本同時編譯在1.3 和1.4 版本下面。別急,在我結識解決辦法之前,讓我先具體的解釋一下問題的描述。
  
  可憐人的連接池(PS:Poor man's connection pool ,很有意思的一句話)
  兩年前,我的公司需要一個連接池,但是又不肯出錢買一個。當時并沒有什么免費的東東可以使用,所以我們自己寫了一個連接池。為了能更好的跟蹤在整個應用中連接的情況,我們寫了一個com.icentris.sql.ConnectionWrapper類,它實現了java.sql.Connection 接口以及其他的一些包裝類(實現了另外的一些的java.sql 接口)。這些包裝類僅僅是跟蹤我們應用中的數據庫使用,以及通過方法調用真正的數據庫資源。
  當J2SE1.4來的時候,我們自然而然的想到升級我們提供給客戶的應用,使這些應用的性能得到很多提升。當然,我們也需要保留1.3版本,因為有些客戶根本不需要升級到1.4。我們氣惱的發現,假如我們不修改,我們的ConnectionWrapper 以及其他JDBC封裝類根本通不過J2SE1.4的編譯。
  
  為了文章的簡明,我通過使用ConnectionWrapper 這個類來演示我對所有其他不能夠通過J2SE1.4的類所使用的技術。假如我按照新的API標準,那么我不得不添加幾個方法到ConnectionWrapper中去,接下來2個大問題擺在了面前:
  
  1.因為我的包裝類需要經歷方法調用,我將不得不調用在J2SE1.3 sql類中并不存在的方法。
  2.因為一些新的方法涉及到一些新出現的類,我將不得不在編譯中面對那些在J2SE1.3中并不存在的類。
  
  反射提供了援助
  一些代碼可以很方便的解釋第一個問題。但是我的ConnectionWrapper 封裝了java.sql.Connection , 所有的我的例子
  依靠于在構造方法中的變量 realConnection :
  
  PRivate java.sql.Connection realConnection = null;
  
  public ConnectionWrapper(java.sql.Connection connection) {
  realConnection = connection;
  }
  
  為了看清楚我怎么做到解決版本不兼容問題,讓我們仔細看一下setHoldability(int)(這個在J2SE1.4被聲明的新方法)
  public void setHoldability(int holdability) throws SQLException {
  realConnection.setHoldability( holdability );
  }
  
  很不幸,這個方法在J2SE1.3中顯然通不過編譯,這就陷入了2難的尷尬境地。為了解決這一情況,我假定setHoldability() 將只會在J2SE1.4
  下面被調用,所以我使用了反射機制來調用該方法。
  
  public void setHoldability(int holdability) throws SQLException {
  Class[] argTypes = new Class[] { Integer.TYPE };
  Object[] args = new Object[] {new Integer(holdability)};
  callJava14Method("setHoldability", realConnection, argTypes, args);
  }
  
  public static Object callJava14Method(String methodName, Object instance,
  Class[] argTypes, Object[] args)
  throws SQLException
  {
  try {
  Method method = instance.getClass().getMethod(methodName, argTypes);
  return method.invoke(instance, args );
  } catch (NoSUChMethodException e) {
  e.printStackTrace();
  throw new SQLException("Error Invoking method (" + methodName + "): "
  + e);
  } catch (IllegalaccessException e) {
  e.printStackTrace();
  throw new SQLException("Error Invoking method (" + methodName + "): "
  + e);
  } catch (InvocationTargetException e) {
  e.printStackTrace();
  throw new SQLException("Error Invoking method (" + methodName + "): "
  + e);
  }
  }
  
  現在我有了setHoldability() 方法,因此能順利通過J2SE1.4的編譯。原理是我并不直接調用J2SE1.3中間java.sql.Connection并不存在的方法,
  而是轉為通過讓setHoldability調用callJava14Method這個通用方法來調用,然后在一個SQLException 里封裝所有的異常。這樣就達到我預期的效果。
  現在所有的在J2SE1.4中新方法都工作的很好,在J2SE1.3的老版本下也能順利編譯而且工作正常。現在我來著手解決第二個問題。
  就是如何在應用中能夠找到一個方法能夠使用J2SE1.3中并不存在的新的類。
  
  Ant 是答案
  在J2SE1.4中,java.sql.Connection 依靠于一個新的類java.sql.Savepoint。因為這個類在java.sql 包中,所以你不可能把它加入到J2SE1.3中去。Java不答應任何的第三方擴展包加入它的核心包(java.* 以及 javax.* )中去。 因此挑戰來了,在J2SE1.4下調用這個新的java.sql.Savepoint 類,但同時需要代碼能夠在J2SE1.3下面得到編譯以及能夠運行。很簡單,不是嗎?所有回答"Yes"的人都會得到一個榛仁巧克力餅(PS:哈哈,我回答了,可是沒有:P)。至少現在我找到了答案,使問題變得很簡單了。
  首先我插入了下面一條有條件的import語句
  // Comment_next_line_to_compile_with_Java_1.3
  import java.sql.Savepoint;
  
  然后我找到了一個能夠在J2SE1.3下面注釋掉import的方法。非常簡單,使用如下Ant 語句就可以了:
  <replace>
  <replacetoken>Comment_next_line_for_Java_1.3
  </replacetoken>
  <replacevalue>Comment_next_line_for_Java_1.3
  //</replacevalue>
  </replace>
  
  這個Ant 的 replace 標簽 有好幾個標簽選項,在以后我給出的全部例子里有很多。在這里面最重要的是使用<replacevalue>來替換<replacetoken> 。
  在xml里面的意思是換行。在J2SE1.4下,沒什么會發生, 但是在J2SE1.3下面一個import聲明被注釋掉了。
  // Comment_next_line_to_compile_with_Java_1.3
  //import java.sql.Savepoint;
  
  但是我在代碼中Savepoint仍在使用public Savepoint setSavepoint(String name) throws SQLException { . . .}。不過我只在J2SE1.4使用這些方法類,在J2SE1.3中只要能編譯就可以了。我發現只要我有一個我自己的Savepoint 類在我的包中,我的代碼就能夠通過編譯,而且不用任何的import包。但是我又要同時在這條import 語句不被注釋的同時我自己的Savepoint類被忽略掉。因此我造了一個空的com.icentris.sql.Savepoint類,這個可能(除了JavaDoc)是最短的有效類:
  package com.icentris.sql;
  
  /** Dummy class to allow ConnectionWrapper to implement java.sql.Connection
  * and still compile under J2SE 1.3 and J2SE 1.4. When compiled
  * under J2SE 1.3, this class compiles as a placeholder instead of the
  * missing java.sql.Savepoint (not in J2SE 1.3). When compiled
  * under J2SE 1.4, this class is ignored and ConnectionWrapper uses the
  * java.sql.Savepoint that is new in J2SE 1.4.
  */
  public class Savepoint {}
  
  在J2SE1.4下我能夠正確的import java.sql.Savepoint類,而在J2SE1.3下面Ant注釋了這條import語句。因此這個Savepoint就被替換成了我這個包里面寫的一個空的Savepoint類。所以我現在就能加入任何引用到Savepoint類的方法,同樣的在這些新方法中使用剛才所說的反射方法。
  // Comment_next_line_to_compile_with_Java_1.3
  import java.sql.Savepoint;
  
  . . .
  public Savepoint setSavepoint() throws SQLException {
  Class[] argTypes = new Class[0];
  Object[] args = new Object[0];<

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 精品国产一区二区三区四区阿崩 | 亚洲一级毛片 | h视频免费在线观看 | 免费黄色入口 | 日韩视频在线观看免费视频 | 99精品视频免费 | 亚洲va在线 | 欧美 日本 在线 | 成人污在线 | 国产精品久久久久国产精品三级 | 中文字幕 欧美 日韩 | 免费毛片在线视频 | 久久国产亚洲精品 | 激情宗合 | 精品国产一区二区在线 | 一本色道久久综合亚洲精品图片 | 亚洲福利在线观看视频 | 国产69精品99久久久久久宅男 | lutube成人福利在线观看污 | 91久久综合 | 羞羞色院91精品网站 | 国产精品久久久久久久久久 | 91看片在线观看视频 | 精品国产99久久久久久宅男i | 成人在线免费观看网址 | 国产亚洲精品视频中文字幕 | h视频免费在线观看 | 娇妻被各种姿势c到高潮小说 | 欧美日韩a∨毛片一区 | 好吊色37pao在线观看 | 国产91中文字幕 | 日韩视频中文 | 深夜免费观看视频 | aaaaa国产欧美一区二区 | 国产精品午夜一区 | 亚洲精品3 | 国产精品久久久久久久久久三级 | 91在线视频免费观看 | 性感美女一级毛片 | 亚洲第一成人久久网站 | 日韩视频在线不卡 |