爪哇島

          關注java
          posts - 6, comments - 0, trackbacks - 0, articles - 1

          2011年2月21日

          var myDate = new Date();
          myDate.getYear();        //獲取當前年份(2位)
          myDate.getFullYear();    //獲取完整的年份(4位,1970-????)
          myDate.getMonth();       //獲取當前月份(0-11,0代表1月)
          myDate.getDate();        //獲取當前日(1-31)
          myDate.getDay();         //獲取當前星期X(0-6,0代表星期天)
          myDate.getTime();        //獲取當前時間(從1970.1.1開始的毫秒數)
          myDate.getHours();       //獲取當前小時數(0-23)
          myDate.getMinutes();     //獲取當前分鐘數(0-59)
          myDate.getSeconds();     //獲取當前秒數(0-59)
          myDate.getMilliseconds();    //獲取當前毫秒數(0-999)
          myDate.toLocaleDateString();     //獲取當前日期
          var mytime=myDate.toLocaleTimeString();     //獲取當前時間
          myDate.toLocaleString( );        //獲取日期與時間


          日期時間腳本庫方法列表

          Date.prototype.isLeapYear 判斷閏年
          Date.prototype.Format 日期格式化
          Date.prototype.DateAdd 日期計算
          Date.prototype.DateDiff 比較日期差
          Date.prototype.toString 日期轉字符串
          Date.prototype.toArray 日期分割為數組
          Date.prototype.DatePart 取日期的部分信息
          Date.prototype.MaxDayOfDate 取日期所在月的最大天數
          Date.prototype.WeekNumOfYear 判斷日期所在年的第幾周
          StringToDate 字符串轉日期型
          IsValidDate 驗證日期有效性
          CheckDateTime 完整日期時間檢查
          daysBetween 日期天數差

          js代碼:

          //--------------------------------------------------- 
          // 判斷閏年 
          //--------------------------------------------------- 
          Date.prototype.isLeapYear = function()  
          {  
              return (0==this.getYear()%4&&((this.getYear()%100!=0)||(this.getYear()%400==0)));  
          }  
           
          //--------------------------------------------------- 
          // 日期格式化 
          // 格式 YYYY/yyyy/YY/yy 表示年份 
          // MM/M 月份 
          // W/w 星期 
          // dd/DD/d/D 日期 
          // hh/HH/h/H 時間 
          // mm/m 分鐘 
          // ss/SS/s/S 秒 
          //--------------------------------------------------- 
          Date.prototype.Format = function(formatStr)  
          {  
              var str = formatStr;  
              var Week = ['日','一','二','三','四','五','六']; 
           
              str=str.replace(/yyyy|YYYY/,this.getFullYear());  
              str=str.replace(/yy|YY/,(this.getYear() % 100)>9?(this.getYear() % 100).toString():'0' + (this.getYear() % 100));  
           
              str=str.replace(/MM/,this.getMonth()>9?this.getMonth().toString():'0' + this.getMonth());  
              str=str.replace(/M/g,this.getMonth());  
           
              str=str.replace(/w|W/g,Week[this.getDay()]);  
           
              str=str.replace(/dd|DD/,this.getDate()>9?this.getDate().toString():'0' + this.getDate());  
              str=str.replace(/d|D/g,this.getDate());  
           
              str=str.replace(/hh|HH/,this.getHours()>9?this.getHours().toString():'0' + this.getHours());  
              str=str.replace(/h|H/g,this.getHours());  
              str=str.replace(/mm/,this.getMinutes()>9?this.getMinutes().toString():'0' + this.getMinutes());  
              str=str.replace(/m/g,this.getMinutes());  
           
              str=str.replace(/ss|SS/,this.getSeconds()>9?this.getSeconds().toString():'0' + this.getSeconds());  
              str=str.replace(/s|S/g,this.getSeconds());  
           
              return str;  
          }  
           
          //+--------------------------------------------------- 
          //| 求兩個時間的天數差 日期格式為 YYYY-MM-dd  
          //+--------------------------------------------------- 
          function daysBetween(DateOne,DateTwo) 
          {  
              var OneMonth = DateOne.substring(5,DateOne.lastIndexOf ('-')); 
              var OneDay = DateOne.substring(DateOne.length,DateOne.lastIndexOf ('-')+1); 
              var OneYear = DateOne.substring(0,DateOne.indexOf ('-')); 
           
              var TwoMonth = DateTwo.substring(5,DateTwo.lastIndexOf ('-')); 
              var TwoDay = DateTwo.substring(DateTwo.length,DateTwo.lastIndexOf ('-')+1); 
              var TwoYear = DateTwo.substring(0,DateTwo.indexOf ('-')); 
           
              var cha=((Date.parse(OneMonth+'/'+OneDay+'/'+OneYear)- Date.parse(TwoMonth+'/'+TwoDay+'/'+TwoYear))/86400000);  
              return Math.abs(cha); 

           
           
          //+--------------------------------------------------- 
          //| 日期計算 
          //+--------------------------------------------------- 
          Date.prototype.DateAdd = function(strInterval, Number) {  
              var dtTmp = this; 
              switch (strInterval) {  
                  case 's' :return new Date(Date.parse(dtTmp) + (1000 * Number)); 
                  case 'n' :return new Date(Date.parse(dtTmp) + (60000 * Number)); 
                  case 'h' :return new Date(Date.parse(dtTmp) + (3600000 * Number)); 
                  case 'd' :return new Date(Date.parse(dtTmp) + (86400000 * Number)); 
                  case 'w' :return new Date(Date.parse(dtTmp) + ((86400000 * 7) * Number)); 
                  case 'q' :return new Date(dtTmp.getFullYear(), (dtTmp.getMonth()) + Number*3, dtTmp.getDate(), dtTmp.getHours(), dtTmp.getMinutes(), dtTmp.getSeconds()); 
                  case 'm' :return new Date(dtTmp.getFullYear(), (dtTmp.getMonth()) + Number, dtTmp.getDate(), dtTmp.getHours(), dtTmp.getMinutes(), dtTmp.getSeconds()); 
                  case 'y' :return new Date((dtTmp.getFullYear() + Number), dtTmp.getMonth(), dtTmp.getDate(), dtTmp.getHours(), dtTmp.getMinutes(), dtTmp.getSeconds()); 
              } 

           
          //+--------------------------------------------------- 
          //| 比較日期差 dtEnd 格式為日期型或者 有效日期格式字符串 
          //+--------------------------------------------------- 
          Date.prototype.DateDiff = function(strInterval, dtEnd) {  
              var dtStart = this; 
              if (typeof dtEnd == 'string' )//如果是字符串轉換為日期型 
              {  
                  dtEnd = StringToDate(dtEnd); 
              } 
              switch (strInterval) {  
                  case 's' :return parseInt((dtEnd - dtStart) / 1000); 
                  case 'n' :return parseInt((dtEnd - dtStart) / 60000); 
                  case 'h' :return parseInt((dtEnd - dtStart) / 3600000); 
                  case 'd' :return parseInt((dtEnd - dtStart) / 86400000); 
                  case 'w' :return parseInt((dtEnd - dtStart) / (86400000 * 7)); 
                  case 'm' :return (dtEnd.getMonth()+1)+((dtEnd.getFullYear()-dtStart.getFullYear())*12) - (dtStart.getMonth()+1); 
                  case 'y' :return dtEnd.getFullYear() - dtStart.getFullYear(); 
              } 

           
          //+--------------------------------------------------- 
          //| 日期輸出字符串,重載了系統的toString方法 
          //+--------------------------------------------------- 
          Date.prototype.toString = function(showWeek) 
          {  
              var myDate= this; 
              var str = myDate.toLocaleDateString(); 
              if (showWeek) 
              {  
                  var Week = ['日','一','二','三','四','五','六']; 
                  str += ' 星期' + Week[myDate.getDay()]; 
              } 
              return str; 

           
          //+--------------------------------------------------- 
          //| 日期合法性驗證 
          //| 格式為:YYYY-MM-DD或YYYY/MM/DD 
          //+--------------------------------------------------- 
          function IsValidDate(DateStr)  
          {  
              var sDate=DateStr.replace(/(^\s+|\s+$)/g,''); //去兩邊空格;  
              if(sDate=='') return true;  
              //如果格式滿足YYYY-(/)MM-(/)DD或YYYY-(/)M-(/)DD或YYYY-(/)M-(/)D或YYYY-(/)MM-(/)D就替換為''  
              //數據庫中,合法日期可以是:YYYY-MM/DD(2003-3/21),數據庫會自動轉換為YYYY-MM-DD格式  
              var s = sDate.replace(/[\d]{ 4,4 }[\-/]{ 1 }[\d]{ 1,2 }[\-/]{ 1 }[\d]{ 1,2 }/g,'');  
              if (s=='') //說明格式滿足YYYY-MM-DD或YYYY-M-DD或YYYY-M-D或YYYY-MM-D  
              {  
                  var t=new Date(sDate.replace(/\-/g,'/'));  
                  var ar = sDate.split(/[-/:]/);  
                  if(ar[0] != t.getYear() || ar[1] != t.getMonth()+1 || ar[2] != t.getDate())  
                  {  
                      //alert('錯誤的日期格式!格式為:YYYY-MM-DD或YYYY/MM/DD。注意閏年。');  
                      return false;  
                  }  
              }  
              else  
              {  
                  //alert('錯誤的日期格式!格式為:YYYY-MM-DD或YYYY/MM/DD。注意閏年。');  
                  return false;  
              }  
              return true;  
          }  
           
          //+--------------------------------------------------- 
          //| 日期時間檢查 
          //| 格式為:YYYY-MM-DD HH:MM:SS 
          //+--------------------------------------------------- 
          function CheckDateTime(str) 
          {  
              var reg = /^(\d+)-(\d{ 1,2 })-(\d{ 1,2 }) (\d{ 1,2 }):(\d{ 1,2 }):(\d{ 1,2 })$/;  
              var r = str.match(reg);  
              if(r==null)return false;  
              r[2]=r[2]-1;  
              var d= new Date(r[1],r[2],r[3],r[4],r[5],r[6]);  
              if(d.getFullYear()!=r[1])return false;  
              if(d.getMonth()!=r[2])return false;  
              if(d.getDate()!=r[3])return false;  
              if(d.getHours()!=r[4])return false;  
              if(d.getMinutes()!=r[5])return false;  
              if(d.getSeconds()!=r[6])return false;  
              return true;  
          }  
           
          //+--------------------------------------------------- 
          //| 把日期分割成數組 
          //+--------------------------------------------------- 
          Date.prototype.toArray = function() 
          {  
              var myDate = this; 
              var myArray = Array(); 
              myArray[0] = myDate.getFullYear(); 
              myArray[1] = myDate.getMonth(); 
              myArray[2] = myDate.getDate(); 
              myArray[3] = myDate.getHours(); 
              myArray[4] = myDate.getMinutes(); 
              myArray[5] = myDate.getSeconds(); 
              return myArray; 

           
          //+--------------------------------------------------- 
          //| 取得日期數據信息 
          //| 參數 interval 表示數據類型 
          //| y 年 m月 d日 w星期 ww周 h時 n分 s秒 
          //+--------------------------------------------------- 
          Date.prototype.DatePart = function(interval) 
          {  
              var myDate = this; 
              var partStr=''; 
              var Week = ['日','一','二','三','四','五','六']; 
              switch (interval) 
              {  
                  case 'y' :partStr = myDate.getFullYear();break; 
                  case 'm' :partStr = myDate.getMonth()+1;break; 
                  case 'd' :partStr = myDate.getDate();break; 
                  case 'w' :partStr = Week[myDate.getDay()];break; 
                  case 'ww' :partStr = myDate.WeekNumOfYear();break; 
                  case 'h' :partStr = myDate.getHours();break; 
                  case 'n' :partStr = myDate.getMinutes();break; 
                  case 's' :partStr = myDate.getSeconds();break; 
              } 
              return partStr; 

           
          //+--------------------------------------------------- 
          //| 取得當前日期所在月的最大天數 
          //+--------------------------------------------------- 
          Date.prototype.MaxDayOfDate = function() 
          {  
              var myDate = this; 
              var ary = myDate.toArray(); 
              var date1 = (new Date(ary[0],ary[1]+1,1)); 
              var date2 = date1.dateAdd(1,'m',1); 
              var result = dateDiff(date1.Format('yyyy-MM-dd'),date2.Format('yyyy-MM-dd')); 
              return result; 

           
          //+--------------------------------------------------- 
          //| 取得當前日期所在周是一年中的第幾周 
          //+--------------------------------------------------- 
          Date.prototype.WeekNumOfYear = function() 
          {  
              var myDate = this; 
              var ary = myDate.toArray(); 
              var year = ary[0]; 
              var month = ary[1]+1; 
              var day = ary[2]; 
              document.write('< script language=VBScript\> \n'); 
              document.write('myDate = DateValue(''+month+'-'+day+'-'+year+'') \n'); 
              document.write('result = DatePart('ww', myDate) \n'); 
              document.write(' \n'); 
              return result; 

           
          //+--------------------------------------------------- 
          //| 字符串轉成日期類型  
          //| 格式 MM/dd/YYYY MM-dd-YYYY YYYY/MM/dd YYYY-MM-dd 
          //+--------------------------------------------------- 
          function StringToDate(DateStr) 
          {  
           
              var converted = Date.parse(DateStr); 
              var myDate = new Date(converted); 
              if (isNaN(myDate)) 
              {  
                  //var delimCahar = DateStr.indexOf('/')!=-1?'/':'-'; 
                  var arys= DateStr.split('-'); 
                  myDate = new Date(arys[0],--arys[1],arys[2]); 
              } 
              return myDate; 

           
          //今天

           function countDay(){
             
              myDate = new Date();
              var todadys= myDate.getFullYear()+'-'+(parseInt(myDate.getMonth())+1)+'-'+myDate.getDate()+' 00:00:00';
              var todadye= myDate.getFullYear()+'-'+(parseInt(myDate.getMonth())+1)+'-'+myDate.getDate()+' 23:59:59';
              
              ...
              
              
              }

          //本月

           function countMonth(){

          var d  = new  Date();   
          d =  new  Date(d.getFullYear(),d.getMonth()+1,0);
                 
               var months= myDate.getFullYear()+'-'+(parseInt(myDate.getMonth())+1)+'-'+'01 00:00:00';
               var monthe= myDate.getFullYear()+'-'+(parseInt(myDate.getMonth())+1)+'-'+d.getDate()+' 23:59:59';

          ...

          }

          posted @ 2011-03-09 10:12 edsion 閱讀(184) | 評論 (0)編輯 收藏

          1.什么是動態代理?

          答:動態代理可以提供對另一個對象的訪問,同時隱藏實際對象的具體事實。代理一般會實現 它所表示的實際對象的接口。代理可以訪問實際對象,但是延遲實現實際對象的部分功能,實際對象實現系統的實際功能,代理對象對客戶隱藏了實際對象。客戶不 知道它是與代理打交道還是與實際對象打交道。
          2.為什么使用動態代理?

          答:因為動態代理可以對請求進行任何處理

          3.使用它有哪些好處?

          答:因為動態代理可以對請求進行任何處理
          4.哪些地方需要動態代理?

          答:不允許直接訪問某些類;對訪問要做特殊處理等

           

          目前Java開發包中包含了對動態代理的支持,但是其實現只支持對接口的的實現。 其實現主要通過java.lang.reflect.Proxy類和java.lang.reflect.InvocationHandler接口。 

          Proxy類主要用來獲取動態代理對象,InvocationHandler接口用來約束調用者實現

          以下為模擬案例,通過動態代理實現在方法調用前后向控制臺輸出兩句字符串

          目錄結構

          <br/>

          定義一個HelloWorld接口

          1 package com.ljq.test;

          2

          3  /**

          4 * 定義一個HelloWorld接口

          5 *

          6 * @author jiqinlin

          7 *

          8 */

          9  public interface HelloWorld {

          10 public void sayHelloWorld();

          11 }

          <br/>

          類HelloWorldImpl是HelloWorld接口的實現

          1 package com.ljq.test;

          2
          3  /**
          4 * 類HelloWorldImpl是HelloWorld接口的實現
          5 *
          6 * @author jiqinlin
          7 *
          8 */
          9  public class HelloWorldImpl implements HelloWorld{
          10
          11 public void sayHelloWorld() {

          12 System.out.println("HelloWorld!");

          13 }

          14

          15 }

          HelloWorldHandler是 InvocationHandler接口實現

          1 package com.ljq.test;

          2

          3  import java.lang.reflect.InvocationHandler;

          4  import java.lang.reflect.Method;

          5

          6  /**

          7 * 實現在方法調用前后向控制臺輸出兩句字符串

          8 *

          9 * @author jiqinlin

          10 *

          11 */

          12  public class HelloWorldHandler implements InvocationHandler{

          13 //要代理的原始對象

          14   private Object obj;

          15

          16 public HelloWorldHandler(Object obj) {

          17 super();

          18 this.obj = obj;

          19 }

          20

          21 /**

          22 * 在代理實例上處理方法調用并返回結果

          23 *

          24 * @param proxy 代理類

          25 * @param method 被代理的方法

          26 * @param args 該方法的參數數組

          27 */

          28 public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

          29 Object result = null;

          30 //調用之前

          31   doBefore();

          32 //調用原始對象的方法

          33 result=method.invoke(obj, args);

          34 //調用之后

          35 doAfter();

          36 return result;

          37 }

          38

          39 private void doBefore(){

          40 System.out.println("before method invoke");

          41 }

          42

          43 private void doAfter(){

          44 System.out.println("after method invoke");

          45 }

          46

          47 }

          測試類

          package com.ljq.test;



          import java.lang.reflect.InvocationHandler;

          import java.lang.reflect.Proxy;





          public class HelloWorldTest {



          public static void main(String[] args) {

          HelloWorld helloWorld
          =new HelloWorldImpl();

          InvocationHandler handler
          =new HelloWorldHandler(helloWorld);



          //創建動態代理對象

          HelloWorld proxy=(HelloWorld)Proxy.newProxyInstance(

          helloWorld.getClass().getClassLoader(),

          helloWorld.getClass().getInterfaces(),

          handler);

          proxy.sayHelloWorld();

          }

          }

          運行結果為:

          posted @ 2011-02-24 16:18 edsion 閱讀(260) | 評論 (0)編輯 收藏

               摘要:   新建項目 2 學習建立user-library-hibernate,并加入相應的jar包   a項目右鍵-build path-configure build path-add library   b選擇user-library,在其中新建library,命命為hibernate   c 在該library中加入hibernate所需要的j...  閱讀全文

          posted @ 2011-02-24 10:12 edsion 閱讀(1320) | 評論 (0)編輯 收藏

          引言

              設計模式是面向對象思想的集大成,GOF在其經典著作中總結了23種設計模式,又可分為:創建型、結構型和行為型3個大類。對于軟件設計者來說,一般的過程就是在熟練掌握語言背景的基礎上,了解類庫的大致框架和常用的函數和接口等,然后多再在百般錘煉中,提高對軟件設計思想的認識。

              軟件設計者要清楚自己的定位和方向,一味的沉溺于技術細節的思路是制約個人技術走向成熟的毒藥。因此,學習軟件設計,了解軟件工程,是每個開發人員必備的一課。筆者在此不想詳細的描述各個設計模式的細節,我想google和baidu上的資料已經多如牛毛了。而且,爭取的學習方法也不是了解所有的設計模式就可以無敵于天下。我所強調的學習方法就是在熟練掌握基本要素的基礎上,了解大致的框架。這一條不僅是學習類庫的方法,對設計模式來說是可行的。同時,切記的是在平時的積累中,不斷的體會和實踐。因此,本文的目的就是將23種模式中,必須掌握的幾個最關鍵、最常用的設計模式,做以總結和簡述。 

              1 Factory Pattern 

              上榜理由:將程序中創建對象的操作,單獨出來處理,大大提高了系統擴展的柔性,接口的抽象化處理給相互依賴的對象創建提供了最好的抽象模式。 

              2 Facade Pattern 

              上榜理由:將表現層和邏輯層隔離,封裝底層的復雜處理,為用戶提供簡單的接口,這樣的例子隨處可見。門面模式很多時候更是一種系統架構的設計,在我所做的項目中,就實現了門面模式的接口,為復雜系統的解耦提供了最好的解決方案。 

              3 Command Pattern 

              上榜理由:將請求封裝為對象,從而將命令的執行和責任分開。通常在隊列中等待命令,這和現實多么的相似呀。如果你喜歡發號施令,請考慮你的ICommond吧。

              4 Strategy Pattern 

              上榜理由:策略模式,將易于變化的部分封裝為接口,通常Strategy 封裝一些運算法則,使之能互換。Bruce Zhang在他的博客中提到策略模式其實是一種“面向接口”的編程方法,真是恰如其分。 

              5 Iterator Pattern 

              上榜理由:相信任何的系統中,都會用到數組、集合、鏈表、隊列這樣的類型吧,那么你就不得不關心迭代模式的來龍去脈。在遍歷算法中,迭代模式提供了遍歷的順序訪問容器,GOF給出的定義為:提供一種方法訪問一個容器(container)對象中各個元素,而又不需暴露該對象的內部細節。.NET中就是使用了迭代器來創建用于foreach的集合。 

              6 Adapter Pattern 

              上榜理由:在原類型不做任何改變的情況下,擴展了新的接口,靈活且多樣的適配一切舊俗。這種打破舊框框,適配新格局的思想,是面向對象的精髓。以繼承方式實現的類的Adapter模式和以聚合方式實現的對象的Adapter模式,各有千秋,各取所長。看來,把它叫做包裝器一點也不為過,

           7 Observer Pattern

           上榜理由:定義對象間的一種一對多的依賴關系,當一個對象的狀態發生改變時, 所有依賴于它的對象都得到通知并被自動更新。觀察者和被觀察者的分開,為模塊劃分提供了清晰的界限。在.NET中使用委托和事件可以更好的實現觀察者模式,事件的注冊和撤銷不就對應著觀察者對其對象的觀察嗎? 

              8 Bridge Pattern 

              上榜理由:把實現和邏輯分開,對于我們深刻理解面向對象的聚合復用的思想甚有助益。
            
              9 Singleton Pattern 

              上榜理由:改善全局變量和命名空間的沖突,可以說是一種改良了的全局變量。這種一個類只有一個實例,且提供一個訪問全局點的方式,更加靈活的保證了實例的創建和訪問約束。.NET Frameeork已經封裝了Singleton類,我們拿來即可。

              總結

              仁者見仁。以上只是筆者一家之言,更重要的真知灼見皆來源于實踐,設計思想和模式的應用也來源于不斷的學習和反復,我也將一如既往。此文只是開端,未來才是不斷的探索。

          posted @ 2011-02-22 10:19 edsion 閱讀(324) | 評論 (0)編輯 收藏

               摘要: 概述 本章講述Struts2的工作原理。 讀者如果曾經學習過Struts1.x或者有過Struts1.x的開發經驗,那么千萬不要想當然地以為這一章可以跳過。實際上Struts1.x與Struts2并無我們想象的血緣關系。雖然Struts2的開發小組極力保留Struts1.x的習慣,但因為Struts2的核心設計完全改變,從思想到設計到工作流程,都有了很大的不同。 Struts2是Struts...  閱讀全文

          posted @ 2011-02-22 09:32 edsion 閱讀(1058) | 評論 (0)編輯 收藏

        1. 1.  各種不同數據庫間類型與 Java 類型映射關系不同導致 E-R 設計的難度,  使得 o/r 的數據庫可移植性大打折扣。   
        2. 2.   business 發生變化 需要修改 表結構時,  需要修改的地方有 表, 映射文件, 實體, E-R 圖,  這些工作都非常繁瑣且容易出錯,  如果先設計 Entity Object,  不會存在上述問題.
        3. 3.  非 OO 的設計思路可能會導致涉及到多表查詢的復雜度增加. 
        4. posted @ 2011-02-21 11:02 edsion 閱讀(145) | 評論 (0)編輯 收藏

              前些天,我負責調研了JPA,持久化,主要是結合EJB3.0;這個項目還用了JIDE、TWaver ;相對jpa來講,這個如果你之前做過Hibernate項目,你將會很容易理解JPA的應用。調研內容如下: 
               4.   JPA 1.0 
                     4.1 ORM: 單表,1:1, 1:n, n:m 
                     4.2 其他: PK生成策略,延遲加載策略,級聯(cascading)選項
                     4.3 JPQL: 單表查詢,多表結合查詢
          1.  概述 
                  Java Persistence API作為JavaEE5.0平臺的標準的ORM規范,將得到所有JavaEE所有服務器的支持。
                  JPA由EJB3.0軟件開發組專家開發,作為JSR-220的實現的一部分。
                  目前hibernate、Toplink以及OpenJPA都提供了JPA的支持。
          JPA包括三個方面技術:
                 ORM映射元數據,JPA支持XML與JDK5.0注釋,元數據描述對象與表之間的映射關系,框架可以將實體對象持久化到數據庫當中。
                 JPA持久化API,用來操作實體對象,執行curd操作,框架在后臺替我們完成了所有的事情,開發者可以從JDBC和SQL代碼中解脫出來。
                 查詢語言,這是持久化操作很重要的一個概念。通過面向對象而非面向數據庫的查詢數據,避免程序與SQL的緊密耦合。
          2.  實體對象
                  訪問數據庫之前,我們總是要設計在應用層承載數據的領域對象,ORM將它們持久化到數據庫當中。
                  按照JPA規范實體必須具備以下要求:
                  必須使用javax.persistence.Entity注解或者使用XML映射文件中相對應的元素。
                  必須具備一個不帶參數的構造函數,類不能聲明成final,方法和持久化的屬性也不能聲明成final,并且被持久化的屬性訪問修飾符不能為public。
                  如果游離狀的實體需要以值的方式進行傳遞,如EJB的Session bean,則必須實現Serializable接口。

          3.  基本注解

                  @Entity:將領域模型標注為一個實體,默認情況下類為表名,可以通過name屬性重新指定。
                  @Id:屬性對應表的主鍵。

                  @Column:屬性對應表的字段。

                  @GenaratedValue:主鍵生成策略,通過startage屬性指定。在4.0會解釋主鍵生成策略。

           

          4.  主鍵生成策略

                  @Target({METHOD, FIELD}) @Retention(RUNTIME)
                            public @interface GeneratedValue {
                             GenerationType strategy() default AUTO;
                             String generator() default "";
                    }

                   IDENTITY:主鍵由數據庫自動生成。

                   AUTO:JPA自動選擇合適的主鍵生成策略,默認選線。

                   SEQUENCE:通過序列產生主鍵,條件是數據庫必須支持序列。

                   TABLE:通過表產生主鍵,使用該策略更易于數據庫的移值。

          5.  實體關系

                  實體之間的關系有一對一,一對多,多對一,多多對。關系是多態的。

                  多對一

                   @Target({METHOD, FIELD}) @Retention(RUNTIME)
                  public @interface ManyToOne {
                        Class targetEntity() default void.class;
                        CascadeType[] cascade() default {};
                        FetchType fetch() default EAGER;
                        boolean optional() default true;
                  }

                  一對一

                 @Target({METHOD, FIELD}) @Retention(RUNTIME)
                    public @interface OneToOne {
                    Class targetEntity() default void.class;
                    CascadeType[] cascade() default {};
                    FetchType fetch() default EAGER;
                    boolean optional() default true;
                    String mappedBy() default "";
                }

           

          6.  雙向關聯特性

                  雙向關系的的反向端必須通過OnToOne,OnToMany或者ManyToMany注解的MappedBy元素必須指向它的持久端。MappendBy表述主控端的屬性或字段。

                   一對多,多對一雙向關系中的多端必須是持久端,因此不能再ManyToOne中使用MappedBy屬性。

                   對于一對一雙向關聯關系,包括對應的外鍵(foreign key)那一段時持久化。

                   對于多對多雙向關聯關系都是持久端。

          7.  延遲加載

                   FetchType.EAGER:代表立即加載。

                   FetchType.LAYZ:代表延遲加載策略。

          8.  多級級聯

                  REFRESH指定當你在訪問期間,如果數據庫數據發生改變時,你的數據是否更 新。

                  PERSIST指定在保存1數據的時是否會同時保存級聯的n數據。

                  MERGE指定當1處于游離狀態被修改了,n的數據也會被修改。

                  REMOVE指定在刪除1的同時是否刪除與之級聯的n數據。

                  ALL指定包含所有的級聯的N方。

          9. Java Persistence Query language

                 基于首次在EJB2.0引用的EJB查詢語言,java持久化查詢語言是一種可移植的查詢語言。旨在以面向對象表達式語言的表達式,將SQL語法和簡單查詢語義綁定在一起.使用這種語言編寫的查詢是可移植的,可以被編譯成所有主流數據庫服務器上的SQL.

          10JPA2.0與JPA1.0

                 JPA2.0也跟Hibernate一樣,也分為一級緩存,二級緩存,而JPA1.0不支持二級緩存,二級緩存是跨事物場景的。當二級緩存有效時,你就不能通過事物來保護并發數據,而只能依靠鎖的策略。

          11 事物與EntityManager

                 EntityManager對象事物管理分為二種,由JTA和RESOURCE_LOCAL來控制。

                 JTA EntityManager一般在容器里面使用,而RESOURCE_LOCAL(應用托管的EntityManager)數據庫本地化事物,一般在J2SE環境下使用。

                  通過容器來傳遞PersistenceContext而不是通過程序來自己傳遞EntityManager,它的生命周期由容器進行管理。

                 在J2SE環境下,RESOURCE_LOCAL由程序來創建EntityManagerFactory,并由EntityManagerFactory來創建EntityManger,這種方式就是RESOURCE_LOCAL的基本使用方式。

           12  監聽實體調用

                  監聽實體主要有PostLoad , PostPersist , PostRemove , PostUpdate, pre-persist , pre-remove 等。調用如下:
          public class UserListeners {
           @PostRemove
           public void dothing(User user) {
            System.out.println("加載..........." + user.getUserName());
           }
          } @Entity
          @org.hibernate.annotations.Entity(dynamicInsert = true)
          @EntityListeners(UserListeners.class)
          public class User implements Serializable{
          ………………..
          }

          13  樂觀鎖定

                  樂觀鎖定它可以確保在實體的狀態從數據庫讀取出來,在沒有中間插入的其它事物更改了與這個實體對應數據庫記錄的情況下,才把更新后實體的狀態更新到數據庫。它確保對數據的更新和刪除與數據庫的當前狀態保持一致,并且不會丟失中間的修改。如果應用程序想啟用實體的樂觀鎖,就必須為實體指定version屬性。如果沒有將樂觀鎖作為實體狀態的一部分進行定義,應用程序就要自己去維護數據的完整性。 

          直接映射

           

          實體的生命周期

             EntityManager用于管理實體實例的生命周期,一個實體實例可以分為new,managed,detached,或remove
             一個新創立的實體實例沒有持久化標識,并且還沒有跟持久化上下相關聯。
             一個受管的實體實例有一個正與持久化上下文相關聯的持久化標識。
             一個脫管的實體實例有持久化標識,但是這個標識沒有跟持久化上下文相關聯。
             被刪除的實體實例有持久化標示,并且與持久化上下文相關聯,但是已經計劃從數據庫中刪除。

          @Enumerated      
          @Temporal      
          @Lob      
          @Transient    


           關系映射      
           @OneToOne      
          @ManyToOne      
          @OneToMany      
          @ManyToMany      
          @MapKey      
          @OrderBy      

          繼承

          @DiscriminatorColumn      
          @DiscriminatorValue      
          @MappedSuperclass      
          @AssociationOverride      
          @AssociationOverrides      
          @AttributeOverride      
          @AttributeOverrides

          鎖定

          @PostPersist      
          @PreRemove      
          @PostRemove      
          @PreUpdate      
          @PostUpdate      
          @PostLoad      
          @EntityListeners      
          @ExcludeDefaultListeners      
          @ExcludeSuperclassListeners

          查詢

          @NamedQueries      
          @NamedNativeQuery      
          @NamedNativeQueries      
          @QueryHint      
          @ColumnResult      
          @EntityResult      
          @FieldResult      
          @SqlResultSetMapping      
          @SqlResultSetMappings 

           

                綜合上面調研的內容,上傳一個基本(簡單)的例子。該例子主要描述了JPA雙向一對一,雙向一對多,雙向多對多,單向一對一,單向一對多,單向多對一與單向多對多,測試代碼中使用了級聯、延遲加載策略,單表查詢、多表查詢JPQL。JPA版本采用Hibernate3.2

          posted @ 2011-02-21 10:22 edsion 閱讀(758) | 評論 (0)編輯 收藏

          主站蜘蛛池模板: 四子王旗| 台州市| 平顶山市| 马公市| 太保市| 兴城市| 神木县| 邢台市| 怀化市| 金秀| 绥棱县| 马山县| 江口县| 大理市| 额尔古纳市| 白水县| 红桥区| 滦南县| 乌苏市| 阿图什市| 同心县| 竹山县| 江源县| 灵武市| 临沧市| 木兰县| 田林县| 固安县| 潼南县| 海伦市| 丹江口市| 稷山县| 东阿县| 卢龙县| 辽宁省| 灵山县| 嘉黎县| 格尔木市| 唐山市| 庆安县| 当涂县|