sql 2008r2存储过程怎么打断点 R2存储过程中怎么输出调试日志信息
一、sql server 2008中存储过程怎么解密
SQLServer2005里使用with encryption选项创建的存储过程仍然和sqlserver2000里一样,都是使用XOR进行了的加密。和2000不一样的是,在2005的系统表syscomments里已经查不到加密过的密文了。要查密文必须使用DAC(专用管理员连接)连接到数据库后,在系统表 sys.sysobjvalues查询,该表的列imageval存储了相应的密文。具体可以使用下面的查询:
SELECT imageval FROM sys.sysobjvalues WHERE objid= object_id(@procedure) AND
valclass= 1 AND subobjid= 1
针对SQL Server 2005的解密存储过程
create PROCEDURE [dbo].[sp__windbi$decrypt]
(@procedure sysname= NULL,@revfl int= 1)
AS
/*
王成辉翻译整理,转贴请注明出自微软BI开拓者www.windbi.com
目前这个存储过程只能解密存储过程,至于解密函数、触发器、视图的存储过程本网站会进一步关注,调用形式为:
exec dbo.sp__windbi$decrypt@procedure,0
如果第二个参数使用1的话,会给出该存储过程的一些提示。
--版本2.0
*/
SET NOCOUNT ON
IF@revfl= 1
BEGIN
PRINT'警告:该存储过程会删除并重建原始的存储过程。'
PRINT'在运行该存储过程之前确保你的数据库有一个备份。'
PRINT'该存储过程通常应该运行在产品环境的一个备份的非产品环境下。'
PRINT'为了运行这个存储过程,将参数@refl的值更改为0。'
RETURN 0
END
DECLARE@intProcSpace bigint,@t bigint,@maxColID smallint,@intEncrypted
tinyint,@procNameLength int
select@maxColID= max(subobjid),@intEncrypted= imageval FROM
sys.sysobjvalues WHERE objid= object_id(@procedure)
GROUP BY imageval
--select@maxColID as'Rows in sys.sysobjvalues'
select@procNameLength= datalength(@procedure)+ 29
DECLARE@real_01 nvarchar(max)
DECLARE@fake_01 nvarchar(max)
DECLARE@fake_encrypt_01 nvarchar(max)
DECLARE@real_decrypt_01 nvarchar(max),@real_decrypt_01a nvarchar(max)
declare@objtype varchar(2),@ParentName nvarchar(max)
select@real_decrypt_01a=''
--提取对象的类型如是存储过程还是函数,如果是触发器,还要得到其父对象的名称
select@objtype=type,@parentname=object_name(parent_object_id)
from sys.objects where [object_id]=object_id(@procedure)
--从sys.sysobjvalues里提出加密的imageval记录
SET@real_01=(SELECT top 1 imageval FROM sys.sysobjvalues WHERE objid=
object_id(@procedure) and valclass= 1 order by subobjid)
--创建一个临时表
create table#output( [ident] [int] IDENTITY(1, 1) NOT NULL,
[real_decrypt] NVARCHAR(MAX))
--开始一个事务,稍后回滚
BEGIN TRAN
--更改原始的存储过程,用短横线替换
if@objtype='P'
SET@fake_01='ALTER PROCEDURE'+@procedure+' WITH ENCRYPTION AS
'+REPLICATE('-', 40003-@procNameLength)
else if@objtype='FN'
SET@fake_01='ALTER FUNCTION'+@procedure+'() RETURNS INT WITH ENCRYPTION AS BEGIN RETURN 1
/*'+REPLICATE('*', datalength(@real_01)/2-@procNameLength)+'*/ END'
else if@objtype='V'
SET@fake_01='ALTER view'+@procedure+' WITH ENCRYPTION AS select 1 as col
/*'+REPLICATE('*', datalength(@real_01)/2-@procNameLength)+'*/'
else if@objtype='TR'
SET@fake_01='ALTER trigger'+@procedure+' ON'+@parentname+' WITH ENCRYPTION AFTER INSERT AS RAISERROR(''N'',16,10)
/*'+REPLICATE('*', datalength(@real_01)/2-@procNameLength)+'*/'
EXECUTE(@fake_01)
--从sys.sysobjvalues里提出加密的假的
SET@fake_encrypt_01=(SELECT top 1 imageval FROM sys.sysobjvalues WHERE objid=
object_id(@procedure) and valclass= 1 order by subobjid)
if@objtype='P'
SET@fake_01='Create PROCEDURE'+@procedure+' WITH ENCRYPTION AS
'+REPLICATE('-', 40003-@procNameLength)
else if@objtype='FN'
SET@fake_01='CREATE FUNCTION'+@procedure+'() RETURNS INT WITH ENCRYPTION AS BEGIN RETURN 1
/*'+REPLICATE('*', datalength(@real_01)/2-@procNameLength)+'*/ END'
else if@objtype='V'
SET@fake_01='Create view'+@procedure+' WITH ENCRYPTION AS select 1 as col
/*'+REPLICATE('*', datalength(@real_01)/2-@procNameLength)+'*/'
else if@objtype='TR'
SET@fake_01='Create trigger'+@procedure+' ON'+@parentname+' WITH ENCRYPTION AFTER INSERT AS RAISERROR(''N'',16,10)
/*'+REPLICATE('*', datalength(@real_01)/2-@procNameLength)+'*/'
--开始计数
SET@intProcSpace=1
--使用字符填充临时变量
SET@real_decrypt_01= replicate(N'A',(datalength(@real_01)/2))
--循环设置每一个变量,创建真正的变量
--每次一个字节
SET@intProcSpace=1
--如有必要,遍历每个@real_xx变量并解密
WHILE@intProcSpace<=(datalength(@real_01)/2)
BEGIN
--真的和假的和加密的假的进行异或处理
SET@real_decrypt_01= stuff(@real_decrypt_01,@intProcSpace, 1,
NCHAR(UNICODE(substring(@real_01,@intProcSpace, 1)) ^
(UNICODE(substring(@fake_01,@intProcSpace, 1)) ^
UNICODE(substring(@fake_encrypt_01,@intProcSpace, 1)))))
SET@intProcSpace=@intProcSpace+1
END
--通过sp_helptext逻辑向表#output里插入变量
insert#output(real_decrypt) select@real_decrypt_01
-- select real_decrypt AS'#output chek' from#output--测试
---------------------------------------
--开始从sp_helptext提取
---------------------------------------
declare@dbname sysname
,@BlankSpaceAdded int
,@BasePos int
,@CurrentPos int
,@TextLength int
,@LineId int
,@AddOnLen int
,@LFCR int--回车换行的长度
,@DefinedLength int
,@SyscomText nvarchar(4000)
,@Line nvarchar(255)
Select@DefinedLength= 255
SELECT@BlankSpaceAdded= 0--跟踪行结束的空格。注意Len函数忽略了多余的空格
CREATE TABLE#CommentText
(LineId int
,Text nvarchar(255) collate database_default)
--使用#output代替sys.sysobjvalues
DECLARE ms_crs_syscom CURSOR LOCAL
FOR SELECT real_decrypt from#output
ORDER BY ident
FOR READ ONLY
--获取文本
SELECT@LFCR= 2
SELECT@LineId= 1
OPEN ms_crs_syscom
FETCH NEXT FROM ms_crs_syscom into@SyscomText
WHILE@@fetch_status>= 0
BEGIN
SELECT@BasePos= 1
SELECT@CurrentPos= 1
SELECT@TextLength= LEN(@SyscomText)
WHILE@CurrentPos!= 0
BEGIN
--通过回车查找行的结束
SELECT@CurrentPos= CHARINDEX(char(13)+char(10),@SyscomText,
@BasePos)
--如果找到回车
IF@CurrentPos!= 0
BEGIN
--如果@Lines的长度的新值比设置的大就插入@Lines目前的内容并继续
While(isnull(LEN(@Line),0)+@BlankSpaceAdded+
@CurrentPos-@BasePos+@LFCR)>@DefinedLength
BEGIN
SELECT@AddOnLen=@DefinedLength-(isnull(LEN(@Line),0)+
@BlankSpaceAdded)
INSERT#CommentText VALUES
(@LineId,
isnull(@Line, N'')+ isnull(SUBSTRING(@SyscomText,
@BasePos,@AddOnLen), N''))
SELECT@Line= NULL,@LineId=@LineId+ 1,
@BasePos=@BasePos+@AddOnLen,@BlankSpaceAdded= 0
END
SELECT@Line= isnull(@Line, N'')+
isnull(SUBSTRING(@SyscomText,@BasePos,@CurrentPos-@BasePos+@LFCR), N'')
SELECT@BasePos=@CurrentPos+2
INSERT#CommentText VALUES(@LineId,@Line)
SELECT@LineId=@LineId+ 1
SELECT@Line= NULL
END
ELSE
--如果回车没找到
BEGIN
IF@BasePos<=@TextLength
BEGIN
--如果@Lines长度的新值大于定义的长度
While(isnull(LEN(@Line),0)+@BlankSpaceAdded+
@TextLength-@BasePos+1)>@DefinedLength
BEGIN
SELECT@AddOnLen=@DefinedLength-
(isnull(LEN(@Line),0)+@BlankSpaceAdded)
INSERT#CommentText VALUES
(@LineId,
isnull(@Line, N'')+ isnull(SUBSTRING(@SyscomText,
@BasePos,@AddOnLen), N''))
SELECT@Line= NULL,@LineId=@LineId+ 1,
@BasePos=@BasePos+@AddOnLen,@BlankSpaceAdded=
0
END
SELECT@Line= isnull(@Line, N'')+
isnull(SUBSTRING(@SyscomText,@BasePos,@TextLength-@BasePos+1), N'')
if LEN(@Line)<@DefinedLength and charindex('',
@SyscomText,@TextLength+1)> 0
BEGIN
SELECT@Line=@Line+'',@BlankSpaceAdded= 1
END
END
END
END
FETCH NEXT FROM ms_crs_syscom into@SyscomText
END
IF@Line is NOT NULL
INSERT#CommentText VALUES(@LineId,@Line)
select Text from#CommentText order by LineId
CLOSE ms_crs_syscom
DEALLOCATE ms_crs_syscom
DROP TABLE#CommentText
---------------------------------------
--结束从sp_helptext提取
---------------------------------------
--删除用短横线创建的存储过程并重建原始的存储过程
ROLLBACK TRAN
DROP TABLE#output
二、SQL Server的存储过程怎么写
SQL server中如何存储:
首先准备数据,测试存储过程
use ssqadm;
创建测试books表
create table books_test( book_id int identity(1,1) primary key,
book_name varchar(20),book_price float,book_auth varchar(10));
插入测试数据
insert into books_test(book_name,book_price,book_auth)values
('论语',25.6,'孔子'),
('天龙八部',25.6,'金庸'),
('雪山飞狐',32.7,'金庸'),
('平凡的世界',35.8,'路遥'),
('史记',54.8,'司马迁');
select* from books_test;*/
创建无参存储过程
if(exists(select* from sys.objects where name='getAllBooks'))
drop proc getAllBooks
go
create procedure getAllBooks
as
begin
select* from books_test;
调用,执行存储过程
exec getAllBooks;
end
go
修改存储过程
alter procedure getallbooks
as
select book_name from books_test;
修改存储过程的名称
sp_rename getallbooks,proc_get_allbooks;
go
exec proc_get_allbooks;
go
创建带参数的存储过程
use ssqadm
go
if(exists(select* from sys.objects where name='searchbooks'))
drop proc searchbooks
exec searchbooks
执行存储searchbooks得到如下结果:
go
create procedure searchbooks(@bookid int)--括号里面是
as
begin
declare@book_id int;定义一个标量变量,只是保证存储过程的完整性,在本存储是多此一举的。
set@book_id=@bookid;
select* from books_test where book_id=@book_id;
end;
go
-- exec searchbooks
执行存储searchbooks得到如下结果:
创建带两个参数的存储过程
use ssqadm
go
if(exists(select* from sys.objects where name='book_test2'))
drop proc book_test2
exec book_test2
执行存储book_test2得到如下结果:
go
create procedure book_test2
(@bookid int,@bookname varchar(20))括号里面是
as
begin
declare@book_id int;
定义一个标量变量,只是保证存储过程的完整性,在本存储是多此一举的。
declare@book_name varchar(20);
set@book_id=@bookid;
set@book_name=@bookname;
select* from books_test where book_id=
@book_id and book_name=@book_name;
end;
go
exec book_test2
扩展资料:
SQL Server中查询存储命令子句:
USE [SSQADM]
Use是跳转到哪个数据库,对这个数据库进行操作。
GO
GO向 SQL Server实用工具发出一批 Transact-SQL语句结束的信号,相当于提交上面的SQL语句。
GO是把t-sql语句分批次执行
(一步成功了才会执行下一步,即一步一个GO)
/****** Object: StoredProcedure [dbo].[PROC_four_five_hr]
Script Date: 07/30/2018 13:44:55******/
SET ANSI_NULLS ONGOSET QUOTED_IDENTIFIER ON
三、SQL***Server***2008***R2存储过程中怎么输出调试日志信息
使用PRINT一般是写程序时观察中间结果,对于你说的这种情况,可以直接用SELECT输出结果,在存储过程里也可以这样。如果是在函数或者触发器里,可以建个测试表,如果是表值函数则可以定义一个表变量。
如果一定要print,那么这样吧:
delcare@number int
declare@course nvarchar(30)--for example
select@number=number,@course=course from choice where studentnumber='20100001'
print@number
print@course
这是楼上的朋友的劳动成果,望勿见怪
如果一定要PRINT全部的,那么用游标吧:
declare@info varchar(200)
declare@curs cursor
set@curs=cursor scroll dynamic
for
select'number='+convert(varchar(20),number)+'; course='+convert(varchar(20),course) from choice where sex='F'
open@curs
fetch next from@curs into@info
while(@@fetch_status=0)
begin
print@info
fetch next from@curs into@info
end