3.3 最长公共子序列

news2025/1/13 10:22:00

 


  • 博主简介:一个爱打游戏的计算机专业学生
  • 博主主页: @夏驰和徐策
  • 所属专栏:算法设计与分析

1.什么是子序列?

我的理解:

在字符串或序列中,子序列是指从原始序列中删除零个或多个元素后得到的序列,而且相对顺序保持不变。换句话说,子序列是从原始序列中选择出的元素,它们保持相对顺序,但可以跳过一些元素或者不选择任何元素。

举个例子,考虑序列 "abcd",它的一些子序列包括:"a","b","c","d","ab","ac","ad","bc","bd","cd","abc","abd","acd","bcd","abcd"。注意,这些子序列可能不一定是连续的,可以跳过某些元素。

子序列的长度可以是从0到原始序列长度的任意值。一个长度为 n 的序列,它有 2^n 个不同的子序列,包括空序列(长度为0的子序列)和原始序列本身(长度为 n 的子序列)。

子序列在算法和字符串处理中经常出现,它们可以用于各种问题,如最长公共子序列、最长递增子序列、字符串匹配等。

2.什么是公共子序列?

公共子序列是指在两个或多个序列中都存在的一个序列,它不要求在原序列中是连续的,但要求在每个序列中的相对顺序保持一致。

换句话说,给定两个序列,一个公共子序列是通过从每个序列中选择零个或多个元素形成的序列,而且这些元素在原序列中的相对顺序保持一致。公共子序列中的元素不一定是连续的,可以跳过某些元素。

举个例子,考虑序列 "ABCD" 和 "ACDF",其中一个公共子序列是 "AD"。在第一个序列中,我们选择了第1个和第4个元素;在第二个序列中,我们选择了第1个和第3个元素。这样形成的序列 "AD" 是两个序列的一个公共子序列。

在计算机科学和算法中,最长公共子序列(Longest Common Subsequence,简称 LCS)是公共子序列问题中最经典和常见的形式。在最长公共子序列问题中,我们要找到两个序列中的最长的公共子序列的长度或具体的序列。该问题在字符串处理、比较和匹配等领域中具有广泛应用。

3.什么是最长公共子序列?

最长公共子序列(Longest Common Subsequence,LCS)是指在两个或多个序列中找到的具有最长长度的公共子序列。

给定两个序列,最长公共子序列是通过从每个序列中选择零个或多个元素形成的一个序列,而且这些元素在原序列中的相对顺序保持一致。最长公共子序列中的元素不一定是连续的,可以跳过某些元素。

举个例子,考虑序列 "ABCD" 和 "ACDF",其中一个最长公共子序列是 "AD"。在第一个序列中,我们选择了第1个和第4个元素;在第二个序列中,我们选择了第1个和第3个元素。这样形成的序列 "AD" 是两个序列的一个最长公共子序列。

最长公共子序列问题是计算机科学中经典的问题之一,它具有广泛的应用,如字符串比较、版本控制、DNA序列比对、文本相似性等。解决最长公共子序列问题的常见方法是使用动态规划算法,通过填充一个二维表格来求解最长公共子序列的长度或找到具体的序列。

4.如何证明最长公共子序列问题具有最优子结构性质 

 证明 :

① 用反证法。若z!=Xm,则{z1,z2,....,zk,Xm}是X和Y的长度为k+1的公共子序列。这与 Z是X和Y的最长公共子序列矛盾,因此必有zk=Xm=yn。由此可知,Zk-1是Xm-1和Yn-1的长度大于k-1的公共子序列。若Xm-1和Yn-1,有长度大于k-1的公共子序列W,则将 Xm加在其尾部产生X和Y的长度大于k的公共子序列。此为示盾。所以,Zk-1是Xm-1和Yn-1的最长公共子序列。

② 由于Zk!=Xm,Z是Xm-1和Y的公共子序列,若Xm-1和Y有长度大于k的公共子序列w,则w也是X和Y的长度大于k的公共子序列。这与Z是X和丫的最长公共子序列矛盾。由此可知,Z是 X和Y的最长公共子序列。

③ 証明方法与②类似。 由此可见,两个序列的最长公共子序列包含了这两个序列的前级的最长公共子序列。因此,最长公共子序列问题具有最优子结构性质。

我对于这个证明的理解:

这个证明是关于最长公共子序列(LCS)问题的最优子结构性质的证明。最长公共子序列问题是在两个序列中找到一个最长的共同子序列的问题,而最优子结构性质是指问题的最优解可以通过子问题的最优解来构建。

证明中使用了反证法的思想来说明最优子结构性质。我将按照证明的步骤来解释其意义:

1. 第一步是通过反证法证明 Zk-1 是 Xm-1 和 Yn-1 的最长公共子序列。假设 Zk-1 不是 Xm-1 和 Yn-1 的最长公共子序列,即存在 Xm-1 和 Yn-1 的长度大于 k-1 的公共子序列 W。这是一个假设。

2. 接下来,通过将 Xm 加在 W 的尾部,构造出一个长度大于 k 的公共子序列。这是基于假设得出的结论。

3. 但是,我们知道 Z 是 X 和 Y 的最长公共子序列,所以根据这个事实,这个推导出的长度大于 k 的公共子序列与 Z 的长度相矛盾。因此,假设是错误的,可以推出 Zk-1 是 Xm-1 和 Yn-1 的最长公共子序列。

4. 类似地,通过反证法证明 Z 是 X 和 Y 的最长公共子序列。假设 Z 不是 X 和 Y 的最长公共子序列,即存在 X 和 Y 的长度大于 k 的公共子序列 w。然后通过推导,得出 Z 是 X 和 Y 的最长公共子序列,与假设矛盾。

综上所述,通过反证法的推理过程,证明了最长公共子序列问题具有最优子结构性质。也就是说,可以通过子问题的最优解来构建原问题的最优解。

这个证明表明,对于最长公共子序列问题,我们可以使用动态规划算法来求解,通过解决子问题并利用子问题的最优解来构建整体的最优解。

5.子问题的递归结构(要记)

 

 6.计算最优值的算法实现

功能代码:

void LCSLength(int m,int n,char *x,char *y,int **c,int **b)
{
	int i,j;
	for(i=1;i<=m;i++)
	{
		c[i][0]=0;
	}
	for(i=1;i<=n;i++)
	{
		c[0][i]=0;
	}
	for(i=1;i<=m;i++)
	{
		for(j=1;j<=n;j++)
		{
			if(x[i]==y[j])
			{
				c[i][j]=c[i-1][j-1]+1;
				b[i][j]=1;
			}
			else if(c[i-1][j]>=c[i][j-1])
			{
				c[i][j]=c[i-1][j];
				b[i][j]=2;
			} 
			else 
			{
				c[i][j]=c[i][j-1];
				b[i][j]=3;
			}
		}
	}
}

7.我对这段代码的理解:

这段代码实现了最长公共子序列(LCS)问题的动态规划解法。

函数`LCSLength`接受以下参数:
- `m`:第一个序列的长度。
- `n`:第二个序列的长度。
- `x`:第一个序列的字符数组。
- `y`:第二个序列的字符数组。
- `c`:二维数组,用于存储最长公共子序列的长度。
- `b`:二维数组,用于指示最长公共子序列的构造方式。

代码中的主要逻辑如下:
1. 初始化边界条件:将`c[i][0]`和`c[0][i]`设置为0,表示一个序列为空时,最长公共子序列的长度为0。
2. 遍历两个序列的字符数组,对于每对字符`x[i]`和`y[j]`:
   - 如果`x[i]`等于`y[j]`,说明这两个字符可以作为最长公共子序列的一部分,所以将`c[i][j]`设置为`c[i-1][j-1]+1`,表示在去掉最后一个字符之前的最长公共子序列的基础上加上当前字符。
   - 如果`x[i]`不等于`y[j]`,则需要根据之前计算得到的结果来决定最长公共子序列的长度。比较`c[i-1][j]`和`c[i][j-1]`的大小,如果前者大于等于后者,则将`c[i][j]`设置为`c[i-1][j]`,表示最长公共子序列来自于序列`x`去掉最后一个字符的子序列;否则,将`c[i][j]`设置为`c[i][j-1]`,表示最长公共子序列来自于序列`y`去掉最后一个字符的子序列。
   - 根据`x[i]`和`y[j]`的关系,更新`b[i][j]`的值,用于指示构造最长公共子序列时的操作方式:1表示选择当前字符,2表示选择`x[i-1]`,3表示选择`y[j-1]`。
3. 最终,`c[m][n]`中存储的就是最长公共子序列的长度,`b`数组可以用于构造最长公共子序列。

这段代码通过填充`c`和`b`数组,使用自底向上的动态规划方法,计算出了最长公共子序列的长度和构造方式。

 8.构造最长公共子序列的算法实现

void LCS(int i, int j, char *x, int **b)
{
    if (i == 0 || j == 0) {
        return;
    }
    if (b[i][j] == 1) {
        LCS(i - 1, j - 1, x, b);
        cout << x[i];
    } else if (b[i][j] == 2) {
        LCS(i - 1, j, x, b);
    } else {
        LCS(i, j - 1, x, b);
    }
}

我对这段代码的理解:

- 第2行开始的函数定义,接受参数:整数 `i` 和 `j`,字符指针 `x`,以及二维整数指针 `b`。
- 第4行进行递归终止条件的判断。如果 `i` 或 `j` 为0,即其中一个序列为空,说明已经遍历完了一个序列的所有字符,所以终止递归。
- 第7行开始的 `if` 语句处理当 `b[i][j]` 的值为1的情况。这表示当前字符 `x[i]` 是最长公共子序列的一部分。
- 第8行递归调用 `LCS` 函数,并将索引 `i-1` 和 `j-1` 作为参数,向前移动到上一个位置。
- 第9行输出当前字符 `x[i]`,即最长公共子序列中的一个字符。
- 第11行开始的 `else if` 语句处理当 `b[i][j]` 的值为2的情况。这表示最长公共子序列来自于 `x` 的前一个位置。
- 第12行递归调用 `LCS` 函数,并将索引 `i-1` 和 `j` 作为参数,向前移动到上一个位置。
- 第14行开始的 `else` 语句处理当 `b[i][j]` 的值既不是1也不是2的情况。这表示最长公共子序列来自于 `y` 的前一个位置。
- 第15行递归调用 `LCS` 函数,并将索引 `i` 和 `j-1` 作为参数,向前移动到上一个位置。

这段代码通过递归的方式,根据 `b` 数组的值不断向前移动,从而构造出最长公共子序列。递归终止条件是当其中一个序列为空时,说明已经遍历完了所有的字符。在每一步递归中,根据 `b` 数组的值选择相应的操作,递归调用 `LCS` 函数,最终输出构造完成的最长公共子序列。

 5.算法的改进

如果我们省略数组 `b`,可以改进算法以减少空间复杂度。以下是改进后的算法:


void LCSLength(int m, int n, char *x, char *y, int **c)
{
    int i, j;
    
    for (i = 1; i <= m; i++) {
        c[i][0] = 0;
    }
    for (j = 0; j <= n; j++) {
        c[0][j] = 0;
    }
    
    for (i = 1; i <= m; i++) {
        for (j = 1; j <= n; j++) {
            if (x[i] == y[j]) {
                c[i][j] = c[i - 1][j - 1] + 1;
            } else {
                c[i][j] = max(c[i - 1][j], c[i][j - 1]);
            }
        }
    }
}

void LCS(int i, int j, char *x, char *y, int **c)
{
    if (i == 0 || j == 0) {
        return;
    }
    if (x[i] == y[j]) {
        LCS(i - 1, j - 1, x, y, c);
        cout << x[i];
    } else {
        if (c[i - 1][j] >= c[i][j - 1]) {
            LCS(i - 1, j, x, y, c);
        } else {
            LCS(i, j - 1, x, y, c);
        }
    }
}

改进后的算法中,我们省略了数组 `b`,直接使用一个二维数组 `c` 来记录最长公共子序列的长度。在 `LCSLength` 函数中,我们计算并填充了数组 `c`,而在 `LCS` 函数中,我们根据 `c` 数组的值进行回溯。

在 `LCS` 函数中,我们首先检查当前位置的字符是否相等。如果相等,则说明当前字符是最长公共子序列的一部分,我们进行递归调用并输出该字符。如果不相等,则根据 `c` 数组的值选择向左移动或向上移动,继续进行递归调用。通过这种方式,我们可以构造出最长公共子序列。这种改进后的算法在空间复杂度上更为高效,但仍能达到相同的结果。

 

算法实现:

C语言实现

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

int max(int a, int b) {
    return (a > b) ? a : b;
}

void LCS(char *X, char *Y, int m, int n) {
    int L[m + 1][n + 1];
    int i, j;

    // 初始化第一行和第一列为0
    for (i = 0; i <= m; i++) {
        L[i][0] = 0;
    }
    for (j = 0; j <= n; j++) {
        L[0][j] = 0;
    }

    // 填充 L 数组
    for (i = 1; i <= m; i++) {
        for (j = 1; j <= n; j++) {
            if (X[i - 1] == Y[j - 1]) {
                L[i][j] = L[i - 1][j - 1] + 1;
            } else {
                L[i][j] = max(L[i - 1][j], L[i][j - 1]);
            }
        }
    }

    // 打印最长公共子序列
    int index = L[m][n];
    char lcs[index + 1];
    lcs[index] = '\0';  // 设置字符串结尾
    i = m;
    j = n;
    while (i > 0 && j > 0) {
        if (X[i - 1] == Y[j - 1]) {
            lcs[index - 1] = X[i - 1];
            i--;
            j--;
            index--;
        } else if (L[i - 1][j] > L[i][j - 1]) {
            i--;
        } else {
            j--;
        }
    }

    // 打印最长公共子序列
    printf("Longest Common Subsequence: %s\n", lcs);
}

int main() {
    char X[] = "AGGTAB";
    char Y[] = "GXTXAYB";
    int m = strlen(X);
    int n = strlen(Y);

    LCS(X, Y, m, n);

    return 0;
}

 C++实现:


#include <iostream>
#include <vector>
#include <string>

using namespace std;

string LCS(string X, string Y) {
    int m = X.length();
    int n = Y.length();

    vector<vector<int>> dp(m + 1, vector<int>(n + 1, 0));

    // 填充 dp 数组
    for (int i = 1; i <= m; i++) {
        for (int j = 1; j <= n; j++) {
            if (X[i - 1] == Y[j - 1]) {
                dp[i][j] = dp[i - 1][j - 1] + 1;
            } else {
                dp[i][j] = max(dp[i - 1][j], dp[i][j - 1]);
            }
        }
    }

    // 构造最长公共子序列
    string lcs;
    int i = m, j = n;
    while (i > 0 && j > 0) {
        if (X[i - 1] == Y[j - 1]) {
            lcs = X[i - 1] + lcs;
            i--;
            j--;
        } else if (dp[i - 1][j] > dp[i][j - 1]) {
            i--;
        } else {
            j--;
        }
    }

    return lcs;
}

int main() {
    string X = "AGGTAB";
    string Y = "GXTXAYB";

    string longestCommonSubsequence = LCS(X, Y);

    cout << "Longest Common Subsequence: " << longestCommonSubsequence << endl;

    return 0;
}

这个示例代码中,我们使用了一个二维向量 `dp` 来保存最长公共子序列的长度。我们首先初始化向量的大小为 `(m+1) x (n+1)`,并将所有元素初始化为0。然后使用两个嵌套的循环来填充向量 `dp`。在填充过程中,我们比较两个字符是否相等,如果相等则当前位置的值为左上角位置的值加1,否则取左方和上方两个位置的较大值。最后,我们根据 `dp` 向量的结果构造出最长公共子序列,并将其返回。 

这个示例代码同样可以作为一个基础的起点,你可以根据具体的需求和问题进行修改和优化。

原理图:

 

总结:

最长公共子序列问题的重点、难点和易错点可以总结如下:

重点:
1. 理解问题:理解最长公共子序列问题的定义和要求,即在给定的两个序列中找到最长的共同子序列。
2. 动态规划思想:掌握动态规划的基本思想,将问题分解为子问题,并通过存储和重复利用子问题的解来提高效率。
3. 最优子结构性质:最长公共子序列问题具有最优子结构性质,即最优解可以由子问题的最优解构成。
4. 递推关系:找到递推关系式,将问题的解表示为子问题的解的组合。

难点:
1. 状态定义:确定问题的状态,即如何定义动态规划的状态。在最长公共子序列问题中,常用的状态是两个序列的前缀,或者是两个序列的某个位置。
2. 状态转移方程:确定状态之间的转移关系,即如何从一个状态转移到下一个状态。在最长公共子序列问题中,状态转移方程通常涉及比较两个序列的元素是否相等,并根据比较结果进行转移。
3. 边界条件:确定边界条件,即初始状态的值或边界状态的处理。在最长公共子序列问题中,通常需要初始化第一行和第一列的状态值。

易错点:
1. 状态索引:注意索引的起始位置和边界条件的处理。索引错误可能导致数组越界或计算错误的状态值。
2. 状态转移方程的正确性:在确定状态转移方程时,要确保它能正确地推导出问题的最优解,包括考虑相等和不相等两种情况。
3. 递归与迭代的区别:最长公共子序列问题可以通过递归或迭代的方式解决。理解它们之间的差异和实现细节是避免出错的关键。

解决最长公共子序列问题需要综合运用动态规划的思想和技巧,同时注意处理好状态定义、状态转移方程和边界条件,以确保算法的正确性和有效性。通过充分理解重点、克服难点并避免易错点,可以更好地解决这类问题。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/598355.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Spring Boot如何实现分布式事务的协调和管理

Spring Boot如何实现分布式事务的协调和管理 在分布式系统中&#xff0c;事务是非常重要的一部分&#xff0c;可以保证多个操作在一个原子性的操作中完成&#xff0c;确保数据的一致性和可靠性。在分布式系统中&#xff0c;分布式事务需要考虑多个服务之间的数据一致性和事务提…

论文排版!

目录 Visio画图后&#xff0c;粘贴到word白边太宽&#xff1f; 【IEEE论文投稿word中双栏情况下插入单栏效果图片】 论文排版之Word双栏排版问题解决 Word公式居中&#xff0c;公式序号靠右&#xff08;制表位实现&#xff09; 请问word如何实现这种长公式的排版&#xff1…

MVCC 实现原理

&#x1f49f;这里是CS大白话专场&#xff0c;让枯燥的学习变得有趣&#xff01; &#x1f49f;没有对象不要怕&#xff0c;我们new一个出来&#xff0c;每天对ta说不尽情话&#xff01; &#x1f49f;好记性不如烂键盘&#xff0c;自己总结不如收藏别人&#xff01; &#x1f…

Pycharm 安装PyQt5

第一步: 新建项目文件夹 新建项目 新建一个 Pyqt5Tools 的文件夹 这里放到了 D盘 安装成功 第二步: 安装相关包 安装PyQt5 pip3 install -i https://pypi.douban.com/simple PyQt5,PyQt5-tools时间根据网速 更新下pip python.exe -m pip install --upgrade pip -i https:…

tensorRT推理相关及遇到的坑

tebsorRT是什么 tensorRT是NVIDIA出的一个高性能深度学习推理&#xff08;inference&#xff09;优化器&#xff0c;可以为深度学习应用提供低延迟、高吞吐率的部署推理。TensorRT可用于对超大规模数据中心、嵌入式平台或自动驾驶平台进行推理加速。TensorRT现已能支持TensorF…

数据库隔离级别

1. 数据库隔离级别 1.1 事务 事务只是一个改变&#xff0c;是一些操作的集合&#xff1b;用专业的术语讲&#xff0c;他就是一个程序的执行单元&#xff1b;事务本身其实并不包含这4个特性&#xff0c;只是我们需要通过某些手段&#xff0c;尽可能的让这个执行单元满足这四个特…

Java学习路线(20)——多线程

一、线程&#xff08;Thread&#xff09; 1、概念&#xff1a; 是一个程序内部的一条执行路径 2、分类 单线程&#xff1a; 程序中只有一条执行路径多线程&#xff1a; 程序中有多条执行路径 二、多线程的创建 1、Thread的概念&#xff1a; Java通过java.lang.Thread类代表…

HTTP 教程2

HTTP 消息结构 HTTP是基于客户端/服务端&#xff08;C/S&#xff09;的架构模型&#xff0c;通过一个可靠的链接来交换信息&#xff0c;是一个无状态的请求/响应协议。 一个HTTP"客户端"是一个应用程序&#xff08;Web浏览器或其他任何客户端&#xff09;&#xff…

什么是C/S架构?与B/S架构有什么区别?

1、 1.1、C/S架构的全称是Client/Server&#xff0c;即客户端/服务器体系结构&#xff0c;主要应用于局域网内。 1.2、B/S架构的全称为Browser/Server&#xff0c;即浏览器/服务器结构。百度安全验证https://baijiahao.baidu.com/s?id1742761249590653499&wfrspider&f…

微信小程序——CSS限制文字宽度和行数(溢出显示省略号)

手把手教你学会判断用户在做向上滑动还是向下滑动 知识回调&#xff08;不懂就看这儿&#xff01;&#xff09;场景复现核心干货单行文本溢出显示省略号多行文本溢出显示省略号 知识回调&#xff08;不懂就看这儿&#xff01;&#xff09; 知识专栏专栏链接微信小程序专栏http…

基于HTML+CSS+JavaScript的在线图书阅读网页设计

目 录 1.项目总体设计 2 1.1需求分析 2 1.2网站结构分析 3 1.2.1导航栏 3 1.2.2主体部分 3 1.3网络风格分析 3 1.4网站结构图 3 2.项目详细设计 4 2.1登录页面设计 4 2.2主页页面设计 5 2.3在线读书页面设计 6 3.项目总结 7 4.参考文献 7 此网页能够满足喜欢看书的书友&#x…

深度学习 - 53.Bert 简介与 Keras-Bert 常用示例

目录 一.引言 二.Bert 简介 1.Embedding Layer 2.Encoder layer 3.Pre-training 与 Fine-Tuning 三.Keras-Bert 常用 Demo 1.获取预训练模型 2.加载预训练模型 3.Fill Text 4.IsCorrelation 5.Get Embedding 6.完整代码 四. Fine-Tuning 五.Bert VS OpenAI GPT …

chatgpt赋能python:Python中等待一秒的语句:让你的程序暂停与等待

Python中等待一秒的语句&#xff1a;让你的程序暂停与等待 当编写Python程序时&#xff0c;经常需要添加暂停或延迟功能&#xff0c;以使程序能够在执行某些操作之前或之后等待一段时间。Python拥有一个内置的语句可以实现这种延迟&#xff1a;time.sleep()。 什么是 time.sl…

四信大气环保远程监测平台,实现大气网格化、精准化监测

近年来&#xff0c;随着国民经济快速发展&#xff0c;我国工业化、城镇化进程加快&#xff0c;随之造成的大气污染问题日益严峻&#xff0c;严重影响人们日常生活幸福指数与身体健康。为此&#xff0c;中共中央、国务院先后发布一系列文件&#xff0c;要求加强大气环境监测&…

价格限制与经济福利

价格控制 实行价格控制通常是政府相信市场价格对买方或卖方不公平两种价格控制&#xff1a;价格上限和价格下限 价格上限&#xff1a; 法定最高价格&#xff0c;任何人不得收取或付出高于此的价格。 价格下限&#xff1a; 法定最低价格&#xff0c;任何人不得收取或付出低于…

为什么亚马逊股价会在今年上涨?亚马逊股价2023年还会继续上涨吗?

来源&#xff1a;猛兽财经 作者&#xff1a;猛兽财经 为什么亚马逊的股价会上涨&#xff1f; 今年以来&#xff0c;亚马逊&#xff08;AMZN&#xff09;的股价已经上涨了20%以上&#xff0c;涨幅达到了23.3%。而同期标普500指数今年以来仅上涨了8.2%。 猛兽财经认为&#xff0c…

数据链路层:媒体接入控制

1.数据链路层&#xff1a;媒体接入控制 笔记来源&#xff1a; 湖科大教书匠&#xff1a;媒体接入控制的基本概念 湖科大教书匠&#xff1a;随机接入–CSMA/CD协议 湖科大教书匠&#xff1a;随机接入–CSMA/CA协议 声明&#xff1a;该学习笔记来自湖科大教书匠&#xff0c;笔记…

微信小程序——监听页面滑动(二)判断用户在做向上滑动还是向下滑动(onScrollPage scroll-view)

手把手教你学会判断用户在做向上滑动还是向下滑动 知识回调&#xff08;不懂就看这儿&#xff01;&#xff09;场景复现核心干货onPageScroll结合scrollTop实现了解touch方法 知识回调&#xff08;不懂就看这儿&#xff01;&#xff09; 知识专栏专栏链接微信小程序专栏https:…

1.WebGL与Shader介绍

webgl介绍 WebGL是一种用于在网页浏览器中创建交互式3D图形的技术。它基于OpenGL ES 2.0&#xff0c;这是一个广泛使用的嵌入式系统3D图形API。以下是webgl的发展史&#xff1a; WebGL允许开发人员使用JavaScript编写代码来控制GPU&#xff08;图形处理单元&#xff09;&…

字符集、字符编码格式检测和转码

目录 1 locale与字符集 1.1 locale 1.2 字符集 2 常见字符集 2.1 Native ANSI 字符集 2.1.1 ASCII 2.1.2 ISO-8859-1 2.1.3 GB2312&#xff0c;GBK&#xff0c;GB18030 2.2 Unicode 字符集 2.2.1 UCS 2.2.2 UTF - Unicode Transformation Format 2.2.3 UTF-8 2.2.4 B…