解决SQL注入问题

  • 只要用户提供的信息不参与sql语句的编译过程,那么尽管用户输入的信息中含有sql关键字那么也不会起作用了

  • 要想使用户提供信息不参与sql语句的编译过程,那么必须使用 java.sql.PreparedStatement 预编译的数据库操作对象,提前编译sql语句框架

  • PreparedStatement 继承了 Statement

  • PreparedStatement的原理是预先对sql语句框架进行编译,然后再给sql语句传值。

  • 将Statement(数据库操作对象) 换为 PreparedStatement(预编译的数据库操作对象)

  • 获取预编译数据库连接对象preparedStatement,将sql语句框架预先进行编译。 其中SQL语句中的 一个 ?表示一个占位符,占位符不能用单引号括起来。

    • preparedStatement=connection.prepareStatement("select * from t_user where loginName=? and loginPwd=? "); 使用 ?进行占位。
  • 给占位符 ? 传值 注意:第一个问号的下标是1 ,第二个问号下标是 2

      - preparedStatement.setString(1,loginName);   第一个问号传   loginName
    
      - preparedStatement.setString(2,loginPwd);    第二个问号传   loginPwd
  • 代码实例:

           
    
         import java.sql.*;
    import java.util.HashMap;
    import java.util.Map;
    import java.util.Scanner; /**
    * 解决sql注入问题
    * 只要用户提供的信息不参与sql语句的编译过程,那么尽管用户输入的信息中含有sql关键字那么也不会起作用了
    * 要想使用户提供信息不参与sql语句的编译过程,那么必须使用java.sql. PreparedStatement 预编译的数据库操作对象
    * PreparedStatement 继承了 Statement
    * PreparedStatement的原理是预先对sql语句框架进行编译,然后再给sql语句传值
    * 将Statement(数据库操作对象) 换为 PreparedStatement(预编译的数据库操作对象)
    *
    *
    */ public class 解决sql注入 {
    public static void main(String[] args) throws SQLException { //掉用initUI()获取用户名密码
    Map<String,String> userLoginInfo=initUI(); //调用register()方法。
    //boolean registerSuccess=register(userLoginInfo); //输出结果
    //System.out.println(registerSuccess?"注册成功":"注册失败"); //调用logIn()方法。
    boolean loginSuccess=login(userLoginInfo);
    System.out.println(loginSuccess?"登陆成功":"密码错误"); } /**
    * 用户注册
    * @param userLoginInfo 用户登录信息
    * @return true成功, false失败
    */ private static boolean register(Map<String, String> userLoginInfo) {
    //创建连接对象
    Connection connection=null;
    Statement statement=null;
    int count=0; try {
    //注册驱动
    Class.forName("com.mysql.cj.jdbc.Driver"); //获取连接
    connection= DriverManager.getConnection("jdbc:mysql://127.0.0.1:3306/mydb?useUnicode=true&characterEncoding=utf8&serverTimezone=Asia/Shanghai",
    "root","123456"); //获取数据库操作对象
    statement=connection.createStatement(); //执行SQL语句
    count=statement.executeUpdate("insert into t_user(loginName,loginPwd,realName)values('"+userLoginInfo.get("loginName")+"','"+userLoginInfo.get("loginPwd")+"',' ') "); } catch (ClassNotFoundException | SQLException e) {
    e.printStackTrace();
    }finally {
    if(statement!=null){
    try {
    statement.close();
    } catch (SQLException e) {
    e.printStackTrace();
    }
    } if(connection!=null){
    try {
    connection.close();
    } catch (SQLException e) {
    e.printStackTrace();
    }
    }
    }
    if(count==1){
    return true;
    }
    return false;
    } /**
    * 用户登录
    * @param userLoginInfo 用户信息
    * @return 成功true 失败fslse
    */ private static boolean login(Map<String, String> userLoginInfo) throws SQLException { //标识
    boolean flag=false; //获取用户信息
    String loginName=userLoginInfo.get("loginName");
    String loginPwd=userLoginInfo.get("loginPwd"); //创建连接对象
    Connection connection=null;
    PreparedStatement preparedStatement=null; //这里使用PreparedStatement(预编译的数据库操作对象)
    ResultSet resultSet=null; try {
    //注册驱动
    Class.forName("com.mysql.cj.jdbc.Driver"); //获取连接
    connection=DriverManager.getConnection("jdbc:mysql://127.0.0.1:3306/mydb?useUnicode=true&characterEncoding=utf8&serverTimezone=Asia/Shanghai",
    "root","123456"); //获取预编译数据库连接对象,将sql语句框架预先进行编译。其中一个?表示一个占位符,占位符不能用单引号括起来
    preparedStatement=connection.prepareStatement("select * from t_user where loginName=? and loginPwd=?"); //使用?进行占位 //给占位符? 传值 注意:第一个问号的下标是1 ,第二个问号下标是 2
    preparedStatement.setString(1,loginName); //第一个问号传 loginName
    preparedStatement.setString(2,loginPwd); //第二个问号传 loginPwd //执行SQL语句 使用 PreparedStatement,执行sql的使用,不用再传入sql语句,因为前面已经编译过了。
    resultSet=preparedStatement.executeQuery(); //处理查询结果集
    if(resultSet.next()){
    flag=true;
    }
    } catch (ClassNotFoundException | SQLException e) {
    e.printStackTrace();
    }finally {
    if(resultSet!=null){
    resultSet.close();
    }
    if(preparedStatement!=null){
    preparedStatement.close();
    }
    if (connection!=null){
    connection.close();
    }
    } return flag;
    } /**
    * 初始化用户界面
    * @return 返回用户登录信息
    */
    private static Map<String, String> initUI() { //创建用户信息接收对象
    Scanner scanner=new Scanner(System.in);
    System.out.println("请输入您的用户名:");
    String userName=scanner.nextLine(); System.out.println("请输入您的密码:");
    String pwd=scanner.nextLine(); //创建一个Map集合用来存放用户输入得用户名和密码
    Map<String,String> userLoginInfo=new HashMap<String,String>();
    userLoginInfo.put("loginName",userName);
    userLoginInfo.put("loginPwd",pwd);
    return userLoginInfo;
    }
    }

JDBC_08_解决SQL注入问题 (登录和注册)的更多相关文章

  1. MySQL_(Java)使用preparestatement解决SQL注入的问题

    MySQL_(Java)使用JDBC向数据库发起查询请求 传送门 MySQL_(Java)使用JDBC创建用户名和密码校验查询方法 传送门 MySQL数据库中的数据,数据库名garysql,表名gar ...

  2. jdbc 07: 解决sql注入

    jdbc连接mysql,解决sql注入问题 package com.examples.jdbc.o7_解决sql注入; import java.sql.*; import java.util.Hash ...

  3. IBatis.Net使用总结(一)-- IBatis解决SQL注入(#与$的区别)

    IBatis解决SQL注入(#与$的区别) 在IBatis中,我们使用SqlMap进行Sql查询时,需要引用参数,在参数引用中可以使用两种占位符#和$.这两种占位符有什么区别呢? (1):#***#, ...

  4. 利用SQL注入漏洞登录后台的实现方法

    利用SQL注入漏洞登录后台的实现方法 作者: 字体:[增加 减小] 类型:转载 时间:2012-01-12我要评论 工作需要,得好好补习下关于WEB安全方面的相关知识,故撰此文,权当总结,别无它意.读 ...

  5. PreparedStatement解决sql注入问题

    总结 PreparedStatement解决sql注入问题 :sql中使用?做占位符 2.得到PreparedStatement对象 PreparedStatement pst=conn.prepar ...

  6. 使用过滤器解决SQL注入和跨站点脚本编制

    1 SQL注入.盲注 1.1 SQL注入.盲注概述 Web 应用程序通常在后端使用数据库,以与企业数据仓库交互.查询数据库事实上的标准语言是 SQL(各大数据库供应商都有自己的不同版本).Web 应用 ...

  7. 解决 SQL 注入的另类方法

    本文是翻译,版权归原作者所有 原文地址(original source):https://bitcoinrevolt.wordpress.com/2016/03/08/solving-the-prob ...

  8. MyBatis是如何解决Sql注入的

    转:[转]mybatis如何防止sql注入 java中预处理PrepareStatement为什么能起到防止SQL注入的作用??!! 一.SQL注入 sql注入大家都不陌生,是一种常见的攻击方式,攻击 ...

  9. 防止SQL注入的登录页面

    <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN""http://www.w3.org/T ...

随机推荐

  1. ffmpeg第4篇:为视频添加动态水印

    动态分为三种: 水印本身变化 水印显示时间变化 水印位置变化 水印本身变化 看过上一篇的小伙伴可能觉得第一种很简单,把jpg格式的图片换成gif格式的不就可以了吗,然而亲自试一下就会发现,把gif图片 ...

  2. springboot框架里的pom.xml文件里的m不显示,只有标红和<>符号的解决方法

    这是因为没有把pom.xml文件加入到maven工程中,所以需要如图所示 亲测有效,原文链接:https://blog.csdn.net/qq_41026946/article/details/107 ...

  3. Centos8.2安装Mongodb4.4.2(社区版)

    1:下载 wget https://fastdl.mongodb.org/linux/mongodb-linux-x86_64-rhel80-4.4.2.tgz 官网地址: 2:解压 tar -zxv ...

  4. ERROR 1040 (HY000) Too many connections

    C:\Users\Jilil>mysql -u root -pEnter password: *************ERROR 1040 (HY000): Too many connecti ...

  5. docker mysql数据备份xtrabackup

    一.概述 线上有一个mysql,是在docker里面运行的. 关于docker封装mysql镜像,请参考链接:https://www.cnblogs.com/xiao987334176/p/11984 ...

  6. springboot整合jsp,完成公交车站路线图

    转: springboot整合jsp,完成公交车站路线图 点赞再看,养成习惯 开发环境: jdk 8 intellij idea tomcat 8 mysql 5.7 maven 3.6 所用技术: ...

  7. LeetCode-宝石与石头

    宝石与石头 LeetCode-771 使用哈希表. 这里使用内置算法库中的map /** * 给定字符串J 代表石头中宝石的类型,和字符串 S代表你拥有的石头.  * S 中每个字符代表了一种你拥有的 ...

  8. IT求职 非技术面试题汇总

    原文链接:https://blog.csdn.net/weixin_40845165/article/details/89852397 说明:原文是浏览网页时无意间看到的.扫了一眼,总结得还不错,感谢 ...

  9. C# 应用 - 使用 WebClient 发起 Http 请求

    1. 需要的库类 \Reference Assemblies\Microsoft\Framework\.NETFramework\v4.7.2\System.dll System.Net.WebCli ...

  10. C语言编程 菜鸟练习100题(21-30)

    [练习21]计算自然数的和 0. 题目: 计算自然数的和 1. 分析: 练习使用 for 循环结构.for 循环允许一个执行指定次数的循环控制结构. 2. 程序: #include <stdio ...