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

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

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

    隨筆-25  評(píng)論-6  文章-0  trackbacks-0
      2006年6月13日

    Question:
    The restore command works fine, but while attempting the rolforward of the sql logs, it complains about database falling short on bufferpool,and then terminates the rollforward process. One close look at the db2diag.log file, I noticed that db2 tried to start the database with the hidden bufferpool. But, apparently, that hidden bufferpool was pretty small to rollforward the logs and bring it online.

    I cannot reduce the bufferpool size as the database cannot be connected to.I cannot connect to the database as it is in rollforward pending state. I cannot rollforward the database, as it's not happy with the size of the *hidden* bufferpool and terminates.

    Answer:

    db2set DB2_OVERRIDE_BPF=50000

    This will bring up all configured bufferpools using 50000 pages each. You
    can choose a smaller/larger value that is suitable for your number of
    bufferpools and available system memory. You can also configure each
    bufferpool individually if you want

    値: 正數(shù)のページ數(shù)
    ???? OR
    ?<entry>[;<entry>...] (<entry>=<バッファー?プール ID>,<ページ數(shù)> )

    There is sitiuation that above solution does not work.

    				When you try to create a bufferpool or alter a bufferpool to a
    very large size and there is not enough memory in the system
    , DB2 may occupies all pagespace and overrall performance of
    the system become slow and may get hang at last because no
    pagespace is available. Then after you connect the database
    next time, DB2 will do crash recovery and still try to
    allocate memory for this big bufferpool and occupy all
    pagespace again. when you specify DB2_OVERRIDE_BPF parameter
    to override bufferpool size, it doesn't take effect in this
    situation. This is becasue DB2_OVERRIDE_BPF only applies for
    bufferpools that already exist at startup, and not to
    bufferpools that are created during crash recovery or roll
    forward. We will fix this problem and check DB2_OVERRIDE_BPF
    registry when creating bufferpools during crash recovery or
    roll forward. This problem only occurs in a 64 bit instance.
    
    		

    Local fix

    				You can use db2iupdt -w 32 <INSTNAME> to update the instance to
    a 32 bit instance, then you can connect to the database
    succefully. After that, use db2iupdt -w 64 <INSTNAME> to update
    the instance to 64 bit again. Please take a backup at this
    point, so we don't have to roll forward through bufferpool
    creation that fails if we need restore and roll forward
    database.
    
    		

    Problem summary

    				Users Affected:
    All users using 64 bit instance
    Problem Description:
    When you try to create a bufferpool or alter a bufferpool to a
    very large size and there is not enough memory in the system
    , DB2 may occupies all pagespace and overrall performance
    of the system become slow and may get hang at last
    because no pagespace is available. Then after you connect the
    database at next time, DB2 will do crash recovery and still try
    to allocate
    memory for this big bufferpool and occupy all pagespace again.
    In this situation, when you specify DB2_OVERRIDE_BPF parameter
    to override bufferpool size, it doesn't take effect in this
    situation. This
    is becasue DB2_OVERRIDE_BPF only applies for bufferpools that
    already exist at startup, and not to bufferpools that are
    created during crash recovery or roll forward. We will fix
    this problem and check
    DB2_OVERRIDE_BPF registry when creating bufferpools during crash
    recovery or roll forward. This problem only occurs in 64 bit
    instance.
    Problem Summary:
    This problem is caused by a misoperation, when you create or alt
    er a bufferpool, the size is too large to allocated from OS and
    cause overrall performance of the system is very slow and seems
    hang. Another problem is that DB2_OVERRIDE_BPF registry only app
    lies to bufferpools that already exists, and not bufferpools wil
    l be created in crash recovery or roll forward. So DB2 will try
    to create this big bufferpool again when doing crash recovery or
     roll forward. We will fix this problem and check DB2_OVERRIDE_B
    PF registry in our crash recovery and roll forward code.
    
    		

    Problem conclusion

    				
    				
    		

    Temporary fix

    				You can use db2iupdt -w 32 <INSTNAME> to update the instance to
    a 32 bit instance, then you can connect to the database
    succefully. After that, use db2iupdt -w 64 <INSTNAME> to update
    the instance to 64 bit again. Please take a backup at this
    point, so we don't have to roll forward through bufferpool
    creation that fails if we need restore and roll
    forward database.Please also notice that be careful when creati
    ng bufferpool, don't specify a too large value.
    
    		


    posted @ 2007-03-15 15:21 MyJavaWorld 閱讀(740) | 評(píng)論 (0)編輯 收藏

    Change db2 password in db2 level

    DB2 depends on operation system level authorization to control DB2 system access.

    In some case, our db2 account only allows to connect to instance or database, but not log on the system. In other words, we are not able to use the account to get a shell on unix or logon locally on windows.

    As usual, when we need to change db2 password, we take action on the system level because it's more frank. But in the case above that we can't get logon, we have to do it on db2 level. There are 2 ways.

    • Use the GUI tool DB2 Configuration Assistance
    • Use the command line tools DB2 CLP or CE

    Using DB2 Configuration Assistance

    Right click on a database name, then choose "Change password".

    CA tool interface

    Note: When more than one databases live in same system, we might share the same account in multi-database. Changing password for one of them will affect all databases in this system. In other words, we don't need to do the change on each database, we could choose any we have privileges on the system.

    Limitation: We are not able to change our password in the case of that we are only allowed to attach to the node but not connect to any database, or we don't get the catalog information for any database in the node.

    For this limitation, it works out with the 2nd way.?

    Using DB2 CLP and CE

    We could use CLP and CE to change password for databases and nodes that we connected or attached to.

    • Change by connecting to database
    C:\> db2 connect to SAMPLE user TEST using OLDPWD new NEWPWD confirm NEWPWD
    C:\> db2 connect to SAMPLE user TEST using OLDPWD change password
    • Change by attaching to node, this way works out for the limitation in Configuration Assistance
    C:\> db2 attach to NODE user TEST using OLDPWD new NEWPWD confirm NEWPWD
    C:\> db2 attach to NODE user TEST using OLDPWD change password


    alter table xxx VOLATILE

    valatile

    • a. 反復(fù)無常的,揮發(fā)性的

    VOLATILE CARDINALITY or NOT VOLATILE CARDINALITY
    Indicates to the optimizer whether or not the cardinality of table table-name can
    vary significantly at run time. Volatility applies to the number of rows in the
    table, not to the table itself. CARDINALITY is an optional keyword. The default
    is NOT VOLATILE.

    VOLATILE
    Specifies that the cardinality of table table-name can vary significantly at
    run time, from empty to large. To access the table, the optimizer will use
    an index scan (rather than a table scan, regardless of the statistics) if that
    index is index-only (all referenced columns are in the index), or that index
    is able to apply a predicate in the index scan. The list prefetch access method
    will not be used to access the table. If the table is a typed table, this option
    is only supported on the root table of the typed table hierarchy (SQLSTATE 428DR).

    NOT VOLATILE
    Specifies that the cardinality of table-name is not volatile.
    Access plans to this table will continue to be based on existing statistics and
    on the current optimization level.

    NOTE: The keyword could be specified within ALTER TABLE, but not CREATE TABLE.

    ?

    Difference between Local and System Database Directory

    DB2 automatically catalogs databases when they are created. It catalogs an entry
    for the database in the local database directory and another entry in the system
    database directory. If the database is created from a remote client (or a client which
    is executing from a different instance on the same machine), an entry is also made
    in the system database directory at the client instance.

    Databases on the same node as the database manager instance are cataloged as
    indirect entries. Databases on other nodes are cataloged as remote entries.

    List DBs in Local Database Directory

    $ db2 list db directory on /path_to_where_db_created?

    List DBs in System Database Directory

    $ db2 list db directory

    ?

    Illustration of standard tables

    Standard Tables Overview

    Query the status for one specified tablespace?

    As we know, we could issue following command to query status for all of tablespaces in current connected DB.

    $ db2 list tablespaces show detail > /tmp/ts.list

    Then find the status for the tablespaces we concerned.

    Is there's a way to query the status for one specified tablespace?

    My answer is No.?

    1. First, I don't find the related column defination in the table or view:
      • sysibm.systablespaces
      • syscat.tablespaces
    2. Second, I don't think DB2 stores the status flag within syscata tables. The reason is that once a tablespace comes into a special status, it will not be allowed changing any longer(this may also happen on a system catalog tablespace.) that will lead to a conflict on changing the tablespace status flag if stores it within system catalog tables.

    Illustration of the DMS table-space address map

    address map for DMS TS

    DB2 directories and files

    • http://publib.boulder.ibm.com/infocenter/db2luw/v8//topic/com.ibm.db2.udb.doc/admin/c0005420.htm

    Illustration of DB2 architechure and process

    overview for tuning

    Reference

    • http://publib.boulder.ibm.com/infocenter/db2luw/v8//topic/com.ibm.db2.udb.doc/admin/c0005418.htm?

    Tablespaces are put into backup pending after a load

    DB2 sets tablespace as Backup Pending state after loading data into a table in linear logging database(that is, logretain or userexit is on), whatever the load is successful or failed. the reason is that the load process will not write log file. from the begin time of loading to the end time of loading, there will be a blank segment in log file, and the rollforward recovery can Not jump over the blank segment to apply the later log file.

    so database needs a backup after loading to make sure db2 have recovery capicibility.

    it's able to use [COPY YES|NO]? or [NONRECOVERABLE] to prevent tablespace go into Backup Pending state. COPY YES will do the backup automatically after loading, and COPY NO and NONRECOVERABLE will give up this backup that means db2 will be not able to recover the database once an serious error occured.


    繼續(xù)閱讀 "Tablespaces are put into backup pending after a load" 的剩余內(nèi)容

    Come pending when droping Procedures

    Problem Description?

    There's no any response for creating or droping any procedures in DB2. Seems it's pending there, without any error returned. This time UPDATE statement on tables could be issued successfully.

    Check Process

    Check OS disk available

    $ df -kl?

    Check database configure

    db2 => get db cfg |more?

    Check system catalog tablespace?

    db2 => list tablespaces show detail |more
    syscatspace 0x0000?

    Cause?

    It caused by issuing CREATE or DROP statement without COMMIT or ROLLBACK statement in CLP which is in non-Autocommit mode.

    Solution?

    After you issue a COMMIT or a ROLLBACK in the CLP or close it, the pending is gone.

    We must be very careful when we use client in non-Autocommit mode just like CLPs in this case.

    To check the Autocommit mode in CLP issues. Refer to:

    set COMMIT mode in db2 clp

    db2 => list command options

    -c 自動(dòng)落實(shí) OFF

    More?

    From this point, we could get more. Most of strange things like this(pending there and no error return) are caused by locking.

    In this case, it was only the opertation on procedures being pended, so track on this thread to suppose the syscat tables related with procedures are locked then find the what probably causes it.
    ?

    db2 can not create index on local view

    There's an object called index view since MS SQL Server 2000, it's an index which is created on an view which is based on a local table. it is attent to imporve the select performence on the view when this view is based on multi-table and has complex logic.

    I try to find a simular thing in DB2, but there's no the simular solution in DB2 v8.2 for now. Instead, I found another thing instested.

    DB2 allows creating index on the tables or views in remote data source, such as a database on another host or an XML data source. To be exactly, that's Not true Index, it's Index Specification. It requests that the tables in remote data source have created index in its own system. And it only repeat the index description in local system.

    										Local DB2 Env??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? Remote DB2 Env
    										

    Index Specification??? ->??? Nickname?? - - ->?? True Index??? ->??? Table

    See following example in DB2 info center.

    • CREATE INDEX statement
      http://publib.boulder.ibm.com/infocenter/db2luw/v8//topic/com.ibm.db2.udb.doc/admin/r0000919.htm?resultof=%22%63%72%65%61%74%65%22%20%22%63%72%65%61%74%22%20%22%69%6e%64%65%78%22%20%22%73%74%61%74%65%6d%65%6e%74%22%20

    Example 3:? The nickname EMPLOYEE references a data source table called CURRENT_EMP. After this nickname was created, an index was defined on CURRENT_EMP. The columns chosen for the index key were WORKDEBT and JOB. Create an index specificationindex. Through this specification, the optimizer will know that the index exists and what its key is. With this information, the optimizer can improve its strategy to access the table. that describes this

       CREATE UNIQUE INDEX JOB_BY_DEPT
    ON EMPLOYEE (WORKDEPT, JOB)
    SPECIFICATION ONLY

    Note: the Nickname is only allowed to be created on tables in local database and on much kind of objects at remote data source. It's not allowed to be created on views in local database.

    set COMMIT mode in db2 clp

    As default, DB2 CLP will do commit after you issue each DB2 statements or SQL automatically.

    If you want to change it instead of issuing COMMIT or ROLLBACK manually, do following.

    SQLLIB\BIN> db2
    db2 => list command options
    ??? -c?? ON
    db2 => update command options using c off
    db2 => list command options
    ??? -c?? OFF

    This change will only exists during this session. The setting will lose when you close this CLP and open CLP next time.

    following cmd will get the same result with above.

    SQLLIB\BIN> db2 list command options
    ??? -c??? ON
    SQLLIB\BIN> db2 +c
    db2 =>
    db2 => list command options
    ??? -c??? OFF
    db2 => quit
    SQLLIB\BIN> db2 list command options
    ??? -c??? ON

    There's also the way to keep the settings forever, do following.

    SQLLIB\BIN\> db2 list command options
    ??? -c?? ON
    SQLLIB\BIN\> db2set db2options=+c
    SQLLIB\BIN\> db2 list command options
    ??? -c?? OFF

    This change will take effection immediately even for others having been opened CLP windows. And will keep along.

    Use following cmd set it back to ON.?

    SQLLIB\BIN> db2set db2opptions=-c

    Reference

    表與索引的重命名 RENAME

    在DB2 中重命名表或者索引

    db2=> RENAME TABLE EMP TO EMPLOYEE
    db2=> RENAME TABLE ABC.EMP TO EMPLOYEE
    db2=> RENAME INDEX NEW-IND TO IND
    db2=> RENAME INDEX ABC.NEW-IND TO IND
    posted @ 2007-03-15 15:00 MyJavaWorld 閱讀(528) | 評(píng)論 (0)編輯 收藏
    三個(gè)UNIX文件時(shí)間ctime、mtime、atime
    ?????? 我曾經(jīng)根據(jù)文件的狀態(tài)在指定時(shí)間內(nèi)是否改變寫過一個(gè)WatchDog來對(duì)服務(wù)進(jìn)行監(jiān)控,其間曾被這三個(gè)時(shí)間搞混淆,所以覺得很有必要和大家分享我對(duì)這三個(gè)術(shù)語的理解。
    ????? ctime(change time)改變時(shí)間:是指文件狀態(tài)最后一次被改變的時(shí)間;
    ????? mtime(modification time)修改時(shí)間:是指文件內(nèi)容最后一次被改變的時(shí)間;
    ????? atime(access time)訪問時(shí)間:是指文件最后一次被讀取的時(shí)間。
    ????? 前兩者的區(qū)別就在于文件狀態(tài)的改變既包括文件索引節(jié)點(diǎn)的改變,也包括文件內(nèi)容的改變。也就是說如果你改變了文件內(nèi)容,則同時(shí)更新了ctime和mtime,但是如果你只改變了文件索引節(jié)點(diǎn)則只是改變了ctime。atime只有在文件被讀取的時(shí)侯才會(huì)改變。它的改變與文件狀態(tài)以及文件內(nèi)容的改變沒有直接的聯(lián)系。
    ???? 例如:echo “Hello World” >> myfile 則同時(shí)改變了ctime和mtime,atime不變;
    ???? chmod u+x myfile 則只改變了ctime,mtime和atime不變。
    ???? cat myfile,則只改變了atime,ctime和mtime不變
    ???? ps:以上操作均在redhat linux下驗(yàn)證通過
    posted @ 2007-03-13 17:51 MyJavaWorld 閱讀(1174) | 評(píng)論 (0)編輯 收藏
    ?
    ?
    查看文章
    ?
    DB2/SQL命令大全
    2006-12-25 13:21

    連接數(shù)據(jù)庫:

    ??connect?to?[數(shù)據(jù)庫名]?user?[操作用戶名]?using?[密碼]?

    創(chuàng)建緩沖池(8K):

    ??create?bufferpool?ibmdefault8k?IMMEDIATE??SIZE?5000?PAGESIZE?8?K?;
    創(chuàng)建緩沖池(16K)(OA_DIVERTASKRECORD):
    ??create?bufferpool?ibmdefault16k?IMMEDIATE??SIZE?5000?PAGESIZE?16?K?;
    創(chuàng)建緩沖池(32K)(OA_TASK):
    ??create?bufferpool?ibmdefault32k?IMMEDIATE??SIZE?5000?PAGESIZE?32?K?;

    創(chuàng)建表空間:

    ??CREATE?TABLESPACE?exoatbs?IN?DATABASE?PARTITION?GROUP?IBMDEFAULTGROUP?PAGESIZE?8K?MANAGED?BY?SYSTEM?USING?('/home/exoa2/exoacontainer')?EXTENTSIZE?32?PREFETCHSIZE?16??BUFFERPOOL?IBMDEFAULT8K??OVERHEAD?24.10?TRANSFERRATE?0.90??DROPPED?TABLE?RECOVERY?OFF;

    ??CREATE?TABLESPACE?exoatbs16k??IN?DATABASE?PARTITION?GROUP?IBMDEFAULTGROUP?PAGESIZE?16K?MANAGED?BY?SYSTEM?USING?('/home/exoa2/exoacontainer16k'???)?EXTENTSIZE?32??PREFETCHSIZE?16??BUFFERPOOL?IBMDEFAULT16K??OVERHEAD?24.1?TRANSFERRATE?0.90??DROPPED?TABLE?RECOVERY?OFF;

    ??CREATE?TABLESPACE?exoatbs32k??IN?DATABASE?PARTITION?GROUP?IBMDEFAULTGROUP?PAGESIZE?32K?MANAGED?BY?SYSTEM?USING?('/home/exoa2/exoacontainer32k'???)?EXTENTSIZE?32??PREFETCHSIZE?16??BUFFERPOOL?IBMDEFAULT32K??OVERHEAD?24.1?TRANSFERRATE?0.90??DROPPED?TABLE?RECOVERY?OFF;

    GRANT?USE?OF?TABLESPACE?exoatbs?TO?PUBLIC;
    GRANT?USE?OF?TABLESPACE?exoatbs16k?TO?PUBLIC;
    GRANT?USE?OF?TABLESPACE?exoatbs32k?TO?PUBLIC;

    創(chuàng)建系統(tǒng)表空間:

    ??CREATE?TEMPORARY?TABLESPACE?exoasystmp?IN?DATABASE?PARTITION?GROUP?IBMTEMPGROUP?PAGESIZE?8K??MANAGED?BY?SYSTEM?USING?('/home/exoa2/exoasystmp'???)?EXTENTSIZE?32?PREFETCHSIZE?16?BUFFERPOOL?IBMDEFAULT8K??OVERHEAD?24.10?TRANSFERRATE?0.90??DROPPED?TABLE?RECOVERY?OFF;

    ??CREATE?TEMPORARY?TABLESPACE?exoasystmp16k?IN?DATABASE?PARTITION?GROUP?IBMTEMPGROUP?PAGESIZE?16K?MANAGED?BY?SYSTEM?USING?('/home/exoa2/exoasystmp16k'??)?EXTENTSIZE?32?PREFETCHSIZE?16?BUFFERPOOL?IBMDEFAULT16K?OVERHEAD?24.10?TRANSFERRATE?0.90??DROPPED?TABLE?RECOVERY?OFF;

    ??CREATE?TEMPORARY?TABLESPACE?exoasystmp32k?IN?DATABASE?PARTITION?GROUP?IBMTEMPGROUP?PAGESIZE?32K?MANAGED?BY?SYSTEM?USING?('/home/exoa2/exoasystmp32k')?EXTENTSIZE?32?PREFETCHSIZE?16?BUFFERPOOL?IBMDEFAULT32K?OVERHEAD?24.10?TRANSFERRATE?0.90??DROPPED?TABLE?RECOVERY?OFF;

    1.?啟動(dòng)實(shí)例(db2inst1):

    db2start

    2.?停止實(shí)例(db2inst1):

    db2stop

    3.?列出所有實(shí)例(db2inst1)

    db2ilist

    5.列出當(dāng)前實(shí)例:

    db2?get?instance

    4.?察看示例配置文件:

    db2?get?dbm?cfg|more

    5.?更新數(shù)據(jù)庫管理器參數(shù)信息:

    db2?update?dbm?cfg?using?para_name?para_value

    6.?創(chuàng)建數(shù)據(jù)庫:

    db2?create?db?test

    7.?察看數(shù)據(jù)庫配置參數(shù)信息

    db2?get?db?cfg?for?test|more

    8.?更新數(shù)據(jù)庫參數(shù)配置信息

    db2?update?db?cfg?for?test?using?para_name?para_value

    10.刪除數(shù)據(jù)庫:

    db2?drop?db?test

    11.連接數(shù)據(jù)庫

    db2?connect?to?test

    12.列出所有表空間的詳細(xì)信息。

    db2?list?tablespaces?show?detail

    13.查詢數(shù)據(jù):

    db2?select?*?from?tb1

    14.刪除數(shù)據(jù):

    db2?delete?from?tb1?where?id=1

    15.創(chuàng)建索引:

    db2?create?index?idx1?on?tb1(id);

    16.創(chuàng)建視圖:

    db2?create?view?view1?as?select?id?from?tb1

    17.查詢視圖:

    db2?select?*?from?view1

    18.節(jié)點(diǎn)編目

    db2?catalog?tcp?node?node_name?remote?server_ip?server?server_port

    19.察看端口號(hào)

    db2?get?dbm?cfg|grep?SVCENAME

    20.測(cè)試節(jié)點(diǎn)的附接

    db2?attach?to?node_name

    21.察看本地節(jié)點(diǎn)

    db2?list?node?direcotry

    22.節(jié)點(diǎn)反編目

    db2?uncatalog?node?node_name

    23.數(shù)據(jù)庫編目

    db2?catalog?db?db_name?as?db_alias?at?node?node_name

    24.察看數(shù)據(jù)庫的編目

    db2?list?db?directory

    25.連接數(shù)據(jù)庫

    db2?connect?to?db_alias?user?user_name?using?user_password

    26.數(shù)據(jù)庫反編目

    db2?uncatalog?db?db_alias

    27.導(dǎo)出數(shù)據(jù)

    db2?export?to?myfile?of?ixf?messages?msg?select?*?from?tb1

    28.導(dǎo)入數(shù)據(jù)

    db2?import?from?myfile?of?ixf?messages?msg?replace?into?tb1

    29.導(dǎo)出數(shù)據(jù)庫的所有表數(shù)據(jù)

    db2move?test?export

    30.生成數(shù)據(jù)庫的定義

    db2look?-d?db_alias?-a?-e?-m?-l?-x?-f?-o?db2look.sql

    31.創(chuàng)建數(shù)據(jù)庫

    db2?create?db?test1

    32.生成定義

    db2?-tvf?db2look.sql

    33.導(dǎo)入數(shù)據(jù)庫所有的數(shù)據(jù)

    db2move?db_alias?import

    34.重組檢查

    db2?reorgchk

    35.重組表tb1

    db2?reorg?table?tb1

    36.更新統(tǒng)計(jì)信息

    db2?runstats?on?table?tb1

    37.備份數(shù)據(jù)庫test

    db2?backup?db?test

    38.恢復(fù)數(shù)據(jù)庫test

    db2?restore?db?test

    399\.列出容器的信息

    db2?list?tablespace?containers?for?tbs_id?show?detail

    40.創(chuàng)建表:

    db2?ceate?table?tb1(id?integer?not?null,name?char(10))

    41.列出所有表

    db2?list?tables

    42.插入數(shù)據(jù):

    db2?insert?into?tb1?values(1,’sam’);

    db2?insert?into?tb2?values(2,’smitty’);

    .?建立數(shù)據(jù)庫DB2_GCB?

    CREATE?DATABASE?DB2_GCB?ON?G:?ALIAS?DB2_GCB?

    USING?CODESET?GBK?TERRITORY?CN?COLLATE?USING?SYSTEM?DFT_EXTENT_SZ?32?

    2.?連接數(shù)據(jù)庫?

    connect?to?sample1?user?db2admin?using?8301206?

    3.?建立別名?

    create?alias?db2admin.tables?for?sysstat.tables;?

    CREATE?ALIAS?DB2ADMIN.VIEWS?FOR?SYSCAT.VIEWS?

    create?alias?db2admin.columns?for?syscat.columns;?

    create?alias?guest.columns?for?syscat.columns;?

    4.?建立表?

    create?table?zjt_tables?as?

    (select?*?from?tables)?definition?only;?

    create?table?zjt_views?as?

    (select?*?from?views)?definition?only;?

    5.?插入記錄?

    insert?into?zjt_tables?select?*?from?tables;?

    insert?into?zjt_views?select?*?from?views;?

    6.?建立視圖?

    create?view?V_zjt_tables?as?select?tabschema,tabname?from?zjt_tables;?

    7.?建立觸發(fā)器?

    CREATE?TRIGGER?zjt_tables_del?

    AFTER?DELETE?ON?zjt_tables?

    REFERENCING?OLD?AS?O?

    FOR?EACH?ROW?MODE?DB2SQL?

    Insert?into?zjt_tables1?values(substr(o.tabschema,1,8),substr(o.tabname,1,10))?

    8.?建立唯一性索引?

    CREATE?UNIQUE?INDEX?I_ztables_tabname?

    [size=3]ON?zjt_tables(tabname);?

    9.?查看表?

    select?tabname?from?tables?

    where?tabname='ZJT_TABLES';?

    10.?查看列?

    select?SUBSTR(COLNAME,1,20)?as?列名,TYPENAME?as?類型,LENGTH?as?長(zhǎng)度?

    from?columns?

    where?tabname='ZJT_TABLES';?

    11.?查看表結(jié)構(gòu)?

    db2?describe?table?user1.department?

    db2?describe?select?*?from?user.tables?

    12.?查看表的索引?

    db2?describe?indexes?for?table?user1.department?

    13.?查看視圖?

    select?viewname?from?views?

    where?viewname='V_ZJT_TABLES';?

    14.?查看索引?

    select?indname?from?indexes?

    where?indname='I_ZTABLES_TABNAME';?

    15.?查看存貯過程?

    SELECT?SUBSTR(PROCSCHEMA,1,15),SUBSTR(PROCNAME,1,15)?

    FROM?SYSCAT.PROCEDURES;?

    16.?類型轉(zhuǎn)換(cast)?

    ip?datatype:varchar?

    select?cast(ip?as?integer)+50?from?log_comm_failed?

    17.?重新連接?

    connect?reset?

    18.?中斷數(shù)據(jù)庫連接?

    disconnect?db2_gcb?

    19.?view?application?

    LIST?APPLICATION;?

    20.?kill?application?

    FORCE?APPLICATION(0);?

    db2?force?applications?all?(強(qiáng)迫所有應(yīng)用程序從數(shù)據(jù)庫斷開)?

    21.?lock?table

    lock?table?test?in?exclusive?mode?

    22.?共享?

    lock?table?test?in?share?mode?

    23.?顯示當(dāng)前用戶所有表?

    list?tables?

    24.?列出所有的系統(tǒng)表?

    list?tables?for?system?

    25.?顯示當(dāng)前活動(dòng)數(shù)據(jù)庫?

    list?active?databases?

    26.?查看命令選項(xiàng)?

    list?command?options?

    27.?系統(tǒng)數(shù)據(jù)庫目錄?

    LIST?DATABASE?DIRECTORY?

    28.?表空間?

    list?tablespaces?

    29.?表空間容器?

    LIST?TABLESPACE?CONTAINERS?FOR?

    Example:?LIST?TABLESPACE?CONTAINERS?FOR?1?

    30.?顯示用戶數(shù)據(jù)庫的存取權(quán)限?

    GET?AUTHORIZATIONS?

    31.?啟動(dòng)實(shí)例?

    DB2START?

    32.?停止實(shí)例?

    db2stop?

    33.?表或視圖特權(quán)?

    grant?select,delete,insert,update?on?tables?to?user?

    grant?all?on?tables?to?user?WITH?GRANT?OPTION?

    34.?程序包特權(quán)?

    GRANT?EXECUTE?

    ON?PACKAGE?PACKAGE-name?

    TO?PUBLIC?

    35.?模式特權(quán)?

    GRANT?CREATEIN?ON?SCHEMA?SCHEMA-name?TO?USER?

    36.?數(shù)據(jù)庫特權(quán)?

    grant?connect,createtab,dbadm?on?database?to?user?

    37.?索引特權(quán)?

    grant?control?on?index?index-name?to?user?

    38.?信息幫助?(??XXXnnnnn?)?

    例:??SQL30081?

    39.?SQL?幫助(說明?SQL?語句的語法)?

    help?statement?

    例如,help?SELECT?

    40.?SQLSTATE?幫助(說明?SQL?的狀態(tài)和類別代碼)?

    ??sqlstate?或???class-code?

    41.?更改與"管理服務(wù)器"相關(guān)的口令?

    db2admin?setid?username?password?

    42.?創(chuàng)建?SAMPLE?數(shù)據(jù)庫?

    db2sampl?

    db2sampl?F:(指定安裝盤)?

    43.?使用操作系統(tǒng)命令?

    !?dir?

    44.?轉(zhuǎn)換數(shù)據(jù)類型?(cast)?

    SELECT?EMPNO,?CAST(RESUME?AS?VARCHAR(370))?

    FROM?EMP_RESUME?

    WHERE?RESUME_FORMAT?=?'ascii'?

    45.?UDF

    要運(yùn)行?DB2?Java?存儲(chǔ)過程或?UDF,還需要更新服務(wù)器上的?DB2?數(shù)據(jù)庫管理程序配置,以包括在該機(jī)器上安裝?JDK?的路徑?

    db2?update?dbm?cfg?using?JDK11_PATH?d:sqllibjavajdk?

    TERMINATE?

    update?dbm?cfg?using?SPM_NAME?sample?

    46.?檢查?DB2?數(shù)據(jù)庫管理程序配置?

    db2?get?dbm?cfg?

    47.?檢索具有特權(quán)的所有授權(quán)名?

    SELECT?DISTINCT?GRANTEE,?GRANTEETYPE,?'DATABASE'?FROM?SYSCAT.DBAUTH?

    UNION?

    SELECT?DISTINCT?GRANTEE,?GRANTEETYPE,?'TABLE?'?FROM?SYSCAT.TABAUTH?

    UNION?

    SELECT?DISTINCT?GRANTEE,?GRANTEETYPE,?'PACKAGE?'?FROM?SYSCAT.PACKAGEAUTH?

    UNION?

    SELECT?DISTINCT?GRANTEE,?GRANTEETYPE,?'INDEX?'?FROM?SYSCAT.INDEXAUTH?

    UNION?

    SELECT?DISTINCT?GRANTEE,?GRANTEETYPE,?'COLUMN?'?FROM?SYSCAT.COLAUTH?

    UNION?

    SELECT?DISTINCT?GRANTEE,?GRANTEETYPE,?'SCHEMA?'?FROM?SYSCAT.SCHEMAAUTH?

    UNION?

    SELECT?DISTINCT?GRANTEE,?GRANTEETYPE,?'SERVER?'?FROM?SYSCAT.PASSTHRUAUTH?

    ORDER?BY?GRANTEE,?GRANTEETYPE,?3?

    create?table?yhdab?

    (id?varchar(10),?

    password?varchar(10),?

    ywlx?varchar(10),?

    kh?varchar(10));?

    create?table?ywlbb?

    (ywlbbh?varchar(8),?

    ywmc?varchar(60))?

    48.?修改表結(jié)構(gòu)?

    alter?table?yhdab?ALTER?kh?SET?DATA?TYPE?varchar(13);?

    alter?table?yhdab?ALTER?ID?SET?DATA?TYPE?varchar(13);?

    alter?table?lst_bsi?alter?bsi_money?set?data?type?int;?

    insert?into?yhdab?values?

    ('20000300001','123456','user01','20000300001'),?

    ('20000300002','123456','user02','20000300002');?

    49.?業(yè)務(wù)類型說明?

    insert?into?ywlbb?values?

    ('user01','業(yè)務(wù)申請(qǐng)'),?

    ('user02','業(yè)務(wù)撤消'),?

    ('user03','費(fèi)用查詢'),?

    ('user04','費(fèi)用自繳'),?

    ('user05','費(fèi)用預(yù)存'),?

    ('user06','密碼修改'),?

    ('user07','發(fā)票打印'),?

    ('gl01','改用戶基本信息'),?

    ('gl02','更改支付信息'),?

    ('gl03','日統(tǒng)計(jì)功能'),?

    ('gl04','沖帳功能'),?

    ('gl05','對(duì)帳功能'),?

    ('gl06','計(jì)費(fèi)功能'),?

    ('gl07','綜合統(tǒng)計(jì)')?

    備份數(shù)據(jù)庫:
    CONNECT?TO?EXOA;
    QUIESCE?DATABASE?IMMEDIATE?FORCE?CONNECTIONS;
    CONNECT?RESET;
    BACKUP?DATABASE?EXOA?TO?"/home/exoa2/db2bak/"?WITH?2?BUFFERS?BUFFER?1024?PARALLELISM?1?WITHOUT?PROMPTING;
    CONNECT?TO?EXOA;
    UNQUIESCE?DATABASE;
    CONNECT?RESET;

    以下是小弟在使用db2move中的一些經(jīng)驗(yàn),希望對(duì)大家有所幫助。?

    ?db2???connect???to??YOURDB???
    連接數(shù)據(jù)庫?

    ?db2look?-d??YOURDB??-a?-e?-x?-o?creatab.sql?
    導(dǎo)出建庫表的SQL?

    ?db2move???YOURDB??export?
    用db2move將數(shù)據(jù)備份出來?

    ?vi???creatab.sql?
    如要導(dǎo)入的數(shù)據(jù)庫名與原數(shù)據(jù)庫不同,要修改creatab.sql中CONNECT?項(xiàng)?
    如相同則不用更改?

    ?db2move??NEWDB??load?
    將數(shù)據(jù)導(dǎo)入新庫中?

    在導(dǎo)入中可能因?yàn)榉N種原因發(fā)生中斷,會(huì)使數(shù)據(jù)庫暫掛?
    db2????list?tablespaces???show???detail?
    如:?
    ??????詳細(xì)說明:?
    ?????裝入暫掛?
    ?總頁數(shù)??????????????????????????=?1652?
    ?可用頁數(shù)????????????????????????=?1652?
    ?已用頁數(shù)?????????????????????????=?1652?
    ?空閑頁數(shù)?????????????????????????=?不適用?
    ?高水位標(biāo)記(頁)?????????????????=?不適用?
    ?頁大小(字節(jié))???????????????????=?4096?
    ?盤區(qū)大?。摚???????????????????=?32?
    ?預(yù)讀取大?。摚?????????????????=?32?
    ?容器數(shù)???????????????????????????=?1?
    ?狀態(tài)更改表空間標(biāo)識(shí)????????????????????=?2?
    ?狀態(tài)更改對(duì)象標(biāo)識(shí)??????????????????????=?59?

    ?db2?select?tabname,tableid?from?syscat.tables?where?tableid=59?
    查看是哪張表掛起?

    表名知道后到db2move.lst(在db2move??YOURDB??export的目錄中)中找到相應(yīng)的.ixf文件?
    ?db2?load?from?tab11.ixf?of?ixf?terminate?into?db2admin.xxxxxxxxx?
    tab11.ixf對(duì)應(yīng)的是xxxxxxxxx表?

    數(shù)據(jù)庫會(huì)恢復(fù)正常,可再用db2?list?tablespaces?show?detail查看

    30.不能通過GRANT授權(quán)的權(quán)限有哪種?

    SYSAM

    SYSCTRL

    SYSMAINT

    要更該述權(quán)限必須修改數(shù)據(jù)庫管理器配置參數(shù)

    31.表的類型有哪些?

    永久表(基表)

    臨時(shí)表(說明表)

    臨時(shí)表(派生表)

    32.如何知道一個(gè)用戶有多少表?

    SELECT*FROMSYSIBM.SYSTABLESWHERECREATOR='USER'

    33.如何知道用戶下的函數(shù)?

    select*fromIWH.USERFUNCTION

    select*fromsysibm.SYSFUNCTIONS

    34.如何知道用戶下的VIEW數(shù)?

    select*fromsysibm.sysviewsWHERECREATOR='USER'

    35.如何知道當(dāng)前DB2的版本?

    select*fromsysibm.sysvERSIONS

    36.如何知道用戶下的TRIGGER數(shù)?

    select*fromsysibm.SYSTRIGGERSWHERESCHEMA='USER'

    37.如何知道TABLESPACE的狀況?

    select*fromsysibm.SYSTABLESPACES

    38.如何知道SEQUENCE的狀況?

    select*fromsysibm.SYSSEQUENCES

    39.如何知道SCHEMA的狀況?

    select*fromsysibm.SYSSCHEMATA

    40.如何知道INDEX的狀況?

    select*fromsysibm.SYSINDEXES

    41.如何知道表的字段的狀況?

    select*fromsysibm.SYSCOLUMNSWHERETBNAME='AAAA'

    42.如何知道DB2的數(shù)據(jù)類型?

    select*fromsysibm.SYSDATATYPES

    43.如何知道BUFFERPOOLS狀況?

    select*fromsysibm.SYSBUFFERPOOLS

    44.DB2表的字段的修改限制?

    只能修改VARCHAR2類型的并且只能增加不能減少.

    45.如何查看表的結(jié)構(gòu)?

    DESCRIBLETABLETABLE_NAME

    OR

    DESCRIBLESELECT*FROMSCHEMA.TABLE_NAME

    ?
    ?

    ? ? ?
    ?
    posted @ 2006-12-28 16:57 MyJavaWorld 閱讀(2762) | 評(píng)論 (0)編輯 收藏

    DB2 UDB中用戶出口程序是如何工作的?

    ?

    采用用戶出口程序與DB2 UDB共同工作背后的基本思想是提供一種歸檔和檢索數(shù)據(jù)庫日志文件的方法以實(shí)現(xiàn)日志冗余處理并從易失性介質(zhì)上轉(zhuǎn)儲(chǔ)出來。重要的,值得注意的是,根據(jù)你的特殊需求,在用戶出口程序當(dāng)中,除了能實(shí)現(xiàn)歸檔和檢索日志之外,你還可以實(shí)現(xiàn)其他的操作。

    ?

    如果一個(gè)數(shù)據(jù)庫需要采用歸檔日志文件來進(jìn)行恢復(fù),在DB2 UDB中實(shí)行用戶出口程序策略將不能恢復(fù)100%的事務(wù)。用戶出口程序只是一種通過拷貝已經(jīng)存在的日志文件到一個(gè)安全的地方來為其提供更多保護(hù)的方法。它是數(shù)據(jù)完整性策略中的一部分,而且是重要的一部分。

    ?

    編譯了用戶出口程序之后,可執(zhí)行程序db2uext2被放置在數(shù)據(jù)庫管理器可以找到的目錄當(dāng)中。在UNIX?中,這個(gè)目錄是/sqllib/adm;在Windows?中,是Program FilesIBMSQLLIBBIN。

    ?

    除非數(shù)據(jù)庫管理器知道用戶出口程序是可用的,不然它不會(huì)調(diào)用db2uext2。讓數(shù)據(jù)庫管理器知道db2uext2可以被調(diào)用的唯一方法是將數(shù)據(jù)庫配置參數(shù)userexit設(shè)置為on。一旦這個(gè)參數(shù)被設(shè)定好而且DB2實(shí)例被重復(fù)利用了,數(shù)據(jù)庫管理器將每五分鐘調(diào)用一次用戶出口程序來檢查那些可以被歸檔到程序相關(guān)的歸檔目錄的日志文件。

    ?

    如果數(shù)據(jù)庫的恢復(fù)是必要的,在前滾操作期間,數(shù)據(jù)庫管理器將調(diào)用db2uext2把歸檔日志文件拷貝回活動(dòng)的日志目錄當(dāng)中。然后,日志文件被再次運(yùn)用到重建的數(shù)據(jù)庫中。

    ?

    讓我們看看被數(shù)據(jù)庫管理器生成的到用戶出口程序的調(diào)用的格式。注意,這一信息也可以在用戶出口示例程序的注釋部分找到。

    db2uext2 -OS -RL

    -RQ -DB -NN

    -LP -LN [-AP]

    ?

    其中:

    os=操作系統(tǒng)

    release=DB2發(fā)行版本

    request= 'ARCHIVE' 或 'RETRIEVE'

    dbname=數(shù)據(jù)庫名

    nodenumber=節(jié)點(diǎn)號(hào)

    logpath=日志文件目錄

    logname=日志文件名

    logsize=日志文件大小(可選)

    startingpage=以4k頁為單位的起始偏移量(可選)

    adsmpasswd=ADSM密碼(可選)

    ?

    注意:只有當(dāng)logpath為裸設(shè)備時(shí)才使用logsize和startingpage。

    ?

    歸檔或者從磁盤檢索日志文件遵從以下命名規(guī)則:

    歸檔:歸檔路徑+數(shù)據(jù)庫名+節(jié)點(diǎn)號(hào)+日志文件名

    檢索:檢索路徑+數(shù)據(jù)庫名+節(jié)點(diǎn)號(hào)+日志文件名

    ?

    比如:如果歸檔的路徑為”c:mylogs”,

    檢索路徑是“c:mylogs”,

    數(shù)據(jù)庫名是“SAMPLE”,

    節(jié)點(diǎn)號(hào)是 NODE0000,

    文件名是 S0000001.LOG,

    日志文件將是:

    歸檔到 - c:mylogsSAMPLENODE0000S0000001.LOG

    檢索自 - c:mylogsSAMPLENODE0000S0000001.LOG

    以下描述了用戶出口程序中的邏輯是如何運(yùn)轉(zhuǎn)的:

    1)? 安裝信號(hào)處理程序。

    2)? 驗(yàn)證傳遞的參數(shù)個(gè)數(shù)。

    3)? 驗(yàn)證操作請(qǐng)求。

    4)? 開始審計(jì)跟蹤(如果有此請(qǐng)求)。

    5)? 根據(jù)操作需求的不同獲取以下路徑中的一個(gè):

    a)? 如果操作需求為歸檔一個(gè)文件,將日志文件從日志路徑拷貝到歸檔路徑中。

    i) 如果沒有找到日志文件,執(zhí)行第6點(diǎn)。

    ?

    b)? 如果操作需求是檢索一個(gè)文件,將日志文件從檢索路徑拷貝到日志路徑中。

    i) 如果沒有找到日志文件,執(zhí)行第6點(diǎn)。

    6) 將錯(cuò)誤記入日志(如果要求或者有需要的話)。

    7)? 結(jié)束審計(jì)跟蹤(如果有請(qǐng)求)。

    8)? 以適當(dāng)?shù)姆祷卮a退出。

    ?

    手工調(diào)用用戶出口程序來歸檔日志文件是可以的,但最好還是使用ARCHIVE LOG命令以便在定義以上參數(shù)時(shí)不會(huì)因?yàn)槟愕脑驅(qū)е洛e(cuò)誤。在這篇文章的末尾,可以找到關(guān)于ARCHIVE LOG命令的鏈接。

    ?

    日志文件術(shù)語

    ?

    DB2 中用戶出口程序的基本功能是將日志文件拷貝到活動(dòng)日志目錄或反之。這里,值得指出一些術(shù)語來闡明活動(dòng)的日志目錄被置于何處以及數(shù)據(jù)庫日志文件的狀態(tài)如何。

    ?

    活動(dòng)日志目錄:

    ?

    該目錄被置于你的數(shù)據(jù)庫目錄中。在Windows中,如果在C:中創(chuàng)建了一個(gè)叫做SAMPLE的單一數(shù)據(jù)庫且實(shí)例名為db2inst1,則將會(huì)出現(xiàn)以下的目錄結(jié)構(gòu):

    ?

    C:DB2INST1NODE0000SQL000001SQLOGDIR

    ?

    SQL00001是SAMPLE數(shù)據(jù)庫的數(shù)據(jù)庫目錄,SQLOGDIR 是活動(dòng)日志目錄。

    ?

    以下的圖1顯示了Windows操作系統(tǒng)上的活動(dòng)日志目錄:

    PATH o:extrusionok="f" gradientshapeok="t" o:connecttype="rect" />SHAPE id=_x0000_i1025 style="WIDTH: 299.25pt; HEIGHT: 180.75pt" type="#_x0000_t75" />/P>

    ?

    圖 1活動(dòng)日志目錄

    日志文件狀態(tài)

    ?

    在活動(dòng)日志目錄中,日志文件可以是活動(dòng)日志,也可以是聯(lián)機(jī)歸檔日志?;顒?dòng)日志是那些被 DB2 用于當(dāng)前事務(wù)處理和崩潰恢復(fù)的日志。聯(lián)機(jī)歸檔日志是那些 DB2 UDB 進(jìn)行常規(guī)處理時(shí)不再需要,而進(jìn)行數(shù)據(jù)庫恢復(fù)時(shí)可能還會(huì)需要的日志。當(dāng)實(shí)現(xiàn)用戶出口程序時(shí),這些聯(lián)機(jī)歸檔日志將最終以歸檔日志目錄中的副本形式出現(xiàn)。

    ?

    既然 DB2 UDB 中用戶出口程序的目的是將數(shù)據(jù)庫日志拷貝到歸檔目錄中,你將最終在活動(dòng)日志目錄(缺省為 SQLOGDIR)中得到重復(fù)的日志文件。你可能考慮刪除這些重復(fù)的聯(lián)機(jī)歸檔日志以釋放文件系統(tǒng)空間。在從數(shù)據(jù)庫目錄中除去這些日志之前,要十分小心地確認(rèn)它們是否已經(jīng)將成功地被復(fù)制到歸檔目錄中。還必須確保數(shù)據(jù)庫管理器進(jìn)行崩潰恢復(fù)時(shí)不再需要它們。要確定你的活動(dòng)日志目錄中哪些日志文件不為正常處理所需要,可用通過以下命令檢查數(shù)據(jù)庫配置:

    ????????

    ???? db2 "get db cfg for sample"

    ?

    這條命令的數(shù)據(jù)庫配置輸出結(jié)果將包含第一個(gè)活動(dòng)日志文件,如:

    First active log file = S000009.LOG

    ?

    在如上所示的輸出當(dāng)中,日志文件S000009.LOG是數(shù)據(jù)庫當(dāng)前的活動(dòng)日志。任何編號(hào)比該日志文件小的日志文件都被看作聯(lián)機(jī)歸檔日志。

    ?

    這里有一個(gè)例子:

    ?

    在這個(gè)場(chǎng)景中,活動(dòng)日志目錄中有日志文件 S000000.LOG - S000009.LOG,歸檔日志目錄中有 S000000.LOG - S000008.LOG。因?yàn)?S000009.LOG 是第一個(gè)活動(dòng)日志文件,所以,可以從活動(dòng)日志目錄中移走 S000001.LOG - S000008.LOG 以釋放磁盤空間。S000009.LOG 文件必須保留在活動(dòng)日志目錄中,因?yàn)樗匀槐划?dāng)前事務(wù)使用。

    ?

    同樣可以通過檢查數(shù)據(jù)庫歷史文件來查看活動(dòng)日志目錄中哪些日志文件不再有用。以下命令將列出數(shù)據(jù)庫備份信息:

    ?

    ???? db2 "list history backup all for database sample"

    ?

    以下是該命令的輸出結(jié)果的一個(gè)例子:

    ?

    List History File for sample

    Number of matching file entries = 4

    Op Obj Timestamp+Sequence Type Dev Earliest Log Current? Log

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

    B D 20030416162026001 F D S0000010.LOGS0000014.LOG

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

    Contains 2 tablespace(s):

    00001 SYSCATSPACE

    00002 USERSPACE1

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

    ?

    在上面的輸出中,最早的日志將表示,需要 S0000010.LOG 及其之后的任何日志。 S00000010.LOG 之前的任何日志可以被安全的刪除。再次提醒,在從活動(dòng)日志目錄中刪除日志文件之前,驗(yàn)證在活動(dòng)日志目錄中存在這些日志文件的拷貝是十分重要的。

    ?

    雖然可以手動(dòng)的從活動(dòng)日志目錄中刪除日志文件,刪除聯(lián)機(jī)歸檔日志文件的安全方法是通過prune logfile命令。該命令可以用來刪除活動(dòng)歸檔目錄中的日志文件。在下面的示例中,以下命令將刪除日志文件 S000000.LOG - S000008.LOG:

    ???? db2 "prune logfile prior to S000009.LOG"

    ?

    注意:根據(jù)您的恢復(fù)策略,可能會(huì)出現(xiàn)之前的前滾操作在數(shù)據(jù)庫上執(zhí)行的場(chǎng)景。歸檔目錄中的老日志文件可能被同名的新日志文件所覆蓋,從而阻止你使用舊日志文件對(duì)數(shù)據(jù)庫進(jìn)行時(shí)間點(diǎn)恢復(fù)。對(duì)用戶出口程序的設(shè)計(jì)程序員來說,考慮類似這樣的情況是很重要的。

    ?

    設(shè)置用戶出口

    ?

    在本文中,我們將采用DB2提供的db2uext2.cdisk樣例c程序,它位于你的c目錄當(dāng)中。在UNIX中,c目錄位于/sqllib/samples。Windows中,這目錄位于Program Files/IBM//samples。

    ?

    在Windows中設(shè)置用戶出口

    ?

    修改和編譯用戶出口程序

    1.? 創(chuàng)建名為C:mylogs的目錄

    2.? 將C:Program filesIBMSQLLIBsamplescdb2uext2.cdisk拷貝到一個(gè)工作目錄當(dāng)中

    3.? 對(duì)于本示例,用戶出口程序的以下部分應(yīng)該得以驗(yàn)證以反映路徑 c:\mylogs\

    ?

    #define ARCHIVE_PATH "c:\mylogs\"

    #define RETRIEVE_PATH "c:\mylogs\"

    #define AUDIT_ACTIVE 1 /* enable audit trail logging */

    #define ERROR_ACTIVE 1 /* enable error trail logging */

    #define AUDIT_ERROR_PATH "c:\mylogs\"

    ???????????????? /* path must end with a slash */

    #define AUDIT_ERROR_ATTR "a" /* append to text file */

    #define BUFFER_SIZE 32

    ??? /* # of 4K pages for output buffer */

    ?

    4.? 確定在你的系統(tǒng)中安裝了被支持的C編譯器(比如,Microsoft Visual Studio)而且環(huán)境中有該編譯器的路徑。

    5.? 通過命令行,將db2uext2.cdisk更名為db2uext2.c并構(gòu)建:

    cl db2uext2.c

    一旦程序被編譯,將創(chuàng)建db2uext2.exe和db2uext2.obj文件。

    6.? 將可執(zhí)行文件db2uext2.exe放到/SQLLIB/BIN目錄當(dāng)中從而使數(shù)據(jù)庫管理器能夠定位并執(zhí)行它用以歸檔和檢索日志。

    ?

    為用戶出口創(chuàng)建并準(zhǔn)備數(shù)據(jù)庫

    7.? 在 DB2 命令窗口中用 db2sampl 命令創(chuàng)建 SAMPLE 數(shù)據(jù)庫。這將使你對(duì)下面的示例使用樣本表。

    db2sampl

    8.? 更新數(shù)據(jù)庫配置文件以便為數(shù)據(jù)庫打開用戶出口。請(qǐng)注意:只能為一個(gè)數(shù)據(jù)庫分配用戶出口程序,因?yàn)?bin 目錄被所有 DB2 實(shí)例共享。

    db2 "update db cfg for sample using userexit on"

    db2stop force

    <span lang="EN-US" style="FONT-SIZE: 10pt; COLOR: black; FONT-FAMILY: 宋體; mso-hansi-font-family: 'Times New Roman'; mso-bidi-font-family: 宋體; mso-

    /P>
    posted @ 2006-12-08 18:12 MyJavaWorld 閱讀(418) | 評(píng)論 (0)編輯 收藏
    其實(shí)主要的就是要?jiǎng)?chuàng)建一個(gè)密鑰倉庫以管理您的公鑰 / 私鑰對(duì)來自您所信任實(shí)體的證書。
    ?
    第一步:生成密鑰對(duì)
    您首先要做的是創(chuàng)建一個(gè)密鑰倉庫和生成密鑰對(duì)。您可以使用以下命令: ?
    keytool -genkey -keyalg RSA -keysize 512 -dname "cn=hyq,o=eagle,c=cn" -alias weblogic -keypass 123456 -keystore C:/mykeystore/weblogic.jks -storepass 123456 -validity 365

    (請(qǐng)注意:鍵入該命令時(shí)必須使其成為一行。此處用多行來顯示,主要是為了可讀性。)如下圖:
    Snap2.gif

    該命令將在 ?C? 盤的 “mykeystore” 目錄中創(chuàng)建名為 “weblogic.jks” 的密鑰倉庫,并賦予它口令 123456 。它將為實(shí)體生成公鑰 / 私鑰對(duì),該實(shí)體的 特征名 為:常用名 “hyq” 、組織 “eagle” 和兩個(gè)字母的國(guó)家代碼 “cn” ?!?/span> -keyalg ”指定它使用的是那種密鑰生成算法來創(chuàng)建密鑰,缺省的是 “DSA” 密鑰生成算法(會(huì)使用缺省的 ?DSA? ?SHA1” 簽名算法),兩個(gè)密鑰(公鑰與私鑰)的長(zhǎng)度是 512 位,由 -keysize 來指定,默認(rèn)的是 1024? 位。 ? 該證書包括公鑰和特征名信息。該證書的有效期為 365 天,由 -validity 來指定,且與別名 “business” 所代表的密鑰倉庫項(xiàng)關(guān)聯(lián)。私鑰被賦予口令 123456

    ?

    命令行里 DName 信息注解

    ?

    DN 信息域

    含義

    CN

    域名或 IP

    OU

    部門,沒有部門的可不要此項(xiàng)

    O

    單位名稱

    L

    單位地址

    S

    省份的拼音(第一個(gè)字母大寫)

    C

    國(guó)家的簡(jiǎn)寫 ( CN 代表中國(guó))


    如果采用選項(xiàng)的缺省值,可以大大縮短該命令。實(shí)際上,這些選項(xiàng)并不是必需的;對(duì)于有缺省值的選項(xiàng),未指定時(shí)將使用缺省值,對(duì)于任何被要求的值,您將會(huì)得到要求輸入它的提示。例如:輸入命令 keytool -genkey -keystore "C:/tone.jks" -storepass 123456 -keyalg RSA ,就會(huì)有如下提示:
    Snap3.gif


    注意:這里的
    密鑰倉庫路徑一定要存在,如果不存在的話,它就會(huì)拋如下的異常

    Snap4.gif
    第二步
    : 產(chǎn)生證書請(qǐng)求certreq.pem 文件

    使用如下命令:

    keytool -certreq -alias weblogic -sigalg "MD5withRSA" -file C:/mykeystore/certreq.pem -keypass 123456 -keystore C:/mykeystore/weblogic.jks -storepass 123456

    Snap5.gif
    這樣在
    C:/mykeystore/ 目錄下 就會(huì)產(chǎn)生一個(gè) certreq.pem 文件,內(nèi)容如下:

    -----BEGIN NEW CERTIFICATE REQUEST-----

    MIHlMIGQAgEAMCsxCzAJBgNVBAYTAmNuMQ4wDAYDVQQKEwVlYWdsZTEMMAoGA1UEAxMDaHlxMFww

    DQYJKoZIhvcNAQEBBQADSwAwSAJBAMhaIG2Ki7+RwZUP4gPBdTbnY38bisW16u1XUyysPxdNwSie

    aSd6E3Hm277E7NjHoz56ZoaYdPPDmdiTkMrS9rcCAwEAAaAAMA0GCSqGSIb3DQEBBAUAA0EAYRNl

    l5dyGgV9hhu++ypcJNQTrDIwjx1QT4fgVubrtIaHU0fzHamD5QG6PYddw9TL51XQHvu6tOS0NUc/

    ItNKJw==

    -----END NEW CERTIFICATE REQUEST-----

    第三步:這就相對(duì)來說簡(jiǎn)單多了,就是CA提交證書請(qǐng)求。
    ??? 你可以隨便從網(wǎng)上找一家免費(fèi)的CA認(rèn)證適用機(jī)構(gòu)(很多的),然后按照上面的提示進(jìn)行操作就可以了,這一步就要用到前面生成的certreq.pem 文件了。(注意:一定要下載根證書)
    ??? 將生成的證書和下載的根證書放至你比較容易找到的位置,我一般將它們和生成的jks文件放到一起。

    第四步:導(dǎo)入證書
    ??? 通過命令:keytool -import -alias RootCA -trustcacerts -file C:/mykeystore/RootCADemo.cer -keystore C:/mykeystore/weblogic.jks -storepass 123456將根證書導(dǎo)入第一步生成的weblogic.jks中,接著將所有其它的證書按照此命令全部導(dǎo)入。(注意證書的別名不能重復(fù),同時(shí)一定注意要用上 -trustcacerts,否則,你在以后使用時(shí),它將會(huì)認(rèn)為你導(dǎo)入的這些證書是不可信任的 ,就會(huì)導(dǎo)致你在配置SSL時(shí)不能正常的工作。)
    ??? 這就全部完成了密鑰倉庫的創(chuàng)建。然后就可以在支持這些證書格式的服務(wù)器上使用了。下一篇將會(huì)寫一下在weblogic上如何配置雙向SSL

    posted @ 2006-10-31 17:18 MyJavaWorld 閱讀(454) | 評(píng)論 (0)編輯 收藏
    現(xiàn)在的DB2 UDB系統(tǒng)中, 主要通過鎖和隔離級(jí)別這兩個(gè)主要的工具來控制并發(fā)連接,維護(hù)數(shù)據(jù)庫中的數(shù)據(jù)在高并發(fā)的環(huán)境下的安全。

    我們?cè)谶@里將簡(jiǎn)要的闡述一下鎖和隔離級(jí)別。

    鎖:

    DB2 UDB中, 鎖的主要作用對(duì)象是表和行, 其他如表空間和索引也是鎖的對(duì)象, 但是因?yàn)槠涠酁橄到y(tǒng)控制, 管理員和用戶使用較少,在這里就不涉及了。

    對(duì)于行級(jí)鎖和表級(jí)鎖, 它們的區(qū)別不言而喻, 主要是鎖的對(duì)象不同。 當(dāng)然鎖對(duì)象的不同連帶也會(huì)影響DB2的并發(fā)能力。

    DB2中的表級(jí)鎖主要有以下幾種:

    1. IS鎖: 此鎖作用于整個(gè)表,表示擁有此鎖的應(yīng)用連接要讀取表中的某些數(shù)據(jù), 但是在此應(yīng)用連接讀取具體的數(shù)據(jù)時(shí), 還必須獲得該行的行級(jí)鎖;

    2. IX鎖: 此鎖作用于整個(gè)表,表示擁有此鎖的應(yīng)用連接需要獨(dú)占使用表中的某些數(shù)據(jù), 但是在此應(yīng)用連接獨(dú)占使用具體的數(shù)據(jù)時(shí), 還必須獲得該行上相應(yīng)的行級(jí)鎖;

    3. SIX鎖: 此鎖是鎖轉(zhuǎn)換的產(chǎn)物,表示應(yīng)用連接擁有S和IX鎖的特性;

    4. S鎖: 此鎖作用于整個(gè)表, 擁有此鎖的應(yīng)用連接可以讀取表中的任何紀(jì)錄;

    5. U鎖: 此鎖作用于整個(gè)表, 擁有此鎖的應(yīng)用連接可以讀取表中的任何紀(jì)錄,也可以更新表中的紀(jì)錄, 但是更新時(shí)需要再獲得X鎖; 此鎖主要在“select … with update”語句建立的可更新游標(biāo)中起作用, 其他的應(yīng)用可以讀取表中的紀(jì)錄, 但是不能更新它;

    6. X鎖: 此鎖作用于整個(gè)表, 擁有此鎖的應(yīng)用連接獨(dú)占的使用表中的任何紀(jì)錄;可以進(jìn)行更新或其他操作;

    7. Z鎖: 此鎖作用于整個(gè)表, 也稱超級(jí)獨(dú)占鎖,主要是在象修改表的定義、 刪除表這一類的語句中會(huì)使用。 擁有此鎖的應(yīng)用連接對(duì)該表有完全的控制權(quán)。 其他的任何應(yīng)用不能讀取或更新表中的任何紀(jì)錄。

    在這里我們主要要看一下 IS/IX/SIX這三個(gè)鎖。 在這三個(gè)鎖中IS/IX本身并不具備使得應(yīng)用連接可以讀取或更新紀(jì)錄的能力,應(yīng)用連接要讀取和更新紀(jì)錄時(shí), 需要再得到相應(yīng)的行級(jí)鎖; 反之亦然, 任何應(yīng)用要獲得行級(jí)鎖操作數(shù)據(jù)記錄之前, 也必須獲得某個(gè)相應(yīng)的表級(jí)鎖。 SIX鎖也是類似的情況。這就是為什麼在很多情況下我們使用的是行級(jí)鎖, 但是用快照(SNAPSHOT)等工具卻能夠看到有表級(jí)鎖存在的原因。

    那麼DB2中又有哪些行級(jí)鎖呢? 讓我們來看下面的這張圖:

    那麼DB2中又有哪些行級(jí)鎖呢?

    此圖中列出了DB2中包含的行級(jí)鎖。 表中的第三列指出, 要獲得此行級(jí)鎖之前, 需要預(yù)先獲得的表級(jí)鎖, 這里列出的是最低要求。

    這六個(gè)行級(jí)鎖的主要功能如下:

    1. S鎖:此行級(jí)鎖的擁有者可以讀取該行的信息;

    2. U鎖:此行級(jí)鎖的擁有者可以讀取該行的信息,如果要更新該行,則仍然需要一個(gè)行級(jí)的X鎖;其他的應(yīng)用只能讀取該行的信息;此鎖主要是用于FOR UPDATE的游標(biāo)。

    3. X鎖:此行級(jí)鎖的擁有者可以更新該行的紀(jì)錄,其他的應(yīng)用不能連接此行的信息;

    4. W鎖:此鎖和X鎖類似,不同之處是此鎖和NW鎖兼容;

    5. NS鎖:類似于S鎖,用于Next Key;

    6. NW鎖:類似于W鎖,用于Next Key;

    在DB2數(shù)據(jù)庫中, 是通過行級(jí)鎖和表級(jí)鎖協(xié)調(diào)作用來提供較好的并發(fā)性, 同時(shí)保證數(shù)據(jù)庫中數(shù)據(jù)的安全。 在DB2中缺省情況下使用行級(jí)鎖(當(dāng)然需要IS/IX鎖配合),只有當(dāng)出現(xiàn)鎖資源不足, 或者是用命令指定使用表級(jí)鎖的情況下, 才會(huì)在應(yīng)用連接中使用表級(jí)鎖。 對(duì)鎖資源分配有興趣的讀者可以參考DB2的管理手冊(cè), 查找其中關(guān)于locklist和maxlocks參數(shù)的論述。對(duì)于用命令指定表級(jí)鎖的情況, 可以參考DB2的命令手冊(cè)中的lock table命令, 此命令用于直接鎖表。

    隔離級(jí)別:

    下面讓我們來看一下隔離級(jí)別。 隔離級(jí)別主要用于控制在DB2根據(jù)應(yīng)用提交的SQL語句向DB2數(shù)據(jù)庫中的相應(yīng)對(duì)象加鎖時(shí), 會(huì)鎖住哪些紀(jì)錄, 也就是鎖定的范圍。 隔離級(jí)別的不同, 鎖定的紀(jì)錄的范圍可能會(huì)有很大的差別。

    隔離級(jí)別分為RR/RS/CS/UR這四個(gè)級(jí)別。 下面讓我們來逐一論述:

    1. RR隔離級(jí)別: 在此隔離級(jí)別下, DB2會(huì)鎖住所有相關(guān)的紀(jì)錄。 在一個(gè)SQL語句執(zhí)行期間, 所有執(zhí)行此語句掃描過的紀(jì)錄都會(huì)被加上相應(yīng)的鎖。 具體的鎖的類型還是由操作的類型來決定, 如果是讀取,則加共享鎖; 如果是更新, 則加獨(dú)占鎖。 由于會(huì)鎖定所有為獲得SQL語句的結(jié)果而掃描的紀(jì)錄, 所以鎖的數(shù)量可能會(huì)很龐大, 這個(gè)時(shí)候, 索引的增加可能會(huì)對(duì)SQL語句的執(zhí)行有很大的影響,因?yàn)樗饕龝?huì)影響SQL語句掃描的紀(jì)錄數(shù)量。

    2. RS隔離級(jí)別: 此隔離級(jí)別的要求比RR隔離級(jí)別稍弱,此隔離級(jí)別下會(huì)鎖定所有符合條件的紀(jì)錄。 不論是讀取, 還是更新, 如果SQL語句中包含查詢條件, 則會(huì)對(duì)所有符合條件的紀(jì)錄加相應(yīng)的鎖。 如果沒有條件語句, 也就是對(duì)表中的所有記錄進(jìn)行處理,則會(huì)對(duì)所有的紀(jì)錄加鎖。

    3. CS隔離級(jí)別: 此隔離級(jí)別僅鎖住當(dāng)前處理的紀(jì)錄。

    4. UR隔離級(jí)別:此隔離級(jí)別下,如果是讀取操作,不會(huì)出現(xiàn)任何的行級(jí)鎖。對(duì)于非只讀的操作,它的鎖處理和CS相同。

    在這四種隔離級(jí)別中, CS是缺省值。 這四種隔離級(jí)別均可以保證DB2數(shù)據(jù)庫在并發(fā)的環(huán)境下不會(huì)有數(shù)據(jù)丟失的情況發(fā)生。 要注意的是如果對(duì)紀(jì)錄進(jìn)行了修改,需要在相應(yīng)的紀(jì)錄上加獨(dú)占類型的鎖, 這些獨(dú)占類型的鎖直到交易結(jié)束時(shí)才會(huì)被釋放, 這一點(diǎn)在四種隔離級(jí)別下都是相同的。

    posted @ 2006-08-22 12:07 MyJavaWorld 閱讀(885) | 評(píng)論 (0)編輯 收藏

    REORG TABLE

    REORG TABLE 語句壓縮與指定的表相關(guān)聯(lián)的數(shù)據(jù)。

    調(diào)用

    此語句可以在使用 DB2 CLI 函數(shù)的應(yīng)用程序中使用,也可以通過 CLP 發(fā)出。

    語法

    >>-REORG TABLE--table-name--+------------+---------------------><
                                '-int1--int2-'
     
     
    

    描述

    REORG TABLE table-name
    標(biāo)識(shí)重組操作的表。名稱必須標(biāo)識(shí)現(xiàn)有的表。
    int1
    需要恢復(fù)的字節(jié)的可選最小百分比。
    int2
    需要為將要執(zhí)行的表壓縮恢復(fù)的最小字節(jié)數(shù)。

    規(guī)則

    • 可選的值 int1int2 必須一起使用,或全都不用。
    • 可選的值 int1 必須是非負(fù)數(shù)。
    • 可選的值 int1 必須介于 0 與 100 之間。

    注意事項(xiàng)

    • DB2 Everyplace 可以以內(nèi)部方式調(diào)用表重組。
    • 第一個(gè)可選參數(shù)是表必須包含的不可用的字節(jié)的百分比(即百分之十(10)意味“至少百分之十的空間不可用”。)第二個(gè)可選參數(shù)是表必須包含的不可用的字節(jié)數(shù)(即 1000 將意味“至少 1000 個(gè)字節(jié)必須是不可用的空間”。)必須符合兩個(gè)條件,才可以進(jìn)行表的實(shí)際重組。
    • 如果沒有指定參數(shù),DB2 Everyplace 對(duì)這此選項(xiàng)使用缺省值。缺省百分比是 30 且缺省字節(jié)是 6144。因此,“reorg table t1”與“reorg table t1 30 6144”相同。
    • 如果重組方式設(shè)置為已啟用,則 DB2 Everyplace 將自動(dòng)重組表。如果在 DELETE 或 UPDATE 上啟用了重組,則在執(zhí)行語句之后,會(huì)對(duì)目標(biāo)表執(zhí)行“reorg table table_name 50 30270”。如果在 DROP TABLE 上啟用了重組,則在刪除表處理結(jié)束時(shí)執(zhí)行“reorg table DB2eSYSTABLES 30 10240”(對(duì)于 DB2eSYSCOLUMNS 和 DB2eSYSRELS 也是如此)。
    • 在 C/C++ 程序中,通過使用具有屬性 SQL_ATTR_REORG_MODE 的 CLI/ODBC 函數(shù) SQLSetStmtAttr 設(shè)置重組方式。在 JAVA 程序中,通過 DB2eStatement 接口 enableReorg 方法設(shè)置重組方式。缺省值是啟用重組。
    • 重組表時(shí),通過物理上回收刪除和更新創(chuàng)建的不可用空間來壓縮包含表的數(shù)據(jù)文件。然后將表的索引更新為指向行的新物理位置。
    • 可以重組“DB2 Everyplace 系統(tǒng)目錄”基本表。
    • 在執(zhí)行 REORG TABLE 語句時(shí),數(shù)據(jù)庫中不應(yīng)發(fā)生任何其它活動(dòng)。

    示例

    使用缺省值壓縮 VNNURSE 表。

    REORG TABLE VNNURSE?
    
    posted @ 2006-07-26 22:10 MyJavaWorld 閱讀(888) | 評(píng)論 (0)編輯 收藏
         摘要: 在數(shù)據(jù)庫應(yīng)用程序中使用存儲(chǔ)過程有許多好處,包括減少對(duì)網(wǎng)絡(luò)的使用、提高性能以及降低開發(fā)成本。Java 存儲(chǔ)過程是 DB2 支持的最流行的例程之一。原因之一是,由于 Java 編程語言非常流行,所以 Java 開發(fā)人員非常多。因此,在有多種語言可供選擇時(shí),Java 例程往往是首選的。 DB2 存儲(chǔ)過程不一定非用 Java 來編寫。如果業(yè)務(wù)邏輯只需要簡(jiǎn)單的存儲(chǔ)過程,那么可以考慮用 ...  閱讀全文
    posted @ 2006-07-21 10:03 MyJavaWorld 閱讀(5620) | 評(píng)論 (0)編輯 收藏

    J***A正則表達(dá)式

    關(guān)鍵詞正則表達(dá)式 ?? ??????????????????????????????????????

    正則表達(dá)式

    。正則表達(dá)式(regular expression)是JDK 1.4的新功能,但是對(duì)sed和awk這樣的Unix的標(biāo)準(zhǔn)實(shí)用工具,以及Python,Perl之類的語言來講,它早就已經(jīng)成為其不可或缺的組成部分了(有人甚至認(rèn)為,它還是Perl能大獲成功的最主要的原因)。單從技術(shù)角度來講,正則表達(dá)式只是一種處理字符串的工具(過去Java這個(gè)任務(wù)是交由String,StringBuffer以及StringTokenizer處理的),但是它常常和I/O一起使用,所以放到這里來講也不算太離題吧。

    正則表達(dá)式是一種功能強(qiáng)大但又非常靈活的文本處理工具。它能讓你用編程的方式來描述復(fù)雜的文本模式,然后在字符串里把它找出來。一旦你找到了這種模式,你就能隨心所欲地處理這些文本了。雖然初看起來正則表達(dá)式的語法有點(diǎn)讓人望而生畏,但它提供了一種精練的動(dòng)態(tài)語言,使我們能用一種通用的方式來解決各種字符串的問題,包括匹配,選擇,編輯以及校驗(yàn)。

    創(chuàng)建正則表達(dá)式

    你可以從比較簡(jiǎn)單的東西入手學(xué)習(xí)正則表達(dá)式。要想全面地掌握怎樣構(gòu)建正則表達(dá)式,可以去看JDK文檔的java.util.regexPattern類的文檔。

    字符
    B字符B
    \xhh16進(jìn)制值0xhh所表示的字符
    \uhhhh16進(jìn)制值0xhhhh所表示的Unicode字符
    \tTab
    \n換行符
    \r回車符
    \f換頁符
    \eEscape

    正則表達(dá)式的強(qiáng)大體現(xiàn)在它能定義字符集(character class)。下面是一些最常見的字符集及其定義的方式,此外還有一些預(yù)定義的字符集:

    字符集
    .表示任意一個(gè)字符
    [abc]表示字符ab,c中的任意一個(gè)(與a|b|c相同)
    [^abc]a,bc之外的任意一個(gè)字符(否定)
    [a-zA-Z]azAZ當(dāng)中的任意一個(gè)字符(范圍)
    [abc[hij]]a,b,c,h,i,j中的任意一個(gè)字符(與a|b|c|h|i|j相同)(并集)
    [a-z&&[hij]]h,i,j中的一個(gè)(交集)
    \s空格字符(空格鍵, tab, 換行, 換頁, 回車)
    \S非空格字符([^\s])
    \d一個(gè)數(shù)字,也就是[0-9]
    \D一個(gè)非數(shù)字的字符,也就是[^0-9]
    \w一個(gè)單詞字符(word character),即[a-zA-Z_0-9]
    \W一個(gè)非單詞的字符,[^\w]

    如果你用過其它語言的正則表達(dá)式,那么你一眼就能看出反斜杠的與眾不同。在其它語言里,"\\"的意思是"我只是要在正則表達(dá)式里插入一個(gè)反斜杠。沒什么特別的意思。"但是在Java里,"\\"的意思是"我要插入一個(gè)正則表達(dá)式的反斜杠,所以跟在它后面的那個(gè)字符的意思就變了。"舉例來說,如果你想表示一個(gè)或更多的"單詞字符",那么這個(gè)正則表達(dá)式就應(yīng)該是"\\w+"。如果你要插入一個(gè)反斜杠,那就得用"\\\\"。不過像換行,跳格之類的還是只用一根反斜杠:"\n\t"。

    這里只給你講一個(gè)例子;你應(yīng)該JDK文檔的java.util.regex.Pattern加到收藏夾里,這樣就能很容易地找到各種正則表達(dá)式的模式了。

    邏輯運(yùn)算符
    XYX 后面跟著 Y
    X|YX或Y
    (X)一個(gè)"要匹配的組(capturing group)". 以后可以用\i來表示第i個(gè)被匹配的組。

    邊界匹配符
    ^一行的開始
    $一行的結(jié)尾
    \b一個(gè)單詞的邊界
    \B一個(gè)非單詞的邊界
    \G前一個(gè)匹配的結(jié)束

    舉一個(gè)具體一些的例子。下面這些正則表達(dá)式都是合法的,而且都能匹配"Rudolph":

    Rudolph
    [rR]udolph
    [rR][aeiou][a-z]ol.*
    R.*

    數(shù)量表示符

    "數(shù)量表示符(quantifier)"的作用是定義模式應(yīng)該匹配多少個(gè)字符。

    • Greedy(貪婪的): 除非另有表示,否則數(shù)量表示符都是greedy的。Greedy的表達(dá)式會(huì)一直匹配下去,直到匹配不下去為止。(如果你發(fā)現(xiàn)表達(dá)式匹配的結(jié)果與預(yù)期的不符),很有可能是因?yàn)?,你以為表達(dá)式會(huì)只匹配前面幾個(gè)字符,而實(shí)際上它是greedy的,因此會(huì)一直匹配下去。
    • Reluctant(勉強(qiáng)的): 用問號(hào)表示,它會(huì)匹配最少的字符。也稱為lazy, minimal matching, non-greedy, 或ungreedy。
    • Possessive(占有的): 目前只有Java支持(其它語言都不支持)。它更加先進(jìn),所以你可能還不太會(huì)用。用正則表達(dá)式匹配字符串的時(shí)候會(huì)產(chǎn)生很多中間狀態(tài),(一般的匹配引擎會(huì)保存這種中間狀態(tài),)這樣匹配失敗的時(shí)候就能原路返回了。占有型的表達(dá)式不保存這種中間狀態(tài),因此也就不會(huì)回頭重來了。它能防止正則表達(dá)式的失控,同時(shí)也能提高運(yùn)行的效率。
    GreedyReluctantPossessive匹配
    X?X??X?+匹配一個(gè)或零個(gè)X
    X*X*?X*+匹配零或多個(gè)X
    X+X+?X++匹配一個(gè)或多個(gè)X
    X{n}X{n}?X{n}+匹配正好n個(gè)X
    X{n,}X{n,}?X{n,}+匹配至少n個(gè)X
    X{n,m}X{n,m}?X{n,m}+匹配至少n個(gè),至多m個(gè)X

    再提醒一下,要想讓表達(dá)式照你的意思去運(yùn)行,你應(yīng)該用括號(hào)把'X'括起來。比方說:

    abc+

    似乎這個(gè)表達(dá)式能匹配一個(gè)或若干個(gè)'abc',但是如果你真的用它去匹配'abcabcabc'的話,實(shí)際上只會(huì)找到三個(gè)字符。因?yàn)檫@個(gè)表達(dá)式的意思是'ab'后邊跟著一個(gè)或多個(gè)'c'。要想匹配一個(gè)或多個(gè)完整的'abc',你應(yīng)該這樣:

    (abc)+

    正則表達(dá)式能輕而易舉地把你給耍了;這是一種建立在Java之上的新語言。

    CharSequence

    JDK 1.4定義了一個(gè)新的接口,叫CharSequence。它提供了StringStringBuffer這兩個(gè)類的字符序列的抽象:

    interface CharSequence {
      charAt(int i);
      length();
      subSequence(int start, int end);
      toString();
    }

    為了實(shí)現(xiàn)這個(gè)新的CharSequence接口,String,StringBuffer以及CharBuffer都作了修改。很多正則表達(dá)式的操作都要拿CharSequence作參數(shù)。

    PatternMatcher

    先給一個(gè)例子。下面這段程序可以測(cè)試正則表達(dá)式是否匹配字符串。第一個(gè)參數(shù)是要匹配的字符串,后面是正則表達(dá)式。正則表達(dá)式可以有多個(gè)。在Unix/Linux環(huán)境下,命令行下的正則表達(dá)式還必須用引號(hào)。

    當(dāng)你創(chuàng)建正則表達(dá)式時(shí),可以用這個(gè)程序來判斷它是不是會(huì)按照你的要求工作。

    //: c12:TestRegularExpression.java// Allows you to easly try out regular expressions.// {Args: abcabcabcdefabc "abc+" "(abc)+" "(abc){2,}" }import java.util.regex.*;
    publicclass TestRegularExpression {
      publicstaticvoid main(String[] args) {
        if(args.length < 2) {
          System.out.println("Usage:\n" +
            "java TestRegularExpression " +
            "characterSequence regularExpression+");
          System.exit(0);
        }
        System.out.println("Input: \"" + args[0] + "\"");
        for(int i = 1; i < args.length; i++) {
          System.out.println(
            "Regular expression: \"" + args[i] + "\"");
          Pattern p = Pattern.compile(args[i]);
          Matcher m = p.matcher(args[0]);
          while(m.find()) {
            System.out.println("Match \"" + m.group() +
              "\" at positions " +
              m.start() + "-" + (m.end() - 1));
          }
        }
      }
    } ///:~

    Java的正則表達(dá)式是由java.util.regexPatternMatcher類實(shí)現(xiàn)的。Pattern對(duì)象表示經(jīng)編譯的正則表達(dá)式。靜態(tài)的compile( )方法負(fù)責(zé)將表示正則表達(dá)式的字符串編譯成Pattern對(duì)象。正如上述例程所示的,只要給Patternmatcher( )方法送一個(gè)字符串就能獲取一個(gè)Matcher對(duì)象。此外,Pattern還有一個(gè)能快速判斷能否在input里面找到regex的(注意,原文有誤,漏了方法名)

    staticboolean matches(?regex, ?input)

    以及能返回String數(shù)組的split( )方法,它能用regex把字符串分割開來。

    只要給Pattern.matcher( )方法傳一個(gè)字符串就能獲得Matcher對(duì)象了。接下來就能用Matcher的方法來查詢匹配的結(jié)果了。

    boolean matches()
    boolean lookingAt()
    boolean find()
    boolean find(int start)

    matches( )的前提是Pattern匹配整個(gè)字符串,而lookingAt( )的意思是Pattern匹配字符串的開頭。

    find( )

    Matcher.find( )的功能是發(fā)現(xiàn)CharSequence里的,與pattern相匹配的多個(gè)字符序列。例如:

    //: c12:FindDemo.javaimport java.util.regex.*;
    import com.bruceeckel.simpletest.*;
    import java.util.*;
    publicclass FindDemo {
      privatestatic Test monitor = new Test();
      publicstaticvoid main(String[] args) {
        Matcher m = Pattern.compile("\\w+")
          .matcher("Evening is full of the linnet's wings");
        while(m.find())
          System.out.println(m.group());
        int i = 0;
        while(m.find(i)) {
          System.out.print(m.group() + " ");
          i++;
        }
        monitor.expect(new String[] {
          "Evening",
          "is",
          "full",
          "of",
          "the",
          "linnet",
          "s",
          "wings",
          "Evening vening ening ning ing ng g is is s full " +
          "full ull ll l of of f the the he e linnet linnet " +
          "innet nnet net et t s s wings wings ings ngs gs s "
        });
      }
    } ///:~

    "\\w+"的意思是"一個(gè)或多個(gè)單詞字符",因此它會(huì)將字符串直接分解成單詞。find( )像一個(gè)迭代器,從頭到尾掃描一遍字符串。第二個(gè)find( )是帶int參數(shù)的,正如你所看到的,它會(huì)告訴方法從哪里開始找——即從參數(shù)位置開始查找。

    Groups

    Group是指里用括號(hào)括起來的,能被后面的表達(dá)式調(diào)用的正則表達(dá)式。Group 0 表示整個(gè)表達(dá)式,group 1表示第一個(gè)被括起來的group,以此類推。所以;

    A(B(C))D

    里面有三個(gè)group:group 0是ABCD, group 1是BC,group 2是C。

    你可以用下述Matcher方法來使用group:

    public int groupCount( )返回matcher對(duì)象中的group的數(shù)目。不包括group0。

    public String group( ) 返回上次匹配操作(比方說find( ))的group 0(整個(gè)匹配)

    public String group(int i)返回上次匹配操作的某個(gè)group。如果匹配成功,但是沒能找到group,則返回null。

    public int start(int group)返回上次匹配所找到的,group的開始位置。

    public int end(int group)返回上次匹配所找到的,group的結(jié)束位置,最后一個(gè)字符的下標(biāo)加一。

    下面我們舉一些group的例子:

    //: c12:Groups.javaimport java.util.regex.*;
    import com.bruceeckel.simpletest.*;
    publicclass Groups {
      privatestatic Test monitor = new Test();
      staticpublicfinal String poem =
        "Twas brillig, and the slithy toves\n" +
        "Did gyre and gimble in the wabe.\n" +
        "All mimsy were the borogoves,\n" +
        "And the mome raths outgrabe.\n\n" +
        "Beware the Jabberwock, my son,\n" +
        "The jaws that bite, the claws that catch.\n" +
        "Beware the Jubjub bird, and shun\n" +
        "The frumious Bandersnatch.";
      publicstaticvoid main(String[] args) {
        Matcher m =
          Pattern.compile("(?m)(\\S+)\\s+((\\S+)\\s+(\\S+))$")
            .matcher(poem);
        while(m.find()) {
          for(int j = 0; j <= m.groupCount(); j++)
            System.out.print("[" + m.group(j) + "]");
          System.out.println();
        }
        monitor.expect(new String[]{
          "[the slithy toves]" +
          "[the][slithy toves][slithy][toves]",
          "[in the wabe.][in][the wabe.][the][wabe.]",
          "[were the borogoves,]" +
          "[were][the borogoves,][the][borogoves,]",
          "[mome raths outgrabe.]" +
          "[mome][raths outgrabe.][raths][outgrabe.]",
          "[Jabberwock, my son,]" +
          "[Jabberwock,][my son,][my][son,]",
          "[claws that catch.]" +
          "[claws][that catch.][that][catch.]",
          "[bird, and shun][bird,][and shun][and][shun]",
          "[The frumious Bandersnatch.][The]" +
          "[frumious Bandersnatch.][frumious][Bandersnatch.]"
        });
      }
    } ///:~

    這首詩是Through the Looking Glass的,Lewis Carroll的"Jabberwocky"的第一部分。可以看到這個(gè)正則表達(dá)式里有很多用括號(hào)括起來的group,它是由任意多個(gè)連續(xù)的非空字符('\S+')和任意多個(gè)連續(xù)的空格字符('\s+')所組成的,其最終目的是要捕獲每行的最后三個(gè)單詞;'$'表示一行的結(jié)尾。但是'$'通常表示整個(gè)字符串的結(jié)尾,所以這里要明確地告訴正則表達(dá)式注意換行符。這一點(diǎn)是由'(?m)'標(biāo)志完成的(模式標(biāo)志會(huì)過一會(huì)講解)。

    start( )和end( )

    如果匹配成功,start( )會(huì)返回此次匹配的開始位置,end( )會(huì)返回此次匹配的結(jié)束位置,即最后一個(gè)字符的下標(biāo)加一。如果之前的匹配不成功(或者沒匹配),那么無論是調(diào)用start( )還是end( ),都會(huì)引發(fā)一個(gè)IllegalStateException。下面這段程序還演示了matches( )lookingAt( )

    //: c12:StartEnd.javaimport java.util.regex.*;
    import com.bruceeckel.simpletest.*;
    publicclass StartEnd {
      privatestatic Test monitor = new Test();
      publicstaticvoid main(String[] args) {
        String[] input = new String[] {
          "Java has regular expressions in 1.4",
          "regular expressions now expressing in Java",
          "Java represses oracular expressions"
        };
        Pattern
          p1 = Pattern.compile("re\\w*"),
          p2 = Pattern.compile("Java.*");
        for(int i = 0; i < input.length; i++) {
          System.out.println("input " + i + ": " + input[i]);
          Matcher
            m1 = p1.matcher(input[i]),
            m2 = p2.matcher(input[i]);
          while(m1.find())
            System.out.println("m1.find() '" + m1.group() +
              "' start = "+ m1.start() + " end = " + m1.end());
          while(m2.find())
            System.out.println("m2.find() '" + m2.group() +
              "' start = "+ m2.start() + " end = " + m2.end());
          if(m1.lookingAt()) // No reset() necessary
            System.out.println("m1.lookingAt() start = "
              + m1.start() + " end = " + m1.end());
          if(m2.lookingAt())
            System.out.println("m2.lookingAt() start = "
              + m2.start() + " end = " + m2.end());
          if(m1.matches()) // No reset() necessary
            System.out.println("m1.matches() start = "
              + m1.start() + " end = " + m1.end());
          if(m2.matches())
            System.out.println("m2.matches() start = "
              + m2.start() + " end = " + m2.end());
        }
        monitor.expect(new String[] {
          "input 0: Java has regular expressions in 1.4",
          "m1.find() 'regular' start = 9 end = 16",
          "m1.find() 'ressions' start = 20 end = 28",
          "m2.find() 'Java has regular expressions in 1.4'" +
          " start = 0 end = 35",
          "m2.lookingAt() start = 0 end = 35",
          "m2.matches() start = 0 end = 35",
          "input 1: regular expressions now " +
          "expressing in Java",
          "m1.find() 'regular' start = 0 end = 7",
          "m1.find() 'ressions' start = 11 end = 19",
          "m1.find() 'ressing' start = 27 end = 34",
          "m2.find() 'Java' start = 38 end = 42",
          "m1.lookingAt() start = 0 end = 7",
          "input 2: Java represses oracular expressions",
          "m1.find() 'represses' start = 5 end = 14",
          "m1.find() 'ressions' start = 27 end = 35",
          "m2.find() 'Java represses oracular expressions' " +
          "start = 0 end = 35",
          "m2.lookingAt() start = 0 end = 35",
          "m2.matches() start = 0 end = 35"
        });
      }
    } ///:~

    注意,只要字符串里有這個(gè)模式,find( )就能把它給找出來,但是lookingAt( )matches( ),只有在字符串與正則表達(dá)式一開始就相匹配的情況下才能返回true。matches( )成功的前提是正則表達(dá)式與字符串完全匹配,而lookingAt( )[67]成功的前提是,字符串的開始部分與正則表達(dá)式相匹配。

    匹配的模式(Pattern flags)

    compile( )方法還有一個(gè)版本,它需要一個(gè)控制正則表達(dá)式的匹配行為的參數(shù):

    Pattern Pattern.compile(String regex, int flag)
    flag的取值范圍如下:
    編譯標(biāo)志效果
    Pattern.CANON_EQ當(dāng)且僅當(dāng)兩個(gè)字符的"正規(guī)分解(canonical decomposition)"都完全相同的情況下,才認(rèn)定匹配。比如用了這個(gè)標(biāo)志之后,表達(dá)式"a\u030A"會(huì)匹配"?"。默認(rèn)情況下,不考慮"規(guī)范相等性(canonical equivalence)"。
    Pattern.CASE_INSENSITIVE
    (?i)
    默認(rèn)情況下,大小寫不明感的匹配只適用于US-ASCII字符集。這個(gè)標(biāo)志能讓表達(dá)式忽略大小寫進(jìn)行匹配。要想對(duì)Unicode字符進(jìn)行大小不明感的匹配,只要將UNICODE_CASE與這個(gè)標(biāo)志合起來就行了。
    Pattern.COMMENTS
    (?x)
    在這種模式下,匹配時(shí)會(huì)忽略(正則表達(dá)式里的)空格字符(譯者注:不是指表達(dá)式里的"\\s",而是指表達(dá)式里的空格,tab,回車之類)。注釋從#開始,一直到這行結(jié)束。可以通過嵌入式的標(biāo)志來啟用Unix行模式。
    Pattern.DOTALL
    (?s)
    在這種模式下,表達(dá)式'.'可以匹配任意字符,包括表示一行的結(jié)束符。默認(rèn)情況下,表達(dá)式'.'不匹配行的結(jié)束符。
    Pattern.MULTILINE
    (?m)
    在這種模式下,'^'和'$'分別匹配一行的開始和結(jié)束。此外,'^'仍然匹配字符串的開始,'$'也匹配字符串的結(jié)束。默認(rèn)情況下,這兩個(gè)表達(dá)式僅僅匹配字符串的開始和結(jié)束。
    Pattern.UNICODE_CASE
    (?u)
    在這個(gè)模式下,如果你還啟用了CASE_INSENSITIVE標(biāo)志,那么它會(huì)對(duì)Unicode字符進(jìn)行大小寫不明感的匹配。默認(rèn)情況下,大小寫不明感的匹配只適用于US-ASCII字符集。
    Pattern.UNIX_LINES
    (?d)
    在這個(gè)模式下,只有'\n'才被認(rèn)作一行的中止,并且與'.','^',以及'$'進(jìn)行匹配。

    在這些標(biāo)志里面,Pattern.CASE_INSENSITIVEPattern.MULTILINE,以及Pattern.COMMENTS是最有用的(其中Pattern.COMMENTS還能幫我們把思路理清楚,并且/或者做文檔)。注意,你可以用在表達(dá)式里插記號(hào)的方式來啟用絕大多數(shù)的模式。這些記號(hào)就在上面那張表的各個(gè)標(biāo)志的下面。你希望模式從哪里開始啟動(dòng),就在哪里插記號(hào)。

    可以用"OR" ('|')運(yùn)算符把這些標(biāo)志合使用:

    //: c12:ReFlags.javaimport java.util.regex.*;
    import com.bruceeckel.simpletest.*;
    publicclass ReFlags {
      privatestatic Test monitor = new Test();
      publicstaticvoid main(String[] args) {
        Pattern p =  Pattern.compile("^java",
          Pattern.CASE_INSENSITIVE | Pattern.MULTILINE);
        Matcher m = p.matcher(
          "java has regex\nJava has regex\n" +
          "J***A has pretty good regular expressions\n" +
          "Regular expressions are in Java");
        while(m.find())
          System.out.println(m.group());
        monitor.expect(new String[] {
          "java",
          "Java",
          "J***A"
        });
      }
    } ///:~

    這樣創(chuàng)建出來的正則表達(dá)式就能匹配以"java","Java","J***A"...開頭的字符串了。此外,如果字符串分好幾行,那它還會(huì)對(duì)每一行做匹配(匹配始于字符序列的開始,終于字符序列當(dāng)中的行結(jié)束符)。注意,group( )方法僅返回匹配的部分。

    split( )

    所謂分割是指將以正則表達(dá)式為界,將字符串分割成String數(shù)組。

    String[] split(CharSequence charseq)
    String[] split(CharSequence charseq, int limit)

    這是一種既快又方便地將文本根據(jù)一些常見的邊界標(biāo)志分割開來的方法。

    //: c12:SplitDemo.javaimport java.util.regex.*;
    import com.bruceeckel.simpletest.*;
    import java.util.*;
    publicclass SplitDemo {
      privatestatic Test monitor = new Test();
      publicstaticvoid main(String[] args) {
        String input =
          "This!!unusual use!!of exclamation!!points";
        System.out.println(Arrays.asList(
          Pattern.compile("!!").split(input)));
        // Only do the first three:
        System.out.println(Arrays.asList(
          Pattern.compile("!!").split(input, 3)));
        System.out.println(Arrays.asList(
          "Aha! String has a split() built in!".split(" ")));
        monitor.expect(new String[] {
          "[This, unusual use, of exclamation, points]",
          "[This, unusual use, of exclamation!!points]",
          "[Aha!, String, has, a, split(), built, in!]"
        });
      }
    } ///:~

    第二個(gè)split( )會(huì)限定分割的次數(shù)。

    正則表達(dá)式是如此重要,以至于有些功能被加進(jìn)了String類,其中包括split( )(已經(jīng)看到了),matches( )replaceFirst( )以及replaceAll( )。這些方法的功能同PatternMatcher的相同。

    替換操作

    正則表達(dá)式在替換文本方面特別在行。下面就是一些方法:

    replaceFirst(String replacement)將字符串里,第一個(gè)與模式相匹配的子串替換成replacement

    replaceAll(String replacement),將輸入字符串里所有與模式相匹配的子串全部替換成replacement

    appendReplacement(StringBuffer sbuf, String replacement)對(duì)sbuf進(jìn)行逐次替換,而不是像replaceFirst( )replaceAll( )那樣,只替換第一個(gè)或全部子串。這是個(gè)非常重要的方法,因?yàn)樗梢哉{(diào)用方法來生成replacement(replaceFirst( )replaceAll( )只允許用固定的字符串來充當(dāng)replacement)。有了這個(gè)方法,你就可以編程區(qū)分group,從而實(shí)現(xiàn)更強(qiáng)大的替換功能。

    調(diào)用完appendReplacement( )之后,為了把剩余的字符串拷貝回去,必須調(diào)用appendTail(StringBuffer sbuf, String replacement)。

    下面我們來演示一下怎樣使用這些替換方法。說明一下,這段程序所處理的字符串是它自己開頭部分的注釋,是用正則表達(dá)式提取出來并加以處理之后再傳給替換方法的。

    //: c12:TheReplacements.javaimport java.util.regex.*;
    import java.io.*;
    import com.bruceeckel.util.*;
    import com.bruceeckel.simpletest.*;
    /*! Here's a block of text to use as input to
        the regular expression matcher. Note that we'll
        first extract the block of text by looking for
        the special delimiters, then process the
        extracted block. !*/publicclass TheReplacements {
      privatestatic Test monitor = new Test();
      publicstaticvoid main(String[] args) throws Exception {
        String s = TextFile.read("TheReplacements.java");
        // Match the specially-commented block of text above:
        Matcher mInput =
          Pattern.compile("/\\*!(.*)!\\*/", Pattern.DOTALL)
            .matcher(s);
        if(mInput.find())
          s = mInput.group(1); // Captured by parentheses// Replace two or more spaces with a single space:
        s = s.replaceAll(" {2,}", " ");
        // Replace one or more spaces at the beginning of each// line with no spaces. Must enable MULTILINE mode:
        s = s.replaceAll("(?m)^ +", "");
        System.out.println(s);
        s = s.replaceFirst("[aeiou]", "(VOWEL1)");
        StringBuffer sbuf = new StringBuffer();
        Pattern p = Pattern.compile("[aeiou]");
        Matcher m = p.matcher(s);
        // Process the find information as you// perform the replacements:while(m.find())
          m.appendReplacement(sbuf, m.group().toUpperCase());
        // Put in the remainder of the text:
        m.appendTail(sbuf);
        System.out.println(sbuf);
        monitor.expect(new String[]{
          "Here's a block of text to use as input to",
          "the regular expression matcher. Note that we'll",
          "first extract the block of text by looking for",
          "the special delimiters, then process the",
          "extracted block. ",
          "H(VOWEL1)rE's A blOck Of tExt tO UsE As InpUt tO",
          "thE rEgUlAr ExprEssIOn mAtchEr. NOtE thAt wE'll",
          "fIrst ExtrAct thE blOck Of tExt by lOOkIng fOr",
          "thE spEcIAl dElImItErs, thEn prOcEss thE",
          "ExtrActEd blOck. "
        });
      }
    } ///:~

    我們用前面介紹的TextFile.read( )方法來打開和讀取文件。mInput的功能是匹配'/*!' 和 '!*/' 之間的文本(注意一下分組用的括號(hào))。接下來,我們將所有兩個(gè)以上的連續(xù)空格全都替換成一個(gè),并且將各行開頭的空格全都去掉(為了讓這個(gè)正則表達(dá)式能對(duì)所有的行,而不僅僅是第一行起作用,必須啟用多行模式)。這兩個(gè)操作都用了StringreplaceAll( )(這里用它更方便)。注意,由于每個(gè)替換只做一次,因此除了預(yù)編譯Pattern之外,程序沒有額外的開銷。

    replaceFirst( )只替換第一個(gè)子串。此外,replaceFirst( )replaceAll( )只能用常量(literal)來替換,所以如果你每次替換的時(shí)候還要進(jìn)行一些操作的話,它們是無能為力的。碰到這種情況,你得用appendReplacement( ),它能讓你在進(jìn)行替換的時(shí)候想寫多少代碼就寫多少。在上面那段程序里,創(chuàng)建sbuf的過程就是選group做處理,也就是用正則表達(dá)式把元音字母找出來,然后換成大寫的過程。通常你得在完成全部的替換之后才調(diào)用appendTail( ),但是如果要模仿replaceFirst( )(或"replace n")的效果,你也可以只替換一次就調(diào)用appendTail( )。它會(huì)把剩下的東西全都放進(jìn)sbuf。

    你還可以在appendReplacement( )replacement參數(shù)里用"$g"引用已捕獲的group,其中'g' 表示group的號(hào)碼。不過這是為一些比較簡(jiǎn)單的操作準(zhǔn)備的,因而其效果無法與上述程序相比。

    reset( )

    此外,還可以用reset( )方法給現(xiàn)有的Matcher對(duì)象配上個(gè)新的CharSequence。

    //: c12:Resetting.javaimport java.util.regex.*;
    import java.io.*;
    import com.bruceeckel.simpletest.*;
    publicclass Resetting {
      privatestatic Test monitor = new Test();
      publicstaticvoid main(String[] args) throws Exception {
        Matcher m = Pattern.compile("[frb][aiu][gx]")
          .matcher("fix the rug with bags");
        while(m.find())
          System.out.println(m.group());
        m.reset("fix the rig with rags");
        while(m.find())
          System.out.println(m.group());
        monitor.expect(new String[]{
          "fix",
          "rug",
          "bag",
          "fix",
          "rig",
          "rag"
        });
      }
    } ///:~

    如果不給參數(shù),reset( )會(huì)把Matcher設(shè)到當(dāng)前字符串的開始處。

    正則表達(dá)式與Java I/O

    到目前為止,你看到的都是用正則表達(dá)式處理靜態(tài)字符串的例子。下面我們來演示一下怎樣用正則表達(dá)式掃描文件并且找出匹配的字符串。受Unix的grep啟發(fā),我寫了個(gè)JGrep.java,它需要兩個(gè)參數(shù):文件名,以及匹配字符串用的正則表達(dá)式。它會(huì)把匹配這個(gè)正則表達(dá)式那部分內(nèi)容及其所屬行的行號(hào)打印出來。

    //: c12:JGrep.java// A very simple version of the "grep" program.// {Args: JGrep.java "\\b[Ssct]\\w+"}import java.io.*;
    import java.util.regex.*;
    import java.util.*;
    import com.bruceeckel.util.*;
    publicclass JGrep {
      publicstaticvoid main(String[] args) throws Exception {
        if(args.length < 2) {
          System.out.println("Usage: java JGrep file regex");
          System.exit(0);
        }
        Pattern p = Pattern.compile(args[1]);
        // Iterate through the lines of the input file:
        ListIterator it = new TextFile(args[0]).listIterator();
        while(it.hasNext()) {
          Matcher m = p.matcher((String)it.next());
          while(m.find())
            System.out.println(it.nextIndex() + ": " +
              m.group() + ": " + m.start());
        }
      }
    } ///:~

    文件是用TextFile打開的(本章的前半部分講的)。由于TextFile會(huì)把文件的各行放在ArrayList里面,而我們又提取了一個(gè)ListIterator,因此我們可以在文件的各行當(dāng)中自由移動(dòng)(既能向前也可以向后)。

    每行都會(huì)有一個(gè)Matcher,然后用find( )掃描。注意,我們用ListIterator.nextIndex( )跟蹤行號(hào)。

    測(cè)試參數(shù)是JGrep.java和以[Ssct]開頭的單詞。

    還需要StringTokenizer嗎?

    看到正則表達(dá)式能提供這么強(qiáng)大的功能,你可能會(huì)懷疑,是不是還需要原先的StringTokenizer。JDK 1.4以前,要想分割字符串,只有用StringTokenizer。但現(xiàn)在,有了正則表達(dá)式之后,它就能做得更干凈利索了。

    //: c12:ReplacingStringTokenizer.javaimport java.util.regex.*;
    import com.bruceeckel.simpletest.*;
    import java.util.*;
    publicclass ReplacingStringTokenizer {
      privatestatic Test monitor = new Test();
      publicstaticvoid main(String[] args) {
        String input = "But I'm not dead yet! I feel happy!";
        StringTokenizer stoke = new StringTokenizer(input);
        while(stoke.hasMoreElements())
          System.out.println(stoke.nextToken());
        System.out.println(Arrays.asList(input.split(" ")));
        monitor.expect(new String[] {
          "But",
          "I'm",
          "not",
          "dead",
          "yet!",
          "I",
          "feel",
          "happy!",
          "[But, I'm, not, dead, yet!, I, feel, happy!]"
        });
      }
    } ///:~

    有了正則表達(dá)式,你就能用更復(fù)雜的模式將字符串分割開來——要是交給StringTokenizer的話,事情會(huì)麻煩得多。我可以很有把握地說,正則表達(dá)式可以取代StringTokenizer。

    要想進(jìn)一步學(xué)習(xí)正則表達(dá)式,建議你看Mastering Regular Expression, 2nd Edition,作者Jeffrey E. F. Friedl (O’Reilly, 2002)。

    總結(jié)

    Java的I/O流類庫應(yīng)該能滿足你的基本需求:你可以用它來讀寫控制臺(tái),文件,內(nèi)存,甚至是Internet。你還可以利用繼承來創(chuàng)建新的輸入和輸出類型。你甚至可以利用Java會(huì)自動(dòng)調(diào)用對(duì)象的toString( )方法的特點(diǎn)(Java僅有的"自動(dòng)類型轉(zhuǎn)換"),通過重新定義這個(gè)方法,來對(duì)要傳給流的對(duì)象做一個(gè)簡(jiǎn)單的擴(kuò)展。

    但是Java的I/O流類庫及其文檔還是留下了一些缺憾。比方說你打開一個(gè)文件往里面寫東西,但是這個(gè)文件已經(jīng)有了,這么做會(huì)把原先的內(nèi)容給覆蓋了 。這時(shí)要是能有一個(gè)異常就好了——有些編程語言能讓你規(guī)定只能往新建的文件里輸出??磥鞪ava是要你用File對(duì)象來判斷文件是否存在,因?yàn)槿绻阌?span id="qokueu4" class="original_words">FileOutputStream或FileWriter的話,文件就會(huì)被覆蓋了。

    我對(duì)I/O流類庫的評(píng)價(jià)是比較矛盾的;它確實(shí)能干很多事情,而且做到了跨平臺(tái)。但是如果你不懂decorator模式,就會(huì)覺得這種設(shè)計(jì)太難理解了,所以無論是對(duì)老師還是學(xué)生,都得多花精力。此外這個(gè)類庫也不完整,否則我也用不著去寫TextFile了。此外它沒有提供格式化輸出的功能,而其他語言都已經(jīng)提供了這種功能。

    posted @ 2006-07-20 15:48 MyJavaWorld 閱讀(327) | 評(píng)論 (0)編輯 收藏
         摘要: Why is RUNSTATS important? Almost every major database today uses some method of updating catalog statistics to provide the best information possi...  閱讀全文
    posted @ 2006-06-23 15:40 MyJavaWorld 閱讀(454) | 評(píng)論 (0)編輯 收藏
    Abstract
    When you update table statistics using the RUNSTATS command, it requires that you run the command against each table, one by one. REORGCHK gives you the capability to update the statistics of a group of tables or all the tables in the database, with only one command.
    .
    Contents
    _
    DB2 UDB uses the statistics information in the catalog table to derive the best access plan. The DBA should regularly run the RUNSTATS command to keep the database statistics updated. This procedure will ensure the best performance on queries.

    If you want to update statistics on all the tables or a group of tables in the database, you can use the REORGCHK command with the UPDATE STATISTICS option. This would be run instead of running the RUNSTATS command against each table. The UPDATE STATISTICS option will first call the RUNSTATS routine to update the table statistic. You can then use these new statistics to determine if a reorg is needed.

    The REORGCHK command allows you to update statistics on a group of tables with one command.

    To update all the user and system tables use:

    REORGCHK UPDATE STATISTICS ON TABLE ALL

    To update all the tables of a particular schema use:

    REORGCHK UPDATE STATISTICS on SCHEMA schema_name

    To update all the tables of a particular table system use:

    REORGCHK UPDATE STATISTICS on TABLE SYSTEM
    posted @ 2006-06-23 15:19 MyJavaWorld 閱讀(476) | 評(píng)論 (0)編輯 收藏
    <html:radio idName=" idName "? value="value"? name=" name" property=" property " />
    表示在輸出時(shí),html:radio輸出為input type="radio";name輸出為name="name";對(duì)
    于value的輸出,當(dāng)不指定idName時(shí),value="value",當(dāng)指定idName時(shí),輸出是由bean
    名為" idName ",屬性名為"value"的屬性值;當(dāng)bean名為" name ",屬性名為" property "的屬性值等于上述value的輸出值時(shí),輸出checked="checked"。
    我的例子:
    <logic:iterate id = "answer" name = "answerList" scope = "page">
    <html:radio idName = "answer" value = "answerItem" name = "question" property = "rightAnswer" ></html:radio>
    </logic:iterate>
    如果從bean answer的answerItem屬性中取出的值等于從bean question的rightAnswer
    屬性中取出的值相等,那么該radio將被選中

    posted @ 2006-06-21 16:54 MyJavaWorld 閱讀(1422) | 評(píng)論 (0)編輯 收藏

    1. 一個(gè)表如果建有大量索引會(huì)影響 INSERT、UPDATE 和 DELETE 語句的性能,因?yàn)樵诒碇械臄?shù)據(jù)更改時(shí),所有索引都須進(jìn)行適當(dāng)?shù)恼{(diào)整。另一方面,對(duì)于不需要修改數(shù)據(jù)的查詢(SELECT 語句),大量索引有助于提高性能,因?yàn)閿?shù)據(jù)庫有更多的索引可供選擇,以便確定以最快速度訪問數(shù)據(jù)的最佳方法。

    2. 組合索引:組合索引即多列索引,指一個(gè)索引含有多個(gè)列。一個(gè)組合索引相當(dāng)于多個(gè)單列索引,如索引(ColA, ColB, ColC)至少相當(dāng)于(ColA)、(ColA, ColB)、(ColA, ColB, ColC)三個(gè)索引。

    2. 覆蓋的查詢可以提高性能。覆蓋的查詢是指查詢中所有指定的列都包含在同一個(gè)索引(組合索引)中。例如,如果在一個(gè)表的 a、b 和 c 列上創(chuàng)建了組合索引,則從該表中檢索 a 和 b 列的查詢被視為覆蓋的查詢。創(chuàng)建覆蓋一個(gè)查詢的索引可以提高性能,因?yàn)樵摬樵兊乃袛?shù)據(jù)都包含在索引自身當(dāng)中;檢索數(shù)據(jù)時(shí)只需引用表的索引頁,不必引用數(shù)據(jù)頁,因而減少了 I/O 總量。盡管給索引添加列以覆蓋查詢可以提高性能,但在索引中額外維護(hù)更多的列會(huì)產(chǎn)生更新和存儲(chǔ)成本。

    3. 對(duì)小型表進(jìn)行索引可能不會(huì)產(chǎn)生優(yōu)化效果,因?yàn)閿?shù)據(jù)庫在遍歷索引以搜索數(shù)據(jù)時(shí),花費(fèi)的時(shí)間可能會(huì)比簡(jiǎn)單的表掃描還長(zhǎng)。

    4. 應(yīng)使用 SQL 事件探查器和索引優(yōu)化向?qū)椭治霾樵?,確定要?jiǎng)?chuàng)建的索引。為數(shù)據(jù)庫及其工作負(fù)荷選擇正確的索引是非常復(fù)雜的,需要在查詢速度和更新成本之間取得平衡。窄索引(搜索關(guān)鍵字中只有很少的列的索引)需要的磁盤空間和維護(hù)開銷都更少。而另一方面,寬索引可以覆蓋更多的查詢。確定正確的索引集沒有簡(jiǎn)便的規(guī)則。經(jīng)驗(yàn)豐富的數(shù)據(jù)庫管理員常常能夠設(shè)計(jì)出很好的索引集,但是,即使對(duì)于不特別復(fù)雜的數(shù)據(jù)庫和工作負(fù)荷來說,這項(xiàng)任務(wù)也十分復(fù)雜、費(fèi)時(shí)和易于出錯(cuò)。可以使用索引優(yōu)化向?qū)惯@項(xiàng)任務(wù)自動(dòng)化。有關(guān)更多信息,請(qǐng)參見索引優(yōu)化向?qū)А?/p>

    5. 可以在視圖上指定索引。

    6. 可以在計(jì)算列上指定索引。

    7. 避免在索引列上使用IS NULL和IS NOT NULL。避免在索引中使用任何可以為空的列,數(shù)據(jù)庫將無法使用該索引。對(duì)于單列索引,如果列包含空值,索引中將不存在此記錄;對(duì)于復(fù)合索引,如果每個(gè)列都為空,索引中同樣不存在此記錄. 如果至少有一個(gè)列不為空,則記錄存在于索引中。

    8. 如果經(jīng)常檢索包含大量數(shù)據(jù)的表中的少于15%的行則需要?jiǎng)?chuàng)建索引。

    9. 衡量索引效率的 95/5 規(guī)則:如果查詢的結(jié)果返回的行數(shù)少于表中所有行的5%,則索引是檢索數(shù)據(jù)的最快方法,如果查詢的結(jié)果超過5%,那么通常使用索引就不是最快的方式。

    10.主關(guān)鍵字和唯一關(guān)鍵字所在的列自動(dòng)具有索引,但外部關(guān)鍵字沒有自動(dòng)索引。


    二、索引的特征
    ??? 在確定某一索引適合某一查詢之后,可以自定義最適合具體情況的索引類型。索引特征包括:

    ●聚集還是非聚集
    ●唯一還是不唯一
    ●單列還是多列
    ●索引中的列順序?yàn)樯蜻€是降序(索引缺省為升序,但目前多數(shù)大型數(shù)據(jù)庫已經(jīng)能夠支持反向索引)
    ●覆蓋還是非覆蓋
    ●還可以自定義索引的初始存儲(chǔ)特征,通過設(shè)置填充因子優(yōu)化其維護(hù),并使用文件和文件組自定義其位置以優(yōu)化性能。
    ●位映射索引(bitmap)

    posted @ 2006-06-21 16:50 MyJavaWorld 閱讀(2281) | 評(píng)論 (0)編輯 收藏

    基礎(chǔ)
    要使用 sql 獲得當(dāng)前的日期、時(shí)間及時(shí)間戳記,請(qǐng)參考適當(dāng)?shù)?db2 寄存器:

    																				
    select current date from sysibm.sysdummy1
    select current time from sysibm.sysdummy1
    select current timestamp from sysibm.sysdummy1
    
    																		

    sysibm.sysdummy1表是一個(gè)特殊的內(nèi)存中的表,用它可以發(fā)現(xiàn)如上面演示的 db2 寄存器的值。您也可以使用關(guān)鍵字 values 來對(duì)寄存器或表達(dá)式求值。例如,在 db2 命令行處理器(command line processor,clp)上,以下 sql 語句揭示了類似信息:

    																				
    values current date
    values current time
    values current timestamp
    
    																		

    在余下的示例中,我將只提供函數(shù)或表達(dá)式,而不再重復(fù) select ... from sysibm.sysdummy1 或使用 values 子句。

    要使當(dāng)前時(shí)間或當(dāng)前時(shí)間戳記調(diào)整到 gmt/cut,則把當(dāng)前的時(shí)間或時(shí)間戳記減去當(dāng)前時(shí)區(qū)寄存器:

    																				
    current time - current timezone
    current timestamp - current timezone
    
    																		

    給定了日期、時(shí)間或時(shí)間戳記,則使用適當(dāng)?shù)暮瘮?shù)可以單獨(dú)抽取出(如果適用的話)年、月、日、時(shí)、分、秒及微秒各部分:

    																				
    year (current timestamp)
    month (current timestamp)
    day (current timestamp)
    hour (current timestamp)
    minute (current timestamp)
    second (current timestamp)
    microsecond (current timestamp)
    
    																		

    從時(shí)間戳記單獨(dú)抽取出日期和時(shí)間也非常簡(jiǎn)單:

    																				
    date (current timestamp)
    time (current timestamp)
    
    																		

    因?yàn)闆]有更好的術(shù)語,所以您還可以使用英語來執(zhí)行日期和時(shí)間計(jì)算:

    																				
    current date + 1 year
    current date + 3 years + 2 months + 15 days
    current time + 5 hours - 3 minutes + 10 seconds
    
    																		

    要計(jì)算兩個(gè)日期之間的天數(shù),您可以對(duì)日期作減法,如下所示:

    																				
    days (current date) - days (date(''1999-10-22''))
    
    																		

    而以下示例描述了如何獲得微秒部分歸零的當(dāng)前時(shí)間戳記:

    																				
    current timestamp - microsecond (current timestamp) microseconds
    
    																		

    如果想將日期或時(shí)間值與其它文本相銜接,那么需要先將該值轉(zhuǎn)換成字符串。為此,只要使用 char() 函數(shù):

    																				
    char(current date)
    char(current time)
    char(current date + 12 hours)
    
    																		

    要將字符串轉(zhuǎn)換成日期或時(shí)間值,可以使用:

    																				
    timestamp (''2002-10-20-12.00.00.000000'')
    timestamp (''2002-10-20 12:00:00'')
    	date (''2002-10-20'')
    	date (''10/20/2002'')
    	time (''12:00:00'')
    	time (''12.00.00'')
    
    																		

    timestamp()、date() 和 time() 函數(shù)接受更多種格式。上面幾種格式只是示例,我將把它作為一個(gè)練習(xí),讓讀者自己去發(fā)現(xiàn)其它格式。

    警告:
    摘自 db2 udb v8.1 sql cookbook,作者 graeme birchall(see http://ourworld.compuserve.com/homepages/graeme_birchall).

    如果你在日期函數(shù)中偶然地遺漏了引號(hào),那將如何呢?結(jié)論是函數(shù)會(huì)工作,但結(jié)果會(huì)出錯(cuò):

    																												
    
    select date(2001-09-22) from sysibm.sysdummy1;
    
    																										

    結(jié)果:

    																												
    ======
    05/24/0006
    
    																										

    為什么會(huì)產(chǎn)生將近 2000 年的差距呢?當(dāng) date 函數(shù)得到了一個(gè)字符串作為輸入?yún)?shù)的時(shí)候,它會(huì)假定這是一個(gè)有效的 db2 日期的表示,并對(duì)其進(jìn)行適當(dāng)?shù)剞D(zhuǎn)換。相反,當(dāng)輸入?yún)?shù)是數(shù)字類型時(shí),函數(shù)會(huì)假定該參數(shù)值減 1 等于距離公元第一天(0001-01-01)的天數(shù)。在上面的例子中,我們的輸入是 2001-09-22,被理解為 (2001-9)-22, 等于 1970 天,于是該函數(shù)被理解為 date(1970)。

    日期函數(shù)
    有時(shí),您需要知道兩個(gè)時(shí)間戳記之間的時(shí)差。為此,db2 提供了一個(gè)名為 timestampdiff() 的內(nèi)置函數(shù)。但該函數(shù)返回的是近似值,因?yàn)樗豢紤]閏年,而且假設(shè)每個(gè)月只有 30 天。以下示例描述了如何得到兩個(gè)日期的近似時(shí)差:

    																				
    timestampdiff (<n>, char(
    	timestamp(''2002-11-30-00.00.00'')-
    	timestamp(''2002-11-08-00.00.00'')))
    
    																		

    對(duì)于 <n>,可以使用以下各值來替代,以指出結(jié)果的時(shí)間單位:

    • 1 = 秒的小數(shù)部分
    • 2 = 秒
    • 4 = 分
    • 8 = 時(shí)
    • 16 = 天
    • 32 = 周
    • 64 = 月
    • 128 = 季度
    • 256 = 年

    當(dāng)日期很接近時(shí)使用 timestampdiff() 比日期相差很大時(shí)精確。如果需要進(jìn)行更精確的計(jì)算,可以使用以下方法來確定時(shí)差(按秒計(jì)):

    																				
    (days(t1) - days(t2)) * 86400 +  
    (midnight_seconds(t1) - midnight_seconds(t2))
    
    																		

    為方便起見,還可以對(duì)上面的方法創(chuàng)建 sql 用戶定義的函數(shù):

    																				
    create function secondsdiff(t1 timestamp, t2 timestamp)
    returns int
    return (
    (days(t1) - days(t2)) * 86400 +  
    (midnight_seconds(t1) - midnight_seconds(t2))
    )
    @
    
    																		

    如果需要確定給定年份是否是閏年,以下是一個(gè)很有用的 sql 函數(shù),您可以創(chuàng)建它來確定給定年份的天數(shù):

    																				
    create function daysinyear(yr int)
    returns int
    return (case (mod(yr, 400)) when 0 then 366 else 
            case (mod(yr, 4))   when 0 then 
            case (mod(yr, 100)) when 0 then 365 else 366 end 
            else 365 end
    	end)@
    
    																		

    最后,以下是一張用于日期操作的內(nèi)置函數(shù)表。它旨在幫助您快速確定可能滿足您要求的函數(shù),但未提供完整的參考。有關(guān)這些函數(shù)的更多信息,請(qǐng)參考 sql 參考大全。

    sql 日期和時(shí)間函數(shù)
    dayname 返回一個(gè)大小寫混合的字符串,對(duì)于參數(shù)的日部分,用星期表示這一天的名稱(例如,friday)。
    dayofweek 返回參數(shù)中的星期幾,用范圍在 1-7 的整數(shù)值表示,其中 1 代表星期日。
    dayofweek_iso 返回參數(shù)中的星期幾,用范圍在 1-7 的整數(shù)值表示,其中 1 代表星期一。
    dayofyear 返回參數(shù)中一年中的第幾天,用范圍在 1-366 的整數(shù)值表示。
    days 返回日期的整數(shù)表示。
    julian_day 返回從公元前 4712 年 1 月 1 日(儒略日歷的開始日期)到參數(shù)中指定日期值之間的天數(shù),用整數(shù)值表示。
    midnight_seconds 返回午夜和參數(shù)中指定的時(shí)間值之間的秒數(shù),用范圍在 0 到 86400 之間的整數(shù)值表示。
    monthname 對(duì)于參數(shù)的月部分的月份,返回一個(gè)大小寫混合的字符串(例如,january)。
    timestamp_iso 根據(jù)日期、時(shí)間或時(shí)間戳記參數(shù)而返回一個(gè)時(shí)間戳記值。
    timestamp_format 從已使用字符模板解釋的字符串返回時(shí)間戳記。
    timestampdiff 根據(jù)兩個(gè)時(shí)間戳記之間的時(shí)差,返回由第一個(gè)參數(shù)定義的類型表示的估計(jì)時(shí)差。
    to_char 返回已用字符模板進(jìn)行格式化的時(shí)間戳記的字符表示。to_char 是 varchar_format 的同義詞。
    to_date 從已使用字符模板解釋過的字符串返回時(shí)間戳記。to_date 是 timestamp_format 的同義詞。
    week 返回參數(shù)中一年的第幾周,用范圍在 1-54 的整數(shù)值表示。以星期日作為一周的開始。
    week_iso 返回參數(shù)中一年的第幾周,用范圍在 1-53 的整數(shù)值表示。

    改變?nèi)掌诟袷?/span>
    在日期的表示方面,這也是我經(jīng)常碰到的一個(gè)問題。用于日期的缺省格式由數(shù)據(jù)庫的地區(qū)代碼決定,該代碼在數(shù)據(jù)庫創(chuàng)建的時(shí)候被指定。例如,我在創(chuàng)建數(shù)據(jù)庫時(shí)使用 territory=us 來定義地區(qū)代碼,則日期的格式就會(huì)像下面的樣子:

    																				
    
    values current date
    1
    ----------
    05/30/2003
    
    1 record(s) selected.
    
    																		

    也就是說,日期的格式是 mm/dd/yyyy. 如果想要改變這種格式,你可以通過綁定特定的 db2 工具包來實(shí)現(xiàn). 其他被支持的日期格式包括:

    def 使用與地區(qū)代碼相匹配的日期和時(shí)間格式。
    eur 使用歐洲日期和時(shí)間的 ibm 標(biāo)準(zhǔn)格式。
    iso 使用國(guó)際標(biāo)準(zhǔn)組織(iso)制訂的日期和時(shí)間格式。
    jis 使用日本工業(yè)標(biāo)準(zhǔn)的日期和時(shí)間格式。
    loc 使用與數(shù)據(jù)庫地區(qū)代碼相匹配的本地日期和時(shí)間格式。
    usa 使用美國(guó)日期和時(shí)間的 ibm 標(biāo)準(zhǔn)格式。

    在 windows 環(huán)境下,要將缺省的日期和時(shí)間格式轉(zhuǎn)化成 iso 格式(yyyy-mm-dd),執(zhí)行下列操作:

    1. 在命令行中,改變當(dāng)前目錄為 sqllib\bnd 。

      例如:
      在 windows 環(huán)境:c:\program files\ibm\sqllib\bnd
      在 unix 環(huán)境:/home/db2inst1/sqllib/bnd

    2. 從操作系統(tǒng)的命令行界面中用具有 sysadm 權(quán)限的用戶連接到數(shù)據(jù)庫:
      
      db2 connect to dbname
      db2 bind @db2ubind.lst datetime iso blocking all grant public
      

      (在你的實(shí)際環(huán)境中, 用你的數(shù)據(jù)庫名稱和想使用的日期格式分別來替換 dbname and iso。)

    現(xiàn)在,你可以看到你的數(shù)據(jù)庫已經(jīng)使用 iso 作為日期格式了:

    																				
    
    values current date
    1
    ----------
    2003-05-30
    
      1 record(s) selected.
    
    
    																		

    定制日期/時(shí)間格式
    在上面的例子中,我們展示了如何將 db2 當(dāng)前的日期格式轉(zhuǎn)化成系統(tǒng)支持的特定格式。但是,如果你想將當(dāng)前日期格式轉(zhuǎn)化成定制的格式(比如‘yyyymmdd’),那又該如何去做呢?按照我的經(jīng)驗(yàn),最好的辦法就是編寫一個(gè)自己定制的格式化函數(shù)。

    下面是這個(gè) udf 的代碼:

    																				
    create function ts_fmt(ts timestamp, fmt varchar(20))
    returns varchar(50)
    return
    with tmp (dd,mm,yyyy,hh,mi,ss,nnnnnn) as
    (
        select
        substr( digits (day(ts)),9),
        substr( digits (month(ts)),9) ,
        rtrim(char(year(ts))) ,
        substr( digits (hour(ts)),9),
        substr( digits (minute(ts)),9),
        substr( digits (second(ts)),9),
        rtrim(char(microsecond(ts)))
        from sysibm.sysdummy1
        )
    select
    case fmt
        when ''yyyymmdd''
            then yyyy || mm || dd
        when ''mm/dd/yyyy''
            then mm || ''/'' || dd || ''/'' || yyyy
        when ''yyyy/dd/mm hh:mi:ss''
            then yyyy || ''/'' || mm || ''/'' || dd || '' '' || 
                   hh || '':'' || mi || '':'' || ss
        when ''nnnnnn''
            then nnnnnn
        else
            ''date format '' || coalesce(fmt,'' <null> '') || 
            '' not recognized.''
        end
    from tmp
    
    																		

    乍一看,函數(shù)的代碼可能顯得很復(fù)雜,但是在仔細(xì)研究之后,你會(huì)發(fā)現(xiàn)這段代碼其實(shí)非常簡(jiǎn)單而且很優(yōu)雅。最開始,我們使用了一個(gè)公共表表達(dá)式(cte)來將一個(gè)時(shí)間戳記(第一個(gè)輸入?yún)?shù))分別剝離為單獨(dú)的時(shí)間元素。然后,我們檢查提供的定制格式(第二個(gè)輸入?yún)?shù))并將前面剝離出的元素按照該定制格式的要求加以組合。

    這個(gè)函數(shù)還非常靈活。如果要增加另外一種模式,可以很容易地再添加一個(gè) when 子句來處理。在使用過程中,如果用戶提供的格式不符合任何在 when 子句中定義的任何一種模式時(shí),函數(shù)會(huì)返回一個(gè)錯(cuò)誤信息。

    使用方法示例:

    																				
    values ts_fmt(current timestamp,''yyyymmdd'')
     ''20030818''
    values ts_fmt(current timestamp,''asa'') 
     ''date format asa not recognized.''
    
    																		

    ?

    posted @ 2006-06-13 11:29 MyJavaWorld 閱讀(368) | 評(píng)論 (0)編輯 收藏
    主站蜘蛛池模板: 亚洲精品无码Av人在线观看国产| 国产成人无码区免费内射一片色欲| 日韩在线免费播放| 亚洲熟妇无码八V在线播放| 亚洲国产精品免费观看| 亚洲一级毛片在线播放| 一级女人18毛片免费| 亚洲乱码在线视频| 成年女人男人免费视频播放| 中文字幕亚洲码在线| 成人免费无遮挡无码黄漫视频| 中文字幕乱码亚洲精品一区| 德国女人一级毛片免费| 亚洲欧美日韩综合俺去了| 香蕉高清免费永久在线视频 | 羞羞漫画登录页面免费| 国产一级大片免费看| 免费无码午夜福利片69| 亚洲性日韩精品一区二区三区| 亚欧国产一级在线免费| 亚洲老妈激情一区二区三区| 可以免费观看的国产视频| 最近2019年免费中文字幕高清 | 美女又黄又免费的视频| 免费又黄又爽的视频| 亚洲av无码片区一区二区三区| 成人女人A级毛片免费软件| 91在线亚洲综合在线| 四虎影视精品永久免费网站| 人人爽人人爽人人片A免费| 久久亚洲2019中文字幕| 国产一级在线免费观看| 亚洲激情中文字幕| 久久久久免费看黄A片APP| 亚洲AV女人18毛片水真多| 亚洲一区无码精品色| 人人揉揉香蕉大免费不卡| 亚洲精品福利网站| 免费看大黄高清网站视频在线| 免费的黄色网页在线免费观看| 亚洲午夜无码久久久久|