SQL註入與防範

来源:https://www.cnblogs.com/dudududu/archive/2018/03/10/8539157.html
-Advertisement-
Play Games

首先給大家看個例子: 1)小編首先在資料庫中建立了一張測試表logintable,表內有一條測試信息: 然後寫了個測試程式: 輸出結果為: 2)然後我們修改main()方法中,login()方法調用時的參數,改為: 執行結果: 3)似乎一切都天經地義,沒什麼問題。但是這時我們再對login()方法調 ...


首先給大家看個例子:

1)小編首先在資料庫中建立了一張測試表logintable,表內有一條測試信息:

然後寫了個測試程式:

package com.java.SqlInject;

import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;

public class SqlInject {

    private static String Driver="com.mysql.jdbc.Driver"; //資料庫驅動
    //連接資料庫的URL地址
    private static String url="jdbc:mysql://localhost:3306/hellojdbc?useUnicode=true&characterEncoding=UTF-8";
    private static String username="root";//資料庫連接用戶名
    private static String password="123456";//資料庫連接密碼
    
    private static Connection conn=null;//資料庫連接對象
    private static Statement stat=null;//語句陳述對象
    private static ResultSet rs=null;//結果數據集
    private static PreparedStatement pst=null;//預編譯語句
    
    
    //使用靜態塊的方式載入驅動
        static {
            try {
                //調用Class對象的靜態forName()方法載入資料庫驅動類
                Class.forName(Driver);
            } catch (ClassNotFoundException e) {
                e.printStackTrace();
            }
            
        }
        
        //使用單例模式返回資料庫連接對象
        public static Connection getConnection() throws SQLException{
            if(conn==null){
                conn=DriverManager.getConnection(url, username, password);
                return conn;
            }
            return conn;

        } 
        
        
        
        public static void login(String name,String password){
            try {
                conn=getConnection();
                stat=conn.createStatement();
                //使用動態拼接的方式拼接sql語句
                rs=stat.executeQuery("select * from logintable where name='"+name+"' and password='"+password+"'");
                if(rs.next()){
                    System.out.println("用戶已註冊");
                }else
                    System.out.println("無記錄");
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }
        
        
        
    
    public static void main(String[] args) {
        login("zhangsan","123456");

    }

}

輸出結果為:

2)然後我們修改main()方法中,login()方法調用時的參數,改為:

public static void main(String[] args) {
        login("zhangsan","123");

    }

執行結果:

3)似乎一切都天經地義,沒什麼問題。但是這時我們再對login()方法調用的參數做一下修改:

public static void main(String[] args) {
        //註意:第一個參數兩個短杠後面有空格
        login("zhangsan';-- ","123");

    }

測試執行結果又變成了:

明明用戶名和密碼都不對,資料庫中也沒有這條記錄,為什麼會出現這種情況?

這就是SQL註入帶來的漏洞問題

惡意用戶通過偽裝請求,來騙過我們的業務程式,達到獲取資料庫核心數據的目的。

通過上面的例子,我們可以看到我們最後一次改寫參數來調用login()方法的時候,java業務程式中接收到的不是我們期望的那個sql語句。

由於分號的存在,使得我們的sql語句拼接後完變成了這樣:

select * from logintable where name='zhangsan';
--  'password='123';

這樣就由一條sql語句變成了兩條sql語句。在第一條的sql語句中去掉了密碼的檢索條件,同時註釋掉了第二條sql語句。

(兩個橫線為註釋符)

總結一下:

SQL註入就是用戶在輸入表單或者URL參數中輸入SQL命令,到達欺騙應用程式的目的,破壞原有SQL的語義,發送惡意的SQL到後端資料庫,導致資料庫信息出現泄露的漏洞。

發生這種漏洞的原因是:

我們的sql語句是通過動態拼接組成的,在拼接完成之前,sql語句是不完整的,所以當在拼接時新加入的參數中有sql命令的註入就有可能改變原有的sql語義。

比方像上面的例子中,密碼被惡意地屏蔽註釋掉了。

解決方法:

傳入外部參數時,不使用動態拼接的方式拼接sql語句;使用參數化方式的sql實現方式(格式化,占位符),即使用預編譯的statement。

然後傳參:

所以上面例子中相關代碼應修改為:

                conn=getConnection();
                //stat=conn.createStatement();
                //使用組合的方式拼接sql語句
                //rs=stat.executeQuery("select * from logintable where name='"+name+"' and password='"+password+"'");
                
                pst=conn.prepareStatement("select * from logintable where name= ? and password= ?");
                pst.setString(1, name);
                pst.setString(2, password);
                rs=pst.executeQuery();
                
                if(rs.next()){
                    System.out.println("用戶已註冊");
                }else
                    System.out.println("無記錄");

其他註意事項:


 

使用嚴格的資料庫管理許可權:

1.僅給予Web應用訪問資料庫的最小許可權;

2.避免Drop table等許可權。

封裝資料庫錯誤:

1.禁止直接將後端資料庫異常信息暴露給用戶;

2.對後端異常信息進行必要的封裝,避免用戶直接查看到後端異常。

機密信息禁止明文存儲:

1.涉密信息需要加密處理;

2.使用AES_ENCRYPT/AES_DECRYPT加密和解密。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 查找最大的N個元素——堆數據結構 給出序列,求出TopK大的元素,使用小頂堆,heapq模塊實現 ...
  • 第一種:點對點 第二種: 發佈者/訂閱者 啟動順序:先訂閱、再發佈 ...
  • 函數:就是讓程式模塊化,把具有獨立功能的代碼塊當成一個整體封裝成一個函數 首先列印一個佛主看看: 那麼這個時候我們想列印很多遍佛主的話,總不能一直複製粘貼,所以我們需要定義一個列印佛主函數: 帶有參數的函數 首先我們來定義一個函數來計算兩個數的總和: 這樣會不會有點太死板了,我們希望能夠從鍵盤輸入a ...
  • 什麼情況下使用ActiveMQ? 1 多個項目之間集成 (1) 跨平臺 (2) 多語言 (3) 多項目 2 降低系統間模塊的耦合度,解耦 軟體擴展性 3 系統前後端隔離 前後端隔離,屏蔽高安全區 安裝步驟: 第一步:安裝jdk,因為activemq依賴jdk來運行 請參照: Linux中安裝jdk ...
  • 安裝pyenv$ git clone git://github.com/yyuu/pyenv.git ~/.pyenv $ echo 'export PYENV_ROOT="$HOME/.pyenv"' >> ~/.bashrc $ echo 'export PATH="$PYENV_ROOT/bi... ...
  • 在python中,有個好用的模塊linecache,該模塊允許從任何文件里得到任何的行,並且使用緩存進行優化,常見的情況是從單個文件讀取多行。linecache.getline(filename,lineno)從名為filename的文件中得到第lineno行示例:從final.txt文件中讀取數據 ...
  • Description 有一棵二叉樹,最大深度為D,且所有的葉子深度都相同。所有結點從上到下從左到右編號為1,2,3,…,2eD-1。在結點1處放一個小球,它會往下落。每個結點上都有一個開關,初始全部關閉,當每次有小球落到一個開關上時,它的狀態都會改變。當小球到達一個內結點時,如果該結點的開關關閉, ...
  • 也是鎖,這個鎖多加了wait(),notify()喚醒一個進程,notifyall()喚醒全部進程方法,創建的時候預設是Rlock類型的鎖,可以設置為lock類型的,預設就ok 1 from random import randint 2 import threading 3 import time ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...