﻿<?xml version="1.0" encoding="utf-8" standalone="yes"?><rss version="2.0" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:trackback="http://madskills.com/public/xml/rss/module/trackback/" xmlns:wfw="http://wellformedweb.org/CommentAPI/" xmlns:slash="http://purl.org/rss/1.0/modules/slash/"><channel><title>BlogJava-海水正蓝-随笔分类-SQL Server</title><link>http://www.blogjava.net/xiaohuzi2008/category/53228.html</link><description>面朝大海，春暖花开</description><language>zh-cn</language><lastBuildDate>Wed, 16 Jan 2013 14:21:11 GMT</lastBuildDate><pubDate>Wed, 16 Jan 2013 14:21:11 GMT</pubDate><ttl>60</ttl><item><title>【转】参数化查询为什么能够防止SQL注入</title><link>http://www.blogjava.net/xiaohuzi2008/archive/2013/01/16/394326.html</link><dc:creator>小胡子</dc:creator><author>小胡子</author><pubDate>Wed, 16 Jan 2013 14:09:00 GMT</pubDate><guid>http://www.blogjava.net/xiaohuzi2008/archive/2013/01/16/394326.html</guid><wfw:comment>http://www.blogjava.net/xiaohuzi2008/comments/394326.html</wfw:comment><comments>http://www.blogjava.net/xiaohuzi2008/archive/2013/01/16/394326.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.blogjava.net/xiaohuzi2008/comments/commentRss/394326.html</wfw:commentRss><trackback:ping>http://www.blogjava.net/xiaohuzi2008/services/trackbacks/394326.html</trackback:ping><description><![CDATA[<div><p>很多人都知道SQL注入，也知道SQL参数化查询可以防止SQL注入，可<strong><span style="color: #ff0000">为什么能防止注入</span></strong>却并不是很多人都知道的。</p>  <p>本文主要讲述的是这个问题，也许你在部分文章中看到过这块内容，当然了看看也无妨。</p>  <p>&nbsp;</p>  <p>首先：我们要了解SQL收到一个指令后所做的事情：</p>  <p>具体细节可以查看文章：<a href="http://blog.csdn.net/babauyang/article/details/7714211"><span style="color: #0000ff">Sql Server 编译、重编译与执行计划重用原理 </span></a></p>  <p>在这里，我简单的表示为： <span style="font-size: large"><strong>收到指令 -&gt; 编译SQL生成执行计划 -&gt;选择执行计划 -&gt;执行执行计划</strong>。</span></p>  <p>具体可能有点不一样，但大致的步骤如上所示。</p>  <p>&nbsp;</p>  <p>接着我们来分析<strong>为什么拼接SQL 字符串会导致SQL注入的风险呢</strong>？</p>  <p>首先创建一张表Users:</p>  <pre>CREATE TABLE [dbo].[Users](  [Id] [uniqueidentifier] NOT NULL,  [UserId] [int] NOT NULL,  [UserName] [varchar](50) NULL,  [Password] [varchar](50) NOT NULL,   CONSTRAINT [PK_Users] PRIMARY KEY CLUSTERED   (  [Id] ASC  )WITH (PAD_INDEX  = OFF, STATISTICS_NORECOMPUTE  = OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS  = ON, ALLOW_PAGE_LOCKS  = ON) ON [PRIMARY]  ) ON [PRIMARY]</pre>   <p><a href="http://images.cnitblog.com/blog/67224/201301/15070752-a7eebda5f28843b09f8ddb5f3a8a2451.jpg"><img style="border-right-width: 0px; display: inline; border-top-width: 0px; border-bottom-width: 0px; border-left-width: 0px" title="3F3ECD42B7A24B139ECA0A7D584CA195" alt="3F3ECD42B7A24B139ECA0A7D584CA195" src="http://images.cnitblog.com/blog/67224/201301/15070753-436bd420a318416896ef5655c20c6451.jpg" border="0" height="134" width="373" /></a></p>  <p>&nbsp;</p>  <p>插入一些数据：</p>  <pre>INSERT INTO [Test].[dbo].[Users]([Id],[UserId],[UserName],[Password])VALUES (NEWID(),1,'name1','pwd1'); INSERT INTO [Test].[dbo].[Users]([Id],[UserId],[UserName],[Password])VALUES (NEWID(),2,'name2','pwd2'); INSERT INTO [Test].[dbo].[Users]([Id],[UserId],[UserName],[Password])VALUES (NEWID(),3,'name3','pwd3'); INSERT INTO [Test].[dbo].[Users]([Id],[UserId],[UserName],[Password])VALUES (NEWID(),4,'name4','pwd4'); INSERT INTO [Test].[dbo].[Users]([Id],[UserId],[UserName],[Password])VALUES (NEWID(),5,'name5','pwd5');</pre>   <p>&nbsp;</p>  <p>假设我们有个用户登录的页面，代码如下：</p>  <p>验证用户登录的sql 如下：</p>  <pre>select COUNT(*) from Users where Password = 'a' and UserName = 'b'&nbsp;</pre>  <p>这段代码返回Password 和UserName都匹配的用户数量，如果大于1的话，那么就代表用户存在。</p>  <p><strong>本文不讨论SQL 中的密码策略，也不讨论代码规范，主要是讲为什么能够防止SQL注入，请一些同学不要纠结与某些代码，或者和SQL注入无关的主题。</strong></p>  <p>&nbsp;</p>  <p>&nbsp;</p>  <p>可以看到执行结果：</p>  <p><a href="http://images.cnitblog.com/blog/67224/201301/15070753-3b926765654b426b85df5c2ef862cae9.jpg"><img style="border-right-width: 0px; display: inline; border-top-width: 0px; border-bottom-width: 0px; border-left-width: 0px" title="15C19A6170754E21A52A79AAA01B9B48" alt="15C19A6170754E21A52A79AAA01B9B48" src="http://images.cnitblog.com/blog/67224/201301/15070754-22351f652be14969af595d2c24916934.jpg" border="0" height="133" width="588" /></a></p>  <p>这个是SQL profile 跟踪的SQL 语句。</p>  <p><a href="http://images.cnitblog.com/blog/67224/201301/15070754-9c3b21b00b2f4f299632f4a60950dd3f.jpg"><img style="border-right-width: 0px; display: inline; border-top-width: 0px; border-bottom-width: 0px; border-left-width: 0px" title="5CB6FB63846740C494C6466FE27D2B3C" alt="5CB6FB63846740C494C6466FE27D2B3C" src="http://images.cnitblog.com/blog/67224/201301/15070755-3fc004dcd8d44187a1726382b6f99545.jpg" border="0" height="98" width="734" /></a></p>  <p>&nbsp;</p>  <p>注入的代码如下：</p>  <pre>select COUNT(*) from Users where Password = 'a' and UserName = 'b' or 1=1&#8212;'</pre>  <p>这里有人将UserName设置为了 &#8220;<span style="font-size: x-large"><strong><em>b' or 1=1 &#8211;</em></strong></span>&#8221;.</p>  <p>&nbsp;</p>  <p>实际执行的SQL就变成了如下：</p>  <p><a href="http://images.cnitblog.com/blog/67224/201301/15070755-f6ca29d19ed9447fa8b2f7cbbde1161b.jpg"><img style="border-right-width: 0px; display: inline; border-top-width: 0px; border-bottom-width: 0px; border-left-width: 0px" title="782A96FEE0784A39B5500CAE267B90EE" alt="782A96FEE0784A39B5500CAE267B90EE" src="http://images.cnitblog.com/blog/67224/201301/15070755-9222f75f6fa8485ebc19156faac5e529.jpg" border="0" height="178" width="674" /></a></p>  <p>&nbsp;</p>  <p><a href="http://images.cnitblog.com/blog/67224/201301/15070756-f58fa71fcb5345e3bf4cf1393a4db2be.jpg"><img style="border-right-width: 0px; display: inline; border-top-width: 0px; border-bottom-width: 0px; border-left-width: 0px" title="5A8FCD361FFE414AB18AEE5C9ED681DE" alt="5A8FCD361FFE414AB18AEE5C9ED681DE" src="http://images.cnitblog.com/blog/67224/201301/15070756-0cc15a288cee48f0a481ab121cc77ac0.jpg" border="0" height="128" width="707" /></a></p>  <p>&nbsp; 可以很明显的看到SQL注入成功了。</p>  <p>&nbsp;</p>  <p>很多人都知道<strong>参数化查询</strong>可以避免上面出现的注入问题，比如下面的代码：</p>  <pre>class Program {     private static string connectionString = "Data Source=.;Initial Catalog=Test;Integrated Security=True";      static void Main(string[] args)     {         Login("b", "a");         Login("b' or 1=1--", "a");     }      private static void Login(string userName, string password)     {         using (SqlConnection conn = new SqlConnection(connectionString))         {             conn.Open();             SqlCommand comm = new SqlCommand();             comm.Connection = conn;             //为每一条数据添加一个参数             comm.CommandText = "select COUNT(*) from Users where Password = @Password and UserName = @UserName";             comm.Parameters.AddRange(             new SqlParameter[]{                                         new SqlParameter("@Password", SqlDbType.VarChar) { Value = password},                 new SqlParameter("@UserName", SqlDbType.VarChar) { Value = userName},             });              comm.ExecuteNonQuery();         }     } }</pre>   <p>&nbsp;</p>  <p>实际执行的SQL 如下所示：</p>  <pre>exec sp_executesql N'select COUNT(*) from Users where Password = @Password and UserName = @UserName',N'@Password varchar(1),@UserName varchar(1)',@Password='a',@UserName='b'</pre>  <pre>exec sp_executesql N'select COUNT(*) from Users where Password = @Password and UserName = @UserName',N'@Password varchar(1),@UserName varchar(11)',@Password='a',@UserName='b'' or 1=1&#8212;'</pre>  <pre>&nbsp;</pre>  <pre>&nbsp;</pre>  <pre>&nbsp;</pre>   <p>可以看到参数化查询主要做了这些事情：</p>  <pre>1：<strong>参数过滤</strong>，可以看到 @UserName='b'' or 1=1&#8212;'</pre>  <pre>2：<strong><span style="color: #ff0000; font-size: x-large">执行计划重用</span></strong></pre>  <p>&nbsp;</p>  <p><span style="font-family: courier new"><strong>因为执行计划被重用，所以可以防止SQL注入。</strong></span></p>  <p>&nbsp;</p>  <p><span style="font-family: courier new">首先分析SQL注入的本质，</span></p>  <p><span style="font-family: courier new">用户写了一段SQL 用来表示查找密码是a的，用户名是b的所有用户的数量。</span></p>  <p><span style="font-family: courier new">通过注入SQL，这段SQL现在表示的含义是查找(密码是a的，并且用户名是b的，) 或者1=1 的所有用户的数量。</span></p>  <p>&nbsp;</p>  <p><span style="font-family: courier new; color: #ff0000"><strong>可以看到SQL的语意发生了改变，为什么发生了改变呢？，因为没有重用以前的执行计划，因为对注入后的SQL语句重新进行了编译，因为重新执行了语法解析。所以要保证SQL语义不变，即我想要表达SQL就是我想表达的意思，不是别的注入后的意思，就应该重用执行计划。</strong></span></p>  <p>&nbsp;</p>  <p><span style="font-family: courier new"><strong>如果不能够重用执行计划，那么就有SQL注入的风险，因为SQL的语意有可能会变化，所表达的查询就可能变化。</strong></span></p>  <p>&nbsp;</p>  <p><span style="font-family: courier new">在SQL Server 中查询执行计划可以使用下面的脚本：</span></p>  <pre>DBCC FreeProccache  select total_elapsed_time / execution_count 平均时间,total_logical_reads/execution_count 逻辑读, usecounts 重用次数,SUBSTRING(d.text, (statement_start_offset/2) + 1,          ((CASE statement_end_offset            WHEN -1 THEN DATALENGTH(text)           ELSE statement_end_offset END              - statement_start_offset)/2) + 1) 语句执行 from sys.dm_exec_cached_plans a cross apply sys.dm_exec_query_plan(a.plan_handle) c ,sys.dm_exec_query_stats b cross apply sys.dm_exec_sql_text(b.sql_handle) d --where a.plan_handle=b.plan_handle and total_logical_reads/execution_count&gt;4000 ORDER BY total_elapsed_time / execution_count DESC;</pre>  <pre>&nbsp;</pre>   <p><a href="http://images.cnitblog.com/blog/67224/201301/15071207-c27987e24254493b997b87412db81da9.jpg"><img style="border-bottom: 0px; border-left: 0px; display: inline; border-top: 0px; border-right: 0px" title="18EFAED775BF4DB9A36C57B39EC6913D" alt="18EFAED775BF4DB9A36C57B39EC6913D" src="http://images.cnitblog.com/blog/67224/201301/15071209-8a6a588dbbf34670b43ed03678e21b1a.jpg" border="0" height="479" width="783" /></a></p>  <p>&nbsp;</p>  <p>博客园有篇文章： <a href="http://www.cnblogs.com/lzrabbit/archive/2012/04/22/2465313.html">Sql Server参数化查询之where in和like实现详解</a></p>  <p>&nbsp;</p>  <p>在这篇文章中有这么一段：</p>  <p><a href="http://images.cnitblog.com/blog/67224/201301/15070800-b97ddd6a59764a8b9a2d6feb5f988aff.png"><img style="border-right-width: 0px; display: inline; border-top-width: 0px; border-bottom-width: 0px; border-left-width: 0px" title="image" alt="image" src="http://images.cnitblog.com/blog/67224/201301/15070801-0420da9fe7314389a853d256fbaea524.png" border="0" height="421" width="741" /></a></p>  <p>&nbsp;</p>  <p>这里作者有一句话：&#8221;<span style="color: #ff0000">不过这种写法和直接拼SQL执行没啥实质性的区别</span>&#8221;</p>  <p>任何拼接SQL的方式都有SQL注入的风险，所以如果没有实质性的区别的话，那么使用exec 动态执行SQL是不能防止SQL注入的。</p>  <p>&nbsp;</p>  <p>比如下面的代码：</p>  <pre>private static void TestMethod() {     using (SqlConnection conn = new SqlConnection(connectionString))     {         conn.Open();         SqlCommand comm = new SqlCommand();         comm.Connection = conn;         //使用exec动态执行SQL　         //实际执行的查询计划为(@UserID varchar(max))select * from Users(nolock) where UserID in (1,2,3,4)　　         //不是预期的(@UserID varchar(max))exec('select * from Users(nolock) where UserID in ('+@UserID+')')             comm.CommandText = "exec('select * from Users(nolock) where UserID in ('+@UserID+')')";         comm.Parameters.Add(new SqlParameter("@UserID", SqlDbType.VarChar, -1) { Value = "1,2,3,4" });         //comm.Parameters.Add(new SqlParameter("@UserID", SqlDbType.VarChar, -1) { Value = "1,2,3,4); delete from Users;--" });         comm.ExecuteNonQuery();     } }</pre>   <p>&nbsp;</p>  <p>执行的SQL 如下：</p>  <pre>exec sp_executesql N'exec(''select * from Users(nolock) where UserID in (''+@UserID+'')'')',N'@UserID varchar(max) ',@UserID='1,2,3,4'</pre>   <pre><a href="http://images.cnitblog.com/blog/67224/201301/15070801-59e2484ae08f4286addd72738aa659ed.jpg"><img style="border-right-width: 0px; display: inline; border-top-width: 0px; border-bottom-width: 0px; border-left-width: 0px" title="D25E99E053D549AF955518AD0A320259" alt="D25E99E053D549AF955518AD0A320259" src="http://images.cnitblog.com/blog/67224/201301/15070803-0171ae9011f143269397408e36b3c539.jpg" border="0" height="428" width="670" /></a></pre>  <pre>&nbsp;</pre>  <pre>可以看到SQL语句并没有参数化查询。</pre>  <pre>&nbsp;</pre>  <pre>如果你将UserID设置为&#8221;</pre>  <p><span style="color: #ff0000">1,2,3,4); delete from Users;&#8212;-</span></p>  <pre>&#8221;,那么执行的SQL就是下面这样：</pre>  <pre>exec sp_executesql N'exec(''select * from Users(nolock) where UserID in (''+@UserID+'')'')',N'@UserID varchar(max) ',@UserID='1,2,3,4); delete from Users;--'</pre>  <p>&nbsp;</p>  <p>不要以为加了个@UserID 就代表能够防止SQL注入，实际执行的SQL 如下：</p>  <p>&nbsp;</p>  <pre><a href="http://images.cnitblog.com/blog/67224/201301/15070804-9db4343666884db082512e3c2c96597e.jpg"><img style="border-right-width: 0px; display: inline; border-top-width: 0px; border-bottom-width: 0px; border-left-width: 0px" title="3C50EFE68418448496BAC7773067AB6F" alt="3C50EFE68418448496BAC7773067AB6F" src="http://images.cnitblog.com/blog/67224/201301/15070806-6c83fe149f0c4951ab6833e9bacfbf43.jpg" border="0" height="431" width="677" /></a></pre>  <pre>&nbsp;</pre>  <pre>任何动态的执行SQL 都有注入的风险，因为动态意味着不重用执行计划，而如果不重用执行计划的话，那么就基本上无法保证你写的SQL所表示的意思就是你要表达的意思。</pre>  <pre>&nbsp;</pre>  <pre>这就好像小时候的填空题，查找密码是(____) 并且用户名是(____)的用户。</pre>  <pre>不管你填的是什么值，我所表达的就是这个意思。</pre>  <pre>&nbsp;</pre>  <pre>最后再总结一句：因为参数化查询可以重用执行计划，并且如果重用执行计划的话，SQL所要表达的语义就不会变化，所以就可以防止SQL注入,如果不能重用执行计划，就有可能出现SQL注入，<br />存储过程也是一样的道理，因为可以重用执行计划。</pre></div>原文出自：<div>http://www.cnblogs.com/LoveJenny/archive/2013/01/15/2860553.html</div><img src ="http://www.blogjava.net/xiaohuzi2008/aggbug/394326.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.blogjava.net/xiaohuzi2008/" target="_blank">小胡子</a> 2013-01-16 22:09 <a href="http://www.blogjava.net/xiaohuzi2008/archive/2013/01/16/394326.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item></channel></rss>