2014年3月5日

          題目描述:
          輸入一個整形數組,數組里有正數也有負數。數組中連續的一個或多個整數組成一個子數組,每個子數組都有一個和。

          求所有子數組的和的最大值。要求時間復雜度為O(n)。

          思路:

          我們可以使用分治法或者減治法來處理這個問題。

          分治法    
          目標:把1個大問題分成2個小問題,2個小問題還可以再分,直到問題規模小的可以簡單解決。

          將該數組等分成兩個子數組,假如知道左右兩側兩個數組的各自的最大子數組和,那么整個數組的最大子數組和可能為三種情況:

          • 右側數組的最大子數組和
          • 左側數組的最大子數組和
          • 左右兩側數組中間毗鄰位置連接形成的子數組的和的最大值(如-6,4,-3,2####5,-6,9,-8,左側最大值為4,右側為9,中間毗鄰位置從2和5向兩側相加,得到中間毗鄰子數組4,-3,2,5,-6,9和為11,三者比較得最大子數組和為11)。

          遞歸到數組中只包含一個數字。

          這種思路也是可行的。進行ln(n)次拆分,每次拆分后進行n次比較,所以算法復雜度為n*ln(n)。但還達不到題目的要求。

          java代碼:

           1 package com.island.info;
           2 /**
           3  * <p>Title: TestMaxArray.java</p>
           4  * <p>Description: 分治法求解連續和最大</p>
           5  * @date 2014-3-05
           6  *
           7  */
           8  
           9 public class MaxSub {
          10         static int arr[] = {4,-3,5,-2,-1,2,6,-2};    //也可以隨機生成
          11         public static void main(String args[]){
          12             System.out.println(max(arr));
          13         }
          14  
          15         //包裝函數
          16         public static int max(final int[] arr){
          17             System.out.println("(1)*****arr.length-1----------------->:"+ (arr.length-1));
          18             return max(arr,0,arr.length-1);
          19         }
          20  
          21         //核心代碼:遞歸調用max()
          22         public static int max(final int[] arr,int leftIndex, int rightIndex){
          23             System.out.println("(2)*****leftIndex--------rightIndex--->:"+leftIndex+"|***************|"+rightIndex);
          24             int sum = 0,leftHalfMax = 0, rightHalfMax = 0;
          25             if (rightIndex-leftIndex==0){
          26                 return arr[rightIndex];
          27             } else {
          28                 int center = (leftIndex+rightIndex)/2;//2分查找中間節點
          29                 int maxLeft = max(arr,leftIndex,center);//左邊最大的
          30                 int maxRight = max(arr,center+1,rightIndex);//右邊最大的
          31                 //以下是查找跨越中間點的最大子序列
          32                 //從中點到左側:
          33                 for (int i=center;i>=leftIndex;--i){
          34                     sum+=arr[i];
          35                     if (sum>leftHalfMax){
          36                         leftHalfMax = sum;
          37                     }
          38                 }
          39                 System.out.println("左邊的sum----------->:"+sum);
          40                 sum=0;
          41                 //從中點到右側
          42                 for (int i=center+1;i<=rightIndex;++i){
          43                     sum+=arr[i];
          44                     if (sum>rightHalfMax){
          45                         rightHalfMax = sum;
          46                     }
          47                 }
          48                 System.out.println("右邊的sum----------->:"+sum);
          49                 return max(maxLeft,maxRight,leftHalfMax+rightHalfMax);
          50             }
          51         }
          52  
          53         //三者取最大值
          54         public static int max(int a,int b,int c){
          55             return a>b?(a>c?a:c):(b>c?b:c);
          56         }
          57     }

          減治法

          目標:將問題規模不斷減小,直到可以簡單處理為止。

          假設我們已知一個數組的最大子數組和,現在在該數組后面增加一個數字,新數組的最大子數組和可能是什么呢:

          • 原數組的最大子數組和;
          • 新增加的數字為正數,和最右側的子數組形成了一個新的最大子數組和(所以為了通過一次遍歷完成,我們需要保存最右側的子數組的最大和)。

          然后將兩個數字進行比較即可。

          所以減治至數組只包含最左側一個數字,我們知道它的最大子數組和和最右側子數組最大和都為還數字,逐次加1個數字直到整個數組即可。

          java代碼:

           1 package com.island.info;
           2  
           3 /**
           4  * <p>Title: TestMaxArray.java</p>
           5  * <p>Description: 分治法求解連續和最大</p>
           6  * @date 2014-3-05
           7  *
           8  */
           9 public class MaxSubArraySum {
          10  
          11     private static long getMax(long a, long b) {
          12         return a > b ? a : b;
          13     }
          14      
          15     /** 
          16     * 獲得連續子數組的最大和 
          17     * @param array 
          18     * @return 最大和,此處用了Long型是為了表示當參數為null或空時,可以返回null,返回其它任何數字都可能引起歧義。 
          19     */ 
          20  
          21     public static Long getMax(int[] array) {
          22          
          23         if (array == null || array.length <= 0) {
          24             return null;
          25         }
          26          
          27         long maxSum = array[0]; //所有子數組中最大的和 
          28         long righteEdge = array[0]; //右側子數組的最大和
          29         for (int i = 1; i < array.length; i++) {
          30             //當右側子數組的最大和為負數時,對于新數組,右側子數組的最大和為新增加的數。  
          31             if (righteEdge < 0) {
          32                 righteEdge = array[i];
          33             } else { //為正數時,對于新數組,右側子數組的最大和為新增加的數 + 原來的最大和。  
          34                 righteEdge += array[i];
          35             }
          36             //所有子數組中最大的和  
          37             System.out.println("righteEdge-------------maxSum:"+righteEdge+"****************"+maxSum);
          38             maxSum = getMax(righteEdge, maxSum);
          39         }
          40         return maxSum;
          41     }
          42  
          43     public static void main(String[] args) {
          44         int[] array = {1-2310-472-5};
          45         //int arr[] = {4,-3,5,-2,-1,2,6,-2};  
          46         System.out.println("Max sum: " + MaxSubArraySum.getMax(array));
          47     }
          48  
          49 }

          posted @ 2014-03-05 11:47 dongisland 閱讀(1713) | 評論 (0)編輯 收藏

          Ajax的原理就是:通過javascript的方式,將前臺數據通過xmlhttp對象傳遞到后臺,后臺在接收到請求后,將需要的結果,再傳回到前臺,這樣就可以實現不需要頁面的回發,頁是數據實現來回傳遞,從頁實現無刷新。 
          Ajax的原理簡單來說,實際上就是通過XmlHttpRequest對象來向服務器發異步請求,從服務器獲得數據,然后用javascript來操作DOM而更新頁面。 
          這其中最關鍵的一步就是從服務器獲得請求數據。要清楚這個過程和原理,我們必須對 XMLHttpRequest有所了解。 
          我們可以看出,XMLHttpRequest對象完全用來向服務器發出一個請求的,它的作用也局限于此,但它的作用是整個ajax實現的關鍵,我們可以把服務器端看成一個數據接口,它返回的是一個純文本流,當然,這個文本流可以是XML格式,可以是Html,可以是Javascript代碼,也可以只是一個字符串。這時候,XMLHttpRequest向服務器端請求這個頁面,服務器端將文本的結果寫入頁面,這和普通的web開發流程是一樣的,不同的是,客戶端在異步獲取這個結果后,不是直接顯示在頁面,而是先由javascript來處理,然后再顯示在頁面。

          posted @ 2014-03-05 11:27 dongisland 閱讀(613) | 評論 (0)編輯 收藏

          內連接:INNER  JOIN或者JOIN,把兩個表中數據對應的數據查出來。 
          外連接:OUTER  JOIN,以某個表為基礎把對應數據查出來,分為左外連接和右外連接。 
          左外連接:LEFT  JOIN或者LEFT  OUTER  JOIN,以某個表為基礎把對應數據查出來。 
          右外連接:RIGHT  JOIN或者RIGHT  OUTER  JOIN,以某個表為基礎把對應數據查出來。 
          全連接:FULL  JOIN,以多個表為基礎

          例子:   
             a表      id   name    
                        1   張3                 
                        2   李四                  
                        3   王武                 

              b表     id     job   parent_id   
                        1     23     1   
                         2     34     2   
                        3     34     4  
            a.id同b.parent_id   存在關系   
          內連接   
            select   a.*,b.*   from   a   inner   join   b     on   a.id=b.parent_id   
           結果是     
            1   張3          1     23     1   
            2   李四         2     34     2   
           左連接   
            select   a.*,b.*   from   a   left   join   b     on   a.id=b.parent_id   
          結果是     
            1   張3           1     23     1   
            2   李四          2     34     2   
            3   王武          null  
          右連接   
            select   a.*,b.*   from   a   right   join   b     on   a.id=b.parent_id   
            結果是     
            1   張3            1     23     1   
            2   李四           2     34     2   
            null                 3     34     4   
            完全連接   
            select   a.*,b.*   from   a   full   join   b     on   a.id=b.parent_id  
            結果是     
            1   張3            1     23     1   
            2   李四           2     34     2   
            null                 3     34     4   
            3   王武           null

          posted @ 2014-03-05 11:26 dongisland 閱讀(196) | 評論 (0)編輯 收藏

          一、Collections類和Collection接口

               Collections是針對集合類的一個幫助類,他提供一系列靜態方法實現對各種集合的搜索、排序、線程安全化等操作。

              Collection是最基本的集合接口,一個Collection代表一組Object,即Collection的元素(Elements)。一些 Collection允許相同的元素而另一些不行。一些能排序而另一些不行。Java SDK不提供直接繼承自Collection的 類,Java SDK提供的類都是繼承自Collection的“子接口”如List和Set。

          二、現在來談談Java集合的一些實現類。

          Collection
          List

          ArreyList 

          Vector

          LinkedList

          │└Stack

          └Set

          HashSet
          LinkedHashSet

          │└TreeSet

          List代表有序、重復的集合

          1.ArrayList類
            ArrayList實現了可變大小的數組。它允許所有元素,包括null。ArrayList沒有同步。
          size,isEmpty,get,set方法運行時間為常數。但是add方法開銷為分攤的常數,添加n個元素需要O(n)的時間。其他的方法運行時間為線性。
            每個ArrayList實例都有一個容量(Capacity),即用于存儲元素的數組的大小。這個容量可隨著不斷添加新元素而自動增加,但是增長算法 并沒有定義。當需要插入大量元素時,在插入前可以調用ensureCapacity方法來增加ArrayList的容量以提高插入效率。
            和LinkedList一樣,ArrayList也是非同步的(unsynchronized)。

          2.Vector類
            Vector非常類似ArrayList,但是Vector是同步的。由Vector創建的Iterator,雖然和ArrayList創建的Iterator是同一接口,但是,因為Vector是同步的,當一個Iterator被創建而且正在被使用,另一個線程改變了Vector的狀態(例如,添加或刪除了一些元素),這時調用Iterator的方法時將拋出ConcurrentModificationException,因此必須捕獲該異常。

          3.LinkedList類
            LinkedList實現了List接口,允許null元素。此外LinkedList提供額外的get,remove,insert方法在 LinkedList的首部或尾部。這些操作使LinkedList可被用作堆棧(stack),隊列(queue)或雙向隊列(deque)。
            注意LinkedList沒有同步方法。如果多個線程同時訪問一個List,則必須自己實現訪問同步。一種解決方法是在創建List時構造一個同步的List:
              List list = Collections.synchronizedList(new LinkedList(...));

          4.Stack 類
            Stack繼承自Vector,實現一個后進先出的堆棧。Stack提供5個額外的方法使得Vector得以被當作堆棧使用。基本的push和pop方法,還有peek方法得到棧頂的元素,empty方法測試堆棧是否為空,search方法檢測一個元素在堆棧中的位置。Stack剛創建后是空棧。

          Set代表無序、不可重復的集合
          HashSet和TreeSet的區別
          HashSet無序,TreeSet有序,兩者均不能有重復的對象

          Map
          HashMap

          Hashtable

          TreeMap
          └WeakHashMap

          Map沒有繼承Collection接口

          1.Hashtable類
            Hashtable繼承Map接口,實現一個key-value映射的哈希表。任何非空(non-null)的對象都可作為key或者value。Hashtable是同步的。

          2.HashMap類
            HashMap和Hashtable類似,不同之處在于 HashMap是非同步的,并且允許null,即null value和null key。,但是將HashMap視為Collection時 (values()方法可返回Collection),其迭代子操作時間開銷和HashMap的容量成比例。因此,如果迭代操作的性能相當重要的話,不要 將HashMap的初始化容量設得過高,或者load factor過低。

          3.WeakHashMap類
            WeakHashMap是一種改進的HashMap,它對key實行“弱引用”,如果一個key不再被外部所引用,那么該key可以被GC回收。

          三、集合類的遍歷

          遍歷通用Collection

          Iterator it = collection.iterator(); // 獲得一個迭代子
            while(it.hasNext()) {
             Object obj = it.next(); // 得到下一個元素
          }

          posted @ 2014-03-05 11:07 dongisland 閱讀(272) | 評論 (0)編輯 收藏

          1) 在Action實現類方面的對比:Struts 1要求Action類繼承一個抽象基類;Struts 1的一個具體問題是使用抽象類編程而不是接口。Struts 2 Action類可以實現一個Action接口,也可以實現其他接口,使可選和定制的服務成為可能。Struts 2提供一個ActionSupport基類去實現常用的接口。即使Action接口不是必須實現的,只有一個包含execute方法的POJO類都可以用作Struts 2的Action。 
          2) 線程模式方面的對比:Struts 1 Action是單例模式并且必須是線程安全的,因為僅有Action的一個實例來處理所有的請求。單例策略限制了Struts 1 Action能做的事,并且要在開發時特別小心。Action資源必須是線程安全的或同步的;Struts 2 Action對象為每一個請求產生一個實例,因此沒有線程安全問題。 即struts1每個action只有1個實例,struts2每一個請求都會new一個實例,所以struts2是線程安全的
          3) Servlet依賴方面的對比:Struts 1 Action依賴于Servlet API,因為Struts 1 Action的execute方法中有HttpServletRequest和HttpServletResponse方法。Struts 2 Action不再依賴于Servlet API,從而允許Action脫離Web容器運行,從而降低了測試Action的難度。 當然,如果Action需要直接訪問HttpServletRequest和HttpServletResponse參數,Struts 2 Action仍然可以訪問它們。但是,大部分時候,Action都無需直接訪問HttpServetRequest和HttpServletResponse,從而給開發者更多靈活的選擇。 
          4) 可測性方面的對比:測試Struts 1 Action的一個主要問題是execute方法依賴于Servlet API,這使得Action的測試要依賴于Web容器。為了脫離Web容器測試Struts 1的Action,必須借助于第三方擴展:Struts TestCase,該擴展下包含了系列的Mock對象(模擬了HttpServetRequest和HttpServletResponse對象),從而可以脫離Web容器測試Struts 1的Action類。Struts 2 Action可以通過初始化、設置屬性、調用方法來測試。 
          5) 封裝請求參數的對比:Struts 1使用ActionForm對象封裝用戶的請求參數,所有的ActionForm必須繼承一個基類:ActionForm。普通的JavaBean不能用作ActionForm,因此,開發者必須創建大量的ActionForm類封裝用戶請求參數。雖然Struts 1提供了動態ActionForm來簡化ActionForm的開發,但依然需要在配置文件中定義ActionForm;Struts 2直接使用Action屬性來封裝用戶請求屬性,避免了開發者需要大量開發ActionForm類的煩瑣,實際上,這些屬性還可以是包含子屬性的Rich對象類型。如果開發者依然懷念Struts 1 ActionForm的模式,Struts 2提供了ModelDriven模式,可以讓開發者使用單獨的Model對象來封裝用戶請求參數,但該Model對象無需繼承任何Struts 2基類,是一個POJO,從而降低了代碼污染。 
          6) 表達式語言方面的對比:Struts 1整合了JSTL,因此可以使用JSTL表達式語言。這種表達式語言有基本對象圖遍歷,但在對集合和索引屬性的支持上則功能不強;Struts 2可以使用JSTL,但它整合了一種更強大和靈活的表達式語言:OGNL(Object Graph Notation Language),因此,Struts 2下的表達式語言功能更加強大。 
          7) — 綁定值到視圖的對比:Struts 1使用標準JSP機制把對象綁定到視圖頁面;Struts 2使用“ValueStack”技術,使標簽庫能夠訪問值,而不需要把對象和視圖頁面綁定在一起。 
          8) 類型轉換的對比:Struts 1 ActionForm 屬性通常都是String類型。Struts 1使用Commons-Beanutils進行類型轉換,每個類一個轉換器,轉換器是不可配置的;Struts 2使用OGNL進行類型轉換,支持基本數據類型和常用對象之間的轉換。 
          9) 數據校驗的對比:Struts 1支持在ActionForm重寫validate方法中手動校驗,或者通過整合Commons alidator框架來完成數據校驗。Struts 2支持通過重寫validate方法進行校驗,也支持整合XWork校驗框架進行校驗。 
          10) Action執行控制的對比:Struts 1支持每一個模塊對應一個請求處理(即生命周期的概念),但是模塊中的所有Action必須共享相同的生命周期。Struts 2支持通過攔截器堆棧(Interceptor Stacks)為每一個Action創建不同的生命周期。開發者可以根據需要創建相應堆棧,從而和不同的Action一起使用。 
          11) 捕獲輸入:Struts1 使用ActionForm對象捕獲輸入。所有的ActionForm必須繼承一個基類。因為其他JavaBean不能用作ActionForm,開發者經常創建多余的類捕獲輸入。動態Bean(DynaBeans)可以作為創建傳統ActionForm的選擇,但是,開發者可能是在重新描述(創建)已經存在的JavaBean(仍然會導致有冗余的javabean)。Struts 2直接使用Action屬性作為輸入屬性,消除了對第二個輸入對象的需求。輸入屬性可能是有自己(子)屬性的rich對象類型。Action屬性能夠通過 web頁面上的taglibs訪問。Struts2也支持ActionForm模式。rich對象類型,包括業務對象,能夠用作輸入/輸出對象。這種 ModelDriven 特性簡化了taglib對POJO輸入對象的引用。

          posted @ 2014-03-05 10:52 dongisland 閱讀(191) | 評論 (0)編輯 收藏


          posts - 5, comments - 0, trackbacks - 0, articles - 0

          Copyright © dongisland

          主站蜘蛛池模板: 潢川县| 喜德县| 封开县| 满城县| 巴马| 泾源县| 梁河县| 巨野县| 澜沧| 南通市| 甘谷县| 枣阳市| 军事| 龙江县| 竹山县| 县级市| 衢州市| 芮城县| 成武县| 宿迁市| 伽师县| 赞皇县| 寻甸| 镇康县| 珠海市| 湘潭县| 星子县| 闸北区| 吉安县| 高台县| 德化县| 东阳市| 呈贡县| 邵阳市| 皮山县| 安徽省| 常宁市| 盐亭县| 潞城市| 蓬莱市| 金阳县|