<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    隨筆 - 18, 文章 - 0, 評論 - 8, 引用 - 0
    數據加載中……

    2005年8月6日

    CyclicBarrier 簡單舉例

    一句話解釋:預備~~~開始

     1 import java.util.concurrent.BrokenBarrierException;
     2 import java.util.concurrent.CyclicBarrier;
     3 
     4 import org.slf4j.Logger;
     5 import org.slf4j.LoggerFactory;
     6 
     7 public class CyclicBarrierLearn {
     8     
     9     private Logger log = LoggerFactory.getLogger(CyclicBarrierLearn.class);
    10     
    11     private class Work extends Thread {
    12         
    13         private String name;
    14         private CyclicBarrier cyclicBarrier;
    15         
    16         public Work(String name, CyclicBarrier cyclicBarrier) {
    17             this.name = name;
    18             this.cyclicBarrier = cyclicBarrier;
    19         }
    20         
    21         @Override
    22         public void run() {
    23             try {
    24                 log.debug("thread name: " + name + " waiting work");
    25                 cyclicBarrier.await();
    26                 log.debug("thread name: " + name + " working");
    27             } catch (InterruptedException e) {
    28                 e.printStackTrace();
    29             } catch (BrokenBarrierException e) {
    30                 e.printStackTrace();
    31             }
    32             
    33         }
    34     }
    35     
    36     public void cyclicBarrier() {
    37         CyclicBarrier cyclicBarrier = new CyclicBarrier(50, new Runnable() {
    38             
    39             @Override
    40             public void run() {
    41                 log.debug("let's begin work");
    42             }
    43         });
    44         
    45         for (int i = 0; i < cyclicBarrier.getParties(); i++) {
    46             Work work = new Work(String.valueOf(i), cyclicBarrier);
    47             work.start();
    48         }
    49         
    50     }
    51 
    52     public static void main(String[] args) {
    53         CyclicBarrierLearn cyclicBarrierLearn = new CyclicBarrierLearn();
    54         cyclicBarrierLearn.cyclicBarrier();
    55 
    56     }
    57 
    58 }
    59 

    posted @ 2017-07-13 11:39 丑男 閱讀(175) | 評論 (0)編輯 收藏

    CountDownLatch 簡單舉例

    一句話解釋:主線程阻塞,其他線程完成后,主線程被喚醒后繼續執行

     1 import java.util.Random;
     2 import java.util.concurrent.CountDownLatch;
     3 
     4 import org.slf4j.Logger;
     5 import org.slf4j.LoggerFactory;
     6 
     7 public class CountDownLatchLearn {
     8     
     9     private Logger log = LoggerFactory.getLogger(CountDownLatchLearn.class);
    10     private CountDownLatch countDownLatch;
    11     
    12     public CountDownLatchLearn() {
    13         countDownLatch = new CountDownLatch(50);
    14     }
    15     
    16     public void countDown() {
    17         Long count = countDownLatch.getCount();
    18         log.debug("countDownLatch count is:" + count.toString());
    19         
    20         for (int i = 0; i < count; i++) {
    21             Work work = new Work(String.valueOf(i), countDownLatch);
    22             work.start();
    23         }
    24         try {
    25             countDownLatch.await();
    26         } catch (InterruptedException e) {
    27             e.printStackTrace();
    28         }
    29         log.debug("work finish!!!");
    30     }
    31     
    32     private class Work extends Thread {
    33         
    34         private String name;
    35         private CountDownLatch countDownLatch;
    36         
    37         public Work(String name, CountDownLatch countDownLatch) {
    38             this.name = name;
    39             this.countDownLatch = countDownLatch;
    40         }
    41         
    42         @Override
    43         public void run() {
    44             Random r = new Random();
    45             int sleep = r.nextInt(2000);
    46             try {
    47                 log.debug("thread sleep: "+ sleep);
    48                 Thread.sleep(sleep);
    49             } catch (InterruptedException e) {
    50                 e.printStackTrace();
    51             }
    52             log.debug("thread: " + name + ": do work");
    53             countDownLatch.countDown();
    54         }
    55     }
    56 
    57     public static void main(String[] args) {
    58         System.out.println("main start!!!");
    59         
    60         CountDownLatchLearn countDownLatchLearn = new CountDownLatchLearn();
    61         countDownLatchLearn.countDown();
    62         
    63         System.out.println("main end!!!");
    64     }
    65 
    66 }

    posted @ 2017-07-13 11:18 丑男 閱讀(308) | 評論 (0)編輯 收藏

    mysql 5.7 for windows安裝過程

    環境說明:

    OSwindows 7 64bit Databasemysql-5.7.18-winx64 Noinstall版本

     

    1. 解壓Mysql安裝目錄

    2. 編寫my.ini配置文件

    3. mysqld --defaults-file=../my.ini --initialize

    4. ALTER USER 'root'@'localhost' IDENTIFIED BY 'password';

    5. mysql –u root –p

    6. 密碼在logs/*.err日志中

     
    my.ini文件內容

     1 # my.ini文件內容
     2 # For advice on how to change settings please see
     3 # http://dev.mysql.com/doc/refman/5.7/en/server-configuration-defaults.html
     4 # *** DO NOT EDIT THIS FILE. It's a template which will be copied to the
     5 # *** default location during install, and will be replaced if you
     6 # *** upgrade to a newer version of MySQL.
     7 
     8 [mysqld]
     9 
    10 # Remove leading # and set to the amount of RAM for the most important data
    11 # cache in MySQL. Start at 70% of total RAM for dedicated server, else 10%.
    12 # innodb_buffer_pool_size = 128M
    13 
    14 # Remove leading # to turn on a very important data integrity option: logging
    15 # changes to the binary log between backups.
    16 # log_bin
    17 
    18 # These are commonly set, remove the # and set as required.
    19 basedir=D:\\mysql-5.7.18-winx64
    20 datadir=D:\\mysql-5.7.18-winx64\\data
    21 # port = ..
    22 # server_id = ..
    23 
    24 
    25 # Remove leading # to set options mainly useful for reporting servers.
    26 # The server defaults are faster for transactions and fast SELECTs.
    27 # Adjust sizes as needed, experiment to find the optimal values.
    28 # join_buffer_size = 128M
    29 # sort_buffer_size = 2M
    30 # read_rnd_buffer_size = 2M 
    31 
    32 sql_mode=NO_ENGINE_SUBSTITUTION,STRICT_TRANS_TABLES
    33 
    34 long_query_time=0.1
    35 slow_query_log=on
    36 slow_query_log_file=D:\\mysql-5.7.18-winx64\\logs\\mysqlslow.log
    37 

    posted @ 2017-07-10 00:53 丑男 閱讀(210) | 評論 (0)編輯 收藏

    添加用戶【備份】

    useradd -g 501 -s /sbin/nologin builder

    posted @ 2013-05-06 17:25 丑男 閱讀(125) | 評論 (0)編輯 收藏

    [ZT]Java虛擬機JVM的調優參數選擇

    在任何一個的生產系統上線前,系統性能調優(Tuning)都是很重要的一步。通常,應用系統的軟硬件的缺省值都是給開發平臺(或小規模系統)設計的,用來跑生產系統幾乎都無法發揮出軟硬件的最佳性能。有時,系統調優前后的性能會差好幾倍。另一方面,由于應用程序的開發人員通常都是針對功能作開發的,因此,開發硬件都是比生產環境要小的機器。例如,生產系統是一臺8個CPU,64GB內存服務器,而開發服務器可能只有1個CPU和4GB內存。所以,在開發人員中常常不具備做性能方面測試的軟硬件環境。另外,有的程序員甚至在開發時都沒有考慮到多用戶并發的環境,程序中存在單點瓶頸等問題。在做壓力測試和調優時,往往就會發現這些關鍵點。

      由于應用系統是個軟硬件的完整統一體,系統調優往往需要涉及硬件、網絡操作系統、中間件,應用程序和數據庫等方面。在調優的過程中,往往需要發現存在瓶頸的地方(也就是導致系統變慢的部分),分析原因,從而改進和確定較優的參數。

      我們在作JVM的調優前,通常先要了解運行的硬件平臺,操作系統和中間件,然后針對這些情況配置相應的系統參數,在測試中不斷完善參數。由于性能調優需要對系統非常了解,并且需要豐富的經驗,因此不是一件容易的事情。這里介紹一些很好的參考資料,就是SPEC.org的網站。這是硬件廠商公布benchmark測試結果的地方,通常硬件廠商會把系統調到最優化才公布結果的,因此很有借鑒意義。常見和JVM有關的benchmark值主要有SPECjAppServer2004和SPECjbb2005。前者是J2EE應用服務器的性能指標,后者是服務器端Java虛擬機的性能指標。給大家介紹這個網站的目的是說大家可以參考硬件廠商給出的JVM配置,在根據自己應用環境的特點,較快的得出較好的參數。例如,這個網頁給出了SUN公司T5120服務器+應用服務器9.1 +JDK1.5的SPECjAppServer2004值是8,439.36:

      http://www.spec.org/jAppServer2004/results/res2007q4/jAppServer2004-20071106-00092.html

      我們現在要關心的不是Benchmark的值(注:實際上,Sun公司的這個值是個很不錯的結果),而是留意在這種環境下JVM的參數配置,可以找到一個欄目“Notes / Tuning Information”:

      JVM Options: -server -XX:+AggressiveHeap

    -Xmx2560m -Xms2560m -Xmn1024m -Xss128k

    -XX:PermSize=256m

    -XX:+DisableExplicitGC

    -XX:ParallelGCThreads=24

    -XX:LargePageSizeInBytes=256m

    -XX:+UseParallelOldGC

    -XX:+AggressiveOpts

    -DAllowManagedFieldsInDefaultFetchGroup=true

    -DAllowMediatedWriteInDefaultFetchGroup=true

    -XX:-UseBiasedLocking

    -Dcom.sun.ejb.containers.readonly.relative.refresh.mode=true

    -Dcom.sun.jts.dblogging.insertquery=insert into

    txn_log_table_0 values (
    ? , ? , ? )

    -Dcom.sun.jts.dblogging.deletequery=delete from

    txn_log_table_0 where localtid
    = ? and servername = ?

    -Dcom.sun.jdo.spi.persistence.support.sqlstore.

    MULTILEVEL_PREFETCH
    =true

      那么上面那些參數是什么意思呢?上述段落中“-XX”的參數是SUN JVM的擴展選項,其中以下的這些都是和垃圾回收(GC)有關:

      -XX:PermSize=256m

    -XX:+DisableExplicitGC

    -XX:ParallelGCThreads=24

    -XX:+UseParallelOldGC

    -XX:+AggressiveHeap

      下面這個選項是選擇大的內存頁面:

      -XX:LargePageSizeInBytes=256m

      "-XX:+AggressiveOpts"是一些試驗性優化參數,“-XX:-UseBiasedLocking”是非競爭性的同步選項。

      而選項“-Xmx2560m -Xms2560m -Xmn1024m -Xss128k”則是初始堆棧的內存值,注意-Xmx和-Xms的值是一樣的,這樣系統性能會較平穩些。

      至于這些參數詳細代表什么意義,大家可以google一下就很容易了解。這是Sun網站上的說明,有興趣的可以讀一下: http://java.sun.com/performance/reference/whitepapers/tuning.html

      如果你的應用系統是JDK1.5,硬件是T5120,操作系統是Solaris,那么這些參數就很有借鑒意義。如果你的硬件系統不是T5120,但是使用SUN的JDK1.5 ,這些參數也是有一定參考作用。當然,最理想的是選擇一個和自己的環境最近似的結果來參考。大多數軟硬件的測試結果都可以在SPEC.org上找到,如果你的系統是J2EE的3層架構,可以用jAppServer2004指標,如果是純JAVA的應用,可用jbb2005的結果:

      http://www.spec.org/jAppServer2004/

      http://www.spec.org/jbb2005/

      需要注意的是,這些調優參數只是提供了一個思路,具體是否合適你的應用還要看實測結果。

    posted @ 2009-04-29 19:24 丑男 閱讀(270) | 評論 (0)編輯 收藏

    [轉帖]Hibernate常用保存

    hibernate對于對象的保存提供了太多的方法,他們之間有很多不同,這里細說一下,以便區別:
    一、預備知識:
    在所有之前,說明一下,對于hibernate,它的對象有三種狀態,transient、persistent、detached
    下邊是常見的翻譯辦法:
    transient:瞬態或者自由態
    persistent:持久化狀態
    detached:脫管狀態或者游離態
    脫管狀態的實例可以通過調用save()、persist()或者saveOrUpdate()方法進行持久化。
    持久化實例可以通過調用 delete()變成脫管狀態。通過get()或load()方法得到的實例都是持久化狀態的。
    脫管狀態的實例可以通過調用 update()、0saveOrUpdate()、lock()或者replicate()進行持久化。
    save() 和persist()將會引發SQL的INSERT,delete()會引發SQLDELETE,而update()或merge()會引發 SQLUPDATE.對持久化(persistent)實例的修改在刷新提交的時候會被檢測到,它也會引起 SQLUPDATE.saveOrUpdate()或者replicate()會引發SQLINSERT或者UPDATE
    二、save 和update區別
    把這一對放在第一位的原因是因為這一對是最常用的。
    save的作用是把一個新的對象保存
    update是把一個脫管狀態的對象保存
    三、update 和saveOrUpdate區別
    這個是比較好理解的,顧名思義,saveOrUpdate基本上就是合成了save和update引用hibernate reference中的一段話來解釋他們的使用場合和區別。
    通常下面的場景會使用update()或saveOrUpdate():
    程序在第一個session中加載對象
    該對象被傳遞到表現層
    對象發生了一些改動
    該對象被返回到業務邏輯層
    程序調用第二個session的update()方法持久這些改動
    saveOrUpdate()做下面的事:
    如果對象已經在本session中持久化了,不做任何事
    如果另一個與本session關聯的對象擁有相同的持久化標識(identifier),拋出一個異常
    如果對象沒有持久化標識(identifier)屬性,對其調用save()
    如果對象的持久標識(identifier)表明其是一個新實例化的對象,對其調用save()
    如果對象是附帶版本信息的(通過或) 并且版本屬性的值表明其是一個新實例化的對象,save()它。
    四、persist和save區別
    這個是最迷離的一對,表面上看起來使用哪個都行,在hibernate reference文檔中也沒有明確的區分他們。
    這里給出一個明確的區分。(可以跟進src看一下,雖然實現步驟類似,但是還是有細微的差別)
    1.persist把一個瞬態的實例持久化,但是并"不保證"標識符被立刻填入到持久化實例中,標識符的填入可能被推遲到flush的時間。
    2.persist" 保證",當它在一個transaction外部被調用的時候并不觸發一個Sql Insert,這個功能是很有用的,當我們通過繼承Session/persistence context來封裝一個長會話流程的時候,一個persist這樣的函數是需要的。
    3.save"不保證"第2條,它要返回標識符,所以它會立即執行Sql insert,不管是不是在transaction內部。
    五、saveOrUpdateCopy,merge和update區別
    首先說明merge是用來代替saveOrUpdateCopy的,然后比較update和merge,update的作用上邊說了,這里說一下merge的作用。
    如果session中存在相同持久化標識(identifier)的實例,用用戶給出的對象的狀態覆蓋舊有的持久實例
    如果session沒有相應的持久實例,則嘗試從數據庫中加載,或創建新的持久化實例,最后返回該持久實例
    用戶給出的這個對象沒有被關聯到session上,它依舊是脫管的
    重點是最后一句:
    當我們使用update的時候,執行完成后,我們提供的對象A的狀態變成持久化狀態
    但當我們使用merge的時候,執行完成,我們提供的對象A還是脫管狀態,hibernate或者new了一個B,或者檢索到一個持久對象,并把我們提供的對象A的所有的值拷貝到這個B,執行完成后B是持久狀態,而我們提供的A還是托管狀態。
    六、flush和update區別
    這兩個的區別好理解
    update操作的是在脫管狀態的對象,而flush是操作的在持久狀態的對象。
    默認情況下,一個持久狀態的對象是不需要update的,只要你更改了對象的值,等待hibernate flush就自動保存到數據庫了。hibernate flush發生再幾種情況下:
    1.調用某些查詢的時候
    2.transaction commit的時候
    3.手動調用flush的時候
    七、lock和update區別
    update是把一個已經更改過的脫管狀態的對象變成持久狀態
    lock是把一個沒有更改過的脫管狀態的對象變成持久狀態
    對應更改一個記錄的內容,兩個的操作不同:
    update的操作步驟是:
    更改脫管的對象->調用update
    lock的操作步驟是:
    調用lock把對象從脫管狀態變成持久狀態——>更改持久狀態的對象的內容——>等待flush或者手動flush

    posted @ 2008-08-23 21:22 丑男 閱讀(242) | 評論 (0)編輯 收藏

    [轉帖]Oracle執行計劃解釋

    一.相關的概念

        Rowid的概念:rowid是一個偽列,既然是偽列,那么這個列就不是用戶定義,而是系統自己給加上的。 對每個表都有一個rowid的偽列,但是表中并不物理存儲ROWID列的值。不過你可以像使用其它列那樣使用它,但是不能刪除改列,也不能對該列的值進行 修改、插入。一旦一行數據插入數據庫,則rowid在該行的生命周期內是唯一的,即即使該行產生行遷移,行的rowid也不會改變。

        Recursive SQL概念:有時為了執行用戶發出的一個sql語句,oracle必須執行一些額外的語句,我們將這些額外的語句稱之為''recursive calls''或''recursive sql statements''.如當一個DDL語句發出后,ORACLE總是隱含的發出一些recursive SQL語句,來修改數據字典信息,以便用戶可以成功的執行該DDL語句。當需要的數據字典信息沒有在共享內存中時,經常會發生Recursive calls,這些Recursive calls會將數據字典信息從硬盤讀入內存中。用戶不比關心這些recursive SQL語句的執行情況,在需要的時候,ORACLE會自動的在內部執行這些語句。當然DML語句與SELECT都可能引起recursive sql.簡單的說,我們可以將觸發器視為recursive sql.

        Row Source(行源):用在查詢中,由上一操作返回的符合條件的行的集合,即可以是表的全部行數據的集合;也可以是表的部分行數據的集合;也可以為對上2個row source進行連接操作(如join連接)后得到的行數據集合。

        Predicate(謂詞):一個查詢中的WHERE限制條件

        Driving table(驅動表):該表又稱為外層表(OUTER table)。 這個概念用于嵌套與HASH連接中。如果該row source返回較多的行數據,則對所有的后續操作有負面影響。注意此處雖然翻譯為驅動表,但實際上翻譯為驅動行源(driving row source)更為確切。一般說來,是應用查詢的限制條件后,返回較少行源的表作為驅動表,所以如果一個大表在WHERE條件有有限制條件(如等值限 制),則該大表作為驅動表也是合適的,所以并不是只有較小的表可以作為驅動表,正確說法應該為應用查詢的限制條件后,返回較少行源的表作為驅動表。在執行 計劃中,應該為靠上的那個row source,后面會給出具體說明。在我們后面的描述中,一般將該表稱為連接操作的row source 1.

        Probed table(被探查表):該表又稱為內層表(INNER table)。在我們從驅動表中得到具體一行的數據后,在該表中尋找符合連接條件的行。所以該表應當為大表(實際上應該為返回較大row source的表)且相應的列上應該有索引。在我們后面的描述中,一般將該表稱為連接操作的row source 2.

        組合索引(concatenated index):由多個列構成的索引,如create index idx_emp on emp(col1, col2, col3, ……),則我們稱idx_emp索引為組合索引。在組合索引中有一個重要的概念:引導列(leading column),在上面的例子中,col1列為引導列。當我們進行查詢時可以使用“where col1 = ? ”,也可以使用“where col1 = ? and col2 = ?”,這樣的限制條件都會使用索引,但是“where col2 = ? ”查詢就不會使用該索引。所以限制條件中包含先導列時,該限制條件才會使用該組合索引。

        可選擇性(selectivity):比較一下列中唯一鍵的數量和表中的行數,就可以判斷該列的可選擇性。 如果該列的“唯一鍵的數量/表中的行數”的比值越接近1,則該列的可選擇性越高,該列就越適合創建索引,同樣索引的可選擇性也越高。在可選擇性高的列上進 行查詢時,返回的數據就較少,比較適合使用索引查詢。

    歡迎進入Oracle社區論壇,與200萬技術人員互動交流 >>進入
    二.oracle訪問數據的存取方法

        1) 全表掃描(Full table Scans, FTS)

        為實現全表掃描,oracle讀 取表中所有的行,并檢查每一行是否滿足語句的WHERE限制條件一個多塊讀 操作可以使一次I/O能讀取多塊數據塊(db_block_multiblock_read_count參數設定),而不是只讀取一個數據塊,這極大的減 少了I/O總次數,提高了系統的吞吐量,所以利用多塊讀的方法可以十分高效地實現全表掃描,而且只有在全表掃描的情況下才能使用多塊讀操作。在這種訪問模 式下,每個數據塊只被讀一次。

        使用FTS的前提條件:在較大的表上不建議使用全表掃描,除非取出數據的比較多,超過總量的5% —— 10%,或你想使用并行查詢功能時。

        使用全表掃描的例子:

        ~~~~~~~~~~~~~~~~~~~~~~~~ sql> explain plan for select * from dual;

        Query Plan

        -----------------------------------------

        SELECT STATEMENT[CHOOSE] Cost=

        table ACCESS FULL DUAL

        2) 通過ROWID的表存取(Table Access by ROWID或rowid lookup)

        行的ROWID指出了該行所在的數據文件、數據塊以及行在該塊中的位置,所以通過ROWID來存取數據可以快速定位到目標數據上,是Oracle存取單行數據的最快方法。

        這種存取方法不會用到多塊讀操作,一次I/O只能讀取一個數據塊。我們會經常在執行計劃中看到該存取方法,如通過索引查詢數據。

        使用ROWID存取的方法: sql> explain plan for select * from dept where rowid = ''AAAAyGAADAAAAATAAF'';

        Query Plan

        ------------------------------------

        SELECT STATEMENT [CHOOSE] Cost=1

        table ACCESS BY ROWID DEPT [ANALYZED]


        3)索引掃描(Index Scan或index lookup)

        我們先通過index查找到數據對應的rowid值(對于非唯一索引可能返回多個rowid值),然后根據rowid直接從表中得到具體的數據,這 種查找方式稱為索引掃描或索引查找(index lookup)。一個rowid唯一的表示一行數據,該行對應的數據塊是通過一次i/o得到的,在此情況下該次i/o只會讀取一個數據庫塊。

        在索引中,除了存儲每個索引的值外,索引還存儲具有此值 的行對應的ROWID值。索引掃描可以由2步組成:(1) 掃描索引得到對應的rowid值。 (2) 通過找到的rowid從表中讀出具體的數據。每步都是單獨的一次I/O,但是對于索引,由于經常使用,絕大多數都已經CACHE到內存中,所以第1步的 I/O經常是邏輯I/O,即數據可以從內存中得到。但是對于第2步來說,如果表比較大,則其數據不可能全在內存中,所以其I/O很有可能是物理I/O,這 是一個機械操作,相對邏輯I/O來說,是極其費時間的。所以如果多大表進行索引掃描,取出的數據如果大于總量的5% —— 10%,使用索引掃描會效率下降很多。如下列所示:SQL> explain plan for select empno, ename from emp where empno=10;

        Query Plan

        ------------------------------------

        SELECT STATEMENT [CHOOSE] Cost=1

        table ACCESS BY ROWID EMP [ANALYZED]

        INDEX UNIQUE SCAN EMP_I1


        但是如果查詢的數據能全在索引中找到,就可以避免進行第2步操作,避免了不必要的I/O,此時即使通過索引掃描取出的數據比較多,效率還是很高的

        sql> explain plan for select empno from emp where empno=10;-- 只查詢empno列值

        Query Plan

        ------------------------------------

        SELECT STATEMENT [CHOOSE] Cost=1

        INDEX UNIQUE SCAN EMP_I1

        進一步講,如果sql語句中對索引列進行排序,因為索引已經預先排序好了,所以在執行計劃中不需要再對索引列進行排序
        sql> explain plan for select empno, ename from emp

        where empno > 7876 order by empno;

        Query Plan

        --------------------------------------------------------------------------------

        SELECT STATEMENT[CHOOSE] Cost=1

        table ACCESS BY ROWID EMP [ANALYZED]

        INDEX RANGE SCAN EMP_I1 [ANALYZED]


        從這個例子中可以看到:因為索引是已經排序了的,所以將按照索引的順序查詢出符合條件的行,因此避免了進一步排序操作。

    根據索引的類型與where限制條件的不同,有4種類型的索引掃描:

        索引唯一掃描(index unique scan)

        索引范圍掃描(index range scan)

        索引全掃描(index full scan)

        索引快速掃描(index fast full scan)

        (1) 索引唯一掃描(index unique scan)

        通過唯一索引查找一個數值經常返回單個ROWID.如果存在UNIQUE 或PRIMARY KEY 約束(它保證了語句只存取單行)的話,Oracle經常實現唯一性掃描。

        使用唯一性約束的例子:

        sql> explain plan for

        select empno,ename from emp where empno=10;

        Query Plan

    ------------------------------------

        SELECT STATEMENT [CHOOSE] Cost=1

        table ACCESS BY ROWID EMP [ANALYZED]

        INDEX UNIQUE SCAN EMP_I1

        (2) 索引范圍掃描(index range scan)

        使用一個索引存取多行數據,在唯一索引上使用索引范圍掃描的典型情況下是在謂詞(where限制條件)中使用了范圍操作符(如>、<、<>、>=、<=、between)

        使用索引范圍掃描的例子:

        sql> explain plan for select empno,ename from emp

        where empno > 7876 order by empno;

        Query Plan

    --------------------------------------------------------------------------------

        SELECT STATEMENT[CHOOSE] Cost=1

        table ACCESS BY ROWID EMP [ANALYZED]

        INDEX RANGE SCAN EMP_I1 [ANALYZED]

        在非唯一索引上,謂詞col = 5可能返回多行數據,所以在非唯一索引上都使用索引范圍掃描。

        使用index rang scan的3種情況:

        (a) 在唯一索引列上使用了range操作符(> < <> >= <= between)

        (b) 在組合索引上,只使用部分列進行查詢,導致查詢出多行

        (c) 對非唯一索引列上進行的任何查詢。

        (3) 索引全掃描(index full scan)

        與全表掃描對應,也有相應的全索引掃描。而且此時查詢出的數據都必須從索引中可以直接得到。

        全索引掃描的例子:

        An Index full scan will not perform single block i/o''s and so it may prove to be inefficient.

        e.g.

        Index BE_IX is a concatenated index on big_emp (empno, ename)

        sql> explain plan for select empno, ename from big_emp order by empno,ename;

        Query Plan

    --------------------------------------------------------------------------------

        SELECT STATEMENT[CHOOSE] Cost=26

        INDEX FULL SCAN BE_IX [ANALYZED]

        (4) 索引快速掃描(index fast full scan)

        掃描索引中的所有的數據塊,與 index full scan很類似,但是一個顯著的區別就是它不對查詢出的數據進行排序,即數據不是以排序順序被返回。在這種存取方法中,可以使用多塊讀功能,也可以使用并行讀入,以便獲得最大吞吐量與縮短執行時間。

        索引快速掃描的例子:

        BE_IX索引是一個多列索引: big_emp (empno,ename)

        sql> explain plan for select empno,ename from big_emp;

        Query Plan

    ------------------------------------------

        SELECT STATEMENT[CHOOSE] Cost=1

        INDEX FAST FULL SCAN BE_IX [ANALYZED]

        只選擇多列索引的第2列:

        sql> explain plan for select ename from big_emp;

        Query Plan

    ------------------------------------------

        SELECT STATEMENT[CHOOSE] Cost=1

        INDEX FAST FULL SCAN BE_IX [ANALYZED]

       三、表之間的連接

        Join是一種試圖將兩個表結合在一起的謂詞,一次只能連接2個表,表連接也可以被稱為表關聯。在后面的敘 述中,我們將會使用“row source”來代替“表”,因為使用row source更嚴謹一些,并且將參與連接的2個row source分別稱為row source1和row source 2.Join過程的各個步驟經常是串行操作,即使相關的row source可以被并行訪問,即可以并行的讀取做join連接的兩個row source的數據,但是在將表中符合限制條件的數據讀入到內存形成row source后,join的其它步驟一般是串行的。有多種方法可以將2個表連接起來,當然每種方法都有自己的優缺點,每種連接類型只有在特定的條件下才會 發揮出其最大優勢。

        row source(表)之間的連接順序對于查詢的效率有非常大的影響。通過首先存取特定的表,即將該表作為驅動表,這樣可以先應用某些限制條件,從而得到一個 較小的row source,使連接的效率較高,這也就是我們常說的要先執行限制條件的原因。一般是在將表讀入內存時,應用where子句中對該表的限制條件。

        根據2個row source的連接條件的中操作符的不同,可以將連接分為等值連接(如WHERE A.COL3 = B.COL4)、非等值連接(WHERE A.COL3 > B.COL4)、外連接(WHERE A.COL3 = B.COL4(+))。上面的各個連接的連接原理都基本一樣,所以為了簡單期間,下面以等值連接為例進行介紹。

        在后面的介紹中,都已:

        SELECT A.COL1, B.COL2

        FROM A, B

        WHERE A.COL3 = B.COL4;

        為例進行說明,假設A表為Row Soruce1,則其對應的連接操作關聯列為COL 3;B表為Row Soruce2,則其對應的連接操作關聯列為COL 4;

        連接類型:

        目前為止,無論連接操作符如何,典型的連接類型共有3種:

        排序 - - 合并連接(Sort Merge Join (SMJ) )

        嵌套循環(Nested Loops (NL) )

        哈希連接(Hash Join)

        排序 - - 合并連接(Sort Merge Join, SMJ)

        內部連接過程:

        1) 首先生成row source1需要的數據,然后對這些數據按照連接操作關聯列(如A.col3)進行排序。

        2) 隨后生成row source2需要的數據,然后對這些數據按照與sort source1對應的連接操作關聯列(如B.col4)進行排序。

        3) 最后兩邊已排序的行被放在一起執行合并操作,即將2個row source按照連接條件連接起來

        下面是連接步驟的圖形表示:

        MERGE

        /"

        SORTSORT

        ||

        Row Source 1Row Source 2

        如果row source已經在連接關聯列上被排序,則該連接操作就不需要再進行sort操作,這樣可以大大提高這種連接操作的連接速度,因為排序是個極其費資源的操 作,特別是對于較大的表。預先排序的row source包括已經被索引的列(如a.col3或b.col4上有索引)或row source已經在前面的步驟中被排序了。盡管合并兩個row source的過程是串行的,但是可以并行訪問這兩個row source(如并行讀入數據,并行排序)。

        SMJ連接的例子:SQL> explain plan for

        select /*+ ordered */ e.deptno, d.deptno

        from emp e, dept d

        where e.deptno = d.deptno

        order by e.deptno, d.deptno;

        Query Plan

    -------------------------------------

        SELECT STATEMENT [CHOOSE] Cost=17

        MERGE JOIN

        SORT JOIN

        table ACCESS FULL EMP [ANALYZED]

        SORT JOIN

        table ACCESS FULL DEPT [ANALYZED]


        排序是一個費時、費資源的操作,特別對于大表。基于這個原因,SMJ經常不是一個特別有效的連接方法,但是如果2個row source都已經預先排序,則這種連接方法的效率也是蠻高的。

        嵌套循環(Nested Loops, NL)

        這個連接方法有驅動表(外部表)的概念。其實,該連接過程就是一個2層嵌套循環,所以外層循環的次數越少越好,這也就是我們為什么將小表或返回較小 row source的表作為驅動表(用于外層循環)的理論依據。但是這個理論只是一般指導原則,因為遵循這個理論并不能總保證使語句產生的I/O次數最少。有時 不遵守這個理論依據,反而會獲得更好的效率。如果使用這種方法,決定使用哪個表作為驅動表很重要。有時如果驅動表選擇不正確,將會導致語句的性能很差、很 差。

        內部連接過程:

        Row source1的Row 1 —— Probe ->Row source 2

        Row source1的Row 2 —— Probe ->Row source 2

        Row source1的Row 3 —— Probe ->Row source 2

        ……。

        Row source1的Row n —— Probe ->Row source 2

        從內部連接過程來看,需要用row source1中的每一行,去匹配row source2中的所有行,所以此時保持row source1盡可能的小與高效的訪問row source2(一般通過索引實現)是影響這個連接效率的關鍵問題。這只是理論指導原則,目的是使整個連接操作產生最少的物理I/O次數,而且如果遵守這 個原則,一般也會使總的物理I/O數最少。但是如果不遵從這個指導原則,反而能用更少的物理I/O實現連接操作,那盡管違反指導原則吧!因為最少的物理 I/O次數才是我們應該遵從的真正的指導原則,在后面的具體案例分析中就給出這樣的例子。

        在上面的連接過程中,我們稱Row source1為驅動表或外部表。Row Source2被稱為被探查表或內部表。

        在NESTED LOOPS連接中,Oracle讀取row source1中的每一行,然后在row sourc2中檢查是否有匹配的行,所有被匹配的行都被放到結果集中,然后處理row source1中的下一行。這個過程一直繼續,直到row source1中的所有行都被處理。這是從連接操作中可以得到第一個匹配行的最快的方法之一,這種類型的連接可以用在需要快速響應的語句中,以響應速度為 主要目標。

        如果driving row source(外部表)比較小,并且在inner row source(內部表)上有唯一索引,或有高選擇性非唯一索引時,使用這種方法可以得到較好的效率。NESTED LOOPS有其它連接方法沒有的的一個優點是:可以先返回已經連接的行,而不必等待所有的連接操作處理完才返回數據,這可以實現快速的響應時間。

        如果不使用并行操作,最好的驅動表是那些應用了where 限制條件后,可以返回較少行數據的的表,所以大表也可能稱為驅動表,關鍵看限制條件。對于并行查詢,我們經常選擇大表作為驅動表,因為大表可以充分利用并 行功能。當然,有時對查詢使用并行操作并不一定會比查詢不使用并行操作效率高,因為最后可能每個表只有很少的行符合限制條件,而且還要看你的硬件配置是否 可以支持并行(如是否有多個CPU,多個硬盤控制器),所以要具體問題具體對待。

        NL連接的例子:

        sql> explain plan for

        select a.dname,b.sql

        from dept a,emp b

        where a.deptno = b.deptno;

        Query Plan

    -------------------------

        SELECT STATEMENT [CHOOSE] Cost=5

        NESTED LOOPS

        table ACCESS FULL DEPT [ANALYZED]

        table ACCESS FULL EMP [ANALYZED]


        哈希連接(Hash Join, HJ)

        這種連接是在oracle 7.3以后引入的,從理論上來說比NL與SMJ更高效,而且只用在CBO優化器中。

        較小的row source被用來構建hash table與bitmap,第2個row source被用來被hansed,并與第一個row source生成的hash table進行匹配,以便進行進一步的連接。Bitmap被用來作為一種比較快的查找方法,來檢查在hash table中是否有匹配的行。特別的,當hash table比較大而不能全部容納在內存中時,這種查找方法更為有用。這種連接方法也有NL連接中所謂的驅動表的概念,被構建為hash table與bitmap的表為驅動表,當被構建的hash table與bitmap能被容納在內存中時,這種連接方式的效率極高。

        HASH連接的例子:

        sql> explain plan for

        select /*+ use_hash(emp) */ empno

        from emp, dept

        where emp.deptno = dept.deptno;

        Query Plan

    ----------------------------

        SELECT STATEMENT[CHOOSE] Cost=3

        HASH JOIN

        table ACCESS FULL DEPT

        table ACCESS FULL EMP


        要使哈希連接有效,需要設置HASH_JOIN_ENABLED=TRUE,缺省情況下該參數為TRUE,另外,不要忘了還要設置 hash_area_size參數,以使哈希連接高效運行,因為哈希連接會在該參數指定大小的內存中運行,過小的參數會使哈希連接的性能比其他連接方式還 要低。

        總結一下,在哪種情況下用哪種連接方法比較好:

        排序 - - 合并連接(Sort Merge Join, SMJ):

        a) 對于非等值連接,這種連接方式的效率是比較高的。

        b) 如果在關聯的列上都有索引,效果更好。

        c) 對于將2個較大的row source做連接,該連接方法比NL連接要好一些。

        d) 但是如果sort merge返回的row source過大,則又會導致使用過多的rowid在表中查詢數據時,數據庫性能下降,因為過多的I/O.

        嵌套循環(Nested Loops, NL):

        a) 如果driving row source(外部表)比較小,并且在inner row source(內部表)上有唯一索引,或有高選擇性非唯一索引時,使用這種方法可以得到較好的效率。

        b) NESTED LOOPS有其它連接方法沒有的的一個優點是:可以先返回已經連接的行,而不必等待所有的連接操作處理完才返回數據,這可以實現快速的響應時間。

        哈希連接(Hash Join, HJ):

        a) 這種方法是在oracle7后來引入的,使用了比較先進的連接理論,一般來說,其效率應該好于其它2種連接,但是這種連接只能用在CBO優化器中,而且需要設置合適的hash_area_size參數,才能取得較好的性能。

        b) 在2個較大的row source之間連接時會取得相對較好的效率,在一個row source較小時則能取得更好的效率。

        c) 只能用于等值連接中

        笛卡兒乘積(Cartesian Product)

        當兩個row source做連接,但是它們之間沒有關聯條件時,就會在兩個row source中做笛卡兒乘積,這通常由編寫代碼疏漏造成(即程序員忘了寫關聯條件)。笛卡爾乘積是一個表的每一行依次與另一個表中的所有行匹配。在特殊情 況下我們可以使用笛卡兒乘積,如在星形連接中,除此之外,我們要盡量使用笛卡兒乘積,否則,自己想結果是什么吧!

        注意在下面的語句中,在2個表之間沒有連接。

        sql> explain plan for

        select emp.deptno,dept,deptno

        from emp,dept

        Query Plan

    ------------------------

        SLECT STATEMENT [CHOOSE] Cost=5

        MERGE JOIN CARTESIAN

        table ACCESS FULL DEPT

        SORT JOIN

        table ACCESS FULL EMP

        CARTESIAN關鍵字指出了在2個表之間做笛卡爾乘積。假如表emp有n行,dept表有m行,笛卡爾乘積的結果就是得到n * m行結果。

    posted @ 2008-08-22 22:33 丑男 閱讀(2504) | 評論 (0)編輯 收藏

    Fedora 7 安裝 nvidia顯卡驅動

    第一步,到nvidia官方下載驅動,根據自己顯卡的型號選擇,一定要看readme
    第二步,驅動只能在命令行方式下安裝,所以要修改/etc/inittab文件
        id:3:initdefault
                    3代表文本方式,5代表圖形方式
        修改后重啟系統
    第三步,執行驅動,不出意外會非常的順利,之后shutdown -r now就完成了

    本人也是第一次安裝,有不對的地方請大家指正,祝好運

    posted @ 2007-07-29 13:48 丑男 閱讀(780) | 評論 (0)編輯 收藏

    apache rewrite 簡單應用

    主要需求,DNSwww.mydomain.com指向www.mydomain.com/app/login.do

    目前是DNS指向某一個IP,不能將IP對應到10.11.xxx.xxx/app/login.do

    生產環境WEB服務器是apache server


    解決辦法
    利用apache rewriter功能,在httpd.conf中找到

    #LoadModule rewrite_module modules/mod_rewrite.so

    將#號去掉,也就是把mod_rewrite.so模塊加載進來

    之后在文件結尾處添加

    rewriteengine on
    rewriterule ^/$ http://10.11.xxx.xxx/app/login.do [R]

    posted @ 2007-07-16 21:47 丑男 閱讀(312) | 評論 (0)編輯 收藏

    [轉貼]jasperreport可以用Collection做為數據源



    jasperreport可以用Collection做為數據源,這種方式比用Connection方式更為靈活方便
     

    posted @ 2005-12-11 18:12 丑男 閱讀(417) | 評論 (0)編輯 收藏

    我的eclipse插件

    1.MyEclipse
       新出的GA4.0,功能非常強大,是我的首先。
       http://www.myeclipseide.com
    2.Log4E
       配合和Log4j日志功能
       http://log4e.jayefem.de/update
    3.i18n
       編輯國際化資源文件,省得寫native2ascii了
       http://propedit.sourceforge.jp/eclipse_plugins/
    4.FreeMen
        隨時知道內存使用情況
        http://www.junginger.biz/eclipse/

    posted @ 2005-09-04 16:13 丑男 閱讀(397) | 評論 (0)編輯 收藏

    [轉帖]學習Reflection

    Reflection 是 Java 程序開發語言的特征之一,它允許運行中的 Java 程序對自身進行檢查,或者說“自審”,并能直接操作程序的內部屬性。例如,使用它能獲得 Java 類中各成員的名稱并顯示出來。
    Java 的這一能力在實際應用中也許用得不是很多,但是在其它的程序設計語言中根本就不存在這一特性。例如,Pascal、C 或者 C++ 中就沒有辦法在程序中獲得函數定義相關的信息。
    JavaBean 是 reflection 的實際應用之一,它能讓一些工具可視化的操作軟件組件。這些工具通過 reflection 動態的載入并取得 Java 組件(類) 的屬性。
     
    1. 一個簡單的例子
    考慮下面這個簡單的例子,讓我們看看 reflection 是如何工作的。
    import java.lang.reflect.*;
    public class DumpMethods {
       public static void main(String args[]) {
           try {
               Class c = Class.forName(args[0]);
               Method m[] = c.getDeclaredMethods();
               for (int i = 0; i < m.length; i++)
                   System.out.println(m[i].toString());
           } catch (Throwable e) {
               System.err.println(e);
           }
       }
    }
    按如下語句執行:
    java DumpMethods java.util.Stack
    它的結果輸出為:
    public java.lang.Object java.util.Stack.push(java.lang.Object)
    public synchronized java.lang.Object java.util.Stack.pop()
    public synchronized java.lang.Object java.util.Stack.peek()
    public boolean java.util.Stack.empty()
    public synchronized int java.util.Stack.search(java.lang.Object)
    這樣就列出了java.util.Stack 類的各方法名以及它們的限制符和返回類型。
    這個程序使用 Class.forName 載入指定的類,然后調用 getDeclaredMethods 來獲取這個類中定義了的方法列表。java.lang.reflect.Methods 是用來描述某個類中單個方法的一個類。
    2.開始使用 Reflection
    用于 reflection 的類,如 Method,可以在 java.lang.relfect 包中找到。使用這些類的時候必須要遵循三個步驟:第一步是獲得你想操作的類的 java.lang.Class 對象。在運行中的 Java 程序中,用 java.lang.Class 類來描述類和接口等。
    下面就是獲得一個 Class 對象的方法之一:
    Class c = Class.forName("java.lang.String");
    這條語句得到一個 String 類的類對象。還有另一種方法,如下面的語句:
    Class c = int.class;
    或者
    Class c = Integer.TYPE;
    它們可獲得基本類型的類信息。其中后一種方法中訪問的是基本類型的封裝類 (如 Integer) 中預先定義好的 TYPE 字段。
    第二步是調用諸如 getDeclaredMethods 的方法,以取得該類中定義的所有方法的列表。
    一旦取得這個信息,就可以進行第三步了——使用 reflection API 來操作這些信息,如下面這段代碼:
    Class c = Class.forName("java.lang.String");
    Method m[] = c.getDeclaredMethods();
    System.out.println(m[0].toString());
    它將以文本方式打印出 String 中定義的第一個方法的原型。
    在下面的例子中,這三個步驟將為使用 reflection 處理特殊應用程序提供例證。
    模擬 instanceof 操作符
    得到類信息之后,通常下一個步驟就是解決關于 Class 對象的一些基本的問題。例如,Class.isInstance 方法可以用于模擬 instanceof 操作符:
    class A {
    }
    public class instance1 {
       public static void main(String args[]) {
           try {
               Class cls = Class.forName("A");
               boolean b1 = cls.isInstance(new Integer(37));
               System.out.println(b1);
               boolean b2 = cls.isInstance(new A());
               System.out.println(b2);
           } catch (Throwable e) {
               System.err.println(e);
           }
       }
    }
    在這個例子中創建了一個 A 類的 Class 對象,然后檢查一些對象是否是 A 的實例。Integer(37) 不是,但 new A() 是。
    3.找出類的方法
    找出一個類中定義了些什么方法,這是一個非常有價值也非常基礎的 reflection 用法。下面的代碼就實現了這一用法:
    import java.lang.reflect.*;
    public class method1 {
       private int f1(Object p, int x) throws NullPointerException {
           if (p == null)
               throw new NullPointerException();
           return x;
       }
       public static void main(String args[]) {
           try {
               Class cls = Class.forName("method1");
               Method methlist[] = cls.getDeclaredMethods();
               for (int i = 0; i < methlist.length; i++) {
                   Method m = methlist[i];
                   System.out.println("name = " + m.getName());
                   System.out.println("decl class = " + m.getDeclaringClass());
                   Class pvec[] = m.getParameterTypes();
                   for (int j = 0; j < pvec.length; j++)
                       System.out.println("param #" + j + " " + pvec[j]);
                   Class evec[] = m.getExceptionTypes();
                   for (int j = 0; j < evec.length; j++)
                       System.out.println("exc #" + j + " " + evec[j]);
                   System.out.println("return type = " + m.getReturnType());
                   System.out.println("-----");
               }
           } catch (Throwable e) {
               System.err.println(e);
           }
       }
    }
    這個程序首先取得 method1 類的描述,然后調用 getDeclaredMethods 來獲取一系列的 Method 對象,它們分別描述了定義在類中的每一個方法,包括 public 方法、protected 方法、package 方法和 private 方法等。如果你在程序中使用 getMethods 來代替 getDeclaredMethods,你還能獲得繼承來的各個方法的信息。
    取得了 Method 對象列表之后,要顯示這些方法的參數類型、異常類型和返回值類型等就不難了。這些類型是基本類型還是類類型,都可以由描述類的對象按順序給出。
    輸出的結果如下:
    name = f1
    decl class = class method1
    param #0 class java.lang.Object
    param #1 int
    exc #0 class java.lang.NullPointerException
    return type = int
    -----
    name = main
    decl class = class method1
    param #0 class [Ljava.lang.String;
    return type = void
    -----

    4.獲取構造器信息
    獲取類構造器的用法與上述獲取方法的用法類似,如:
    import java.lang.reflect.*;
    public class constructor1 {
       public constructor1() {
       }
       protected constructor1(int i, double d) {
       }
       public static void main(String args[]) {
           try {
               Class cls = Class.forName("constructor1");
               Constructor ctorlist[] = cls.getDeclaredConstructors();
               for (int i = 0; i < ctorlist.length; i++) {
                   Constructor ct = ctorlist[i];
                   System.out.println("name = " + ct.getName());
                   System.out.println("decl class = " + ct.getDeclaringClass());
                   Class pvec[] = ct.getParameterTypes();
                   for (int j = 0; j < pvec.length; j++)
                       System.out.println("param #" + j + " " + pvec[j]);
                   Class evec[] = ct.getExceptionTypes();
                   for (int j = 0; j < evec.length; j++)
                       System.out.println("exc #" + j + " " + evec[j]);
                   System.out.println("-----");
               }
           } catch (Throwable e) {
               System.err.println(e);
           }
       }
    }
    這個例子中沒能獲得返回類型的相關信息,那是因為構造器沒有返回類型。
    這個程序運行的結果是:
    name = constructor1
    decl class = class constructor1
    -----
    name = constructor1
    decl class = class constructor1
    param #0 int
    param #1 double
    -----
    5.獲取類的字段(域)
    找出一個類中定義了哪些數據字段也是可能的,下面的代碼就在干這個事情:

    import java.lang.reflect.*;
    public class field1 {
       private double d;
       public static final int i = 37;
       String s = "testing";
       public static void main(String args[]) {
           try {
               Class cls = Class.forName("field1");
               Field fieldlist[] = cls.getDeclaredFields();
               for (int i = 0; i < fieldlist.length; i++) {
                   Field fld = fieldlist[i];
                   System.out.println("name = " + fld.getName());
                   System.out.println("decl class = " + fld.getDeclaringClass());
                   System.out.println("type = " + fld.getType());
                   int mod = fld.getModifiers();
                   System.out.println("modifiers = " + Modifier.toString(mod));
                   System.out.println("-----");
               }
           } catch (Throwable e) {
               System.err.println(e);
           }
       }
    }
    這個例子和前面那個例子非常相似。例中使用了一個新東西 Modifier,它也是一個 reflection 類,用來描述字段成員的修飾語,如“private int”。這些修飾語自身由整數描述,而且使用 Modifier.toString 來返回以“官方”順序排列的字符串描述 (如“static”在“final”之前)。這個程序的輸出是:
    name = d
    decl class = class field1
    type = double
    modifiers = private
    -----
    name = i
    decl class = class field1
    type = int
    modifiers = public static final
    -----
    name = s
    decl class = class field1
    type = class java.lang.String
    modifiers =
    -----
    和獲取方法的情況一下,獲取字段的時候也可以只取得在當前類中申明了的字段信息 (getDeclaredFields),或者也可以取得父類中定義的字段 (getFields) 。

    6.根據方法的名稱來執行方法
    文本到這里,所舉的例子無一例外都與如何獲取類的信息有關。我們也可以用 reflection 來做一些其它的事情,比如執行一個指定了名稱的方法。下面的示例演示了這一操作:
    import java.lang.reflect.*;
    public class method2 {
       public int add(int a, int b) {
           return a + b;
       }
       public static void main(String args[]) {
           try {
               Class cls = Class.forName("method2");
               Class partypes[] = new Class[2];
               partypes[0] = Integer.TYPE;
               partypes[1] = Integer.TYPE;
               Method meth = cls.getMethod("add", partypes);
               method2 methobj = new method2();
               Object arglist[] = new Object[2];
               arglist[0] = new Integer(37);
               arglist[1] = new Integer(47);
               Object retobj = meth.invoke(methobj, arglist);
               Integer retval = (Integer) retobj;
               System.out.println(retval.intvalue());
           } catch (Throwable e) {
               System.err.println(e);
           }
       }
    }
    假如一個程序在執行的某處的時候才知道需要執行某個方法,這個方法的名稱是在程序的運行過程中指定的 (例如,JavaBean 開發環境中就會做這樣的事),那么上面的程序演示了如何做到。
    上例中,getMethod 用于查找一個具有兩個整型參數且名為 add 的方法。找到該方法并創建了相應的 Method 對象之后,在正確的對象實例中執行它。執行該方法的時候,需要提供一個參數列表,這在上例中是分別包裝了整數 37 和 47 的兩個 Integer 對象。執行方法的返回的同樣是一個 Integer 對象,它封裝了返回值 84。
    7.創建新的對象
    對于構造器,則不能像執行方法那樣進行,因為執行一個構造器就意味著創建了一個新的對象 (準確的說,創建一個對象的過程包括分配內存和構造對象)。所以,與上例最相似的例子如下:
    import java.lang.reflect.*;
    public class constructor2 {
       public constructor2() {
       }
       public constructor2(int a, int b) {
           System.out.println("a = " + a + " b = " + b);
       }
       public static void main(String args[]) {
           try {
               Class cls = Class.forName("constructor2");
               Class partypes[] = new Class[2];
               partypes[0] = Integer.TYPE;
               partypes[1] = Integer.TYPE;
               Constructor ct = cls.getConstructor(partypes);
               Object arglist[] = new Object[2];
               arglist[0] = new Integer(37);
               arglist[1] = new Integer(47);
               Object retobj = ct.newInstance(arglist);
           } catch (Throwable e) {
               System.err.println(e);
           }
       }
    }
    根據指定的參數類型找到相應的構造函數并執行它,以創建一個新的對象實例。使用這種方法可以在程序運行時動態地創建對象,而不是在編譯的時候創建對象,這一點非常有價值。
    8.改變字段(域)的值
    reflection 的還有一個用處就是改變對象數據字段的值。reflection 可以從正在運行的程序中根據名稱找到對象的字段并改變它,下面的例子可以說明這一點:
    import java.lang.reflect.*;
    public class field2 {
       public double d;
       public static void main(String args[]) {
           try {
               Class cls = Class.forName("field2");
               Field fld = cls.getField("d");
               field2 f2obj = new field2();
               System.out.println("d = " + f2obj.d);
               fld.setDouble(f2obj, 12.34);
               System.out.println("d = " + f2obj.d);
           } catch (Throwable e) {
               System.err.println(e);
           }
       }
    }
    這個例子中,字段 d 的值被變為了 12.34。
    9.使用數組
    本文介紹的 reflection 的最后一種用法是創建的操作數組。數組在 Java 語言中是一種特殊的類類型,一個數組的引用可以賦給 Object 引用。觀察下面的例子看看數組是怎么工作的:
    import java.lang.reflect.*;
    public class array1 {
       public static void main(String args[]) {
           try {
               Class cls = Class.forName("java.lang.String");
               Object arr = Array.newInstance(cls, 10);
               Array.set(arr, 5, "this is a test");
               String s = (String) Array.get(arr, 5);
               System.out.println(s);
           } catch (Throwable e) {
               System.err.println(e);
           }
       }
    }
    例中創建了 10 個單位長度的 String 數組,為第 5 個位置的字符串賦了值,最后將這個字符串從數組中取得并打印了出來。
    下面這段代碼提供了一個更復雜的例子:
    import java.lang.reflect.*;
    public class array2 {
       public static void main(String args[]) {
           int dims[] = new int[]{5, 10, 15};
           Object arr = Array.newInstance(Integer.TYPE, dims);
           Object arrobj = Array.get(arr, 3);
           Class cls = arrobj.getClass().getComponentType();
           System.out.println(cls);
           arrobj = Array.get(arrobj, 5);
           Array.setInt(arrobj, 10, 37);
           int arrcast[][][] = (int[][][]) arr;
           System.out.println(arrcast[3][5][10]);
       }
    }
    例中創建了一個 5 x 10 x 15 的整型數組,并為處于 [3][5][10] 的元素賦了值為 37。注意,多維數組實際上就是數組的數組,例如,第一個 Array.get 之后,arrobj 是一個 10 x 15 的數組。進而取得其中的一個元素,即長度為 15 的數組,并使用 Array.setInt 為它的第 10 個元素賦值。
    注意創建數組時的類型是動態的,在編譯時并不知道其類型。

    posted @ 2005-08-07 01:02 丑男 閱讀(419) | 評論 (0)編輯 收藏

    [轉帖]Eclipse快捷鍵

    作用域
    功能
    快捷鍵
    全局
    查找并替換
    Ctrl+F
    文本編輯器
    查找上一個
    Ctrl+Shift+K
    文本編輯器
    查找下一個
    Ctrl+K
    全局
    撤銷
    Ctrl+Z
    全局
    復制
    Ctrl+C
    全局
    恢復上一個選擇
    Alt+Shift+↓
    全局
    剪切
    Ctrl+X
    全局
    快速修正
    Ctrl1+1
    全局
    內容輔助
    Alt+/
    全局
    全部選中
    Ctrl+A
    全局
    刪除
    Delete
    全局
    上下文信息
    Alt+?
    Alt+Shift+?
    Ctrl+Shift+Space
    Java編輯器
    顯示工具提示描述
    F2
    Java編輯器
    選擇封裝元素
    Alt+Shift+↑
    Java編輯器
    選擇上一個元素
    Alt+Shift+←
    Java編輯器
    選擇下一個元素
    Alt+Shift+→
    文本編輯器
    增量查找
    Ctrl+J
    文本編輯器
    增量逆向查找
    Ctrl+Shift+J
    全局
    粘貼
    Ctrl+V
    全局
    重做
    Ctrl+Y
     
    查看
    作用域
    功能
    快捷鍵
    全局
    放大
    Ctrl+=
    全局
    縮小
    Ctrl+-
     
    窗口
    作用域
    功能
    快捷鍵
    全局
    激活編輯器
    F12
    全局
    切換編輯器
    Ctrl+Shift+W
    全局
    上一個編輯器
    Ctrl+Shift+F6
    全局
    上一個視圖
    Ctrl+Shift+F7
    全局
    上一個透視圖
    Ctrl+Shift+F8
    全局
    下一個編輯器
    Ctrl+F6
    全局
    下一個視圖
    Ctrl+F7
    全局
    下一個透視圖
    Ctrl+F8
    文本編輯器
    顯示標尺上下文菜單
    Ctrl+W
    全局
    顯示視圖菜單
    Ctrl+F10
    全局
    顯示系統菜單
    Alt+-
     
    導航
    作用域
    功能
    快捷鍵
    Java編輯器
    打開結構
    Ctrl+F3
    全局
    打開類型
    Ctrl+Shift+T
    全局
    打開類型層次結構
    F4
    全局
    打開聲明
    F3
    全局
    打開外部javadoc
    Shift+F2
    全局
    打開資源
    Ctrl+Shift+R
    全局
    后退歷史記錄
    Alt+←
    全局
    前進歷史記錄
    Alt+→
    全局
    上一個
    Ctrl+,
    全局
    下一個
    Ctrl+.
    Java編輯器
    顯示大綱
    Ctrl+O
    全局
    在層次結構中打開類型
    Ctrl+Shift+H
    全局
    轉至匹配的括號
    Ctrl+Shift+P
    全局
    轉至上一個編輯位置
    Ctrl+Q
    Java編輯器
    轉至上一個成員
    Ctrl+Shift+↑
    Java編輯器
    轉至下一個成員
    Ctrl+Shift+↓
    文本編輯器
    轉至行
    Ctrl+L
     
    搜索
    作用域
    功能
    快捷鍵
    全局
    出現在文件中
    Ctrl+Shift+U
    全局
    打開搜索對話框
    Ctrl+H
    全局
    工作區中的聲明
    Ctrl+G
    全局
    工作區中的引用
    Ctrl+Shift+G
     
    文本編輯
    作用域
    功能
    快捷鍵
    文本編輯器
    改寫切換
    Insert
    文本編輯器
    上滾行
    Ctrl+↑
    文本編輯器
    下滾行
    Ctrl+↓
     
    文件
    作用域
    功能
    快捷鍵
    全局
    保存
    Ctrl+X
    Ctrl+S
    全局
    打印
    Ctrl+P
    全局
    關閉
    Ctrl+F4
    全局
    全部保存
    Ctrl+Shift+S
    全局
    全部關閉
    Ctrl+Shift+F4
    全局
    屬性
    Alt+Enter
    全局
    新建
    Ctrl+N
     
    項目
    作用域
    功能
    快捷鍵
    全局
    全部構建
    Ctrl+B
     
    源代碼
    作用域
    功能
    快捷鍵
    Java編輯器
    格式化
    Ctrl+Shift+F
    Java編輯器
    取消注釋
    Ctrl+
    Java編輯器
    注釋
    Ctrl+/
    Java編輯器
    添加導入
    Ctrl+Shift+M
    Java編輯器
    組織導入
    Ctrl+Shift+O
    Java編輯器
    使用try/catch塊來包圍
    未設置,太常用了,所以在這里列出,建議自己設置。
    也可以使用Ctrl+1自動修正。
     
    運行
    作用域
    功能
    快捷鍵
    全局
    單步返回
    F7
    全局
    單步跳過
    F6
    全局
    單步跳入
    F5
    全局
    單步跳入選擇
    Ctrl+F5
    全局
    調試上次啟動
    F11
    全局
    繼續
    F8
    全局
    使用過濾器單步執行
    Shift+F5
    全局
    添加/去除斷點
    Ctrl+Shift+B
    全局
    顯示
    Ctrl+D
    全局
    運行上次啟動
    Ctrl+F11
    全局
    運行至行
    Ctrl+R
    全局
    執行
    Ctrl+U
     
    重構
    作用域
    功能
    快捷鍵
    全局
    撤銷重構
    Alt+Shift+Z
    全局
    抽取方法
    Alt+Shift+M
    全局
    抽取局部變量
    Alt+Shift+L
    全局
    內聯
    Alt+Shift+I
    全局
    移動
    Alt+Shift+V
    全局
    重命名
    Alt+Shift+R
    全局
    重做
    Alt+Shift+Y

    posted @ 2005-08-06 00:02 丑男 閱讀(441) | 評論 (0)編輯 收藏

    主站蜘蛛池模板: 97在线视频免费公开观看| 亚洲国产精品无码一线岛国| 性无码免费一区二区三区在线 | v片免费在线观看| 亚洲一区在线免费观看| 亚洲人成色77777| 免费少妇a级毛片人成网| av无码久久久久不卡免费网站| 波霸在线精品视频免费观看| 亚洲AV无码成人精品区狼人影院| 久久精品国产精品亚洲毛片| 亚洲片一区二区三区| 日韩精品免费电影| 91精品国产免费久久久久久青草| 国产免费AV片在线观看| 一区二区免费电影| 国产精品亚洲专区在线播放| 国产亚洲精品VA片在线播放| 666精品国产精品亚洲| 亚洲毛片αv无线播放一区| 亚洲日本va午夜中文字幕久久| 看全色黄大色大片免费久久| 免费看成人AA片无码视频羞羞网| 久久综合国产乱子伦精品免费| 国产在线观看无码免费视频| 午夜肉伦伦影院久久精品免费看国产一区二区三区| 亚洲人成日本在线观看| 911精品国产亚洲日本美国韩国 | 亚洲欧美中文日韩视频| 亚洲男人的天堂在线| 亚洲激情中文字幕| 亚洲午夜在线电影| 亚洲五月六月丁香激情| 亚洲精品国产福利片| 亚洲第一精品电影网| 亚洲国产夜色在线观看| 亚洲国产精品久久久久秋霞影院| 亚洲图片校园春色| 亚洲国产区男人本色在线观看| 亚洲va成无码人在线观看| 亚洲av极品无码专区在线观看|