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
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...