轉 ------ sql load 詳解

yyp2009發表於2008-12-18
sql load 詳解 (2008-09-01 16:18:08)
 這裡用 excel 資料做個最簡單的測試。

  1)把 excel 資料存為 t.txt 檔案,注意檔案字尾名為 .txt

  1 jhchen 11/07/2005 20:04:00 2005-11-7 20:04
  2 jhchen 11/07/2005 20:04:00 2005-11-7 20:04
  3 jhchen 11/07/2005 20:04:00 2005-11-7 20:04

  
  2)SQL> create table t_load(
    2  id         number,
    3  name       varchar2(10),
    4  dat1       date,
    5  dat2       date,
    6  dat3       date
    7  );

  Table created.

  3)控制檔案 t.ctl 如下

  load data                    
  infile 't.txt'               
  badfile 't.bad'              
  append into table t_load     
  fields terminated by X'09'     
  trailing nullcols            
  (                            
  id   ,                           
  name ,                           
  dat1  date "mm/dd/yyyy hh24:mi:ss",
  dat2  date "yyyy-mm-dd hh24:mi:ss"
  )

  其中,X'09' 是製表符,TAB鍵,trailing nullcols 表示如表的欄位沒有對應的值時允許為空。

  
  4)C:\Documents and Settings\cjh>sqlldr userid=jhchen/oracle control=t.ctl

  SQL*Loader: Release 9.2.0.6.0 - Production on 星期一 11月 7 20:20:00 2005

  Copyright (c) 1982, 2002, Oracle Corporation.  All rights reserved.

  達到提交點,邏輯記錄計數3

  
  5)SQL> select * from t_load;

          ID NAME       DAT1      DAT2      DAT3
  ---------- ---------- --------- --------- ---------
           1 jhchen     07-NOV-05 07-NOV-05
           2 jhchen     07-NOV-05 07-NOV-05
           3 jhchen     07-NOV-05 07-NOV-05

  sqlldr userid=lgone/tiger control=a.ctl
  LOAD DATA
  INFILE 't.dat' // 要匯入的檔案
  // INFILE 'tt.date' // 匯入多個檔案
  // INFILE * // 要匯入的內容就在control檔案裡 下面的BEGINDATA後面就是匯入的內容

  INTO TABLE table_name // 指定裝入的表
  BADFILE 'c:\bad.txt' // 指定壞檔案地址

  ************* 以下是4種裝入表的方式
  APPEND // 原先的表有資料 就加在後面
  // INSERT // 裝載空表 如果原先的表有資料 sqlloader會停止 預設值
  // REPLACE // 原先的表有資料 原先的資料會全部刪除
  // TRUNCATE // 指定的內容和replace的相同 會用truncate語句刪除現存資料

  ************* 指定的TERMINATED可以在表的開頭 也可在表的內部欄位部分
  FIELDS TERMINATED BY ',' OPTIONALLY ENCLOSED BY '"'
  // 裝載這種資料: 10,lg,"""lg""","lg,lg"
  // 在表中結果: 10 lg "lg" lg,lg
  // TERMINATED BY X '09' // 以十六進位制格式 '09' 表示的
  // TERMINATED BY WRITESPACE // 裝載這種資料: 10 lg lg

  TRAILING NULLCOLS ************* 表的欄位沒有對應的值時允許為空

  ************* 下面是表的欄位
  (
  col_1 , col_2 ,col_filler FILLER // FILLER 關鍵字 此列的數值不會被裝載
  // 如: lg,lg,not 結果 lg lg
  )
  // 當沒宣告FIELDS TERMINATED BY ',' 時
  // (
  // col_1 [interger external] TERMINATED BY ',' ,
  // col_2 [date "dd-mon-yyy"] TERMINATED BY ',' ,
  // col_3 [char] TERMINATED BY ',' OPTIONALLY ENCLOSED BY 'lg'
  // )
  // 當沒宣告FIELDS TERMINATED BY ','用位置告訴欄位裝載資料
  // (
  // col_1 position(1:2),
  // col_2 position(3:10),
  // col_3 position(*:16), // 這個欄位的開始位置在前一欄位的結束位置
  // col_4 position(1:16),
  // col_5 position(3:10) char(8) // 指定欄位的型別
  // )

  BEGINDATA // 對應開始的 INFILE * 要匯入的內容就在control檔案裡
  10,Sql,what
  20,lg,show

  =====================================================
  //////////// 注意begindata後的數值前面不能有空格

  1 ***** 普通裝載
  LOAD DATA
  INFILE *
  INTO TABLE DEPT
  REPLACE
  FIELDS TERMINATED BY ',' OPTIONALLY ENCLOSED BY '"'
  (DEPTNO,
  DNAME,
  LOC
  )
  BEGINDATA
  10,Sales,"""USA"""
  20,Accounting,"Virginia,USA"
  30,Consulting,Virginia
  40,Finance,Virginia
  50,"Finance","",Virginia // loc 列將為空
  60,"Finance",,Virginia // loc 列將為空

  2 ***** FIELDS TERMINATED BY WHITESPACE 和 FIELDS TERMINATED BY x'09' 的情況
  LOAD DATA
  INFILE *
  INTO TABLE DEPT
  REPLACE
  FIELDS TERMINATED BY WHITESPACE
  -- FIELDS TERMINATED BY x'09'
  (DEPTNO,
  DNAME,
  LOC
  )
  BEGINDATA
  10 Sales Virginia

  3 ***** 指定不裝載那一列
  LOAD DATA
  INFILE *
  INTO TABLE DEPT
  REPLACE
  FIELDS TERMINATED BY ',' OPTIONALLY ENCLOSED BY '"'
  ( DEPTNO,
  FILLER_1 FILLER, // 下面的 "Something Not To Be Loaded" 將不會被裝載
  DNAME,
  LOC
  )
  BEGINDATA
  20,Something Not To Be Loaded,Accounting,"Virginia,USA"

  4 ***** position的列子
  LOAD DATA
  INFILE *
  INTO TABLE DEPT
  REPLACE
  ( DEPTNO position(1:2),
  DNAME position(*:16), // 這個欄位的開始位置在前一欄位的結束位置
  LOC position(*:29),
  ENTIRE_LINE position(1:29)
  )
  BEGINDATA
  10Accounting Virginia,USA

  5 ***** 使用函式 日期的一種表達 TRAILING NULLCOLS的使用
  LOAD DATA
  INFILE *
  INTO TABLE DEPT
  REPLACE
  FIELDS TERMINATED BY ','
  TRAILING NULLCOLS // 其實下面的ENTIRE_LINE在BEGINDATA後面的資料中是沒有直接對應
  // 的列的值的 如果第一行改為 10,Sales,Virginia,1/5/2000,, 就不用TRAILING NULLCOLS了
  (DEPTNO,
  DNAME "upper(:dname)", // 使用函式
  LOC "upper(:loc)",
  LAST_UPDATED date 'dd/mm/yyyy', // 日期的一種表達方式 還有'dd-mon-yyyy' 等
  ENTIRE_LINE ":deptno||:dname||:loc||:last_updated"
  )
  BEGINDATA
  10,Sales,Virginia,1/5/2000
  20,Accounting,Virginia,21/6/1999
  30,Consulting,Virginia,5/1/2000
  40,Finance,Virginia,15/3/2001

  6 ***** 使用自定義的函式 // 解決的時間問題
  create or replace
  function my_to_date( p_string in varchar2 ) return date
  as
  type fmtArray is table of varchar2(25);

  l_fmts fmtArray := fmtArray( 'dd-mon-yyyy', 'dd-month-yyyy',
  'dd/mm/yyyy',
  'dd/mm/yyyy hh24:mi:ss' );
  l_return date;
  begin
  for i in 1 .. l_fmts.count
  loop
  begin
  l_return := to_date( p_string, l_fmts(i) );
  exception
  when others then null;
  end;
  EXIT when l_return is not null;
  end loop;

  if ( l_return is null )
  then
  l_return :=
  new_time( to_date('01011970','ddmmyyyy') + 1/24/60/60 *
  p_string, 'GMT', 'EST' );
  end if;

  return l_return;
  end;
  /

  LOAD DATA
  INFILE *
  INTO TABLE DEPT
  REPLACE
  FIELDS TERMINATED BY ','
  TRAILING NULLCOLS
  (DEPTNO,
  DNAME "upper(:dname)",
  LOC "upper(:loc)",
  LAST_UPDATED "my_to_date( :last_updated )" // 使用自定義的函式
  )
  BEGINDATA
  10,Sales,Virginia,01-april-2001
  20,Accounting,Virginia,13/04/2001
  30,Consulting,Virginia,14/04/2001 12:02:02
  40,Finance,Virginia,987268297
  50,Finance,Virginia,02-apr-2001
  60,Finance,Virginia,Not a date

  7 ***** 合併多行記錄為一行記錄
  LOAD DATA
  INFILE *
  concatenate 3 // 通過關鍵字concatenate 把幾行的記錄看成一行記錄
  INTO TABLE DEPT
  replace
  FIELDS TERMINATED BY ','
  (DEPTNO,
  DNAME "upper(:dname)",
  LOC "upper(:loc)",
  LAST_UPDATED date 'dd/mm/yyyy'
  )
  BEGINDATA
  10,Sales, // 其實這3行看成一行 10,Sales,Virginia,1/5/2000
  Virginia,
  1/5/2000
  // 這列子用 continueif list="," 也可以
  告訴sqlldr在每行的末尾找逗號 找到逗號就把下一行附加到上一行

LOAD DATA
  INFILE *
  continueif this(1:1) = '-' // 找每行的開始是否有連線字元 - 有就把下一行連線為一行
  // 如 -10,Sales,Virginia,
  // 1/5/2000 就是一行 10,Sales,Virginia,1/5/2000
  // 其中1:1 表示從第一行開始 並在第一行結束 還有continueif next 但continueif list最理想
  INTO TABLE DEPT
  replace
  FIELDS TERMINATED BY ','
  (DEPTNO,
  DNAME "upper(:dname)",
  LOC "upper(:loc)",
  LAST_UPDATED date 'dd/mm/yyyy'
  )
  BEGINDATA // 但是好象不能象右面的那樣使用
  -10,Sales,Virginia, -10,Sales,Virginia,
  1/5/2000 1/5/2000
  -40, 40,Finance,Virginia,13/04/2001
  Finance,Virginia,13/04/2001

來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/13750068/viewspace-517380/,如需轉載,請註明出處,否則將追究法律責任。

相關文章