爪哇島

          關(guān)注java
          posts - 6, comments - 0, trackbacks - 0, articles - 1

          2011年3月9日

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


          日期時(shí)間腳本庫(kù)方法列表

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

          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 時(shí)間 
          // 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;  
          }  
           
          //+--------------------------------------------------- 
          //| 求兩個(gè)時(shí)間的天數(shù)差 日期格式為 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); 

           
           
          //+--------------------------------------------------- 
          //| 日期計(jì)算 
          //+--------------------------------------------------- 
          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' )//如果是字符串轉(zhuǎn)換為日期型 
              {  
                  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(); 
              } 

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

           
          //+--------------------------------------------------- 
          //| 日期合法性驗(yàn)證 
          //| 格式為:YYYY-MM-DD或YYYY/MM/DD 
          //+--------------------------------------------------- 
          function IsValidDate(DateStr)  
          {  
              var sDate=DateStr.replace(/(^\s+|\s+$)/g,''); //去兩邊空格;  
              if(sDate=='') return true;  
              //如果格式滿(mǎn)足YYYY-(/)MM-(/)DD或YYYY-(/)M-(/)DD或YYYY-(/)M-(/)D或YYYY-(/)MM-(/)D就替換為''  
              //數(shù)據(jù)庫(kù)中,合法日期可以是:YYYY-MM/DD(2003-3/21),數(shù)據(jù)庫(kù)會(huì)自動(dòng)轉(zhuǎn)換為YYYY-MM-DD格式  
              var s = sDate.replace(/[\d]{ 4,4 }[\-/]{ 1 }[\d]{ 1,2 }[\-/]{ 1 }[\d]{ 1,2 }/g,'');  
              if (s=='') //說(shuō)明格式滿(mǎn)足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('錯(cuò)誤的日期格式!格式為:YYYY-MM-DD或YYYY/MM/DD。注意閏年。');  
                      return false;  
                  }  
              }  
              else  
              {  
                  //alert('錯(cuò)誤的日期格式!格式為:YYYY-MM-DD或YYYY/MM/DD。注意閏年。');  
                  return false;  
              }  
              return true;  
          }  
           
          //+--------------------------------------------------- 
          //| 日期時(shí)間檢查 
          //| 格式為: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;  
          }  
           
          //+--------------------------------------------------- 
          //| 把日期分割成數(shù)組 
          //+--------------------------------------------------- 
          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; 

           
          //+--------------------------------------------------- 
          //| 取得日期數(shù)據(jù)信息 
          //| 參數(shù) interval 表示數(shù)據(jù)類(lèi)型 
          //| y 年 m月 d日 w星期 ww周 h時(shí) 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; 

           
          //+--------------------------------------------------- 
          //| 取得當(dāng)前日期所在月的最大天數(shù) 
          //+--------------------------------------------------- 
          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; 

           
          //+--------------------------------------------------- 
          //| 取得當(dāng)前日期所在周是一年中的第幾周 
          //+--------------------------------------------------- 
          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; 

           
          //+--------------------------------------------------- 
          //| 字符串轉(zhuǎn)成日期類(lèi)型  
          //| 格式 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) | 評(píng)論 (0)編輯 收藏

          2011年2月24日

          1.什么是動(dòng)態(tài)代理?

          答:動(dòng)態(tài)代理可以提供對(duì)另一個(gè)對(duì)象的訪(fǎng)問(wèn),同時(shí)隱藏實(shí)際對(duì)象的具體事實(shí)。代理一般會(huì)實(shí)現(xiàn) 它所表示的實(shí)際對(duì)象的接口。代理可以訪(fǎng)問(wèn)實(shí)際對(duì)象,但是延遲實(shí)現(xiàn)實(shí)際對(duì)象的部分功能,實(shí)際對(duì)象實(shí)現(xiàn)系統(tǒng)的實(shí)際功能,代理對(duì)象對(duì)客戶(hù)隱藏了實(shí)際對(duì)象。客戶(hù)不 知道它是與代理打交道還是與實(shí)際對(duì)象打交道。
          2.為什么使用動(dòng)態(tài)代理?

          答:因?yàn)閯?dòng)態(tài)代理可以對(duì)請(qǐng)求進(jìn)行任何處理

          3.使用它有哪些好處?

          答:因?yàn)閯?dòng)態(tài)代理可以對(duì)請(qǐng)求進(jìn)行任何處理
          4.哪些地方需要?jiǎng)討B(tài)代理?

          答:不允許直接訪(fǎng)問(wèn)某些類(lèi);對(duì)訪(fǎng)問(wèn)要做特殊處理等

           

          目前Java開(kāi)發(fā)包中包含了對(duì)動(dòng)態(tài)代理的支持,但是其實(shí)現(xiàn)只支持對(duì)接口的的實(shí)現(xiàn)。 其實(shí)現(xiàn)主要通過(guò)java.lang.reflect.Proxy類(lèi)和java.lang.reflect.InvocationHandler接口。 

          Proxy類(lèi)主要用來(lái)獲取動(dòng)態(tài)代理對(duì)象,InvocationHandler接口用來(lái)約束調(diào)用者實(shí)現(xiàn)

          以下為模擬案例,通過(guò)動(dòng)態(tài)代理實(shí)現(xiàn)在方法調(diào)用前后向控制臺(tái)輸出兩句字符串

          目錄結(jié)構(gòu)

          <br/>

          定義一個(gè)HelloWorld接口

          1 package com.ljq.test;

          2

          3  /**

          4 * 定義一個(gè)HelloWorld接口

          5 *

          6 * @author jiqinlin

          7 *

          8 */

          9  public interface HelloWorld {

          10 public void sayHelloWorld();

          11 }

          <br/>

          類(lèi)HelloWorldImpl是HelloWorld接口的實(shí)現(xiàn)

          1 package com.ljq.test;

          2
          3  /**
          4 * 類(lèi)HelloWorldImpl是HelloWorld接口的實(shí)現(xiàn)
          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接口實(shí)現(xiàn)

          1 package com.ljq.test;

          2

          3  import java.lang.reflect.InvocationHandler;

          4  import java.lang.reflect.Method;

          5

          6  /**

          7 * 實(shí)現(xiàn)在方法調(diào)用前后向控制臺(tái)輸出兩句字符串

          8 *

          9 * @author jiqinlin

          10 *

          11 */

          12  public class HelloWorldHandler implements InvocationHandler{

          13 //要代理的原始對(duì)象

          14   private Object obj;

          15

          16 public HelloWorldHandler(Object obj) {

          17 super();

          18 this.obj = obj;

          19 }

          20

          21 /**

          22 * 在代理實(shí)例上處理方法調(diào)用并返回結(jié)果

          23 *

          24 * @param proxy 代理類(lèi)

          25 * @param method 被代理的方法

          26 * @param args 該方法的參數(shù)數(shù)組

          27 */

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

          29 Object result = null;

          30 //調(diào)用之前

          31   doBefore();

          32 //調(diào)用原始對(duì)象的方法

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

          34 //調(diào)用之后

          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 }

          測(cè)試類(lèi)

          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);



          //創(chuàng)建動(dòng)態(tài)代理對(duì)象

          HelloWorld proxy=(HelloWorld)Proxy.newProxyInstance(

          helloWorld.getClass().getClassLoader(),

          helloWorld.getClass().getInterfaces(),

          handler);

          proxy.sayHelloWorld();

          }

          }

          運(yùn)行結(jié)果為:

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

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

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

          2011年2月22日

          引言

              設(shè)計(jì)模式是面向?qū)ο笏枷氲募蟪桑珿OF在其經(jīng)典著作中總結(jié)了23種設(shè)計(jì)模式,又可分為:創(chuàng)建型、結(jié)構(gòu)型和行為型3個(gè)大類(lèi)。對(duì)于軟件設(shè)計(jì)者來(lái)說(shuō),一般的過(guò)程就是在熟練掌握語(yǔ)言背景的基礎(chǔ)上,了解類(lèi)庫(kù)的大致框架和常用的函數(shù)和接口等,然后多再在百般錘煉中,提高對(duì)軟件設(shè)計(jì)思想的認(rèn)識(shí)。

              軟件設(shè)計(jì)者要清楚自己的定位和方向,一味的沉溺于技術(shù)細(xì)節(jié)的思路是制約個(gè)人技術(shù)走向成熟的毒藥。因此,學(xué)習(xí)軟件設(shè)計(jì),了解軟件工程,是每個(gè)開(kāi)發(fā)人員必備的一課。筆者在此不想詳細(xì)的描述各個(gè)設(shè)計(jì)模式的細(xì)節(jié),我想google和baidu上的資料已經(jīng)多如牛毛了。而且,爭(zhēng)取的學(xué)習(xí)方法也不是了解所有的設(shè)計(jì)模式就可以無(wú)敵于天下。我所強(qiáng)調(diào)的學(xué)習(xí)方法就是在熟練掌握基本要素的基礎(chǔ)上,了解大致的框架。這一條不僅是學(xué)習(xí)類(lèi)庫(kù)的方法,對(duì)設(shè)計(jì)模式來(lái)說(shuō)是可行的。同時(shí),切記的是在平時(shí)的積累中,不斷的體會(huì)和實(shí)踐。因此,本文的目的就是將23種模式中,必須掌握的幾個(gè)最關(guān)鍵、最常用的設(shè)計(jì)模式,做以總結(jié)和簡(jiǎn)述。 

              1 Factory Pattern 

              上榜理由:將程序中創(chuàng)建對(duì)象的操作,單獨(dú)出來(lái)處理,大大提高了系統(tǒng)擴(kuò)展的柔性,接口的抽象化處理給相互依賴(lài)的對(duì)象創(chuàng)建提供了最好的抽象模式。 

              2 Facade Pattern 

              上榜理由:將表現(xiàn)層和邏輯層隔離,封裝底層的復(fù)雜處理,為用戶(hù)提供簡(jiǎn)單的接口,這樣的例子隨處可見(jiàn)。門(mén)面模式很多時(shí)候更是一種系統(tǒng)架構(gòu)的設(shè)計(jì),在我所做的項(xiàng)目中,就實(shí)現(xiàn)了門(mén)面模式的接口,為復(fù)雜系統(tǒng)的解耦提供了最好的解決方案。 

              3 Command Pattern 

              上榜理由:將請(qǐng)求封裝為對(duì)象,從而將命令的執(zhí)行和責(zé)任分開(kāi)。通常在隊(duì)列中等待命令,這和現(xiàn)實(shí)多么的相似呀。如果你喜歡發(fā)號(hào)施令,請(qǐng)考慮你的ICommond吧。

              4 Strategy Pattern 

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

              5 Iterator Pattern 

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

              6 Adapter Pattern 

              上榜理由:在原類(lèi)型不做任何改變的情況下,擴(kuò)展了新的接口,靈活且多樣的適配一切舊俗。這種打破舊框框,適配新格局的思想,是面向?qū)ο蟮木琛R岳^承方式實(shí)現(xiàn)的類(lèi)的Adapter模式和以聚合方式實(shí)現(xiàn)的對(duì)象的Adapter模式,各有千秋,各取所長(zhǎng)。看來(lái),把它叫做包裝器一點(diǎn)也不為過(guò),

           7 Observer Pattern

           上榜理由:定義對(duì)象間的一種一對(duì)多的依賴(lài)關(guān)系,當(dāng)一個(gè)對(duì)象的狀態(tài)發(fā)生改變時(shí), 所有依賴(lài)于它的對(duì)象都得到通知并被自動(dòng)更新。觀(guān)察者和被觀(guān)察者的分開(kāi),為模塊劃分提供了清晰的界限。在.NET中使用委托和事件可以更好的實(shí)現(xiàn)觀(guān)察者模式,事件的注冊(cè)和撤銷(xiāo)不就對(duì)應(yīng)著觀(guān)察者對(duì)其對(duì)象的觀(guān)察嗎? 

              8 Bridge Pattern 

              上榜理由:把實(shí)現(xiàn)和邏輯分開(kāi),對(duì)于我們深刻理解面向?qū)ο蟮木酆蠌?fù)用的思想甚有助益。
            
              9 Singleton Pattern 

              上榜理由:改善全局變量和命名空間的沖突,可以說(shuō)是一種改良了的全局變量。這種一個(gè)類(lèi)只有一個(gè)實(shí)例,且提供一個(gè)訪(fǎng)問(wèn)全局點(diǎn)的方式,更加靈活的保證了實(shí)例的創(chuàng)建和訪(fǎng)問(wèn)約束。.NET Frameeork已經(jīng)封裝了Singleton類(lèi),我們拿來(lái)即可。

              總結(jié)

              仁者見(jiàn)仁。以上只是筆者一家之言,更重要的真知灼見(jiàn)皆來(lái)源于實(shí)踐,設(shè)計(jì)思想和模式的應(yīng)用也來(lái)源于不斷的學(xué)習(xí)和反復(fù),我也將一如既往。此文只是開(kāi)端,未來(lái)才是不斷的探索。

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

               摘要: 概述 本章講述Struts2的工作原理。 讀者如果曾經(jīng)學(xué)習(xí)過(guò)Struts1.x或者有過(guò)Struts1.x的開(kāi)發(fā)經(jīng)驗(yàn),那么千萬(wàn)不要想當(dāng)然地以為這一章可以跳過(guò)。實(shí)際上Struts1.x與Struts2并無(wú)我們想象的血緣關(guān)系。雖然Struts2的開(kāi)發(fā)小組極力保留Struts1.x的習(xí)慣,但因?yàn)镾truts2的核心設(shè)計(jì)完全改變,從思想到設(shè)計(jì)到工作流程,都有了很大的不同。 Struts2是Struts...  閱讀全文

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

          2011年2月21日

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

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

          3.  基本注解

                  @Entity:將領(lǐng)域模型標(biāo)注為一個(gè)實(shí)體,默認(rèn)情況下類(lèi)為表名,可以通過(guò)name屬性重新指定。
                  @Id:屬性對(duì)應(yīng)表的主鍵。

                  @Column:屬性對(duì)應(yīng)表的字段。

                  @GenaratedValue:主鍵生成策略,通過(guò)startage屬性指定。在4.0會(huì)解釋主鍵生成策略。

           

          4.  主鍵生成策略

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

                   IDENTITY:主鍵由數(shù)據(jù)庫(kù)自動(dòng)生成。

                   AUTO:JPA自動(dòng)選擇合適的主鍵生成策略,默認(rèn)選線(xiàn)。

                   SEQUENCE:通過(guò)序列產(chǎn)生主鍵,條件是數(shù)據(jù)庫(kù)必須支持序列。

                   TABLE:通過(guò)表產(chǎn)生主鍵,使用該策略更易于數(shù)據(jù)庫(kù)的移值。

          5.  實(shí)體關(guān)系

                  實(shí)體之間的關(guān)系有一對(duì)一,一對(duì)多,多對(duì)一,多多對(duì)。關(guān)系是多態(tài)的。

                  多對(duì)一

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

                  一對(duì)一

                 @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.  雙向關(guān)聯(lián)特性

                  雙向關(guān)系的的反向端必須通過(guò)OnToOne,OnToMany或者M(jìn)anyToMany注解的MappedBy元素必須指向它的持久端。MappendBy表述主控端的屬性或字段。

                   一對(duì)多,多對(duì)一雙向關(guān)系中的多端必須是持久端,因此不能再M(fèi)anyToOne中使用MappedBy屬性。

                   對(duì)于一對(duì)一雙向關(guān)聯(lián)關(guān)系,包括對(duì)應(yīng)的外鍵(foreign key)那一段時(shí)持久化。

                   對(duì)于多對(duì)多雙向關(guān)聯(lián)關(guān)系都是持久端。

          7.  延遲加載

                   FetchType.EAGER:代表立即加載。

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

          8.  多級(jí)級(jí)聯(lián)

                  REFRESH指定當(dāng)你在訪(fǎng)問(wèn)期間,如果數(shù)據(jù)庫(kù)數(shù)據(jù)發(fā)生改變時(shí),你的數(shù)據(jù)是否更 新。

                  PERSIST指定在保存1數(shù)據(jù)的時(shí)是否會(huì)同時(shí)保存級(jí)聯(lián)的n數(shù)據(jù)。

                  MERGE指定當(dāng)1處于游離狀態(tài)被修改了,n的數(shù)據(jù)也會(huì)被修改。

                  REMOVE指定在刪除1的同時(shí)是否刪除與之級(jí)聯(lián)的n數(shù)據(jù)。

                  ALL指定包含所有的級(jí)聯(lián)的N方。

          9. Java Persistence Query language

                 基于首次在EJB2.0引用的EJB查詢(xún)語(yǔ)言,java持久化查詢(xún)語(yǔ)言是一種可移植的查詢(xún)語(yǔ)言。旨在以面向?qū)ο蟊磉_(dá)式語(yǔ)言的表達(dá)式,將SQL語(yǔ)法和簡(jiǎn)單查詢(xún)語(yǔ)義綁定在一起.使用這種語(yǔ)言編寫(xiě)的查詢(xún)是可移植的,可以被編譯成所有主流數(shù)據(jù)庫(kù)服務(wù)器上的SQL.

          10JPA2.0與JPA1.0

                 JPA2.0也跟Hibernate一樣,也分為一級(jí)緩存,二級(jí)緩存,而JPA1.0不支持二級(jí)緩存,二級(jí)緩存是跨事物場(chǎng)景的。當(dāng)二級(jí)緩存有效時(shí),你就不能通過(guò)事物來(lái)保護(hù)并發(fā)數(shù)據(jù),而只能依靠鎖的策略。

          11 事物與EntityManager

                 EntityManager對(duì)象事物管理分為二種,由JTA和RESOURCE_LOCAL來(lái)控制。

                 JTA EntityManager一般在容器里面使用,而RESOURCE_LOCAL(應(yīng)用托管的EntityManager)數(shù)據(jù)庫(kù)本地化事物,一般在J2SE環(huán)境下使用。

                  通過(guò)容器來(lái)傳遞PersistenceContext而不是通過(guò)程序來(lái)自己傳遞EntityManager,它的生命周期由容器進(jìn)行管理。

                 在J2SE環(huán)境下,RESOURCE_LOCAL由程序來(lái)創(chuàng)建EntityManagerFactory,并由EntityManagerFactory來(lái)創(chuàng)建EntityManger,這種方式就是RESOURCE_LOCAL的基本使用方式。

           12  監(jiān)聽(tīng)實(shí)體調(diào)用

                  監(jiān)聽(tīng)實(shí)體主要有PostLoad , PostPersist , PostRemove , PostUpdate, pre-persist , pre-remove 等。調(diào)用如下:
          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  樂(lè)觀(guān)鎖定

                  樂(lè)觀(guān)鎖定它可以確保在實(shí)體的狀態(tài)從數(shù)據(jù)庫(kù)讀取出來(lái),在沒(méi)有中間插入的其它事物更改了與這個(gè)實(shí)體對(duì)應(yīng)數(shù)據(jù)庫(kù)記錄的情況下,才把更新后實(shí)體的狀態(tài)更新到數(shù)據(jù)庫(kù)。它確保對(duì)數(shù)據(jù)的更新和刪除與數(shù)據(jù)庫(kù)的當(dāng)前狀態(tài)保持一致,并且不會(huì)丟失中間的修改。如果應(yīng)用程序想啟用實(shí)體的樂(lè)觀(guān)鎖,就必須為實(shí)體指定version屬性。如果沒(méi)有將樂(lè)觀(guān)鎖作為實(shí)體狀態(tài)的一部分進(jìn)行定義,應(yīng)用程序就要自己去維護(hù)數(shù)據(jù)的完整性。 

          直接映射

           

          實(shí)體的生命周期

             EntityManager用于管理實(shí)體實(shí)例的生命周期,一個(gè)實(shí)體實(shí)例可以分為new,managed,detached,或remove
             一個(gè)新創(chuàng)立的實(shí)體實(shí)例沒(méi)有持久化標(biāo)識(shí),并且還沒(méi)有跟持久化上下相關(guān)聯(lián)。
             一個(gè)受管的實(shí)體實(shí)例有一個(gè)正與持久化上下文相關(guān)聯(lián)的持久化標(biāo)識(shí)。
             一個(gè)脫管的實(shí)體實(shí)例有持久化標(biāo)識(shí),但是這個(gè)標(biāo)識(shí)沒(méi)有跟持久化上下文相關(guān)聯(lián)。
             被刪除的實(shí)體實(shí)例有持久化標(biāo)示,并且與持久化上下文相關(guān)聯(lián),但是已經(jīng)計(jì)劃從數(shù)據(jù)庫(kù)中刪除。

          @Enumerated      
          @Temporal      
          @Lob      
          @Transient    


           關(guān)系映射      
           @OneToOne      
          @ManyToOne      
          @OneToMany      
          @ManyToMany      
          @MapKey      
          @OrderBy      

          繼承

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

          鎖定

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

          查詢(xún)

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

           

                綜合上面調(diào)研的內(nèi)容,上傳一個(gè)基本(簡(jiǎn)單)的例子。該例子主要描述了JPA雙向一對(duì)一,雙向一對(duì)多,雙向多對(duì)多,單向一對(duì)一,單向一對(duì)多,單向多對(duì)一與單向多對(duì)多,測(cè)試代碼中使用了級(jí)聯(lián)、延遲加載策略,單表查詢(xún)、多表查詢(xún)JPQL。JPA版本采用Hibernate3.2

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

          主站蜘蛛池模板: 三亚市| 萨嘎县| 桓台县| 邳州市| 上饶市| 贡山| 黎平县| 乐陵市| 延长县| 昆山市| 大余县| 靖西县| 昌吉市| 舒兰市| 义马市| 文登市| 襄樊市| 淳化县| 巴东县| 罗定市| 柳州市| 钦州市| 慈溪市| 甘孜| 长沙县| 广州市| 三门县| 竹北市| 新龙县| 江阴市| 张掖市| 犍为县| 锡林浩特市| 万州区| 玉环县| 孝感市| 观塘区| 宜丰县| 利川市| 蒲江县| 金塔县|