主頁 > 區塊鏈 > 【資料結構與演算法】->演算法->動態規劃(下)->如何實作搜索引擎的拼寫糾錯功能?

【資料結構與演算法】->演算法->動態規劃(下)->如何實作搜索引擎的拼寫糾錯功能?

2020-09-21 01:33:53 區塊鏈

動態規劃(下)

    • Ⅰ 前言
    • Ⅱ 字串相似度的量化
    • Ⅲ 萊文斯坦距離的計算
    • Ⅳ 最長公共子串長度的計算
    • Ⅴ 如何實作搜索引擎的拼寫糾錯

Ⅰ 前言

如果對動態規劃的基礎內容還有不了解的同學,可以跳轉去看我下面的文章👇

【資料結構與演算法】->演算法->動態規劃(上)->初識動態規劃->怎么精準地幫助女朋友薅羊毛

【資料結構與演算法】->演算法->動態規劃(中)->詳解動態規劃理論

在我前面的一篇文章 Trie 樹 中我提到了搜索引擎的一個功能,關鍵詞提示,除此之外,一般搜索引擎為了優化用戶的體驗,還有個拼寫糾錯的功能,

當你在搜索框中,不小心輸入錯單詞時,搜索引擎就會非常智能地檢測出你的拼寫錯誤,并且用對應的正確單詞來進行搜索,那么,這個功能是如何實作的呢?

在這里插入圖片描述

Ⅱ 字串相似度的量化

要實作拼寫糾錯功能,首先要能識別兩個字串,但是計算機只認識數字,兩個字串之間的相似度要如何量化呢?有一個非常著名的量化方法,那就是 編輯距離(Edit Distance)

顧名思義,編輯距離指 的就是,將一個字串轉化成另一個字串,需要的最少編輯操作次數(比如增加一個字符,洗掉一個字符,替換一個字符),編輯距離越大,說明兩個字串的相似程度越小;相反,編輯距離越小,說明兩個字串的相似程度越大,對于兩個完全相同的字串來說,編輯距離就是 0,

根據所包含的編輯操作種類的不同,編輯距離有多種不同的計算方式,比較著名的有 萊文斯坦距離(Levenshtein distance)最長公共子串長度(Longest common substring length),其中,萊溫斯坦距離允許增加、洗掉、替換字符這三種編輯操作,最長公共子串長度只允許增加、洗掉字符這兩個編輯操作,

而且,萊文斯坦距離和最長公共子串,是從兩個相反的角度來分析字串的相似程度,萊文斯坦距離的大小,表示兩個字串差異的大小;最長公共子串的大小,表示兩個字串相似程度的大小,

關于這兩個計算方法,我舉個例子來說明,下圖中兩個字串,mitcum 和 mtacnu 的萊文斯坦距離是 3,最長公共子串長度是 4,

在這里插入圖片描述
了解了編輯距離的概念之后,我們來看,如何快速計算兩個字串之間的編輯距離,

Ⅲ 萊文斯坦距離的計算

這個問題是求把一個字串變成另一個字串,需要的最少編輯次數,整個求解程序涉及到多個決策階段,我們需要依次考察一個字串中的每個字符,跟另一個字串中的字符是否匹配,匹配的話如何處理,不匹配的話又如何處理,所以,這個問題符合多階決策最優模型

在上一篇文章中的總結里,我們說貪心、回溯、動態規劃可以解決的問題,都可以抽象成這樣一個模型,要解決這個問題,我們可以先用最簡單的回溯演算法試試,

回溯是一個遞回處理的程序,如果 a[i] 與 b[j] 匹配,我們遞回考察 a[i+1] 和 b[j+1],如果 a[i] 與 b[j] 不匹配,那我們有很多種處理方式可選:

  • 可以洗掉 a[i],然后遞回考察 a[i+1] 和 b[j];
  • 可以洗掉 b[j],然后遞回考察 a[i] 和 b[j+1];
  • 可以在 a[i] 前面添加一個跟 b[j] 相同的字符,然后遞回考察 a[i] 和 b[j+1];
  • 可以在 b[j] 前面添加一個跟 a[i] 相同的字符,然后遞回考察 a[i+1] 和 b[j];
  • 可以將 a[i] 替換成 b[j],或者將 b[j] 替換成 a[i],然后遞回考察 a[i+1] 和 b[j+1];

我們用Java實作就是下面這樣👇

package com.tyz.third.core;

/**
 * 萊文斯坦距離的計算
 * @author Tong
 */
public class LevenshteinDistance {
	private char[] strOne = "mitcmu".toCharArray();
	private char[] strTwo = "mtacnu".toCharArray();
	private int lengthOne = strOne.length;
	private int lengthTwo = strTwo.length;
	private int minDist = Integer.MAX_VALUE; //存盤結果
	
	/**
	 * 回溯法計算萊文斯坦距離
	 * @param i 第一個字串的遍歷索引
	 * @param j 第二個字串的遍歷索引
	 * @param edist 編輯次數
	 */
	public void lvstBT(int i, int j, int edist) {
		//一個字串已經遍歷完了,另一個字串剩下的字符數就是需要編輯的次數
		if (i == lengthOne || j == lengthTwo) {
			if (i < lengthOne) {
				edist += (lengthOne - i); 
			}
			if (j < lengthTwo) {
				edist += (lengthTwo - j);
			}
			if (edist < minDist) {
				minDist = edist;
			}
			return;
		}
		if (strOne[i] == strTwo[j]) { //兩個字符匹配
			lvstBT(i+1, j+1, edist);
		} else { //兩個字符不匹配
			lvstBT(i + 1, j, edist + 1); //洗掉a[i]或者b[j]前添加一個字符
			lvstBT(i, j + 1, edist + 1); //洗掉b[j]或者a[i]前添加一個字符
			lvstBT(i + 1, j + 1, edist + 1); //將a[i]和b[j]替換為相同字符
		}
	}
}

根據回溯演算法的實作,我們可以畫出遞回樹,看是否存在重復子問題,如果存在重復子問題,那我們就可以考慮用動態規劃來解決;如果不存在重復子問題,那回溯法就是最好的解決方法,

在這里插入圖片描述
在遞回樹中,每個節點代表一個狀態,狀態包含三個變數 (i, j, edist) ,其中,edist 表示處理到 a[i] 和 b[j] 時,已經執行的編輯次數,

在遞回樹中,(i, j) 兩個變數重復的節點很多,比如 (3, 2) 和 (2, 3),對于 (i, j) 相同的節點,我們只需要保留 edist 最小的,繼續遞回處理就可以了,剩下的節點都可以舍棄,所以,狀態就從 (i, j, edist) 變成了 (i, j, min_edist),其中 min_edist 表示處理到 a[i] 和 b[j],已經執行的最少編輯次數,

到這里我們就發現了,這就是典型的動態規劃的例子啊,不過,這個問題的狀態轉移方式,要比我在動態規劃前兩篇文章里說的還要復雜,我們求二維矩陣的最短路徑,到達狀態 (i, j) 只能通過 (i-1, j) 或 (i, j-1) 兩個狀態轉移過來,但是今天這個問題,狀態可能從 (i-1, j), (i, j-1) 或 (i-1, j-1) 三個狀態中的任意一個遷移過來,

在這里插入圖片描述
基于上面的分析,我們可以把狀態轉移的程序,用公式寫出來,

在這里插入圖片描述
了解了狀態與狀態之間的遞推關系,我們畫出一個二維的狀態圖,按行依次填充狀態表中的每個值,

在這里插入圖片描述
我們現在既有狀態轉移方程,又理清了完整的填表程序,代碼的實作就非常簡單了,

/**
	 * 動態規劃求萊文斯坦距離
	 * 萊文斯坦距離的計算
	 * 如果: a[i] != b[j], 那么: min_edist(i, j) 就等于 
	 * 					min(min_edist(i-1, j)+1, min_edist(i-1, j-1)+1)
	 * 如果: a[i] == b[j], 那么: min_edist(i, j) 就等于 
	 * 					min(min_edist(i-1, j)+1, min_edist(i-1, j-1)+1, min_edist(i-1, j-1))
	 * 其中, min表示求三個數中的最小值
	 */
	public int lvstDB(char[] strOne, int lengthOne, char[] strTwo, int lengthTwo) {
		int[][] minDist = new int[lengthOne][lengthTwo];
		
		//初始化第0列:a[0...i]到b[0...0]的編輯距離
		for (int i = 0; i < lengthOne; i++) {
			if (strOne[i] == strTwo[0]) {
				minDist[i][0] = i; //在第一個字符匹配之前, 前面的全是不匹配的, 所以編輯距離等于i
			} else if (i != 0) { //只要不是第一行的, 下一行的編輯距離就等于上一行的加1
				minDist[i][0] = minDist[i-1][0] + 1; 
			} else {
				minDist[i][0] = 1; //a(0, 0)出現不匹配,編輯距離為1
			}
		}
		
		//初始化第0行:a[0...0]到b[0...j]的編輯距離
		for (int j = 0; j < lengthTwo; j++) {
			if (strOne[0] == strTwo[j]) {
				minDist[0][j] = j; 
			} else if (j != 0) { 
				minDist[0][j] = minDist[0][j-1] + 1; 
			} else {
				minDist[0][j] = 1; 
			} 
		}
		
		for (int i = 1; i < lengthOne; i++) {
			for (int j = 1; j < lengthTwo; j++) {
				if (strOne[i] == strTwo[j]) {
					minDist[i][j] = min(minDist[i-1][j]+1,
										minDist[i][j-1]+1, minDist[i-1][j-1]);
				} else {
					minDist[i][j] = min(minDist[i-1][j]+1,
										minDist[i][j-1]+1, minDist[i-1][j-1]+1);
				}
			}
		}
		return minDist[lengthOne-1][lengthTwo-1];
	}
	
	/**
	 * 求x,y,z中的最小值
	 * @param x
	 * @param y
	 * @param z
	 * @return
	 */
	private int min(int x, int y, int z) {
		int minValue = Integer.MAX_VALUE;
		if (x < minValue) {
			minValue = x;
		}
		if (y < minValue) {
			minValue = y;
		}
		if (z < minValue) {
			minValue = z;
		}
		return minValue;
	}

Ⅳ 最長公共子串長度的計算

這個問題的解決思路和萊文斯坦距離的解決思路非常相似,也可以用動態規劃解決,我們前面已經詳細描述了萊文斯坦距離的動態規劃解決思路,所以針對最長公共子串的計算,我直接定義狀態,然后寫狀態轉移方程,

每個狀態還是包括三個變數 (i, j, max_lcs),max_lcs 表示 a[0…i] 和 b[0…j] 的最長公共子串長度,那 (i, j) 這個狀態都是由哪些狀態轉移過來的呢?

我們先來看回溯的處理思路,我們從 a[0] 和 b[0] 開始,依次考察兩個字串中的字符是否匹配,注意我們前面說的,最長公共子串長度只允許增加、洗掉字符這兩個編輯操作,

  • 如果 a[i] 與 b[j] 互相匹配,我們將最大公共子串長度加 1,并且繼續考察 a[i+1] 和 b[j+1],
  • 如果 a[i] 與 b[j] 不匹配,最長公共子串長度不變,這時候,有兩個不同的決策路線,
    • 洗掉 a[i],或者在 b[j] 前面加上一個字符 a[i],然后繼續考察 a[i+1] 和 b[j];
    • 洗掉 b[j],或者在 a[i] 前面加上一個字符 b[j],然后繼續考察 a[i] 和 b[j+1],

反過來也就是說,如果我們要求 a[0…i] 和 b[0…j] 的公共長度 max_lcs(i, j),我們只有可能通過下面這個三個狀態轉移過來:

  • (i-1, j-1, max_lcs),其中 max_lcs 表示 a[0…i-1] 和 b[0…j-1] 的最長公共子串長度;
  • (i-1, j, max_lcs),其中 max_lcs 表示 a[0…i-1] 和 b[0…j] 的最長公共子串長度;
  • (i, j-1, max_lcs),其中 max_lcs 表示 a[0…i] 和 b[0…j-1] 的最長公共子串長度;

如果我們把這個轉移程序,用狀態轉移方程寫出來,就是下面的樣子:
在這里插入圖片描述
有了狀態轉移方程,我們將可以實作代碼了,大家可以對照著講解做一個參考,

package com.tyz.third.core;

/**
 * 動態規劃計算最長公共子串長度
 * 如果: a[i] == b[j], 那么: max_lcs(i, j)就等于:
 * 							max(max_lcs(i-1, j-1)+1, max_lcs(i-1, j), max_lcs(i, j-1));
 * 如果: a[i] != b[j], 那么: max_lcs(i, j)就等于:
 * 							max(max_lcs(i-1, j-1), max_lcs(i, j-1));
 * 其中max表示求三個數中的最大值
 * @author Tong
 */
public class LongestCommonSubstring {
	
	public int lcs(char[] strOne, int lengthOne, char[] strTwo, int lengthTwo) {
		int[][] maxlcs = new int[lengthOne][lengthTwo];
		//初始化第0行: a[0...0]與b[0...j]的maxlcs
		for (int j = 0; j < lengthTwo; j++) {
			if (strOne[0] == strTwo[j]) {
				maxlcs[0][j] = 1;
			} else if (j != 0) {
				maxlcs[0][j] = maxlcs[0][j-1];
			} else {
				maxlcs[0][j] = 0;
			}
		}
		//初始化第0列: a[0...i]與b[0...0]的maxlcs
		for (int i = 0; i < lengthOne; i++) {
			if (strOne[i] == strTwo[0]) {
				maxlcs[i][0] = 1;
			} else if (i != 0) {
				maxlcs[i][0] = maxlcs[i-1][0];
			} else {
				maxlcs[i][0] = 0;
			}
		}
		
		for (int i = 1; i < lengthOne; i++) {
			for (int j = 1; j < lengthTwo; j++) {
				if (strOne[i] == strTwo[j]) {
					maxlcs[i][j] = max(maxlcs[i-1][j], 
								maxlcs[i][j-1], maxlcs[i-1][j-1]+1);
				} else {
					maxlcs[i][j] = max(maxlcs[i-1][j], 
							maxlcs[i][j-1], maxlcs[i-1][j-1]);
				}
			}
		}
		return maxlcs[lengthOne-1][lengthTwo-1];
	}
	
	private int max(int x, int y, int z) {
		int maxValue = Integer.MIN_VALUE;
		if (x > maxValue) {
			maxValue = x;
		}
		if (y > maxValue) {
			maxValue = y;
		}
		if (z > maxValue) {
			maxValue = z;
		}
		return maxValue;
	}
}

Ⅴ 如何實作搜索引擎的拼寫糾錯

當用戶在搜索框內,輸入一個拼寫錯誤的單詞時,我們就拿這個單詞跟詞庫里的單詞一 一進行比較,計算編輯距離,將編輯距離最小的單詞,作為糾正之后的單詞,提示給用戶,

這就是拼寫糾錯最基本的原理,不過,真正用于商用的搜索引擎,拼寫糾錯功能顯然不會就這么簡單,一方面,單純利用編輯距離來糾錯,效果并不一定好;另一方面,詞庫中的資料量可能非常大,搜索引擎每天要支持海量的搜索,所以對糾錯的性能要求很高,

針對糾錯效果不好的問題,我們有很多種優化思路,我這里介紹幾種,

  • 我們并不僅僅取出編輯距離最小的那個單詞,而是取出編輯距離最小的 TOP 10,然后根據其他引數,決策選擇哪個單詞作為拼寫糾錯單詞,比如使用搜索熱門程度來決定哪個單詞作為拼寫糾錯單詞,
  • 我們還可以用多種編輯距離計算方法,比如這篇文章里講的兩種,然后分別編輯距離最小的 TOP 10,然后求交集,用交集的結果,再繼續優化處理,
  • 我們還可以通過統計用戶的搜索日志,得到最常被拼錯的單詞串列,以及對應的拼寫正確的單詞,搜索引擎在拼寫糾錯的時候,首先在這個最常被拼錯單詞串列中查找,如果一旦找到,直接回傳對應的正確的單詞,這樣糾錯的效果非常好,
  • 我們還有更加高級一點的做法,引入個性化因素,針對每個用戶,維護這個用戶特有的搜索偏好,也就是常用的搜索詞,當用戶輸入錯誤的單詞的時候,我們首先在這個用戶常用的搜索關鍵詞中,計算編輯距離,查找編輯距離最小的單詞,

針對糾錯性能方面,我們也有相應的優化方式,我在這里講兩種分治的優化思路,

  • 如果糾錯功能的 TPS 不高,我們可以部署多臺機器,每臺機器運行一個獨立的糾錯功能,當有一個糾錯請求的時候,我們通過負載均衡,分配到其中一臺機器,來計算編輯距離,得到糾錯單詞,
  • 如果糾錯系統的回應時間太長,也就是,每個糾錯請求處理時間太長,我們可以將糾錯的詞庫,分配到很多臺機器,當有一個糾錯請求的時候,我們就將這個拼寫錯誤的單詞,同時發送到多臺機器,讓多臺機器并行處理,分別得到編輯距離最小的單詞,然后再對比合并,最終決定出一個最優的糾錯單詞,

真正的搜索引擎的拼寫糾錯優化,肯定不會這么簡單,但是萬變不離其宗,我們搞清楚了核心原理,就能更好地去理解它,

轉載請註明出處,本文鏈接:https://www.uj5u.com/qukuanlian/93048.html

標籤:區塊鏈

上一篇:Python筆記:matplotlib 三維圖表繪制方法簡介

下一篇:sublime Text3使用python時input輸入無反應如何解決

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • JAVA使用 web3j 進行token轉賬

    最近新學習了下區塊鏈這方面的知識,所學不多,給大家分享下。 # 1. 關于web3j web3j是一個高度模塊化,反應性,型別安全的Java和Android庫,用于與智能合約配合并與以太坊網路上的客戶端(節點)集成。 # 2. 準備作業 jdk版本1.8 引入maven <dependency> < ......

    uj5u.com 2020-09-10 03:03:06 more
  • 以太坊智能合約開發框架Truffle

    前言 部署智能合約有多種方式,命令列的瀏覽器的渠道都有,但往往跟我們程式員的風格不太相符,因為我們習慣了在IDE里寫了代碼然后打包運行看效果。 雖然現在IDE中已經存在了Solidity插件,可以撰寫智能合約,但是部署智能合約卻要另走他路,沒辦法進行一個快捷的部署與測驗。 如果團隊管理的區塊節點多、 ......

    uj5u.com 2020-09-10 03:03:12 more
  • 谷歌二次驗證碼成為區塊鏈專用安全碼,你怎么看?

    前言 谷歌身份驗證器,前些年大家都比較陌生,但隨著國內互聯網安全的加強,它越來越多地出現在大家的視野中。 比較廣泛接觸的人群是國際3A游戲愛好者,游戲盜號現象嚴重+國外賬號安全應用廣泛,這類游戲一般都會要求用戶系結名為“兩步驗證”、“雙重驗證”等,平臺一般都推薦用谷歌身份驗證器。 后來區塊鏈業務風靡 ......

    uj5u.com 2020-09-10 03:03:17 more
  • 密碼學DAY1

    目錄 ##1.1 密碼學基本概念 密碼在我們的生活中有著重要的作用,那么密碼究竟來自何方,為何會產生呢? 密碼學是網路安全、資訊安全、區塊鏈等產品的基礎,常見的非對稱加密、對稱加密、散列函式等,都屬于密碼學范疇。 密碼學有數千年的歷史,從最開始的替換法到如今的非對稱加密演算法,經歷了古典密碼學,近代密 ......

    uj5u.com 2020-09-10 03:03:50 more
  • 密碼學DAY1_02

    目錄 ##1.1 ASCII編碼 ASCII(American Standard Code for Information Interchange,美國資訊交換標準代碼)是基于拉丁字母的一套電腦編碼系統,主要用于顯示現代英語和其他西歐語言。它是現今最通用的單位元組編碼系統,并等同于國際標準ISO/IE ......

    uj5u.com 2020-09-10 03:04:50 more
  • 密碼學DAY2

    ##1.1 加密模式 加密模式:https://docs.oracle.com/javase/8/docs/api/javax/crypto/Cipher.html ECB ECB : Electronic codebook, 電子密碼本. 需要加密的訊息按照塊密碼的塊大小被分為數個塊,并對每個塊進 ......

    uj5u.com 2020-09-10 03:05:42 more
  • NTP時鐘服務器的特點(京準電子)

    NTP時鐘服務器的特點(京準電子) NTP時鐘服務器的特點(京準電子) 京準電子官V——ahjzsz 首先對時間同步進行了背景介紹,然后討論了不同的時間同步網路技術,最后指出了建立全球或區域時間同步網存在的問題。 一、概 述 在通信領域,“同步”概念是指頻率的同步,即網路各個節點的時鐘頻率和相位同步 ......

    uj5u.com 2020-09-10 03:05:47 more
  • 標準化考場時鐘同步系統推進智能化校園建設

    標準化考場時鐘同步系統推進智能化校園建設 標準化考場時鐘同步系統推進智能化校園建設 安徽京準電子科技官微——ahjzsz 一、背景概述隨著教育事業的快速發展,學校建設如雨后春筍,隨之而來的學校教育、管理、安全方面的問題成了學校管理人員面臨的最大的挑戰,這些問題同時也是學生家長所擔心的。為了讓學生有更 ......

    uj5u.com 2020-09-10 03:05:51 more
  • 位元幣入門

    引言 位元幣基本結構 位元幣基礎知識 1)哈希演算法 2)非對稱加密技術 3)數字簽名 4)MerkleTree 5)哪有位元幣,有的是UTXO 6)位元幣挖礦與共識 7)區塊驗證(共識) 總結 引言 上一篇我們已經知道了什么是區塊鏈,此篇說一下區塊鏈的第一個應用——位元幣。其實先有位元幣,后有的區塊 ......

    uj5u.com 2020-09-10 03:06:15 more
  • 北斗對時服務器(北斗對時設備)電力系統應用

    北斗對時服務器(北斗對時設備)電力系統應用 北斗對時服務器(北斗對時設備)電力系統應用 京準電子科技官微(ahjzsz) 中國北斗衛星導航系統(英文名稱:BeiDou Navigation Satellite System,簡稱BDS),因為是目前世界范圍內唯一可以大面積提供免費定位服務的系統,所以 ......

    uj5u.com 2020-09-10 03:06:20 more
最新发布
  • web3 產品介紹:metamask 錢包 使用最多的瀏覽器插件錢包

    Metamask錢包是一種基于區塊鏈技術的數字貨幣錢包,它允許用戶在安全、便捷的環境下管理自己的加密資產。Metamask錢包是以太坊生態系統中最流行的錢包之一,它具有易于使用、安全性高和功能強大等優點。 本文將詳細介紹Metamask錢包的功能和使用方法。 一、 Metamask錢包的功能 數字資 ......

    uj5u.com 2023-04-20 08:46:47 more
  • Hyperledger Fabric 使用 CouchDB 和復雜智能合約開發

    在上個實驗中,我們已經實作了簡單智能合約實作及客戶端開發,但該實驗中智能合約只有基礎的增刪改查功能,且其中的資料管理功能與傳統 MySQL 比相差甚遠。本文將在前面實驗的基礎上,將 Hyperledger Fabric 的默認資料庫支持 LevelDB 改為 CouchDB 模式,以實作更復雜的資料... ......

    uj5u.com 2023-04-16 07:28:31 more
  • .NET Core 波場鏈離線簽名、廣播交易(發送 TRX和USDT)筆記

    Get Started NuGet You can run the following command to install the Tron.Wallet.Net in your project. PM> Install-Package Tron.Wallet.Net 配置 public reco ......

    uj5u.com 2023-04-14 08:08:00 more
  • DKP 黑客分析——不正確的代幣對比率計算

    概述: 2023 年 2 月 8 日,針對 DKP 協議的閃電貸攻擊導致該協議的用戶損失了 8 萬美元,因為 execute() 函式取決于 USDT-DKP 對中兩種代幣的余額比率。 智能合約黑客概述: 攻擊者的交易:0x0c850f,0x2d31 攻擊者地址:0xF38 利用合同:0xf34ad ......

    uj5u.com 2023-04-07 07:46:09 more
  • Defi開發簡介

    Defi開發簡介 介紹 Defi是去中心化金融的縮寫, 是一項旨在利用區塊鏈技術和智能合約創建更加開放,可訪問和透明的金融體系的運動. 這與傳統金融形成鮮明對比,傳統金融通常由少數大型銀行和金融機構控制 在Defi的世界里,用戶可以直接從他們的電腦或移動設備上訪問廣泛的金融服務,而不需要像銀行或者信 ......

    uj5u.com 2023-04-05 08:01:34 more
  • solidity簡單的ERC20代幣實作

    // SPDX-License-Identifier: GPL-3.0 pragma solidity >=0.7.0 <0.9.0; import "hardhat/console.sol"; //ERC20 同質化代幣,每個代幣的本質或性質都是相同 //ETH 是原生代幣,它不是ERC20代幣, ......

    uj5u.com 2023-03-21 07:56:29 more
  • solidity 參考型別修飾符memory、calldata與storage 常量修飾符C

    在solidity語言中 參考型別修飾符(參考型別為存盤空間不固定的數值型別) memory、calldata與storage,它們只能修飾參考型別變數,比如字串、陣列、位元組等... memory 適用于方法傳參、返參或在方法體內使用,使用完就會清除掉,釋放記憶體 calldata 僅適用于方法傳參 ......

    uj5u.com 2023-03-08 07:57:54 more
  • solidity注解標簽

    在solidity語言中 注釋符為// 注解符為/* 內容*/ 或者 是 ///內容 注解中含有這幾個標簽給予我們使用 @title 一個應該描述合約/介面的標題 contract, library, interface @author 作者的名字 contract, library, interf ......

    uj5u.com 2023-03-08 07:57:49 more
  • 評價指標:相似度、GAS消耗

    【代碼注釋自動生成方法綜述】 這些評測指標主要來自機器翻譯和文本總結等研究領域,可以評估候選文本(即基于代碼注釋自動方法而生成)和參考文本(即基于手工方式而生成)的相似度. BLEU指標^[^?88^^?^]^:其全稱是bilingual evaluation understudy.該指標是最早用于 ......

    uj5u.com 2023-02-23 07:27:39 more
  • 基于NOSTR協議的“公有制”版本的Twitter,去中心化社交軟體Damus

    最近,一個幽靈,Web3的幽靈,在網路游蕩,它叫Damus,這玩意詮釋了什么叫做病毒式營銷,滑稽的是,一個Web3產品卻在Web2的產品鏈上瘋狂傳銷,各方大佬紛紛為其背書,到底發生了什么?Damus的葫蘆里,賣的是什么藥? 注冊和簡單實用 很少有什么產品在用戶注冊環節會有什么噱頭,但Damus確實出 ......

    uj5u.com 2023-02-05 06:48:39 more