存儲過程是一組預先編譯好的sql語句。將他放在服務器上面,由用戶通過指定存儲過程的名字來執行它。
存儲過程的優點:
封裝——可用于操作數據庫對象的方法,用戶只需要知道它的輸入輸出參數并理解其目的即可。
改善性能——已經預先編譯
減少網絡流量——只返回最后的結果集
重要性——針對復雜邏輯,應用已經測試號的存儲過程,不容易發生錯誤。
安全性——如果數據庫擁有者DBO或者系統管理員SA編譯并保存了存儲結構,存儲過程就有了對它使用的數據庫對象的所有訪問權限。因此系統管理員可以向單獨的用于授予對數據對象的最小訪問權限,而不是直接允許用戶使用數據庫對象。
這次項目開發中運用了不少的存儲過程,中間遇到了一個小問題,大家可以看看:
事情是這樣的:我想根據卡號、教師編號、教師姓名、聯系電話這四個條件查詢教師編號,這四個條件可以不存在,可以只存在一個,可以只存在兩個,可以只存在三個,可以存在四個。開始我寫的存儲過程是這樣的
- <span style="font-size:24px;">CREATE PROCEDURE [dbo].[procCardByMore]</span>
- @cardNo varchar (50),
- @TeacherID varchar(50),
- @Name varchar(50),
- @telephone varchar(50)
- AS
- DECLARE @sql varchar(1000)
- SET @sql='SELECT TeacherID as 教師編號,Name as 教師姓名,GradeName as 所屬年級,cardNo as 卡號,cash as 卡內余額,realName as 管理員,OpenTime as 開卡時間,state as 狀態,telephone as 聯系電話 FROM view_CardDetail '
-
- IF(@cardNo <>'')
- BEGIN
- SET @sql=@sql+'and cardNo ='''+@cardNo+''''
- END
-
- IF(@TeacherID <>'')
- BEGIN
- SET @sql=@sql+'and TeacherID ='''+@TeacherID+''''
- END
-
- IF(@Name <>'')
- BEGIN
- SET @sql=@sql+'and Name ='''+@Name +''''
- END
-
- IF(@Name <>'')
- BEGIN
- SET @sql=@sql+'and telephone ='''+@telephone +''''
- END
-
- exec(@sql)
這里面假如卡號不為空的話,那么寫出來的sql語句是這樣的:
- SELECT TeacherID as 教師編號,Name as 教師姓名,GradeName as 所屬年級,cardNo as 卡號,cash as 卡內余額,realName as 管理員,OpenTime as 開卡時間,state as 狀態,telephone as 聯系電話FROM view_CardDetail and cardNo ='@cardNo’
這句話顯然是錯誤的,在查詢語句和條件的過渡的地方沒有where,而是突然來了個and。然而這個時候如果加上個必然條件那就完美了,接下來的存儲過程創建語句是這樣的:
- CREATE PROCEDURE [dbo].[procCardByMore]
- @cardNo varchar (50),
- @TeacherID varchar(50),
- @Name varchar(50),
- @telephone varchar(50)
- AS
- DECLARE @sql varchar(1000)
- SET @sql='SELECT TeacherID as 教師編號,Name as 教師姓名,GradeName as 所屬年級,cardNo as 卡號,cash as 卡內余額,realName as 管理員,OpenTime as 開卡時間,state as 狀態,telephone as 聯系電話FROM view_CardDetail WHERE 1=1 '
-
- IF(@cardNo <>'')
- BEGIN
- SET @sql=@sql+'and cardNo ='''+@cardNo+''''
- END
-
- IF(@TeacherID <>'')
- BEGIN
- SET @sql=@sql+'and TeacherID ='''+@TeacherID+''''
- END
-
- IF(@Name <>'')
- BEGIN
- SET @sql=@sql+'and Name ='''+@Name +''''
- END
-
- IF(@Name <>'')
- BEGIN
- SET @sql=@sql+'and telephone ='''+@telephone +''''
- END
-
- exec(@sql)
這樣的話執行出來的結果是這樣的:
- SELECT TeacherID as 教師編號,Nameas 教師姓名,GradeName as 所屬年級,cardNo as 卡號,cashas 卡內余額,realName as 管理員,OpenTime as 開卡時間,stateas 狀態,telephone as 聯系電話FROM view_CardDetail WHERE 1=1and cardNo ='+@cardNo+'
必然條件在這里充當了橋梁的作用,雖然我們沒有“1=1“的需求,但是它的存在卻恰好解決了兩者互不相容的尷尬局面。
受上面的啟發,下面的情況是這樣解決的:
需要根據教師編號和起始日期查詢教師的消費記錄情況,兩個條件的存在情況各自隨便。
存儲過程代碼如下:
- CREATE PROCEDURE [dbo].[procConsumeByMore]
- @TeacherID varchar(50),
- @TimeStart datetime,
- @TimeEnd datetime
- AS
- DECLARE @sql varchar(1000)
- SET @sql='SELECT TeacherID as 教師編號,Name as 教師姓名,orderID as 訂單編號,adultNum as 成人數量,childNum as 兒童數量,consumeCash as 消費金額,consumeTime as 消費時間,userID as 管理員編號,realName as 管理員姓名FROM view_consumeDetail WHERE 1=1 '
-
-
- IF(@TeacherID <>'')
- BEGIN
- SET @sql=@sql+'and TeacherID ='''+@TeacherID+''''
- END
-
- IF( @TimeStart<>null and @TimeEnd<>null)
- BEGIN
- SET @sql=@sql+'and consumeTime between '''+@TimeStart +''' and '''+@TimeEnd+''''
- END
-
- exec(@sql)
這里當@TimeStart和@TimeEnd為空的時候,空字符串不能轉換成日期類型,這時候假如兩個變量為空,我可以給他們賦一個完全的值:
- exec procConsumeByMore '', '1799-01-01','9999-12-29'
如上面的代碼就可以選擇出所有的消費記錄了。
上面兩件事情中必然事件這個配角起了不可忽視的作用,然而必然事件卻不是我們在查詢過程中需要的條件。
生活中配角并不意味著無趣和呆板,配角一樣可以靈動,只是它把對別人的善意放在第一位,不把張揚作為表達自己的方式。‘上善若水,水善利萬物而不爭’,但是水又無處不在,其實就是這樣一種狀態。創造一個最好的配角,并不比創造一個主角容易。