zoukankan      html  css  js  c++  java
  • SQL注入:Statement与PrepareStatement的区别,#{}和${}的区别

    Statement:使用字符串拼接的方式

    案例:用户登陆

    需求:

    1)、有一张用户表

    2)、添加几条用户记录

     id int primary key auto_increment,
     name varchar(20),
     password varchar(20)
    )
    insert into user values (null,'jack','123'),(null,'rose','456');
    -- 登录, SQL 中大小写不敏感
    select * from user where name='JACK' and password='123';
    -- 登录失败
    select * from user where name='JACK' and password='333';

    3) 、使用 Statement 字符串拼接的方式实现用户的登录, 用户在控制台上输入用户名和密码。

    步骤:

    1)、得到用户从控制台上输入的用户名和密码来查询数据库

    2)、写一个登录的方法

      a)、通过工具类得到连接

      b)、创建语句对象,使用拼接字符串的方式生成 SQL 语句

      c)、查询数据库,如果有记录则表示登录成功,否则登录失败

      d)、释放资源

    public class Demo7Login {
        //从控制台上输入的用户名和密码
        public static void main(String[] args) {
            Scanner sc = new Scanner(System.in);
            System.out.println("请输入用户名:");
            String name = sc.nextLine();
            System.out.println("请输入密码:");
            String password = sc.nextLine();
            login(name, password);
        }
         /**
         * 登录的方法
         */
         public static void login(String name, String password) {
             //a) 通过工具类得到连接
             Connection connection = null;
             Statement statement = null;
             ResultSet rs = null;
             try {
                 connection = JdbcUtils.getConnection();
                 //b) 创建语句对象,使用拼接字符串的方式生成 SQL 语句
                 statement = connection.createStatement();
                 //c) 查询数据库,如果有记录则表示登录成功,否则登录失败
                 String sql = "select * from user where name='" + name + "' and password='" + password
                + "'";
                 System.out.println(sql);
                 rs = statement.executeQuery(sql);
                 if (rs.next()) {
                 System.out.println("登录成功,欢迎您:" + name);
                 } else {
                 System.out.println("登录失败");
                 }
             } catch (SQLException e) {
                e.printStackTrace();
             } finally {
                 //d) 释放资源
                 JdbcUtils.close(connection, statement, rs);
             }
        } 
    }

    SQL 注入问题

    当我们输入以下密码,我们发现我们账号和密码都不对竟然登录成功了
    请输入用户名:
    newboy
    请输入密码:
    a' or '1'='1
    select * from user where name='newboy' and password='a' or '1'='1'
    登录成功,欢迎您:newboy
    问题分析:
    select * from user where name='newboy' and password='a' or '1'='1'
    name='newboy' and password='a' 为假
    '1'='1' 真
    相当于
    select * from user where true; 查询了所有记录

    我们让用户输入的密码和 SQL 语句进行字符串拼接。用户输入的内容作为了 SQL 语句语法的一部分,改变了原有 SQL 真正的意义,以上问题称为 SQL 注入。要解决 SQL 注入就不能让用户输入的密码和我们的 SQL 语句进行简单的字符串拼接。

    PreparedStatement 将 SQL 语句发送给数据库预编译

    PreparedStatement 是 Statement 接口的子接口,继承于父接口中所有的方法。它是一个预编译的 SQL 语句
     
    PreparedSatement 的执行原理

    1)、因为有预先编译的功能,提高 SQL 的执行效率。

    2)、可以有效的防止 SQL 注入的问题,安全性更高。

    Connection 创建 PreparedStatement 对象

    PreparedSatement 的好处

    1、prepareStatement()会先将 SQL 语句发送给数据库预编译。PreparedStatement 会引用着预编译后的结果。可以多次传入不同的参数给 PreparedStatement 对象并执行。减少 SQL 编译次数,提高效率。

    2、安全性更高,没有 SQL 注入的隐患。

    3、提高了程序的可读性

    使用 PreparedStatement 的步骤:

    1)、编写 SQL 语句,未知内容使用?占位:"SELECT * FROM user WHERE name=? AND password=?";

    2)、获得 PreparedStatement 对象

    3)、设置实际参数:setXxx(占位符的位置, 真实的值)

    4)、执行参数化 SQL 语句

    5)、关闭资源

    使用 PreparedStatement 改写上面的登录程序,看有没有 SQL 注入的情况

    /**
    * 使用 PreparedStatement
    */
    public class Demo8Login {
     //从控制台上输入的用户名和密码
     public static void main(String[] args) throws SQLException {
         Scanner sc = new Scanner(System.in);
         System.out.println("请输入用户名:");
         String name = sc.nextLine();
         System.out.println("请输入密码:");
         String password = sc.nextLine();
         login(name, password);
     }
     /**
     * 登录的方法
     * @param name
     * @param password
     */
     private static void login(String name, String password) throws SQLException {
         Connection connection = JdbcUtils.getConnection();
         //写成登录 SQL 语句,没有单引号
         String sql = "select * from user where name=? and password=?";
         //得到语句对象
         PreparedStatement ps = connection.prepareStatement(sql);
         //设置参数
         ps.setString(1, name);
         ps.setString(2,password);
         ResultSet resultSet = ps.executeQuery();
         if (resultSet.next()) {
            System.out.println("登录成功:" + name);
         }
         else {
            System.out.println("登录失败");
         }
         //释放资源,子接口直接给父接口
         JdbcUtils.close(connection,ps,resultSet);
         } 
     }

    Mybatis使用OGNL表达式解析对象字段的值,#{}或者${}括号中的值为pojo属性名称

    #{}与${}的区别:

    ${} 的作用实际上是字符串拼接,所以要特别小心sql注入问题。

    #{}用到了prepareStement,它将 SQL 语句发送给数据库预编译,安全性更高,没有 SQL 注入的隐患,减少 SQL 编译次数,提高效率。

     
     
     
     
  • 相关阅读:
    论文笔记系列-DARTS: Differentiable Architecture Search
    Win10安装TensorFlow1.9-GPU版本
    论文笔记系列-Efficient Neural Architecture Search via Parameter Sharing
    论文笔记模板
    无偏估计
    【TensorFlow】tf.nn.softmax_cross_entropy_with_logits的用法
    Python学习札记(十三) Function3 函数参数二
    LeetCode Add Two Numbers
    Python学习札记(十二) Function3 函数参数一
    Python学习札记(十一) Function2 函数定义
  • 原文地址:https://www.cnblogs.com/zwh0910/p/14680077.html
Copyright © 2011-2022 走看看