從壹個或多個表中檢索數據。SELECT SQL 命令是與其它 Vfp壹樣的內置的 Vfp命令。當妳使用 SELECT 來生成查詢時, Vfp翻譯查詢並從表中獲取指定數據。妳可以從以下地方創建 SELECT 查詢:
“命令”窗口中
帶有其它任何 Vfp命令的 Vfp程序中
查詢設計器中
SELECT [ALL | DISTINCT] [TOP nExpr [PERCENT]] [Alias.] Select_Item
[[AS] Column_Name] [, [Alias.] Select_Item [[AS] Column_Name] ...]
FROM [FORCE] [DatabaseName!] Table [[AS] Local_Alias]
[ [INNER | LEFT [OUTER] | RIGHT [OUTER] | FULL [OUTER] JOIN DatabaseName!]
Table [[AS] Local_Alias] [ON JoinCondition ...]
[[INTO Destination] | [TO FILE FileName [ADDITIVE] | TO PRINTER [PROMPT] | TO SCREEN]]
[PREFERENCE PreferenceName] [NOCONSOLE] [PLAIN] [NOWAIT]
[WHERE JoinCondition [AND JoinCondition ...] [AND | OR FilterCondition [AND | OR FilterCondition ...]]]
[Group By GroupColumn [, GroupColumn ...]] [HAVING FilterCondition] [UNION [ALL] SELECTCommand]
[Order By Order_Item [ASC | DESC] [, Order_Item [ASC | DESC] ...]]
參數
SELECT
在 SELECT 子句中指定在查詢結果中包含的字段、常量和表達式。
ALL
查詢結果中包含所有行 ( 包括重復值 )。ALL 是默認設置。
DISTINCT
在查詢結果中剔除重復的行。每壹個 SELECT 子句只能使用壹次 DISTINCT。
TOP nExpr [PERCENT]
在符合查詢條件的所有記錄中,選取指定數量或百分比的記錄。TOP 子句必須與 ORDER BY 子句同時使用。ORDER BY 子句指定查詢結果中包含的列上由Top字句決定的行數, TOP 子句根據此排序選定最開始的 nExpr個或 nExpr% 的記錄。
您可以指定選取 1 到 32767 個記錄。使用 ORDER BY 子句指定的字段進行排序,會產生並列的情況,比如,可能有多個記錄,它們在選定的字段上相同;所以,如果您指定 nExpr 為 10,在查詢結果中可能多於 10 個記錄,因為可能有幾個記錄位置並列。
如果包含 PERCENT 關鍵字指定查詢結果中的記錄數,得到記錄數的可能是小數,這時進行取整。包含 PERCENT 關鍵字時,nExpr 的範圍是 0.01 到 99.99。
[Alias.] Select_Item
限定匹配項的名稱。Select_Item 指定的每壹項在查詢結果中都生成壹列。壹個項可以是以下壹個
FROM 子句所包含的表中的字段名稱。
壹個常量,查詢結果中每壹行都出現這個常量值。
壹個表達式,可以是用戶自定義函數名。
關於使用用戶定義函數的詳細信息, 參見註釋節中的帶用戶定義函數的 SELECT。
妳用 Select_Item 指定的各項生成壹個查詢結果列。
如果兩個或更多的項具有相同的名稱, 在項名前包含表別名和壹個句點來避免列重復。
[AS] Column_Name
為查詢輸出中的列指定顯示名。Column_Name 可以是表達式但不能包含不允許的字符, 如, 字段名中的空格。
當 Select_Item 是壹個表達式或包含壹個字段函數而且妳想給該列壹個有意義的名字時該選項是有用的。
FROM [FORCE] DatabaseName!
列出所有從中檢索數據的表。
FORCE 指定連接表時按它們出現在 FROM 子句中的順序。如果省略 FORCE, Vfp會試圖對查詢進行優化。但是, 使用 FORCE 子句,避免了優化過程,可能加快查詢執行的速度。
當包含表的數據庫不是當前數據庫時,DatabaseName! 指定這個數據庫的名稱。如果數據庫不是當前數據庫,就必須指定包含表的數據庫名稱。應在數據庫名稱之後表名之前加上感嘆號(!)分隔符。
[[AS] Local_Alias]
為 Table 中的表指定壹個臨時名稱。如果指定了本地別名,那麽在整個SELECT 語句中必須都用這個別名代替表名。本地別名不影響 Visual FoxPro環境。INNER JOIN 只有在其他表中包含對應記錄(壹個或多個)的記錄才出現在查詢結果中。
INNER JOIN 只有在其他表中包含對應記錄(壹個或多個)的記錄才出現在查詢結果中。
LEFT [OUTER] JOIN 在查詢結果中包含:JOIN 左側表中的所有記錄,以及JOIN 右側表中匹配的記錄。OUTER 關鍵字可被省略;包含 OUTER 強調這是壹個外連接 (outer join)。
RIGHT [OUTER] JOIN 在查詢結果中包含:JOIN 右側表中的所有記錄,以及 JOIN 左側表中匹配的記錄。OUTER 關鍵字可被省略;包含 OUTER 強調這是壹個外連接接 (outer join)。
FULL [OUTER] JOIN 在查詢結果中包含:JOIN 兩側所有的匹配記錄,和不匹配的記錄;包含 OUTER 強調這是壹個外連接 (outer join)。
關於連接的詳細信息, 參見備註段中的 Joins。
ON JoinCondition 指定連接條件。
INTO Destination
指定在何處保存查詢結果。Destination 可以是下列子句之壹:
ARRAY ArrayName ,將查詢結果保存到變量數組中。
如果查詢結果中不包含任何記錄,則不創建這個數組。
CURSOR CursorName [NOFILTER | READWRITE] 將查詢結果保存到臨時表中。
要創建壹個查用於子查詢中的遊標, 用 NOFILTER。關於 NOFILTER 的詳細信息, 參見備註節。
要指定遊標是臨時的和可修改的, 使用 READWRITE。如果源表或表使用 autoincrementing, 該設置不會被 READWRITE 遊標繼承。
DBF | TABLE TableName [DATABASE DatabaseName [NAME LongTableName]] 保存查詢結果到壹個表中。
包含 DATABASE DatabaseName 以指定添加了表的數據庫。
包含 NAME LongTableName 可以為該表命壹個最多可包括 128 個字符的並且可以在數據庫中代替短名字的長名。
如果沒有包括 INTO 子句, 查詢結果顯示在壹個“瀏覽”窗口中。也可以用 TO FILE 子句來定向查詢結果到打印機或壹個文件。
TO FILE FileName [ADDITIVE] | TO PRINTER [PROMPT] | TO SCREEN
定向查詢結果到打印機或壹個文件。
ADDITIVE 添加查詢輸出到 TO FILE FileName 中指定的已存在的文本文件內容中。
TO PRINTER 定向查詢輸出到壹個打印機。在打印開始之前,使用可選的 PROMPT 子句顯示壹個對話框。您可以根據當前安裝的打印機驅動程序調整打印機的設置。將 PROMPT 子句放置在緊跟 TO PRINTER 之後。
TO SCREEN 使查詢結果定向輸出到 Vfp主窗口或活動的用戶自定義窗口中。
PREFERENCE PreferenceName
如果查詢結果送往瀏覽窗口,就可以使用 PREFERENCE 保存瀏覽窗口的屬性和選項以備後用。關於 PREFERENCE 功能的詳細信息, 參見備註節。
NOCONSOLE
不顯示送到文件、打印機或 Vfp主窗口的查詢結果。
PLAIN
防止列標題出現在顯示的查詢結果中。不管有無 TO 子句都可使用 PLAIN子句。如果 SELECT 語句中包括 INTO 子句,則忽略 PLAIN 子句。
NOWAIT
打開瀏覽窗口並將查詢結果輸出到這個窗口後繼續程序的執行。程序並不等待關閉瀏覽窗口,而是立即執行緊接在 SELECT 語句後面的程序行。關於如何使用 NOWAIT 的說明, 參見備註節。
WHERE JoinCondition
指定 Vfp的查詢結果中只包括符合指定條件的記錄。JoinCondition 指定位於 FROM 子句中的字段連接表。關於指定連接條件的詳細信息, 參見備註節。
WHERE 支持 JoinCondition 的 ESCAPE 操作符, 讓妳可以執行包含有百分號 (%) 和下劃線 (_) 通配符的 SELECT SQL 命令查詢。ESCAPE 允許妳指定壹個按原字樣處理的 SELECT SQL 命令通配符。在 ESCAPE 子句中, 壹旦壹個字符被放到通配符字符之前,就表示這個通配符被看作壹個文字字符。
FilterCondition
指定將包含在查詢結果中記錄必須符合的條件。使用 AND 或 OR 操作符,您可以包含隨意數目的過濾條件。您還可以使用 NOT 操作符將邏輯表達式的值取反,或使用 EMPTY() 函數以檢查空字段。
SELECT SQL 命令在篩選條件中支持 "<field> IS / IS NOT NULL"。要學習如何使用 FilterCondition。
Group By GroupColumn [, GroupColumn ...]
按列的值對查詢結果的行進行分組。GroupColumn 可以是常規的表字段名,也可以是壹個包含 SQL 字段函數的字段名,還可以是壹個數值表達式,指定查詢結果表中的列位置(最左邊的列編號為 1 )。
HAVING FilterCondition
指定包括在查詢結果中的組必須滿足的篩選條件。HAVING 應該同 GROUP BY壹起使用。它能包含數量不限的篩選條件,篩選條件用 AND 或 OR 連接,還可以使用 NOT 來對邏輯表達式求反。可以在 HAVING 子句中使用本地別名和字段函數。 關於妳可以使用的字段函數的詳細信息, 參見備註節。FilterCondition 不能包含子查詢。
可以使用帶 HAVING 子句的 Group By。使用 HAVING 子句的命令如果沒有使用 GROUP BY 子句,則它的作用與WHERE 子句相同。
如果 HAVING 子句不包含字段函數的話,使用 WHERE 子句可以獲得較快的速度。
HAVING 子句應該出現在 INTO 子句前否則產生錯誤。
[UNION [ALL] SELECTCommand]
把壹個 SELECT 語句的最後查詢結果同另壹個 SELECT 語句最後查詢結果組合起來。默認情況下,UNION 檢查組合的結果並排除重復的行。
要組合多個UNION 子句,可使用括號。可以用 UNION 子句模擬壹個外部聯接。
ALL 防止 UNION 刪除組合結果中重復的行。
當壹個列是備註或通用型時, 不允許連接不同類型的列。
在 Vfp8.0 以前的版本中, 當在兩個不同類型的字段上執行 UNION 操作時妳需要執行明確的轉換。
Vfp現在對支持它的數據類型支持隱含數據類型轉換。關於隱含數據類型轉換和數據類型優先, UNION 子句允許的規則, 以及其它信息的詳細內容, 參見備註節中的數據類型轉換和優先。
Order By Order_Item [ASC | DESC]
根據列的數據對查詢結果進行排序。每個 Order_Item 都必須對應查詢結果中的壹列。它可以是下列之壹:
FROM 子句中表的字段,同時也是 SELECT 主句(不在子查詢中)的壹個選擇項。
壹個數值表達式,表示查詢結果中列的位置(最左邊列編號為 1 )。
ASC 指定查詢結果根據排序項以升序排列。它是 ORDER BY 的默認選項。
DESC 指定查詢結果以降序排列。
備註
在使用 FROM 子句時如果沒有打開表, Vfp顯示“打開”對話框讓妳指定文件位置。壹但打開後, 表在查詢完成後仍然保持打開。
當在 Destination 參數中使用 CURSOR 子句時, 如果妳指定了壹個打開的表的名字, Vfp產生壹條錯誤信息。在 SELECT 執行後, 臨時遊標保持打開並是活動的和只讀的除非妳指定了 READWRITE 選項。當妳關閉該臨時遊標時, 它被刪除。遊標可以指定 SORTWORK 而成為存在於驅動器或卷上的臨時文件。
當在 Destination 參數中使用 CURSOR 子句時, 妳現在可以使用 NOFILTER 來創建壹個可用於後來的查詢的遊標。在早期版本的 Vfp中, 妳需要包括壹個額外的常數或表達式作為篩選。例如, 添加壹個邏輯 true 作為篩選表達式來創建壹個可用於後來的查詢的查詢:
SELECT *, .T. FROM customers INTO CURSOR myquery
但是, 包括 NOFILTER 會降低查詢性能因為要在磁盤上創建壹個臨時表。臨時表在遊標關閉時從磁盤上刪除。
當在 Destination 參數中使用 DBF | TABLE 子句時, 如果妳指定了壹個已經打開的表, 而且 SET SAFETY 是設置為 OFF, Vfp不警告地復寫該表。如果妳沒有指定壹個擴展名, Vfp給表壹個 .dbf 擴展名。在 SELECT 執行後表保持打開並且是活動的。
如果妳在相同查詢中包括 INTO 和 TO 子句, Vfp忽略 TO 子句。如果妳包括 TO 子句但沒有包括 INTO 子句, 妳可以定向查詢結果到壹個名為 FileName 的 ASCII 文本文件, 到打印機, 或到 Vfp主窗口。
PREFERENCE 把特征, 屬性或參數選項長期保存在 FoxUser.dbf 資源文件中。Preferences 可以在任何時候獲取。第壹次執行有 PREFERENCE Preference Name 的 SELECT 命令時創建參數選項。以後執行有相同參數選項名的 SELECT 命令時便將瀏覽窗口恢復到原來的參數選項狀態。當瀏覽窗口關閉時,更新參數選項。如果您按下 CTRL+Q+W 鍵退出“瀏覽”窗口,您對“瀏覽”窗口所做的更改不會保存到資源文件中。
SELECT 命令中包括 TO SCREEN 可以把查詢結果定向輸出到 Vfp主窗口或用戶自定義窗口。如果顯示時 Vfp主窗口或用戶自定義窗口中寫滿了壹屏,就暫停輸出。按任意鍵可以查看查詢結果後面的內容。但是,如果命令中包括了 NOWAIT 子句,顯示查詢結果時就不會暫停,等待按鍵,而是在 Vfp主窗口或用戶自定義窗口中連續滾過所有內容。如果命令中包含有 INTO 子句,忽略 NOWAIT 子句。
在壹個 SQL 查詢的 WHERE 子句中包括 EVALUATE() 函數會返回不正確的數據。
如果包括壹個以上的表在查詢中, 妳應該在第壹個以後為每壹個表指定壹個連接條件。連接條件可以包含篩選條件。
註意 每壹個 SELECT 語句的最大連接數是 9.
必須用 AND 操作符來連接多個連接條件。各連接條件具有以下格式:
當妳在串中使用 = 操作符時, 它的動作根據 SET ANSI 的設置會不同。當 SET ANSI 設置為 OFF 時, Vfp只比較串到較短串結束。當 SET ANSI 設置為 ON 時, Vfp遵循 ANSI 標準的字符串比較。關於 Vfp如果執行字符串比較的額外信息, 參見 SET ANSI 和 SET EXACT。
下列字段函數可以與選定項壹起使用,選定項可以是壹個字段或包含字段的表達式:
AVG(Select_Item), 計算列中數值的平均值。
COUNT(Select_Item), 計算列中選定項的數目。計算查詢輸出的行數。COUNT(*) 計算查詢輸出中的行數。
MIN(Select_Item), 確定列中 Select_Item 的最小值。
MAX(Select_Item), 確定列中 Select_Item 的最大值。
SUM(Select_Item), 計算列中數值的和。
字段函數不能嵌套使用。
UNION 子句遵守下列規則:
不能使用 UNION 來組合子查詢。
兩個 SELECT 命令的查詢結果中的列數必須相同。
兩個 SELECT 查詢結果中的對應列必須有相同的數據類型和寬度。
只有最後的 SELECT 中可以包含 ORDER BY 子句,而且必須按編號指出所輸出的列。如果包含了壹個 ORDER BY 子句,它將影響整個結果。
當妳用 UNION 連接查詢中的兩個表時, 僅匹配連接字段值的記錄會出現在查詢結果中。如果在父表中的記錄在子表中沒有相應的記錄, 父表中的記錄不會出現在查詢結果中。壹個外部聯接允許妳包括父表中的所有記錄到輸出結果中, 連同子表中的匹配記錄壹起。要在 Vfp中創建壹個外部聯接, 妳需要要使用壹個嵌套的 SELECT 命令
註意 確信在每壹個分號前包括壹個空格。否則, Vfp產生壹個錯誤。
上例中, 在 UNION 子句前的部分的命令從兩個表中選擇具有匹配值的記錄。不包括沒有相關的發票的客戶公司。命令中 UNION 子句後的部分選擇客戶表中的在訂單表中無匹配記錄的記錄。
關於第二部分的命令, 註意以下幾點:
包括在園括號中的 SELECT 語句首先處理。該語句的結果是選擇訂單表中的所有客戶編號。
WHERE 子句找出 customer 表中的在 orders 表沒有相關記錄的所有客戶編號。由於第壹節中的命令提供了所在 orders 表中有客戶編號的公司, Customer 表中的所有公司現在都包含在查詢結果中了。
因為在 UNION 中的表的結構必須相同, 有兩個占位符在第二個 SELECT 語句中來代表第壹個 SELECT 語句中的 orders.order_id 和 orders.emp_id。
註意 占位符必須與它們所代表的字段有相同類型。如果字段是日期型, 占位符應該是 。如果字段是壹個字符字段, 占位符應該是壹個空串 ("")。
如果妳沒有在 Order By 子句中指定排序, 查詢結果顯示為未排序。
當妳發出 SET TALK ON 並執行 SELECT 時, Vfp顯示查詢使用的時間和結果中的記錄數。 _TALLY 包含了在查詢結果中的記錄數。
SET FILTER 設置的篩選條件對 SELECT 命令不起作用。
註意 下面部分提到的子查詢, 是指在 SELECT 命令中包含的 SELECT 命令。子查詢必須包括在園括號中。在 SELECT 命令的 WHERE 子句中可以包含最多兩個平級的(非嵌套)的子查詢。子查詢中可以有多個連接條件 (join conditions)。
在妳創建查詢輸出時, 列的命名遵循如下規則:
如果選擇項是具有唯壹名稱的字段,則用字段名作為輸出列名。
如果多個選擇項具有相同名稱。例如,如果名為 Customer 的表有壹個STREET 字段,而名為 Employees 的表也有壹個 STREET 字段,則輸出列命名為 Extension_A 和 Extension_B (STREET_A 和 STREET_B)。如果選擇項名稱有 10 字符長,可以將名稱截短後再加下劃線和字母。例如,DEPARTMENT 變為 DEPARTME_A。
如果選擇項是表達式,它的輸出列命名為 EXP_A。其他表達式分別命名為EXP_B、EXP_C,依此類推。
如果選擇項包含諸如 COUNT() 這樣的字段函數,則輸出列命名為CNT_A。如果另壹個選擇項包含 SUM(),它的輸出列命名為 SUM_B。
用戶定義函數和 在 SELECT 子句中使用用戶自定義函數有明顯優點,但使用時應考慮以下限制:
SELECT 子句的運行速度會受用戶自定義函數執行速度的影響。因此,如果使用戶自定義函數的操作量很大,則這些函數的功能最好調用 C 語言或匯編語言編寫的 API 或用戶自定義函數來完成。
在 SELECT 激活的用戶自定義函數中,很難預測 Vfp輸入/輸出(I/O)和表的環境。壹般來說,不知道選擇的工作區是哪壹個,不知道當前表的名稱,甚至不知道正在處理的字段名。這些變量的值完全取決於用戶自定義函數在優化過程的什麽地方激活。
在 SELECT 子句調用的用戶自定義函數中修改 VfpI/O 或表的環境是很不安全的。壹般來說,這樣做的結果難以預料。
從 SELECT 將值傳遞給用戶自定函數唯壹可靠的方法,是激活用戶自定義函數時以參數的形式傳遞。
經過實踐,有可能發現某種被認為是違法的操作在某種 FoxPro 版本中運行正確,但這並不保證它在以後的版本中也能正確運行。
拋開這些限制不說,用戶自定義函數在 SELECT 語句中還是可接受的。但不要忘記使用 SELECT 可能要降低性能。要學習如何在 SELECT 中使用用戶定義函數, 參見示例節。
連接 Vfp支持 ANSI SQL '92 連接 (Join) 語法,通過比較兩個或多個表中的字段,將它們的記錄連接到壹起,生成查詢。例如,內部連接 (inner join) 是將兩個表中連接字段 (joined field) 值相同的記錄選取到查詢中。Vfp支持嵌套連接(nested joins)
由於 SQL 是派生於數學集合理論, 各表可以代表壹個環。指定連接條件的 ON 子句確定交接點, 它代表匹配的行集合。對於壹個內部聯接, 交接發生在兩個環的內部或 "inner" 部分。壹個外聯接不僅僅包括這些表內部的交叉區域匹配的行, 也包括環的外面的左或右部的交集的行。