c# 剔除sql语句'尾巴'的五种方法

一:背景

1. 讲故事

这几天都在修复bug真的太忙了,期间也遇到了一个挺有趣bug,和大家分享一下,这是一块sql挺复杂的报表相关业务,不知道哪一位大佬在错综复杂的 嵌套 + 平行 if判断中sql拼接在某些UI组合下出问题了,最终的 sql 架构类似这样的。

var sql = "select 1 union all select 2 union all select 3 union all";

这种sql到数据库去肯定是报错的,有些朋友可能想说这还不简单,在相关的 if 判断中不要追加这个 union all 就好了,这确实是一个根治的办法,但现实情况这一块的业务太复杂了,也不太敢改里面的代码,改的没问题还好,改出问题你得兜着走,所以最保险的办法就是怎么去掉 union all 这个大尾巴,所以我干脆思考了一会,想出了如下五种办法。

二:剔除 union all 的五大方式

1. 最原始的 for 循环

最简单的办法就是通过 for 循环搞定,我可以倒序判断最后几个字符是不是关键词 union all 就可以了,如下代码所示:

    static void Main(string[] args)
    {
      var sql = "select 1 union all select 2 union all select 3 union all";

      var keyword = "union all";

      var isall = true;
      int i = 0;

      for (i = 1; i <= keyword.Length; i++)
      {
        if (keyword[keyword.Length - i] != sql[sql.Length - i])
        {
          isall = false;
          break;
        }
      }

      if (isall)
      {
        var query = sql.Substring(0, sql.Length - i + 1);

        Console.WriteLine(query);
      }
    }

从代码中可以看出,只要在倒序的过程中,有一个字符和 keyword 中的不符,那就直接跳出,否则就是全匹配,拿到最后的 i 进行 Substring 截取即可。

2. 使用 Substring 搞定

第一种方式确实可以实现,但实现的并不轻松,毕竟大家都是用 C# 写代码而不是 C,为了这点小功能写了这么多代码,显得太 low 了,所以尽量能用类库的方法就用类库的方法吧,改进措施很简单,可以从 sql 尾部切出 keyword.length 个字符,也就是: start:sql.length - keyword.length,然后判断一下是否和 keyword 相等即可,代码修改如下:

    static void Main(string[] args)
    {
      var sql = "select 1 union all select 2 union all select 3 union all";

      var keyword = "union all";

      var isSucc = sql.Substring(sql.Length - keyword.Length) == keyword;

      if (isSucc)
      {
        var query = sql.Substring(0, sql.Length - keyword.Length);

        Console.WriteLine(query);
      }
    }

3. 使用 LastIndexOf

第二种方式写出来的代码确实比较简洁,但大家有没有发现一个问题,我为了获取最后的 string 做了两次 substring 操作,也就是说在托管堆中生成了两个 string 对象,那能不能免掉一个 substring 呢?给 gc 减轻一些负担,这就可以用到 LastIndexOf 方法了,代码如下:

    static void Main(string[] args)
    {
      var sql = "select 1 union all select 2 union all select 3 union all";

      var keyword = "union all";

      var index = sql.LastIndexOf(keyword);

      if (sql.Length - index == keyword.Length)
      {
        var query = sql.Substring(0, index);

        Console.WriteLine(query);
      }
    }

思想很简单,就是判断最后出现的 union all的位置到尾部的距离 是否恰好和 keyword.length 一致,如果是的话那 keyword 就是 sql 的大尾巴,这里的 if 写的有点难懂,其实还可以使用 EndsWith 再优化一下代码:

    static void Main(string[] args)
    {
      var sql = "select 1 union all select 2 union all select 3 union all";

      var keyword = "union all";

      if (sql.EndsWith(keyword))
      {
        var query = sql.Substring(0, sql.Length - keyword.Length);
        Console.WriteLine(query);
      }
    }

4. 使用 Split 切割

前面几种方式都是在 string 上做文章,要么 substring,要么 LastIndexOf,要么 EndsWith,其实也可以跳出这个定势思维,转换成数组进行处理,用 union all 作为分隔符切割字符串,如果数组的最后一个元素为 string.Empty,那就表明 sql 尾巴就是 keyword, 对吧,代码修改如下:

    static void Main(string[] args)
    {
      var sql = "select 1 union all select 2 union all select 3 union all";

      var keyword = "union all";

      var arr = sql.Split("union all");

      if (string.IsNullOrEmpty(arr[arr.Length - 1]))
      {
        var query = string.Join(keyword, arr.Take(arr.Length - 1));

        Console.WriteLine(query);
      }
    }

5. 使用 TrimEnd

相信很多朋友用这个方法的场景大多在于剔除尾部的空格,哈哈,其实它还有一个隐藏功能,不仅可以剔除空格,还可以剔除任意多个指定的字符,这就🐂👃了,不信的话可以看看 TrimEnd 方法签名即可:

  public sealed class String
  {
    public string TrimEnd(params char[] trimChars)
    {
      if (trimChars == null || trimChars.Length == 0)
      {
        return TrimHelper(1);
      }
      return TrimHelper(trimChars, 1);
    }
  }

可以看到 trimChars 是一个字符数组,你可以灌入你想剔除的任意多个字符,有了这个思想,我可以将 keyword 转成 char[] 再灌入到 TrimEnd 即可,代码如下:

    static void Main(string[] args)
    {
      var sql = "select 1 union all select 2 union all select 3 union all";

      var keyword = "union all";

      var query = sql.TrimEnd(keyword.ToArray());

      Console.WriteLine(query);
    }

可以看出已成功剔除,此时我很好奇底层到底是怎么实现的,源码如下:

    private string TrimHelper(char[] trimChars, int trimType)
    {
      int num = Length - 1;
      int i = 0;
      if (trimType != 1)
      {
        for (i = 0; i < Length; i++)
        {
          int num2 = 0;
          char c = this[i];
          for (num2 = 0; num2 < trimChars.Length && trimChars[num2] != c; num2++)
          {
          }
          if (num2 == trimChars.Length)
          {
            break;
          }
        }
      }
      if (trimType != 0)
      {
        for (num = Length - 1; num >= i; num--)
        {
          int num3 = 0;
          char c2 = this[num];
          for (num3 = 0; num3 < trimChars.Length && trimChars[num3] != c2; num3++)
          {
          }
          if (num3 == trimChars.Length)
          {
            break;
          }
        }
      }
      return CreateTrimmedString(i, num);
    }

    private string CreateTrimmedString(int start, int end)
    {
      int num = end - start + 1;
      if (num == Length)
      {
        return this;
      }
      if (num == 0)
      {
        return Empty;
      }
      return InternalSubString(start, num);
    }

从源码中可以看出,它的代码逻辑是从后往前筛查,一直到遇见的字符不在 char[] 中为止,这里就有一个潜在的坑,大家要注意了,比如下面sql的结果:

    static void Main(string[] args)
    {
      var sql = "select * from english_l union all";

      var keyword = "union all";

      var query = sql.TrimEnd(keyword.ToArray());

      Console.WriteLine(query);
    }

四:总结

五种方式各有利弊,不管是简单粗暴的,基于性能的,灵活巧妙的,都能达到最终的目的,暂时就想到这5种,脑仁已经疼了😂😂😂, 更多好玩的写法,欢迎大家留言讨论哈!

以上就是c# 剔除sql语句'尾巴'的五种方法的详细内容,更多关于c# sql语句的资料请关注我们其它相关文章!

(0)

相关推荐

  • C#实现连接SQL Server2012数据库并执行SQL语句的方法

    本文实例讲述了C#实现连接SQL Server2012数据库并执行SQL语句的方法.分享给大家供大家参考,具体如下: 开发工具:Visual Studio 2012 数据库: SQL Server 2012 使用Visual Studio时还是直接和微软自家的SQL Server数据库连接比较方便,就像使用Eclipse时和MySQL连接便捷一样的道理 无论使用什么工具步骤都一样: 1. 首先保证相关工具都已经正确安装了 2. 开启数据库连接服务 3. 在开发工具中通过用户名和口令与数据库进行关

  • c#实现用SQL池,多线程定时批量执行SQL语句的方法

    在实际项目开发中,业务逻辑层的处理速度往往很快,特别是在开发Socket通信服务的时候,网络传输很快,但是一旦加上数据库操作,性能一落千丈,数据库操作的效率往往成为一个系统整体性能的瓶颈.面对这问题,我们怎么办呢?好,下面我就为大家介绍一种方法:构建SQL池,分离业务逻辑层和数据访问层,让业务逻辑层从低效的数据库操作解脱,以提高系统整体性能. (一)SQL池 SQL池是SQL容器,用于存放业务逻辑层抛过来的SQL语句.SQL池主要提供以下几种方法: 1)internal string Pop()

  • C# 启用事务提交多条带参数的SQL语句实例代码

    具体代码如下所示: /// <summary> /// 启用事务提交多条带参数的SQL语句 /// </summary> /// <param name="mainSql">主表SQL</param> /// <param name="mainParam">主表对应的参数</param> /// <param name="detailSql">明细表SQL语句&l

  • C#中实现一次执行多条带GO的sql语句实例

    本文实例讲述了C#中实现一次执行多条带GO的sql语句.分享给大家供大家参考.具体如下: using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Collections; using System.Text.RegularExpressions; using System.Data.SqlClient; namespace ConsoleApplicati

  • C#中验证sql语句是否正确(不执行语句)

    SET PARSEONLY检查每个 Transact-SQL 语句的语法并返回任何错误消息,但不编译和执行语句.SET PARSEONLY { ON | OFF }当 SET PARSEONLY 为 ON 时,SQL Server 只分析语句.当 SET PARSEONLY 为 OFF 时,SQL Server 编译并执行语句.SET PARSEONLY 的设置是在分析时设置,而不是在执行或运行时设置.在存储过程或触发器中不要使用 PARSEONLY.如果 OFFSETS 选项为 ON 而且没有

  • C#拼接SQL语句 用ROW_NUMBER实现的高效分页排序

    如果项目中要用到数据库,铁定要用到分页排序.之前在做数据库查询优化的时候,通宵写了以下代码,来拼接分页排序的SQL语句 复制代码 代码如下: /// <summary> /// 单表(视图)获取分页SQL语句 /// </summary> /// <param name="tableName">表名或视图名</param> /// <param name="key">唯一键</param> //

  • C#调用SQL语句时乘号的用法

    一般来说,在ADO中,我们需要在SQL语句中使用乘法运算,可是添加'*'以后执行程序总是会出错,这是因为'*'与sql中的'*'关键字重合了,因此编译就会出错. 解决办法为:将乘法运算放到sql语句外面,将结果放入一个变量中,然后在sql语句中引用即可. 如下示例所示: int count = (this.AspNetPager1.CurrentPageIndex-1)*this.AspNetPager1.PageSize; string sql = " select top "+co

  • C#实现String字符串转化为SQL语句中的In后接的参数详解

    实现把String字符串转化为In后可用参数代码: public string StringToList(string aa) { string bb1 = "("; if (!string.IsNullOrEmpty(aa.Trim())) { string[] bb = aa.Split(new string[] { "\r\n", ",", ";", "* " }, StringSplitOption

  • C#使用带like的sql语句时防sql注入的方法

    本文实例叙述了在拼接sql语句的时候,如果遇到Like的情况该怎么办. 一般采用带like的SQL语句进行简单的拼接字符串时,需要开率遇到sql注入的情况.这确实是个需要注意的问题. 这里结合一些查阅的资料做了初步的整理. 如这样一个sql语句: select * from game where gamename like '%张三%' 用c#表示的话: string keywords = "张三"; StringBuilder strSql=new StringBuilder();

  • c# 剔除sql语句'尾巴'的五种方法

    一:背景 1. 讲故事 这几天都在修复bug真的太忙了,期间也遇到了一个挺有趣bug,和大家分享一下,这是一块sql挺复杂的报表相关业务,不知道哪一位大佬在错综复杂的 嵌套 + 平行 if判断中sql拼接在某些UI组合下出问题了,最终的 sql 架构类似这样的. var sql = "select 1 union all select 2 union all select 3 union all"; 这种sql到数据库去肯定是报错的,有些朋友可能想说这还不简单,在相关的 if 判断中不

  • 浅谈基于SQL Server分页存储过程五种方法及性能比较

    在SQL Server数据库操作中,我们常常会用到存储过程对实现对查询的数据的分页处理,以方便浏览者的浏览. 创建数据库data_Test : create database data_Test GO use data_Test GO create table tb_TestTable --创建表 ( id int identity(1,1) primary key, userName nvarchar(20) not null, userPWD nvarchar(20) not null, u

  • PHP判断数组是否为空的常用方法(五种方法)

    本文介绍了PHP开发中遇到的数组问题,这里介绍了判断PHP数组为空的5种方法,有需要的朋友可以借鉴参考一下. 1. isset功能:判断变量是否被初始化 说明:它并不会判断变量是否为空,并且可以用来判断数组中元素是否被定义过 注意:当使用isset来判断数组元素是否被初始化过时,它的效率比array_key_exists高4倍左右 <?php $a = ''; $a['c'] = ''; if (!isset($a)) echo '$a 未被初始化' . ""; if (!iss

  • JavaScript数组去重的五种方法

    javascript数组去重是一个比较常见的需求,解决方法也有很多种,网上都可以找到答案的,下面小编给大家整理了一份关于同类型的数组去重的方法,先给大家介绍下简单实现思路. 思路: 遍历数组,一一比较,比较到相同的就删除后面的 遍历数组,一一比较,比较到相同的,跳过前面重复的,不相同的放入新数组 任取一个数组元素放入新数组,遍历剩下的数组元素任取一个,与新数组的元素一一比较,如果有不同的,放入新数组. 遍历数组,取一个元素,作为对象的属性,判断属性是否存在 1. 删除后面重复的: functio

  • java执行SQL语句实现查询的通用方法详解

    完成SQL查询 并将查询结果放入Vector容器,以便其他程序使用 /* * 执行sql查询语句 */ public static <T> Vector<T> executeQuery(Class<T> clazz, String sql, Object... args) { Connection conn = null; PreparedStatement preparedstatement = null; ResultSet rs = null; Vector<

  • SQL server分页的4种方法示例(很全面)

    这篇博客讲的是SQL server的分页方法,用的SQL server 2012版本.下面都用pageIndex表示页数,pageSize表示一页包含的记录.并且下面涉及到具体例子的,设定查询第2页,每页含10条记录. 首先说一下SQL server的分页与MySQL的分页的不同,mysql的分页直接是用limit (pageIndex-1),pageSize就可以完成,但是SQL server 并没有limit关键字,只有类似limit的top关键字.所以分页起来比较麻烦. SQL serve

  • python 模块重载的五种方法

    环境准备 新建一个 foo 文件夹,其下包含一个 bar.py 文件 $ tree foo foo └── bar.py 0 directories, 1 file bar.py 的内容非常简单,只写了个 print 语句 print("successful to be imported") 只要 bar.py 被导入一次,就被执行一次 print 禁止重复导入 由于有 sys.modules 的存在,当你导入一个已导入的模块时,实际上是没有效果的. >>> from

  • Java字符串拼接的五种方法及性能比较分析(从执行100次到90万次)

    目录 > 源代码,供参考 > 测试结果: > 查看源代码,以及简单分析 > 字符串拼接一般使用"+",但是"+"不能满足大批量数据的处理,Java中有以下五种方法处理字符串拼接,各有优缺点,程序开发应选择合适的方法实现. 1. 加号 "+" 2. String contact() 方法 3. StringUtils.join() 方法 4. StringBuffer append() 方法 5. StringBuilder

  • SQL server分页的四种方法思路详解(最全面教程)

    目录 方法一:三重循环 思路 方法二:利用max(主键) 方法三:利用row_number关键字 第四种方法:offset /fetch next(2012版本及以上才有) 总结   这篇博客讲的是SQL server的分页方法,用的SQL server 2012版本.下面都用pageIndex表示页数,pageSize表示一页包含的记录.并且下面涉及到具体例子的,设定查询第2页,每页含10条记录.   首先说一下SQL server的分页与MySQL的分页的不同,mysql的分页直接是用lim

  • SQL数据去重的3种方法实例详解

    目录 1.使用distinct去重 2.使用group by 3.使用ROW_NUMBER() OVER 或 GROUP BY 和 COLLECT_SET/COLLECT_LIST 3.1 ROW_NUMBER() OVER 3.2 GROUP BY 和 COLLECT_SET/COLLECT_LIST distinct与group by的去重方面的区别 使用去重distinct方法的示例详解 总结 1.使用distinct去重 distinct用来查询不重复记录的条数,用count(disti

随机推荐