資料庫
主語言是面向記錄的,一組主變數一次只能存放一條記錄,僅使用主變數並不能完全滿足SQL語句向應用程式輸出數據的要求。嵌入式SQL引入了游標的概念,用來協調這兩種不同的處理方式。在資料庫開發過程中,當你檢索的數據只是一條記錄時,你所編寫的事務語句代碼往往使用SELECT INSERT 語句。但是我們常常會遇到這樣情況,即從某一結果集中逐一地讀取一條記錄。那么如何解決這種問題呢?游標為我們提供了一種極為優秀的解決方案。
優點
在資料庫中,游標是一個十分重要的概念。游標提供了一種對從表中檢索出的數據進行操作的靈活手段,就本質而言,游標實際上是一種能從包括多條數據記錄的結果集中每次提取一條記錄的機制。游標總是與一條SQL 查詢語句相關聯因為游標由結果集(可以是零條、一條或由相關的選擇語句檢索出的多條記錄)和結果集中指向特定記錄的游標位置組成。當決定對結果集進行處理時,必須聲明一個指向該結果集的游標。如果曾經用C 語言寫過對檔案進行處理的程式,那么游標就像您打開檔案所得到的檔案句柄一樣,只要檔案打開成功,該檔案句柄就可代表該檔案。對於游標而言,其道理是相同的。可見游標能夠實現按與傳統程式讀取平面檔案類似的方式處理來自基礎表的結果集,從而把表中數據以平面檔案的形式呈現給程式。
我們知道關係資料庫管理系統實質是面向集合的,在MS SQL SERVER 中並沒有一種描述表中單一記錄的表達形式,除非使用where 子句來限制只有一條記錄被選中。因此我們必須藉助於游標來進行面向單條記錄的數據處理。由此可見,游標允許應用程式對查詢語句select 返回的行結果集中每一行進行相同或不同的操作,而不是一次對整個結果集進行同一種操作;它還提供對基於游標位置而對表中數據進行刪除或更新的能力;而且,正是游標把作為面向集合的資料庫管理系統和面向行的程式設計兩者聯繫起來,使兩個數據處理方式能夠進行溝通。
種類
MS SQL SERVER 支持三種類型的游標:Transact_SQL 游標,API伺服器游標和客戶游標。
(1)Transact_SQL 游標
Transact_SQL 游標是由DECLARE CURSOR 語法定義、主要用在Transact_SQL腳本、存儲過程和觸發器中。Transact_SQL 游標主要用在伺服器上,由從客戶端傳送給伺服器的Transact_SQL 語句或是批處理、存儲過程、觸發器中的Transact_SQL 進行管理。 Transact_SQL 游標不支持提取數據塊或多行數據。
(2)API游標
API 游標支持在OLE DB, ODBC 以及DB_library 中使用游標函式,主要用在伺服器上。每一次客戶端應用程式調用API 游標函式,MS SQL SEVER 的OLE DB 提供者、ODBC驅動器或DB_library 的動態程式庫(DLL) 都會將這些客戶請求傳送給伺服器以對API游標進行處理。
(3)客戶游標
客戶游標主要是當在客戶機上快取結果集時才使用。在客戶游標中,有一個預設的結果集被用來在客戶機上快取整個結果集。客戶游標僅支持靜態游標而非動態游標。由於伺服器游標並不支持所有的Transact-SQL語句或批處理,所以客戶游標常常僅被用作伺服器游標的輔助。因為在一般情況下,伺服器游標能支持絕大多數的游標操作。由於API 游標和Transact-SQL 游標使用在伺服器端,所以被稱為伺服器游標,也被稱為後台游標,而客戶端游標被稱為前台游標。在本章中我們主要講述伺服器(後台)游標
使用示例(1)
create proc cursorTest
@_id int=0,
@_name varchar(50)=""
as--創建游標
declare @cursor cursor--設定游標欲操作的數據集
set @cursor=cursor for
select _id,_name from users
open @cursor--打開游標
fetch next from @cursor into @_id,@_name--移動游標指向到第一條數據,提取第一條數據存放在變數中
while(@@fetch_status=0)begin--如果上一次操作成功則繼續循環
print @_name--操作提出的數據
fetch next from @cursor into @_id,@_name--繼續提下一行
end
close @cursor--關閉游標
deallocate @cursor--刪除游標
使用示例2
CREATE proc [dbo].[As_Proc_GetPlanDetail]
(
@ids varchar(max)--
)
as
create table #temp
(
dept varchar(100),
sorttitle varchar(200),
title varchar(200),
spec varchar(50),
model varchar(50),
budget decimal(18,2),
amount int,
sgamount int,
htamount int,
ysamount int,
fenfamount int
)
declare @did varchar(32)
declare @sorttitle varchar(200)
declare @title varchar(200)
declare @spec varchar(50)
declare @model varchar(50)
declare @budget decimal(18,2)
declare @amount int
declare @sgamount int
declare @htamount int
declare @ysamount int--驗收數量
declare @fenfamount int--分發數量
declare cursor1 cursor for
select rowid,dbo.clip(ypdept,':',1) from As_year_plan where Charindex(rowid,@ids,0)>0 union select rowid,dbo.clip(department,':',1) from As_assets_requisition where Charindex(rowid,@ids,0)>0 -----------------------------------------------控制計畫
declare @rowid varchar(32)
declare @dept varchar(100)
open cursor1
fetch next from cursor1 into @rowid,@dept
while@@fetch_status=0
begin
declare cursor2 cursor for select a.rowid, b.sorttitle,a.title,a.spec,a.model,a.budget,a.amount from As_assts_planDetai a left join As_assetsSort b on a.sortId=b.rowid whereplanID=@rowid
open cursor2
fetch next from cursor2 into @did, @sorttitle, @title,@spec, @model ,@budget ,@amount
while @@fetch_status=0
begin
select @sgamount= isnull(sum(a.amount),0) from As_assets_buyDetail a inner join As_assets_buy b on a.buyId=b.rowid wherea.planId=@didand b.state=1
--print @did
select @htamount= isnull(sum(a.amount),0) from As_contract_detail a inner join As_stock_contract b on a.contId=b.rowid where b.state=1 and a.bydetId in(
select c.rowid from As_assets_buyDetail c inner join As_assets_buy d on c.buyId=d.rowid wherec.planId=@didand d.state=1)
---驗收數量
select @ysamount=count(1) from As_AssetsInfo a inner join As_contract_detail b on a. contractId=b.rowid where b.bydetId in(select rowid from As_assets_buyDetail whereplanId=@did)
--分發數量
select @fenfamount=count(1) from As_AssetsInfo a inner join As_contract_detail b on a. contractId=b.rowid where b.bydetId in(select rowid from As_assets_buyDetail whereplanId=@did) and a.drawState=1
insert into #temp(dept, sorttitle, title , spec, model , budget ,amount,sgamount,htamount, ysamount,fenfamount) values(@dept,@sorttitle, @title , @spec, @model , @budget ,@amount,@sgamount,@htamount,@ysamount,@fenfamount)
fetch next from cursor2 into @did, @sorttitle, @title,@spec, @model ,@budget ,@amount
end
close cursor2 --關閉游標
deallocate cursor2
fetch next from cursor1 into @rowid,@dept
end
close cursor1 --關閉游標
deallocate cursor1 --釋放游標
--查詢臨時表
select * from #temp
詳細說明
RS.OPEN SQL,CONN,A,B
參數A為設定游標的類型,其取值為: 0 僅向前游標,只能向前瀏覽記錄,不支持分頁、Recordset、BookMark
1 鍵集游標,其他用戶對記錄所做的修改將反映到記錄集中,但其他用戶增加或刪除記錄不會反映到記錄集中。支持分頁、Recordset、BookMark
2動態游標功能最強,但耗資源也最多。用戶對記錄所做的修改,增加或刪除記錄都將反映到記錄集中。支持全功能瀏覽。
3靜態游標,只是數據的一個快照,用戶對記錄所做的修改,增加或刪除記錄都不會反映到記錄集中。支持向前或向後移動
參數B為記錄集的鎖定類型,其取值為:
1 鎖定類型,默認的,唯讀,不能作任何修改
2 當編輯時立即鎖定記錄,最安全的方式
3 只有在調用Update方法時才鎖定記錄集,而在此前的其他操作仍可對當前記錄進行更改、插入和刪除等
4 當編輯時記錄不會被鎖定,而更改、插入和刪除是在批處理方式下完成的
打開數據記錄集方法其實不止一種,但是我們用的最多的就是
rs.open sql,1,1的方法,可是後面的數字參數很多人不解其意,下面我們來介紹一下。
其實open方法後面有多個參數
CursorType LockType CommandType
比如 rs.open sql,1,1
也可以寫成
rs.cursorType = 1
rs.LockType = 1
rs.open sql
其中CursorType代表從一個表或者一個SQL查詢結果返回的記錄。
這個參數有四個值分別是:
adOpenForwardOnly 表示只允許在記錄集內的記錄間往前移動。這個是預設值。
adOpenKeyset 反映由其它用戶所做的對記錄的改變或者刪除動作,但並不反映由其它用戶做作的添加新記錄的動作。
adOpenDynamic 反映由其它用戶所做的對記錄的改變或者刪除動作,包括添加的新記錄
adOpenStatic 不反映其它用戶對記錄所做的修改,添加,刪除動作。
這四個值VBSCRIPT預定義位
adOpenForwardOnly = 0
adOpenKeyset = 1
adOpenDynamic = 2
adOpenStatic = 3
lockType 表示當打開記錄集時,數據提供者用於鎖定資料庫的類型:
adLockReadOnly 數據不能改變,這是預設值!
adLockPessimistic 數據提供者在開始編輯數據的時候鎖定記錄
adLockOptimistic 僅當調用update方法時,數據提供者鎖定記錄
adLockBatchOptimistic 用於批處理修改
他們的常量值定義分別是:
adLockReadOnly = 1
adLockPessimistic = 2
adLockOptimistic = 3
adLockBatchOptimistic = 4
rs.open sql,conn,1,1 讀取記錄 select
rs.open sql,conn,1,3 只更新記錄最好 update
rs.open sql,conn,2,3 插入和刪除最好 insert delete
測量學
vernier
讀取度盤(或直尺)讀數的一種遊動測微標尺。是早期測繪儀器上常用的測微設備。用於量測主尺上不足一個分劃的零數。其上刻有指標、分劃線並註記分劃線號數。可分:順讀游標和逆讀游標兩種。
讀數原理是:游標上幾個分劃的長度,相當於度盤(或直尺)上(n一1)個分劃的長度,故利用每一分劃值與度盤(或直尺)每一分劃值之差(稱游標最小讀數),可求出度盤(或直尺)上不足一分劃的零數。當其指標沿度盤(或直尺)移動到某一分劃之間時,根據指標位置可讀出度盤(或直尺)上的整數;再找出遊標與度盤(或直尺)相重合的分劃線的號數,乘以游標最小讀數,即得不足一分劃的零數;將零數加上整數,即為指標所指度盤(或直尺)的整個讀數。