博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
防SQL注入:生成参数化的通用分页查询语句
阅读量:4358 次
发布时间:2019-06-07

本文共 4751 字,大约阅读时间需要 15 分钟。

原文:

      前些时间看了玉开兄的“”这篇文章,才突然想起某个项目也是使用了累似的通用分页存储过程。使用这种通用的存储过程进行分页查询,想要防SQL注入,只能对输入的参数进行过滤,例如将一个单引号“'”转换成两个单引号“''”,但这种做法是不安全的,厉害的黑客可以通过编码的方式绕过单引号的过滤,要想有效防SQL注入,只有参数化查询才是最终的解决方案。但问题就出在这种通用分页存储过程是在存储过程内部进行SQL语句拼接,根本无法修改为参数化的查询语句,因此这种通用分页存储过程是不可取的。但是如果不用通用的分页存储过程,则意味着必须为每个具体的分页查询写一个分页存储过程,这会增加不少的工作量。

      经过几天的时间考虑之后,想到了一个用代码来生成参数化的通用分页查询语句的解决方案。代码如下:

ContractedBlock.gif
ExpandedBlockStart.gif
Code
    public class PagerQuery
ExpandedBlockStart.gifContractedBlock.gif    
{
        
private int _pageIndex;
        
private int _pageSize = 20;
        
private string _pk;
        
private string _fromClause;
        
private string _groupClause;
        
private string _selectClause;
        
private string _sortClause;
        
private StringBuilder _whereClause;
        
public DateTime DateFilter = DateTime.MinValue;
        
protected QueryBase()
ExpandedSubBlockStart.gifContractedSubBlock.gif        
{
            _whereClause 
= new StringBuilder();
        }
ExpandedSubBlockStart.gifContractedSubBlock.gif        
/**//// <summary>
        
/// 主键
        
/// </summary>
        public string PK
ExpandedSubBlockStart.gifContractedSubBlock.gif        
{
ExpandedSubBlockStart.gifContractedSubBlock.gif            
get return _pk; }
ExpandedSubBlockStart.gifContractedSubBlock.gif            
set { _pk = value; }
        }
        
public string SelectClause
ExpandedSubBlockStart.gifContractedSubBlock.gif        
{
ExpandedSubBlockStart.gifContractedSubBlock.gif            
get return _selectClause; }
ExpandedSubBlockStart.gifContractedSubBlock.gif            
set { _selectClause = value; }
        }
        
public string FromClause
ExpandedSubBlockStart.gifContractedSubBlock.gif        
{
ExpandedSubBlockStart.gifContractedSubBlock.gif            
get return _fromClause; }
ExpandedSubBlockStart.gifContractedSubBlock.gif            
set { _fromClause = value; }
        }
        
public StringBuilder WhereClause
ExpandedSubBlockStart.gifContractedSubBlock.gif        
{
ExpandedSubBlockStart.gifContractedSubBlock.gif            
get return _whereClause; }
ExpandedSubBlockStart.gifContractedSubBlock.gif            
set { _whereClause = value; }
        }
        
public string GroupClause
ExpandedSubBlockStart.gifContractedSubBlock.gif        
{
ExpandedSubBlockStart.gifContractedSubBlock.gif            
get return _groupClause; }
ExpandedSubBlockStart.gifContractedSubBlock.gif            
set { _groupClause = value; }
        }
        
public string SortClause
ExpandedSubBlockStart.gifContractedSubBlock.gif        
{
ExpandedSubBlockStart.gifContractedSubBlock.gif            
get return _sortClause; }
ExpandedSubBlockStart.gifContractedSubBlock.gif            
set { _sortClause = value; }
        }
ExpandedSubBlockStart.gifContractedSubBlock.gif        
/**//// <summary>
        
/// 当前页数
        
/// </summary>
        public int PageIndex
ExpandedSubBlockStart.gifContractedSubBlock.gif        
{
ExpandedSubBlockStart.gifContractedSubBlock.gif            
get return _pageIndex; }
ExpandedSubBlockStart.gifContractedSubBlock.gif            
set { _pageIndex = value; }
        }
ExpandedSubBlockStart.gifContractedSubBlock.gif        
/**//// <summary>
        
/// 分页大小
        
/// </summary>
        public int PageSize
ExpandedSubBlockStart.gifContractedSubBlock.gif        
{
ExpandedSubBlockStart.gifContractedSubBlock.gif            
get return _pageSize; }
ExpandedSubBlockStart.gifContractedSubBlock.gif            
set { _pageSize = value; }
        }
ExpandedSubBlockStart.gifContractedSubBlock.gif        
/**//// <summary>
        
/// 生成缓存Key
        
/// </summary>
        
/// <returns></returns>
        public override string GetCacheKey()
ExpandedSubBlockStart.gifContractedSubBlock.gif        
{
            
const string keyFormat = "Pager-SC:{0}-FC:{1}-WC:{2}-GC:{3}-SC:{4}";
            
return string.Format(keyFormat, SelectClause, FromClause, WhereClause, GroupClause, SortClause);
        }
ExpandedSubBlockStart.gifContractedSubBlock.gif        
/**//// <summary>
        
/// 生成查询记录总数的SQL语句
        
/// </summary>
        
/// <returns></returns>
        public string GenerateCountSql()
ExpandedSubBlockStart.gifContractedSubBlock.gif        
{
            StringBuilder sb 
= new StringBuilder();
            sb.AppendFormat(
" from {0}", FromClause);
            
if (WhereClause.Length > 0)
                sb.AppendFormat(
" where 1=1 {0}", WhereClause);
            
if (!string.IsNullOrEmpty(GroupClause))
                sb.AppendFormat(
" group by {0}", GroupClause);
            
return string.Format("Select count(0) {0}", sb);
        }
ExpandedSubBlockStart.gifContractedSubBlock.gif        
/**//// <summary>
        
/// 生成分页查询语句,包含记录总数
        
/// </summary>
        
/// <returns></returns>
        public string GenerateSqlIncludeTotalRecords()
ExpandedSubBlockStart.gifContractedSubBlock.gif        
{
            StringBuilder sb 
= new StringBuilder();
            
if (string.IsNullOrEmpty(SelectClause))
                SelectClause 
= "*";
            
if (string.IsNullOrEmpty(SortClause))
                SortClause 
= PK;
            
int start_row_num = (PageIndex - 1)*PageSize + 1;
            sb.AppendFormat(
" from {0}", FromClause);
            
if (WhereClause.Length > 0)
                sb.AppendFormat(
" where 1=1 {0}", WhereClause);
            
if (!string.IsNullOrEmpty(GroupClause))
                sb.AppendFormat(
" group by {0}", GroupClause);
            
string countSql = string.Format("Select count(0) {0};", sb);
            
string tempSql =
                
string.Format(
                    
"WITH t AS (SELECT ROW_NUMBER() OVER(ORDER BY {0}) as row_number,{1}{2}) Select * from t where row_number BETWEEN {3} and {4};",
                    SortClause, SelectClause, sb, start_row_num, (start_row_num 
+ PageSize - 1));
            
return tempSql + countSql;
        }
ExpandedSubBlockStart.gifContractedSubBlock.gif        
/**//// <summary>
        
/// 生成分页查询语句
        
/// </summary>
        
/// <returns></returns>
        public override string GenerateSql()
ExpandedSubBlockStart.gifContractedSubBlock.gif        
{
            StringBuilder sb 
= new StringBuilder();
            
if (string.IsNullOrEmpty(SelectClause))
                SelectClause 
= "*";
            
if (string.IsNullOrEmpty(SortClause))
                SortClause 
= PK;
            
int start_row_num = (PageIndex - 1)*PageSize + 1;
            sb.AppendFormat(
" from {0}", FromClause);
            
if (WhereClause.Length > 0)
                sb.AppendFormat(
" where 1=1 {0}", WhereClause);
            
if (!string.IsNullOrEmpty(GroupClause))
                sb.AppendFormat(
" group by {0}", GroupClause);
            
return
                
string.Format(
                    
"WITH t AS (SELECT ROW_NUMBER() OVER(ORDER BY {0}) as row_number,{1}{2}) Select * from t where row_number BETWEEN {3} and {4}",
                    SortClause, SelectClause, sb, start_row_num, (start_row_num 
+ PageSize - 1));
        }
    }

 

使用方法:

 

PagerQuery query = new PagerQuery();

query.PageIndex = 1;
    query.PageSize = 20;
    query.PK = "ID";
    query.SelectClause = "*";
    query.FromClause = "TestTable";
    query.SortClause = "ID DESC";

    if (!string.IsNullOrEmpty(code))

    {
     query.WhereClause.Append(" and ID= @ID");
    }

a) GenerateCountSql ()方法生成的语句为:

Select count(0) from TestTable Where 1=1 and ID= @ID

b) GenerateSql()方法生成的语句为:

WITH t AS (SELECT ROW_NUMBER() OVER(ORDER BY ECID DESC) as row_number, * from TestTable where 1=1 and ID= @ID) Select * from t where row_number BETWEEN 1 and 20

c) GenerateSqlIncludetTotalRecords()方法生成的语句为:

WITH t AS (SELECT ROW_NUMBER() OVER(ORDER BY E.ECID DESC) as row_number,* from TestTable where 1=1 and ID= @ID) Select * from t where row_number BETWEEN 1 and 20;Select count(0) from ECBasicInfo where 1=1 and ID= @ID;

注意:以上代码生成的SQL语句是曾对SQL SERVER 2005以上版本的,希望这些代码对大家有用

posted on
2015-01-05 10:59 阅读(
...) 评论(
...)

转载于:https://www.cnblogs.com/lonelyxmas/p/4203049.html

你可能感兴趣的文章
Android使用Fragment程序崩溃
查看>>
codevs 2822 爱在心中(强连通分量)
查看>>
七:python 对象类型详解三:列表
查看>>
c语言基本数据类型相关
查看>>
SQL Server DATEADD() 函数
查看>>
makefile中的wildcard和patsubst
查看>>
F#基础教程 mutable关键字
查看>>
完全卸载MySQL数据库
查看>>
C#总结项目《影院售票系统》编写总结一
查看>>
Failed to stop iptables.service: Unit iptables.service not loaded.
查看>>
madpaly 移植到 TQ2440 遇到问题madplay not found (2)
查看>>
LISTVIEW显示JPEG缩略图
查看>>
YII中引用自定义类
查看>>
Unity 基础
查看>>
Python字符串切片操作知识详解
查看>>
利用python批量修改word文件名的方法示例
查看>>
【洛谷 P3227】 [HNOI2013]切糕(最小割)
查看>>
web.xml中监听器配置
查看>>
Leeo 智能夜灯:默默守护你的家
查看>>
MVC4网站发布到windows server 2003服务器
查看>>