資料庫鎖相關

来源:https://www.cnblogs.com/joeysh/archive/2019/04/22/10748511.html
-Advertisement-
Play Games

前言 關於資料庫鎖,是一個很重要的知識點; 不少人在開發的時候,應該 很少會註意到 這些鎖的問題,也很少會給程式加鎖(除了 庫存 這些對數量準確性要求極高的情況下); 一般也就聽過常說的樂觀鎖和悲觀鎖,瞭解過基本的含義之後就沒了,沒有去實際的操作過,本文將簡單的整理一下資料庫鎖的知識,希望對大家有所 ...


前言

關於資料庫鎖,是一個很重要的知識點;

不少人在開發的時候,應該很少會註意到這些鎖的問題,也很少會給程式加鎖(除了庫存這些對數量準確性要求極高的情況下);

一般也就聽過常說的樂觀鎖和悲觀鎖,瞭解過基本的含義之後就沒了,沒有去實際的操作過,本文將簡單的整理一下資料庫鎖的知識,希望對大家有所幫助;

引入

本文參考文章:資料庫的兩大神器

資料庫鎖

簡介

在MySQL中鎖看起來是很複雜的,因為有一大堆的東西和名詞:排它鎖,共用鎖,表鎖,頁鎖,間隙鎖,意向排它鎖,意向共用鎖,行鎖,讀鎖,寫鎖,樂觀鎖,悲觀鎖,死鎖。這些名詞有的博客又直接寫鎖的英文的簡寫--->X鎖,S鎖,IS鎖,IX鎖,MMVC等等之類。鎖的相關知識又跟存儲引擎,索引,事務的隔離級別都是關聯的;

以上的一大堆鎖可能很多人都只是知道一些概念,但是我們的程式在一般情況下還是可以跑得好好的。因為這些鎖資料庫隱式幫我們加了:

  • 對於UPDATE、DELETE、INSERT語句,InnoDB自動給涉及數據集加排他鎖(X),也就是我們常說的寫鎖;
  • MyISAM在執行查詢語句SELECT前,會自動給涉及的所有表加讀鎖,在執行更新操作(UPDATE、DELETE、INSERT等)前,會自動給涉及的表加寫鎖,這個過程並不需要用戶干預

表鎖和行鎖

從鎖的粒度我們可以分為兩大類,它們各自的特點如下:

  • 表鎖:開銷小,加鎖快;不會出現死鎖;鎖定力度大,發生鎖衝突概率高,併發度低;
  • 行鎖:開銷大,加鎖慢;會出現死鎖;鎖定力度小,發生鎖衝突的概率低,併發度高;

同樣,不同的存儲引擎支持的鎖的力度也不一樣:

  • InnoDB:表鎖行鎖都支持(InnoDB的行鎖是基於索引的 ,稍後會演示);
  • MyISAM:只支持表鎖;

表鎖

表鎖也分為兩種模式:

  • 表讀鎖(Table Read Lock)
  • 表寫鎖(Table Write Lock)
    • 讀讀不阻塞:當前用戶讀取數據,其他用戶也在讀取數據,不會加鎖;
    • 讀寫阻塞:當前用戶在讀取數據的時候,其他用戶不能修改當前用戶讀的數據;
    • 寫寫阻塞:當前用戶在修改數據,其他用戶不能修改當前用戶正在修改的數據;

總結得到:

  1. 讀讀不阻塞,讀寫阻塞,寫寫阻塞
  2. 讀鎖和寫鎖是互斥的,讀寫操作是串列
  3. 在mysql裡邊,寫鎖是優先於讀鎖的

行鎖

我們使用MySQL一般是使用的InnoDB引擎,上面也提到了InnoDB和MyISAM的一些區別:

  • InnoDB行鎖表鎖都支持,MyISAM只支持表鎖;
  • InnoDB支持事務,MyISAM不支持;

InnoDB實現了以下兩種類型的行鎖:

  • 共用鎖(s鎖):允許一個事務去讀一行,會阻止其他事務獲取相同數據集的排他鎖(讀取數據的時候不允許修改)
    • 也被稱為讀鎖:讀鎖是共用的,多個線程可以同時讀取統一數據集,但是不允許其他線程進行修改(也就是不允許其他事務獲取相同數據集的排他鎖);
  • 排他鎖(x鎖):允許獲取排他鎖去做更新操作,阻止其他事務獲取相通數據的共用鎖和排他鎖(一個事務修改數據的時候,阻止其他事務對相同數據集做更新或者查詢操作);
    • 也被稱為寫鎖:寫鎖是排他的,寫鎖會阻塞其他的寫鎖和讀鎖;

意向鎖

為了允許行鎖和表鎖共存,實現多粒度鎖機制,InnoDB還有兩種內部使用的意向鎖(Intention Locks),這兩種意向鎖都是表鎖

  • 意向共用鎖(IS):事務打算給數據行加共用鎖,事務在給一個數據行加共用鎖前必須先取得該表的IS鎖;
  • 意向排他鎖(IX):事務打算給數據行加排他鎖,事務再給一個數據行加排他鎖前必須先取得該表的IX鎖;

意向鎖也是資料庫隱式幫我們做了,不需要程式員操心

表鎖行鎖測試

準備

上面我們提到了InnoDB支持行鎖,但是是基於索引的情況,下麵我們來實際的看一下:

  • 首先我們用客戶端連接上MySQL資料庫,為了測試鎖的效果,我們需要打開兩個或者兩個以上的客戶端(我打開了兩個)然後創建一個資料庫;

    CREATE DATABASE test CHARACTER SET utf8;
  • 然後我們需要建立一個表:

    CREATE TABLE `user` (
      `id` bigint(20) NOT NULL AUTO_INCREMENT,
      `username` varchar(255) NOT NULL,
      `age` int(11) NOT NULL,
      PRIMARY KEY (`id`)
    ) ENGINE=InnoDB CHARSET=utf8;
  • 我們簡單的建了一個user表,表中有三個欄位,其中id為自增主鍵,大家都知道主鍵是自帶索引的,也就是聚簇索引(主鍵索引),其他的欄位都是不帶索引的。

  • mysql> show tables;
    +----------------+
    | Tables_in_test |
    +----------------+
    | user           |
    +----------------+
    1 row in set (0.01 sec)
  • 現在我們簡單的往裡面添加幾條數據:

    INSERT INTO `user`(username,age) VALUES ('tom',23),('joey',22),('James',21),('William',20),('David',24);
    • mysql> select * from user;
      +----+----------+-----+
      | id | username | age |
      +----+----------+-----+
      |  1 | tom      |  23 |
      |  2 | joey     |  22 |
      |  3 | James    |  21 |
      |  4 | William  |  20 |
      |  5 | David    |  24 |
      +----+----------+-----+
      5 rows in set (0.00 sec)

測試

好的,現在前提都已經弄好了,我們可以開始測試了:

我們知道MySQL的事務是自動提交的,為了測試,我們需要把事務的自動提交關閉;

mysql> set  autocommit = 0;
Query OK, 0 rows affected (0.01 sec)

現在我們來查看一下MySQL的事務提交狀態:

mysql> show VARIABLES like 'autocommit';
+---------------+-------+
| Variable_name | Value |
+---------------+-------+
| autocommit    | OFF   |
+---------------+-------+
1 row in set, 1 warning (0.04 sec)

從上面可以看出,我們把事務的自動提交已經關閉了,下麵我們開始測試(打開的視窗都需要關閉事務的自動提交);

行鎖測試

首先,我打開了兩個視窗,分別為A和B,現在,我們兩個視窗的狀態都已經調整完畢(關閉事務自動提交)。我們在A視窗,輸入以下語句:

mysql> begin;
Query OK, 0 rows affected (0.00 sec)

mysql> select * from user where id = 1 for update;
+----+----------+-----+
| id | username | age |
+----+----------+-----+
|  1 | tom      |  23 |
+----+----------+-----+
1 row in set (0.02 sec)

mysql>

很明顯,以上語句中,打開了事務,然後執行了一條SQL語句,在select 語句後邊加了 for update相當於加了排它鎖(寫鎖),加了寫鎖以後,其他的事務就不能對它修改了!需要等待當前事務修改完提交之後才可以修改;

現在我們在視窗B執行相同的操作:

mysql> begin;
Query OK, 0 rows affected (0.00 sec)

mysql> select * from user where id = 1 for update;

-

註意到了嗎,視窗B並沒有數據出現,因為視窗A執行的時候加了排他鎖,但是視窗A並沒有提交事務,所以鎖也沒有得到釋放,現在我們在視窗A提交事務:

mysql> begin;
Query OK, 0 rows affected (0.00 sec)

mysql> select * from user where id = 1 for update;
+----+----------+-----+
| id | username | age |
+----+----------+-----+
|  1 | tom      |  23 |
+----+----------+-----+
1 row in set (0.02 sec)

mysql> commit;
Query OK, 0 rows affected (0.00 sec)

同時,視窗B出現了以下情況:

mysql> begin;
Query OK, 0 rows affected (0.00 sec)

mysql> select * from user where id = 1 for update;
+----+----------+-----+
| id | username | age |
+----+----------+-----+
|  1 | tom      |  23 |
+----+----------+-----+
1 row in set (4.34 sec)

mysql>

沒錯,因為視窗A提交了事務,釋放的排他鎖,所以視窗B獲取到了數據並重新為該數據添加了排他鎖,所以此時你在A視窗在重覆之前操作的時候還是會阻塞,因為視窗B沒有提交事務,也就是沒有釋放排他鎖;

現在,我們在視窗A執行以下語句:

mysql> begin;
Query OK, 0 rows affected (0.00 sec)

mysql> select * from user where id = 2 for update;
+----+----------+-----+
| id | username | age |
+----+----------+-----+
|  2 | joey     |  22 |
+----+----------+-----+
1 row in set (0.00 sec)

mysql>

有的同學可能會說,不對啊,我視窗B還沒有提交事務,釋放排他鎖啊。

但是,大家註意看我的SQL語句,這次查的是id = 2的數據;

這是InnoDB的一大特性,我上面說了,InnoDB的行鎖是基於索引的 ,因為此時我們的條件是基於主鍵的,而主鍵是自帶索引的,所以加的是行鎖,這個時候視窗A鎖的是id = 2的這條數據,視窗B鎖的是id = 1的這條數據,他們互不幹擾;

表鎖測試

現在,我們再來測試一下,沒有索引,走表鎖的情況;

我們上面有提過,InnoDB的行鎖是基於索引,沒有索引的話,鎖住的就是整張表:

我們在視窗A輸入執行以下操作:

mysql> begin;
Query OK, 0 rows affected (0.00 sec)

mysql> select * from user where age = 20 for update;
+----+----------+-----+
| id | username | age |
+----+----------+-----+
|  4 | William  |  20 |
+----+----------+-----+
1 row in set (0.04 sec)

mysql>

大家註意,這次的條件是使用的age,但是age是沒有索引的,所以我們在B視窗執行相同的操作:

mysql> begin;
Query OK, 0 rows affected (0.00 sec)

mysql> select * from user where age = 20 for update;
-

很清楚的能看到,視窗B處於阻塞狀態,我們換個條件繼續執行:

mysql> begin;
Query OK, 0 rows affected (0.00 sec)

mysql> select * from user where age = 22 for update;
-

同樣,儘管查詢的數據換成了age = 22,但是還是會阻塞住,也就證明看不是鎖的行;

我們再來試試換一個列作為條件:

mysql> begin;
Query OK, 0 rows affected (0.00 sec)

mysql> select * from user where id = 1 for update;
-

同樣的結果,我們現在在A視窗提交事務,再來看一下B視窗:

A:

mysql> begin;
Query OK, 0 rows affected (0.00 sec)

mysql> select * from user where age = 20 for update;
+----+----------+-----+
| id | username | age |
+----+----------+-----+
|  4 | William  |  20 |
+----+----------+-----+
1 row in set (0.04 sec)

mysql> commit;
Query OK, 0 rows affected (0.00 sec)

B:

mysql> begin;
Query OK, 0 rows affected (0.00 sec)

mysql> select * from user where id = 1 for update;
+----+----------+-----+
| id | username | age |
+----+----------+-----+
|  1 | tom      |  23 |
+----+----------+-----+
1 row in set (0.00 sec)

mysql>

當視窗A提交事務後,也就釋放了鎖,這個時候視窗B獲取到了鎖,得到了數據,並鎖住了id = 1的這一行數據;

聯合索引測試

關於聯合索引中,需要註意的一點就是最左匹配原則 ,說白了就是查詢是否走了索引,如果走了索引,同樣加的還是行鎖,否則鎖的還是表,下麵我們來看一下。首先,我們需要把表中的username和age建一個聯合索引:

mysql> create index index_username_age on user(username,age);
Query OK, 0 rows affected (0.04 sec)
Records: 0  Duplicates: 0  Warnings: 0

mysql> show index from user;
+-------+------------+--------------------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+
| Table | Non_unique | Key_name           | Seq_in_index | Column_name | Collation | Cardinality | Sub_part | Packed | Null | Index_type | Comment | Index_comment |
+-------+------------+--------------------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+
| user  |          0 | PRIMARY            |            1 | id          | A         |           5 |     NULL | NULL   |      | BTREE      |         |               |
| user  |          1 | index_username_age |            1 | username    | A         |           4 |     NULL | NULL   |      | BTREE      |         |               |
| user  |          1 | index_username_age |            2 | age         | A         |           5 |     NULL | NULL   |      | BTREE      |         |               |
+-------+------------+--------------------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+
3 rows in set (0.00 sec)

mysql>

上面可以看出,我們建立聯合索引成功,下麵我們開始測試,首先,我們在視窗A執行以下操作:

mysql> begin;
Query OK, 0 rows affected (0.00 sec)

mysql> select * from user where username='tom' and age = 20 for update;
+----+----------+-----+
| id | username | age |
+----+----------+-----+
|  1 | tom      |  20 |
+----+----------+-----+
1 row in set (0.00 sec)

mysql>

可以看出,和我們之前的操作沒啥兩樣,同樣是打開事務進行操作,現在我們在視窗B執行以下操作:

mysql> begin;
Query OK, 0 rows affected (0.00 sec)

mysql>  select * from user where username='tom' and age = 20 for update;
-

很清楚的看到B視窗被鎖住了,但是我們現在確定的是加的鎖,並不知道是行鎖還是表鎖,沒關係,我們換個條件:

mysql> begin;
Query OK, 0 rows affected (0.00 sec)

mysql>  select * from user where username='joey' and age = 22 for update;
+----+----------+-----+
| id | username | age |
+----+----------+-----+
|  2 | joey     |  22 |
+----+----------+-----+
1 row in set (0.00 sec)

mysql>

這樣,我們很清楚的就能看到走的是行鎖了。

只不過大家要註意聯合索引的命中規則也就是最左匹配原則,我們可以試一試單獨使用username作為條件看看走的什麼鎖,也可以看看單獨使用age走的什麼鎖,這裡就不再演示了,大家可以自行的嘗試。

總結

前提:必須在事務裡面

樣例:select * from table where column = condition for update;

結果:

  • 當coulmn是索引列的時候,也就是查詢走的索引的時候,這個時候鎖的就是行(行鎖);
  • 當coulmn不是索引的時候,也就是查詢沒走索引的時候,這個時候鎖的就是整個表(表鎖);

悲觀鎖

含義

悲觀鎖是從資料庫層面加鎖。總是假設最壞的情況,每次去拿數據的時候都認為別人會修改,所以每次在拿數據的時候都會上鎖,這樣別人想拿這個數據就會阻塞直到它釋放鎖;

例子

上面其實關於行鎖和表鎖的測試那裡我們使用的排他鎖也就是悲觀鎖;

select * from table where xxx  for update

在上面我們舉的例子夠多了,這裡不再多說;

樂觀鎖

含義

總是假設最好的情況,每次去拿數據的時候都認為別人不會修改,所以不會上鎖,但是在更新的時候會判斷一下在此期間別人有沒有去更新這個數據;

例子

表中有一個版本欄位,第一次讀的時候,獲取到這個欄位。處理完業務邏輯開始更新的時候,需要再次查看該欄位的值是否和第一次的一樣。如果一樣就更新,反之拒絕。之所以叫樂觀,因為這個模式沒有從資料庫加鎖,等到更新的時候再判斷是否可以更新。

update table set xxx where id = 1 and version =  1;

上面的語句就很清楚的說明瞭樂觀鎖,在對id = 1的數據進行更新的同時添加了version = 1的條件,version是當前事務開始之前查詢出來的版本號,如果這個時候其他事務對id = 1的數據進行了更新會將version+1,所以如果其他事務進行了更新,這條語句是執行不成功的;

參考文章:https://juejin.im/post/5b4977ae5188251b146b2fc8

間隙鎖GAP

當我們用範圍條件檢索數據而不是相等條件檢索數據,並請求共用或排他鎖時,InnoDB會給符合範圍條件的已有數據記錄的索引項加鎖;對於鍵值在條件範圍內但並不存在的記錄,叫做“間隙(GAP)”。InnoDB也會對這個“間隙”加鎖,這種鎖機制就是所謂的間隙鎖。

值得註意的是:間隙鎖只會在Repeatable read隔離級別下使用~

例子:假如emp表中只有101條記錄,其empid的值分別是1,2,...,100,101

Select * from  emp where empid > 100 for update;

上面是一個範圍查詢,InnoDB不僅會對符合條件的empid值為101的記錄加鎖,也會對empid大於101(這些記錄並不存在)的“間隙”加鎖

InnoDB使用間隙鎖的目的有兩個:

  • 為了防止幻讀(上面也說了,Repeatable read隔離級別下再通過GAP鎖即可避免了幻讀)
  • 滿足恢復和複製的需要
    • MySQL的恢復機制要求:在一個事務未提交前,其他併發事務不能插入滿足其鎖定條件的任何記錄,也就是不允許出現幻讀

死鎖

併發的問題就少不了死鎖,在MySQL中同樣會存在死鎖的問題。

但一般來說MySQL通過回滾幫我們解決了不少死鎖的問題了,但死鎖是無法完全避免的,可以通過以下的經驗參考,來儘可能少遇到死鎖:

  • 1)以固定的順序訪問表和行。比如對兩個job批量更新的情形,簡單方法是對id列表先排序,後執行,這樣就避免了交叉等待鎖的情形;將兩個事務的sql順序調整為一致,也能避免死鎖。
  • 2)大事務拆小。大事務更傾向於死鎖,如果業務允許,將大事務拆小。
  • 3)在同一個事務中,儘可能做到一次鎖定所需要的所有資源,減少死鎖概率。
  • 4)降低隔離級別。如果業務允許,將隔離級別調低也是較好的選擇,比如將隔離級別從RR調整為RC,可以避免掉很多因為gap鎖造成的死鎖。
  • 5)為表添加合理的索引。可以看到如果不走索引將會為表的每一行記錄添加上鎖,死鎖的概率大大增大。

總結

本文介紹了MySQL數據鎖以及事務的一些知識點,下麵我們來總結一下;

不同的存儲引擎支持的鎖的力度也不一樣:

  • InnoDB:表鎖行鎖都支持(也做了演示);
    • 當查詢走的索引的時候,這個時候鎖的就是行;
    • 當查詢沒走的索引的時候,這個時候鎖的就是表;
  • MyISAM:只支持表鎖;

資料庫鎖從鎖的粒度我們可以分為兩大類,它們各自的特點如下::

  • 表鎖:開銷小,加鎖快;不會出現死鎖;鎖定力度大,發生鎖衝突概率高,併發度低;
  • 行鎖:開銷大,加鎖慢;會出現死鎖;鎖定力度小,發生鎖衝突的概率低,併發度高;

悲觀鎖:總是假設最好的情況,每次去拿數據的時候都認為別人不會修改,所以不會上鎖,但是在更新的時候會判斷一下在此期間別人有沒有去更新這個數據;

樂觀鎖:總是假設最好的情況,每次去拿數據的時候都認為別人不會修改,所以不會上鎖,但是在更新的時候會判斷一下在此期間別人有沒有去更新這個數據;

最後

最後說一下,本文的參考文章:資料庫的兩大神器

大家可以去看一下原文,本人也是小菜雞一枚,說的有問題還望大家指出來;

大家共同學習,一起進步。


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

-Advertisement-
Play Games
更多相關文章
  • JDBC(Java DataBase Connector) :定義了一系列的介面用於資料庫的操作 要掌握的介面: java.sql.Driver : -->驅動 java.sql.Connection -->連接 java.sql.Statement -->靜態處理塊 java.sql.Prepar ...
  • 更新緩存的時候涉及兩個問題: 刪除(del)還是 修改(set)? 先操作資料庫,還是 先操作緩存? 組合起來就有四種情況: 第一種情況:先刪除緩存,後更新資料庫 如果刪除緩存失敗,則後面的操作都不會執行,沒問題; 如果刪除緩存成功,更新資料庫失敗,則緩存與資料庫不一致,但這種不一致會馬上被修正, ...
  • SQL操作符 算術操作符:+加, 減, 乘,/除 比較操作符: ,=,!=,,= 常用的判斷,和!=相同 between $lower_val$ and $hight_val$ between .. and.. 包括兩端 查詢20 ...
  • 這一篇筆記的mysql優化是註重於查詢優化,根據mysql的執行情況,判斷mysql什麼時候需要優化,關於資料庫開始階段的資料庫邏輯、物理結構的設計結構優化不是本文重點,下次再談 查看mysql語句的執行情況,判斷是否需要進行優化 以下分別通過java程式員可分析的前三個方面來討論mysql語句的查 ...
  • 1.按姓氏筆畫排序:Select * From TableName Order By CustomerName Collate Chinese_PRC_Stroke_ci_as 2.資料庫加密:select encrypt('原始密碼')select pwdencrypt('原始密碼')select ...
  • mysql優化二之鎖機制 mysql提供了鎖機制和MVCC機制來保證併發操作的安全性,這裡主要討論鎖機制, MVCC見下篇文章 mysql的鎖按照鎖粒度可分為行鎖與表鎖,按照操作類型劃分可讀鎖和寫鎖 InnoDB存儲引擎支持表鎖和行鎖,預設鎖為行鎖,MyIsam只支持表鎖 鎖粒度越高則併發性越好 表 ...
  • SQL中只有兩列數據(欄位1,欄位2),將其相同欄位1的行轉列 轉換前: 轉換後: 轉自:https://bbs.csdn.net/topics/392320974 ...
  • 一.Spark2.0的新特性Spark讓我們引以為豪的一點就是所創建的API簡單、直觀、便於使用,Spark 2.0延續了這一傳統,併在兩個方面凸顯了優勢: 1、標準的SQL支持; 2、數據框(DataFrame)/Dataset (數據集)API的統一。 在SQL方面,我們已經對Spark的SQL ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...