JavaWeb框架_Struts2_(六)----->Struts2的輸入校驗

来源:http://www.cnblogs.com/Mairr/archive/2017/12/06/7978684.html
-Advertisement-
Play Games

1. 輸入校驗章節目錄 輸入校驗概述 客戶端校驗 伺服器端校驗 手動編程校驗 重寫validate方法 重寫validateXxx()方法 輸入校驗流程 校驗框架校驗 Struts2 內置的校驗器 常用的內置校驗器的配置 客戶端校驗 伺服器端校驗 重寫validate方法 重寫validateXxx ...


1. 輸入校驗章節目錄

 

  • 輸入校驗概述
    • 客戶端校驗
    • 伺服器端校驗
  • 手動編程校驗
    • 重寫validate方法
    • 重寫validateXxx()方法
    • 輸入校驗流程
  • 校驗框架校驗
    • Struts2 內置的校驗器
    • 常用的內置校驗器的配置

 

2. 詳細內容

 

2.1 輸入校驗概述

 

  輸入校驗分為客戶端校驗和伺服器端校驗,客戶端校驗主要是過濾正常用戶的誤操作,主要通過JavaScript代碼完成;伺服器端校驗是整個應用阻止非法數據的最後防線,主要通過在應用中編程實現。

 

2.1.1 客戶端校驗

 

  大多數情況下,使用JavaScript進行客戶端校驗的步驟如下:

  1. 編寫校驗函數;
  2. 在提交表單的事件中調用校驗函數;
  3. 根據校驗函數來判斷是否進行表單提交;

  下麵通過一個簡單的示例講解使用JavaScript進行客戶端校驗的方法,具體代碼如下所示:

<%--
  Created by IntelliJ IDEA.
  User: mairr
  Date: 17-12-5
  Time: 下午10:07
  To change this template use File | Settings | File Templates.
--%>


<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<%@ page language = "java" import = "java.util.*" pageEncoding="utf-8"%>
<%@ taglib prefix="s" uri="/struts-tags"%>


<html>
  <head>
    <title>註冊界面</title>
    <script type="text/javascript" >

        // 去掉前後的空格
      function deltrim(x) {
          while (x.length > 0 && x.charAt(0) == ' ')
              x = x.substring(1, x.length);
          while (x.length > 0 && x.charAt(x.length - 1) == ' ')
              x = x.substring(0, x.length - 1);
          return x;
      }

        // 非空驗證
      function isNULL(elem,message){
          var va = deltrim(elem.value);
          if(va == " ") {
              alert(message);
              elem.focus();
              return false;
          }
          return true;
          }

          // 驗證帳號,帳號只能是小寫字母數字,並且只能以字母開頭
      function validateId(){
          var first = document.forms[0].id.value.charAt(0);
          var exp = /^[a-z0-9]+$/;
          if(isNULL(document.forms[0].id,"請輸入帳號")){     // 驗證非空
              // 驗證首字元
              if((first >= 'a' && first <= 'z') || (first >= 'A' && first <= 'Z')){}
              else{
                  alert("帳號首字元必須是字母!");
                  document.forms[0].id.focus();
                  return false;
              }
              if(!exp.test(document.forms[0].id.value)){
                  alert("帳號必須是字母或者數字!");
                  document.forms[0].id.focus();
                  return false;
              }
              return true;
          }
          else{
              return false;
          }
      }


      // 驗證密碼,密碼要在8位以上,且需要有字母或者數字之外的字元
      function validatepwd(){
          var exp = /^[a-z0-9]+$/;
          if(isNULL(document.forms[0].pwd," 請輸入密碼")){   //驗證非空
              if(document.forms[0].pwd.value.length <= 8){
                  alert("密碼大於8位");
                  document.forms[0].pwd.focus();
                  return false;
              }else{
                  if(exp.test(document.forms[0].pwd.value)){
                      alert("密碼要有字母和數字之外的字元!");
                      document.forms[0].pwd.focus();
                      return false;
                  }
              }
          }else{
              return false;
          }
          if(document.forms[0].pwd.value != document.forms[0].repwd.value)
          {
              alert("兩次密碼不一樣!");
              document.forms[0].pwd.focus();
              return false;
          }
          return true;
          }


          // 驗證郵箱,右邊為六位數子
          function checkcode() {
              var exp = /^[0-9]+$/;
              if(isNULL(document.forms[0].ecode,"請輸入郵編")){    // 驗證非空
                  if(document.forms[0].ecode.value.length != 6){
                      alert("郵編為6位");
                      document.forms[0].ecode.focus();
                      return false;
                  }else{
                      if(!exp.test(document.forms[0].ecode.value)){
                          alert("郵編為數字");
                          document.forms[0].ecode.focus();
                          return false;
                      }
                  }
                  return true;
              }else{
                  return false;
              }
          }


          // 驗證E-mail的基本格式
          function checkEmail(){
              var exp = /^\w+((-\w+)|(\.\w+))*@[A-Za-z0-9]+((\.|-)[A-Za-z0-9]+)*\.[A-Za-z0-9]+$/;
              if(isNULL(document.forms[0].email,"請輸入Email")){     // 驗證非空
                  if(!exp.test(document.forms[0].email.value)){
                      alert("Email格式錯誤");
                      document.forms[0].email.focus();
                      return false;
                  }
                  return true;
              }else{
                  return false;
              }
          }


          // 提交按鈕
        function gogo(){
              if(validateId() && validatepwd() && checkcode() && checkEmail()){
                  document.forms[0].submit();
                  return true;
              }
              return false;
        }
    </script>
  </head>

<body> <s:form action ="" theme="simple"> <table> <tr> <td>登錄帳號</td> <td><s:textfield name="id"/></td> </tr> <tr> <td>密碼</td> <td><s:password name="pwd"/></td> </tr> <tr> <td>確認密碼</td> <td><s:password name="repwd"/></td> </tr> <tr> <td>郵編</td> <td><s:textfield name = "ecode"/></td> </tr> <tr> <td>Mail</td> <td><s:textfield name = "email"/></td> </tr> <tr> <td><input type="button" value="提交" onclick="return gogo()" /></td> <td><s:reset value = "重置"/></td> </tr> </table> </s:form> </body>

</html>

  上述的校驗過程中,建立了一個Struts2的項目(前面幾篇博文有介紹),JSP校驗頁面中運用了部分struts2的標簽。由上述JSP程式,執行結果有如下幾種情況:

  (1) 頁面:

  (2) 客戶端校驗結果:

 

  客戶端校驗可以過濾用戶的錯誤操作,是第一道防線,一般使用JavaScript代碼實現。僅有客戶端校驗是不夠的。攻擊者可以繞過客戶端校驗直接進行非法輸入,這樣會引起系統的異常,為了確保數據的合法性,防止用戶通過非正常手段提交錯誤信息,所以必須加上伺服器驗證。

 

2.1.2 伺服器端校驗

 

  伺服器對於系統的安全性、完整性、健壯性起到至關重要的作用。Struts2框架提供了一套驗證框架,通過驗證框架能夠非常簡單和快速地完成輸入校驗。

  在伺服器端,對於輸入校驗Struts2提供了兩種實現方法:一是採用手工編寫代碼實現;另外一種是,給予XML配置方式的實現(校驗框架校驗)。接下來詳細介紹這兩種方法。

 

2.2 手動編程校驗

 

  手動編程校驗主要是通過在類中編寫校驗邏輯代碼,有兩種方式i:一是在Action類中重寫validate()方法;二是在Action類中重寫validateXxx()方法。

 

2.2.1 重寫validate方法

 

  validate()方法會校驗Action中所有與execute()方法簽名相同的方法。當某個數據校驗失敗時,在validate()方法中應該調用addFiledError()方法向系統fieldErrors添加校驗失敗信息。為了使用addFileError方法,Action類需要繼承ActionSupport。

  如果系統的fieldErrors包含失敗信息,Struts2會將請求轉發到名為input的result。在input視圖中可以通過<s:fielderror/>標簽失敗信息。

  下麵通過一個簡單示例講解如何重寫validate()方法進行輸入校驗。

(1) 創建一個JSP頁面login.jsp,要求驗證所有輸入項不能為空、密碼長度6~12位以及兩次密碼一樣(不運用javascript的客戶端校驗),具體代碼如下所示:

<%--
  Created by IntelliJ IDEA.
  User: mairr
  Date: 17-12-6
  Time: 下午9:39
  To change this template use File | Settings | File Templates.
--%>


<%@ page language="java" import="java.util.*" pageEncoding="utf-8" %>
<%@ taglib prefix="s" uri="/struts-tags" %>



<%
    String path = request.getContextPath();
    String basePath = request.getScheme() + "://" + request.getServerName() + ":" + request.getServerPort() + path + "/";
%>



<html>
<head>
    <base href="<%=basePath%>">
</head>
<body>
    <s:form action="login" theme="simple">
        <table>
            <tr>
                <td>登陸帳號</td>
                <td><s:textfield name = "id"/></td>
            </tr>
            <tr>
                <td>密碼</td>
                <td><s:password name = "pwd"/></td>
            </tr>
            <tr>
                <td>確認密碼</td>
                <td><s:password name = "repwd"/></td>
            </tr>
            <tr>
                <td><s:submit value= "提交"/></td>
                <td><s:reset value = "重置"/></td>
            </tr>
        </table>


        <!--顯示this.addFieldError("id","id不能為空")的信息-->
        <s:fielderror fieldName="id"/>

        <!--顯示所有校驗失敗信息-->
        <s:fielderror/>


    </s:form>
</body>
</html>

 

(2) 創建Action類LoginAction.java,在該類中重寫validate()方法,具體代碼如下“

package action;
import com.opensymphony.xwork2.ActionSupport;

public class LoginAction extends ActionSupport {
    private String id;
    private String pwd;
    private String repwd;

    public String getId() {
        return id;
    }

    public void setId(String id) {
        this.id = id;
    }

    public String getPwd() {
        return pwd;
    }

    public void setPwd(String pwd) {
        this.pwd = pwd;
    }

    public String getRepwd() {
        return repwd;
    }

    public void setRepwd(String repwd) {
        this.repwd = repwd;
    }


    // 重寫validate方法,進行輸入校驗,該方法在execute方法之前執行
    @Override
    public void validate() {
        if (id == null || id.trim().equals(" ")) {
            this.addFieldError("id", "id不能為空");
        }
        if (pwd == null || pwd.trim().equals(" ")) {
            this.addFieldError("pwd", "密碼不能為空");
        } else {
            if (pwd.length() < 6 || pwd.length() > 12) {
                this.addFieldError("pwdlength", "密碼的長度在6~12位之間");
            }
        }
        if (!pwd.equals(repwd)) {
            this.addFieldError("pwdsame", "兩次密碼不一致");
        }
    }


    // 實現登陸業務處理
    @Override
    public String execute() {
        return SUCCESS;
    }
}

 

(3) 配置Struts2的action,具體代碼如下:

<?xml version="1.0" encoding="UTF-8"?>

<!DOCTYPE struts PUBLIC
        "-//Apache Software Foundation//DTD Struts Configuration 2.5//EN"
        "http://struts.apache.org/dtds/struts-2.5.dtd">

<struts>
    <package name="default" namespace="/" extends="struts-default">
        <action name="login" class="action.LoginAction">
            <result name="input">/login.jsp</result>
        </action>
    </package>
</struts>

 

(4) web.xml配置

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"
         version="3.1">
    <filter>
        <filter-name>struts2</filter-name>
        <filter-class>org.apache.struts2.dispatcher.filter.StrutsPrepareAndExecuteFilter</filter-class>
    </filter>
    <filter-mapping>
        <filter-name>struts2</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>
</web-app>

 

(5) 運行login.jsp頁面,如果輸入非法信息則輸出校驗結果如下所示:

 

2.2.2 重寫validateXxx()方法

 

  validateXxx()只會校驗Action中方法名為Xxx()的方法,其中Xxx的第一個字母要大寫。重寫validateXxx()方法進行輸入校驗與重寫validate()方法基本一樣,唯一不同的就是校驗的方法名不同。

  將上面一個例子重寫為validateXxx()方法進行輸入校驗,則需要修改Action類和配置文件。

  • 修改後的Action類的代碼如下:
package action;
import com.opensymphony.xwork2.ActionSupport;

public class LoginAction extends ActionSupport {
    private String id;
    private String pwd;
    private String repwd;

    public String getId() {
        return id;
    }

    public void setId(String id) {
        this.id = id;
    }

    public String getPwd() {
        return pwd;
    }

    public void setPwd(String pwd) {
        this.pwd = pwd;
    }

    public String getRepwd() {
        return repwd;
    }

    public void setRepwd(String repwd) {
        this.repwd = repwd;
    }


    // 重寫validateLogin方法,進行輸入校驗
    public void validateLogin() {
        if (id == null || id.trim().equals(" ")) {
            this.addFieldError("id", "id不能為空");
        }
        if (pwd == null || pwd.trim().equals(" ")) {
            this.addFieldError("pwd", "密碼不能為空");
        } else {
            if (pwd.length() < 6 || pwd.length() > 12) {
                this.addFieldError("pwdlength", "密碼的長度在6~12位之間");
            }
        }
        if (!pwd.equals(repwd)) {
            this.addFieldError("pwdsame", "兩次密碼不一致");
        }
    }


    // 實現登陸業務處理
    public String login() {
        return SUCCESS;
    }
}
  • 修改後的配置文件struts.xml如下:
<?xml version="1.0" encoding="UTF-8"?>

<!DOCTYPE struts PUBLIC
        "-//Apache Software Foundation//DTD Struts Configuration 2.5//EN"
        "http://struts.apache.org/dtds/struts-2.5.dtd">

<struts>
    <package name="default" namespace="/" extends="struts-default">
        <action name="login" class="action.LoginAction" method="login">
            <result name="input">/login.jsp</result>
        </action>
    </package>
</struts>
  • 結果是一致的:

 

 

2.2.3 輸入校驗的流程

 

  經過前面講解,可以知道,Struts2輸入校驗需要經過一下幾個步驟:

  1. 類型轉器對請求參數執行類型轉換,並且把轉換後的值賦給Action中的屬性。
  2. 如果在執行轉換過程中出現了異常,系統會把異常信息保存到ActionContext , conversionError攔截器將異常信息添加到fieldError里。不管類型轉換是否出現異常,都會進入步驟三;
  3. 系統通過反射技術先調用Action中的validateXxx()方法,Xxx為方法名字。
  4. 再調用Action中的validate()方法;
  5. 經過上面的4個步驟,如果系統中的fieldError存在錯誤信息(即存放錯誤信息的集合的size大於0),系統自動將請求轉發至名稱為input的視圖。如果系統中的fileError沒有任何錯誤信息,系統將執行Action中的處理方法。

 

2.3 校驗框架校驗

 

  使用Struts2校驗框架的好處是將校驗邏輯放到配置文件中,實現校驗邏輯代碼與業務邏輯代碼的分離。使用基於校驗框架校驗方式實現輸入校驗時,Action也需要繼承ActionSupport,並且提供校驗文件。同樣框架校驗的方式也有2種:一是校驗Action中所有與execute方法簽名相同的方法;二是校驗Action中某個與execute方法簽名相同的方法。

 

2.3.1 Struts2內置的校驗器

 

  Struts2框架提供的內置校驗器如下:(多看幾遍,還蠻重要的

  1. required:必填校驗器,要求field的值不能為null;
  2. requiredstring:必填字元串校驗器,要求field的值不能為null,並且長度大於0,預設情況下會對字元串去掉前後空格;
  3. stringlength:字元串長度校驗器,要求field的值必須在指定的範圍內,否則校驗失敗;miniLength參數指定最小長度,maxLength參數指定最大長度,trim參數指定校驗field之前是否去掉字元串前後空格;
  4. regex:正則表達式校驗器,檢查被校驗的field是否匹配一個正則表達式。expression參數指定正則表達式, caseSensitive參數指定進行正則表達式匹配時,是否區分大小寫,預設值為true;
  5. int:整數校驗器,要求field的整數值必須在指定範圍內,mini指定最小值,max最大值;
  6. double:雙精度浮點數校驗器,要求field的雙進度浮點數必須在指定範圍內,mini指定最小值,max最大值;
  7. fieldexpression:欄位OGNL表達式校驗器,要求field滿足一個OGNL表達式,expression參數指定一個OGNL表達式,該邏輯表達式基於ValueStack進行求值,返回true時校驗通過,否則不通過。該校驗器只能用於<filed-validator>;
  8. email:郵件地址校驗器,要求如果field的值非空,則必須是合法的郵件地址;
  9. url:網址校驗器,要求如果field的值非空,則必須是合法的URL地址i;
  10. date:日期校驗器,要求field的日期值必須在指定的範圍內,mini指定最小值,max指定最大值;
  11. conversion:轉換校驗器,指定在類型轉換失敗時,提示錯誤信息;
  12. visitor:用於校驗Action中的複合屬性,指定一個校驗文件用於校驗複合中的屬性;
  13. expression:OGNL表達式表達式校驗,expression參數指定OGNL表達式,該邏輯表達式基於ValueStack進行求值,返回true時校驗通過,否則不通過,該校驗器不用在欄位校驗器的配置中,只能用於<validator>.

 

2.3.2 常用內置校驗器的配置

  

  對於Struts2校驗框架來說,一般有兩種方式來配置校驗器:

  • 使用<validator>
  • 使用<field-validator>

  當<validator>的子節點中配置了<param name="fieldName">用於指定某個屬性進行校驗時,則達到的效果與<field-validator>是一樣的。如下所示:(兩種配置方式是等效的

<!--校驗user.id屬性時,用<validator>來配置-->
<validator type="required">
   <s:param name="fieldName">user.id</s:param>
   <message>用戶的ID不能為空!</message>
</validator>
        
        
<!--------方式二 ----->
<field name="user.id">
    <field-validator type ="required">
       <message>用戶的ID不能為空!</message>
    </field-validator>
</field>

  下麵簡單介紹幾種常用的內置校驗器的配置示例:

(1)  required(必填校驗器)

<field-validator type ="required">
    <message>用戶的ID不能為空!</message>
</field-validator>

(2)  requiredstring(必填字元串校驗器)

<field-validator type ="requiredstring">
     <param name ="trim">true</param>
     <message>用戶的ID不能為空!</message>
</field-validator>

(3)  stringlength(字元串長度校驗器)

<field-validator type ="stringlength">
    <param name = "maxlength">12</param>
    <param name = "minilength">6</param>
    <message>密碼必須在6~12位之間</message>
</field-validator>

(4)  email(郵件地址校驗器)

<field-validator type ="email">
    <message>郵箱格式不正確</message>
</field-validator>

(5)  regex(正則表達式校驗器)

<field-validator type ="regex">
      <param name="expression"><![CDATA[^1[3578]\d{9}$]]></param>
     <message>手機號格式不正確</message>
</field-validator>

(6)  int(整數校驗)

<param type ="int">
     <param name="max">100</param>
     <param name="mini">0</param>
     <message>年齡必須在0~100之間</message>
</field-validator>

(7)  欄位OGNL表達式校驗器

<param type ="int">
     <param name="expression"><![CDATA[imagefile.length() <=0 ]]></param>
     <message>文件不能為空</message>
</field-validator>

 

  基於Struts2框架的校驗主要是以上內容,這是Struts2框架應用比較多的一個版塊,比較重要的內容!

 


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

-Advertisement-
Play Games
更多相關文章
  • 退出程式不用exit() #01代碼 1 #__author: _nbloser 2 #date: 2017/12/5 3 4 5 shaoguan = ['仁化', '始興', '樂昌', '南雄'] 6 jiangmeng = ['開平', '蓬江', '台山', '鶴山', '恩平'] 7 g ...
  • 21342134123 ...
  • 在上文中《Java IO(1)基礎知識——位元組與字元》瞭解到了什麼是位元組和字元,主要是為了對Java IO中有關位元組流和字元流有一個更好的瞭解。 本文所述的輸出輸出指的是Java中傳統的IO,也就是阻塞式輸入輸出(Blocking I/O, BIO),在JDK1.4之後出現了新的輸入輸出API——N ...
  • 貪心演算法(又稱貪婪演算法)是指,在對問題求解時,總是做出在當前看來是最好的選擇。也就是說,不從整體最優上加以考慮,他所做出的僅是在某種意義上的局部最優解。貪心演算法不是對所有問題都能得到整體最優解,但對範圍相當廣泛的許多問題他能產生整體最優解或者是整體最優解的近似解。 ...
  • Spring註解 Spring的對象訪問 Spring面向切麵編程 Spring MVC框架1.spring的優點輕量級:基礎版本的spring框架大約2mb控制反轉(IOC):把生成對象的權利反轉給spring框架面向切麵(AOP):把可重用的功能提取出來,然後再將這些通用的功能在合適的時候織入到 ...
  • 異常一 只開啟一個客戶端,輸入信息後關閉,客戶端出現如下異常 根據異常說明 ChatClientFrame客戶端117行 提示原因是Socket關閉 分析原因 客戶端代碼 while (connected) { String str = dis.readUTF(); 當視窗關閉後,Socket已經關 ...
  • 裝飾器: 定義:本質就是函數,(裝飾其它函數),就是為其它函數添加附加功能。 原則:1.不能修改被裝飾的函數的源代碼。 2.不能修改被裝飾的函數的調用方式。(被修飾函數感知不到) 實現裝飾器知識儲備: 函數即"變數" 高階函數 A:把一個函數名,當做形參傳給另外一個函數。 B:返回值中包含函數名 D... ...
  • Item 1. 考慮用靜態工廠方法替代構造器 獲得一個類的實例時我們都會採取一個公有的構造器。Foo x = new Foo(); 同時我們應該掌握另一種方法就是靜態工廠方法(static factory method)。 一句話總結,靜態工廠方法其實就是一個返回類的實例的靜態方法。 書中給出的例子 ...
一周排行
    -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# ...