【蓝桥杯软件赛 零基础备赛20周】第6周——栈

news2024/11/19 8:52:50

文章目录

  • 1. 基本数据结构概述
    • 1.1 数据结构和算法的关系
    • 1.2 线性数据结构概述
    • 1.3 二叉树简介
  • 2. 栈
    • 2.1 手写栈
    • 2.2 C++STL栈
    • 2.3 Java 栈
    • 2.4 Python栈
  • 3 习题

1. 基本数据结构概述

很多计算机教材提到:程序 = 数据结构 + 算法。

“以数据结构为弓,以算法为箭”

数据结构是是计算机存储、组织数据的方法。常用的数据结构有:数组(Array)、栈(Stack)、队列(Queue)、链表(Linked List)、树(Tree)、图(Graph)、堆(Heap)、散列表(Hash)等。分为两大类:线性表、非线性表。数组、栈、队列、链表是线性表,其他是非线性表。

1.1 数据结构和算法的关系

数据结构和算法往往密不可分。下面以图的存储为例,说明数据结构和算法的关系。这几种存图的数据结构,各有优缺点,也各有自己的应用场景。

(1)边集数组

定义结构体数组:

struct Edge{
     int u, v, w;
}edges[M];

其中(u,v,w)表示一条边,起点是u,终点是v,边长是w。edges[M]可以存M条边。

边集数组的优点:是最节省空间的存图方法,存储空间不可能再少了。n=1000个点,m=5000条边的图,使用的存储空间是12×5000 = 60KB。

边集数组的缺点:不能快速定位某条边。如果要找某点u和哪些点有边连接,得把整个edges[M]从头到尾搜一遍才能知道。

边集数组的的应用场景:如果算法不需要查找特定的边,就用边集数组。例如最小生成树Kruskal算法、最短路径Bellman-ford算法。

(2)邻接矩阵

定义一个二维数组:

	int edge[N][N];

其中edge[i][j]表示点i和点j之间有一条边,边长为edge[i][j]。它可以存N个点的边。若edge[i][j]=0,表示i和j之间没有边;若edge[i][j] != 0,i和j之间有边,边长为edge[i][j]。

邻接矩阵的优点:能极快地查询任意两点之间是否有边。如果edge[i][j] != 0,说明点i和j之间有一条边,边长edge[i][j]。

邻接矩阵的缺点:如果图是一张稀疏图,大部分点和边之间没有边,那么邻接矩阵很浪费空间,因为大多数edge[][]=0,没有用到。n=1000个点,m=5000条边的图,使用的存储空间是12×1000×1000 = 12MB。

邻接矩阵的应用场景:(1)稠密图,几乎所有的点之间都有边,edge[][]数组几乎用满了,很少浪费;(2)算法需要快速查找边,而且计算结果和任意两点的关系有关,例如最短路径算法的Floyd算法。

(3)邻接表

邻接矩阵的优点:十分节省空间,因为只存储存在的边。

邻接矩阵的缺点:没有明显的缺点。它的存储空间只比边集数组大一点点,而查询边的速度只比邻接矩阵慢一点点。

邻接矩阵应用场景:基于稀疏图的大部分算法。

1.2 线性数据结构概述

在所有数据结构中,线性表是最简单的。线性表有数组、链表、队列、栈,它们有一个共同的特征:把同类型的数据一个接一个地串在一起。

下面对线性表做个概述,并比较它们的优缺点。

(1)数组

数组是最最简单的数据结构,它的逻辑结构和物理内存的存储完全一样。例如C语言中定义一个整型数组int a[10],系统会分配一个40字节的存储空间,这100个字符的存储地址是连续的。

#include <bits/stdc++.h>
using namespace std;
int main(){
    int a[10];
    for (int i=0;i<10;i++)  
         cout << &a[i] << " ";  //打印10个整数的存储地址
    return 0;
}

在作者机器上运行,输出10个整数的存储地址:
0x6dfec4 0x6dfec8 0x6dfecc 0x6dfed0 0x6dfed4 0x6dfed8 0x6dfedc 0x6dfee0 0x6dfee4 0x6dfee8

数组的优点:(1)简单,容易编程;(2)访问快捷,要定位到某个数据,只需使用下标即可,例如a[0]是第1个数据,a[i]是第i-1个数据;(3)与某些应用场景直接对应,例如数列是一维数组,可以在一维护数组上排序,矩阵是二维数组,表示空间坐标,等等。

数组的缺点:删除和增加数据很麻烦,非常耗时。例如要删除数组int a[10]的第5个数据,只能采用覆盖的方法,从第6个数据开始,每个往前挪一位。增加数据也麻烦,例如要在第5个位置插入一个数据,只能把原来第5个开始的数据逐个往后挪一位,空出第5个位置给新数据。

(2)链表

链表是为了克服数组的缺点提出的一种线性表,链表的插入和删除操作,不需要挪动其他数据。简单地说,链表是“是用指针串起来的数组”。链表的数据不是连续存放的,而是用指针串起来的。例如下图删除链表的第3个数据,只要把原来连接第3个数据的指针断开,然后连接它前后的数据即可,不用挪动其他的数据。

在这里插入图片描述

链表的优点:增加和删除数据很便捷。这个优点弥补了数组的缺点。

链表的缺点:定位某个数据比较麻烦。例如要输出第5个数据,需要从链表头开始,沿着指针一步步走,找到第5个。链表的这个缺点却是数组的优点。

链表和数组的优缺点正好相反,它们的应用场合不同,数组适合静态数据,链表适合动态数据。

链表如何编程实现?在常见的数据结构教材中,链表的数据节点是动态分配的,各节点之间用指针来连接。但是在算法竞赛中,如果手写链表,一般不用动态分配,而是用静态数组来模拟。

手写链表的代码见题目:排队顺序

题解

#include<bits/stdc++.h> 

using namespace std;

int arr[1000010];
int main()
{
    int n;
    cin >> n;
    for (int i = 1; i <= n; i++) {
        cin >> arr[i];
    }
    int head;
    cin >> head;
    cout << head;
    while (arr[head] != 0) {
        cout << " " << arr[head];
        head = arr[head];
    }
    return 0;
}

当然,除非必要,一般不手写链表,而是用系统提供的链表,例如C++ STL的list,Java LinkedList,Python的list。

链表在蓝桥杯等算法竞赛中不太常用,所以本章没有详细介绍。

(3)队列

队列是线性数据的一种使用方式,模拟现实世界的排队操作。例如排队购物,只能从队头离开队伍,新来的人只能排到队尾,不能插队。队列有一个出口和一个入口,出口是队头,入口是队尾。队列的编程实现,可以用数组,也可以用链表。

队列这种数据结构无所谓优缺点,只有适合不适合。例如宽度优先搜索BFS,就是基于队列的,用其他数据结构都不合适。

(4)栈

栈也是线性数据的一种使用方式,模拟现实世界的单出入口。例如一管泡腾片,先放进去的泡腾片后出来。栈的编程比队列更简单,同样可以用数组或链表实现。

栈有它的使用场合,例如递归使用栈来处理函数的自我调用过程。

1.3 二叉树简介

二叉树是一种高度组织性、高效率的数据结构。例如在一棵有n个节点的满二叉树上定位某个数据,只需走logn步,插入和删除某个数据也只需logn步。在二叉树的基础上发展出了很多高级数据结构和算法。大多数高级数据结构,例如树状数组、线段树、树链剖分、平衡树、动态树等,都是基于二叉树的。

2. 栈

栈(stack)是比队列更简单的数据结构,它的特点是“先进后出”。

队列有两个口,一个入口和一个出口。而栈只有唯一的一个口,既从这个口进入,又从这个口出来。所以如果自己写栈的代码,比队列的代码更简单。

2.1 手写栈

如果使用环境简单,最简单的手写栈代码用数组实现。

const int N = 300008;                        //定义栈的大小
struct mystack{
    int a[N];                                //存放栈元素,从a[0]开始
    int t = -1;                              //栈顶位置,初始栈为空,置初值为-1
    void push(int data){ a[++t] = data; }    //把元素data送入栈
    int top()   { return a[t]; }             //读栈顶元素,不弹出
    void pop()  { if(t>-1) t--;}             //弹出栈顶
    int size()  { return t+1;}               //栈内元素的数量
    int empty() { return t==-1 ? 1:0; }      //若栈为空返回1
};

使用栈时要注意不能超过栈的空间。上面第1行定义了栈的大小是N,栈内的元素数量不要超过它。

用下面的例子给出上述手写代码的应用。

例题:表达式括号匹配

题解:

合法的括号串例如“(())”、“()()()”,像“)(()”这样是非法的。合法括号组合的特点是:左括号先出现,右括号后出现;左括号和右括号一样多。

括号组合的合法检查是栈的经典应用。用一个栈存储所有的左括号。遍历字符串的每一个字符,处理流程是:
(1)若字符是 ‘(’,进栈。
(2)若字符是’)',有两种情况:如果栈不空,说明有一个匹配的左括号,弹出这个左括号,然后继续读下一个字符;如果栈空了,说明没有与右括号匹配的左括号,字符串非法,输出NO,程序退出。
(3)读完所有字符后,如果栈为空,说明每个左括号有匹配的右括号,输出YES,否则输出NO。

C++代码

手写栈:

#include <bits/stdc++.h>
using namespace std;
const int N = 300008;                        //定义栈的大小
struct mystack{
    int a[N];                                //存放栈元素,从a[0]开始
    int t = -1;                              //栈顶位置,初始栈为空,置初值为-1
    void push(int data){ a[++t] = data; }    //把元素data送入栈
    int top()   { return a[t]; }         //读栈顶元素,不弹出
    void pop()  { if(t>-1) t--;     }        //弹出栈顶
    int size()  { return t+1;}            //栈内元素的数量
    int empty() {return t==-1 ? 1:0; }   //若栈为空返回1
}st;
int main(){
    char x;
	while(cin>>x){   //循环输入
	    if(x=='@') break;       //输入为@停止;
		if(x=='(') st.push(x);  //左括号入栈
		if(x==')'){             //遇到一个右括号
            if(st.empty()) {cout<<"NO";return 0;} //栈空,没有左括号与右括号匹配
            else st.pop();      //匹配到一个左括号,出栈
		}
	}
	if(st.empty()) cout<<"YES"; //栈为空,所有左括号已经匹配到右括号,输出yes
	else cout<<"NO";
	return 0;
}

STL:

#include<bits/stdc++.h> 
using namespace std;
string str;
stack<char> ops;
int main()
{
    cin >> str;
    long res = 0;
    for (int i = 0; i < str.length(); i++) {
        char c = str[i];
        if (c == '(') {
            ops.push(c);
        }
        else if (c == ')') {
            if (!ops.empty()) {
                ops.pop();
            }
            else {
                cout << "NO";
                return 0;
            }
        }
    }
    if (!ops.empty()) {
        cout << "NO";
        return 0;
    }
    cout << "YES";
    return 0;
}

Java代码

import java.util.Scanner;
public class Main {
    static final int N = 300008;
    static class mystack {
        int[] a = new int[N];
        int t = -1;
        void push(int data) {  a[++t] = data;    }
        int top() {   return a[t];        }
        void pop(){   if(t > -1) t--;     }
        int size(){   return t + 1;       }
        boolean  empty() {   return t == -1 ? true : false; }
    }
    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        mystack st = new mystack();
        String s = sc.next();
        for (int i = 0; i < s.length(); i++) {
            char x = s.charAt(i);
            if(x == '@') break;
            if(x == '(') st.push(x);
            if(x == ')') {
                if(st.empty()) {
                    System.out.println("NO");
                    return;
                }
                else st.pop();
            }
        }
        if(st.empty()) System.out.println("YES");
        else System.out.println("NO");
    }
}

Python代码

Python的手写栈用到了list。用list模拟栈有一个好处,不用担心栈空间不够大,因为list自动扩展空间。而且list的栈操作非常快,因为栈顶是list的末尾元素,栈只有一个出入口,只在list的末尾进行进栈和出栈操作,操作极为快捷。下面是list实现的栈功能。

在这里插入图片描述

下面是例题的Python代码,栈用list模拟。

st = []             #定义栈,用list实现
flag =  True        #判断左括号和右括号的数量是否一样多
s = input().strip()
for x in s:
    if x=='(':  st.append("(")    #进栈
    if x==")":
        if len(st)!=0:            #len():栈的长度
            st.pop()              #出栈,也可以写成 del st[-1] ,st[-1]是栈顶
        else:                     #栈已空,没有匹配的左括号
           flag = False
           break
if len(st)==0 and flag:  print('YES')
else:                    print('NO')

2.2 C++STL栈

竞赛时一般不自己手写栈,而是用STL stack:https://wyqz.top/p/870124582.html#toc-heading-9。

STL stack的主要操作见下表。

在这里插入图片描述
用下面的例题说明STLqueue的应用

例题:排列

题解:

把符合条件的一对<i, j>称为一个“凹”。首先模拟检查“凹”,了解执行的过程。以“3 1 2 5”为例,其中的“凹”有:“3-1-2”和“3-1-2-5”;还有相邻的“3-1”、“1-2”、“2-5”。一共5个“凹”,总价值13。

像“3-1-2”和“3-1-2-5”这样的“凹”,需要检查连续3个以上的数字。

例如“3 1 2”,从“3”开始,下一个应该比“3”小,例如“1”,再后面的数字比“1”大,才能形成“凹”。

再例如“3-1-2-5”,前面的“3-1-2”已经是“凹”了,最后的“5”也会形成新的“凹”,条件是这个“5”必须比中间的“1-2”大才行。

总结上述过程是:先检查“3”;再检查“1”符合“凹”;再检查“2”,比前面的“1”大,符合“凹”;再检查“5”,比前面的“2”大,符合“凹”。

以上是检查一个“凹”的两头,还有一种是“嵌套”。一旦遇到比前面小的数字,那么以这个数字为头,可能形成新的“凹”。例如“6 4 2 8”,其中的“6-4-2-8”是“凹”,内部的“4-2-8”也是凹。如果学过递归、栈,就会发现这是嵌套,所以本题用栈来做很合适。

以“6 4 2 8”为例,用栈模拟找“凹”。当新的数比栈顶的数小,就进栈;如果比栈顶的数大,就出栈,此时找到了一个“凹”并计算价值。下图中的圆圈数字是数在数组中的下标位置,用于计算题目要求的价值。

在这里插入图片描述
图(1):6进栈。
图(2):4准备进栈,发现比栈顶的6小,说明可以形成凹,4进栈。
图(3):2准备进栈,发现比栈顶的4小,说明可以形成凹,2进栈。
图(4):8准备进栈,发现比栈顶的2大,这是一个凹“4-2-8”,对应下标“②–④”,弹出2,然后计算价值,j-i+1=④-②+1=3。
图(5):8准备进栈,发现比栈顶的4大,这是一个凹“6-4-8”,对应下标“①–④”,弹出4,然后计算价值,j-i+1=④-①+1=4。
图(6):8终于进栈了,数字也处理完了,结束。

在上述过程中,只计算了长度为3和3以上的凹,并没有计算题目中“(3)a[i]-a[j]之间不存在其他数字”的长度为2的凹,所以最后统一加上这种情况的价值(n-1)×2 = 6。

最后统计得“6 4 2 8”的总价值是3+4+6=13。

C++代码

#include <bits/stdc++.h>
using namespace std;
const int N = 300008;
int a[N];                 //这里a[]是题目的数字排列
int main(){
    int n;  cin>>n;
    for(int i = 1;i <= n;i++)  cin>>a[i];   //输入数列
    stack <int> st;                   //定义栈
    long long ans = 0;
    for(int i = 1;i <= n;i++){
        while(!st.empty() && a[st.top()] < a[i]){
            st.pop();
            if(!st.empty()){
                int last = st.top();
                ans += (i - last + 1);
            }
        }
        st.push(i);
    }
    ans += (n - 1) * 2;              //(3)a[i]-a[j]之间不存在其他数字的情况
    cout<<ans;
}

2.3 Java 栈

Java Stack https://docs.oracle.com/en/java/javase/14/docs/api/java.base/java/util/Stack.html

有以下操作。

在这里插入图片描述

例题 排列 的Java代码。

import java.util.Scanner;
import java.util.Stack;
public class Main {
    static final int N = 300008;
    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        int n = sc.nextInt();
        int[] a = new int[N];
        for(int i = 1; i <= n; i++)  a[i] = sc.nextInt();        
        Stack<Integer> st = new Stack<>();
        long ans = 0;
        for(int i = 1; i <= n; i++) {
            while(!st.empty() && a[st.peek()] < a[i]) {
                st.pop();
                if(!st.empty()) {
                    int last = st.peek();
                    ans += (long)(i - last + 1);
                }
            }
            st.push(i);
        }
        ans += (n - 1) * 2;
        System.out.println(ans);
    }
}

2.4 Python栈

python的栈可以用以下三种方法实现:(1)list;(2)deque;(3)LifoQueue。比较它们的运行速度,list和deque一样快,而LifoQueue慢得多,建议使用list。前面已经介绍了用list实现栈的方法。

下面是例题 排列 的代码。

n = int(input())
a = [int(x) for x in input().split()]
st = []                                       #定义栈,用list实现
ans = 0
for i in range(n):
    while len(st) != 0 and a[st[-1]] < a[i]:  #st[-1]是栈顶
        st.pop()                              #弹出栈顶
        if len(st) != 0:
            last = st[-1]                     #读栈顶
            ans += (i - last + 1)
    st.append(i)                              #进栈
ans += (n - 1) * 2
print(ans)

3 习题

表达式括号匹配

表达式求值

小鱼的数字游戏

后缀表达式

【模板】栈

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

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

相关文章

使用YOLOv8训练自己的数据集

&#x1f368; 本文为&#x1f517;365天深度学习训练营 中的学习记录博客&#x1f356; 原作者&#xff1a;K同学啊 | 接辅导、项目定制 拉取项目 git clone https://github.com/ultralytics/ultralytics安装依赖 cd ultralytics pip install -r requirement.txt pip instal…

某60区块链安全之薅羊毛攻击实战二学习记录

区块链安全 文章目录 区块链安全薅羊毛攻击实战二实验目的实验环境实验工具实验原理实验内容薅羊毛攻击实战二 实验步骤EXP利用 薅羊毛攻击实战二 实验目的 学会使用python3的web3模块 学会分析以太坊智能合约复杂场景下薅羊毛攻击漏洞及其利用 找到合约漏洞进行分析并形成利…

springboot+vue志愿者在线报名服务管理系统java毕业设计源码+数据库

vuespringboot志愿服务管理系统 本项目是springbootvueElementuimysql源码 开发工具&#xff0c;idea和eclipse都可以,MySQL 源码下载地址 https://download.csdn.net/download/yibo2022/88401958?spm1003.2166.3001.6637.3https://download.csdn.net/download/yibo2022/884…

深入理解Zookeeper系列-1.初识Zoookeeper

&#x1f44f;作者简介&#xff1a;大家好&#xff0c;我是爱吃芝士的土豆倪&#xff0c;24届校招生Java选手&#xff0c;很高兴认识大家&#x1f4d5;系列专栏&#xff1a;Spring源码、JUC源码、Kafka原理、分布式技术原理&#x1f525;如果感觉博主的文章还不错的话&#xff…

LLM算法工程师面试题总结

一、请简述对大模型的基本原理和架构的理解。 大型语言模型如GPT&#xff08;Generative Pre-trained Transformer&#xff09;系列是基于自注意力机制的深度学习模型&#xff0c;主要用于处理和生成人类语言。下面简要概述了它们的一些基本原理和架构特点&#xff1a; 基本原…

springBoot3.2 + jdk21 + GraalVM上手体验

springBoot3.2 jdk21 GraalVM上手体验 SpringBoot2.x官方已经停止维护了&#xff0c;jdk8这次真的得换了&#x1f923; 可以参考官方文章进行体验&#xff1a;https://spring.io/blog/2023/09/09/all-together-now-spring-boot-3-2-graalvm-native-images-java-21-and-virt…

对el-button封装使用

x改变el-button样式并且多处使用 el-button封装&#xff1a; <template><el-buttonclass"my-btn":style"{ width }":disabled"disabled"click"myClick"mouseenter"person.active true"mouseleave"person.ac…

计算机视觉(OpenCV+TensorFlow)

计算机视觉&#xff08;OpenCVTensorFlow&#xff09; 文章目录 计算机视觉&#xff08;OpenCVTensorFlow&#xff09;前言3.图像金字塔3.1 高斯金字塔3.2 拉普拉斯金字塔 4.图像轮廓图像边缘和图像轮廓的区别检测图像绘制边缘 5.轮廓近似外接矩形外接圆 6. 模板匹配6.1 什么是…

Python实现FA萤火虫优化算法优化BP神经网络分类模型(BP神经网络分类算法)项目实战

说明&#xff1a;这是一个机器学习实战项目&#xff08;附带数据代码文档视频讲解&#xff09;&#xff0c;如需数据代码文档视频讲解可以直接到文章最后获取。 1.项目背景 萤火虫算法&#xff08;Fire-fly algorithm&#xff0c;FA&#xff09;由剑桥大学Yang于2009年提出 , …

系列十五、SpringBoot的启动原理分析

一、概述 所谓SpringBoot的启动原理&#xff0c;翻译成大白话就是"当我们在主启动类上运行run方法时&#xff0c;SpringBoot底层到底做了什么事情&#xff0c;能够帮助我们启动一个Spring的web应用"&#xff0c;上边用大白话解释了一下什么是SpringBoot的启动原理&am…

百度文心一言AI大模型,解读徐礼昭提出的“三体零售”模型

徐礼昭提出的“三体零售”模型是一种创新的零售理论模型&#xff0c;该模型将零售基础物质、零售商业能量和零售数字化系统视为一个相互作用的复杂有机体。这一理论模型为零售行业的发展提供了全新的视角和更高维度的认知。 首先&#xff0c;零售基础物质是零售行业的基石&…

cocos 关于多个摄像机,动态添加节点的显示问题,需要动态修改layer。(跟随摄像机滚动)(神坑官网也不说明一下)

参考文章&#xff1a;Cocos 3.x 层级Layer - 简书 2D镜头跟随应该怎么实现呢 - Creator 3.x - Cocos中文社区 关于多个摄像机&#xff0c;动态添加节点的显示问题&#xff0c;需要动态修改layer&#xff1f; 场景&#xff1a;在制作摄像机跟随角色移动功能时&#xff0c;新增…

RPG项目01_场景及人物动画管理器

基于“RPG项目01_UI登录”&#xff0c;新建一个文件夹名为Model&#xff08;模型&#xff09; 将资源场景拖拽至Model中 找到相应场景双击进入 红色报错部分Clear清掉即可&#xff0c;我们可以重做 接下来另存场景 起名为Game 点击保存 场景就保存至Scene中了 在文件夹下新创建…

jsp前端输入中文数据传到controller变成问号?的解决办法

还是写老师布置的实验的时候&#xff0c;解决了xml文件找不到的问题之后又遇到新的问题&#xff1a;前端登录处输入用户名和密码&#xff0c;结果明明输入的用户名是对的密码也是对的&#xff08;输入的用户名是中文&#xff09;&#xff0c;它就是显示用户名或密码错误。然后我…

2012-2021年银行数字化转型程度数据(根据年报词频计算)

2012-2021年银行数字化转型程度&#xff08;根据年报词频计算&#xff09; 1、时间&#xff1a;2012-2021年 2、指标&#xff1a;银行名称、年份、数字化转型程度 3、范围&#xff1a;52家银行&#xff08;上海银行、中信银行、中国银行、交通银行、光大银行、兰州银行、兴业…

电子学会C/C++编程等级考试2021年06月(四级)真题解析

C/C++等级考试(1~8级)全部真题・点这里 第1题:数字三角形问题 (图1) 图1给出了一个数字三角形。从三角形的顶部到底部有很多条不同的路径。对于每条路径,把路径上面的数加起来可以得到一个和,你的任务就是找到最大的和。 注意:路径上的每一步只能从一个数走到下一层上和它…

熬夜会秃头——Beta冲刺总结随笔

这个作业属于哪个课程2301-计算机学院-软件工程社区-CSDN社区云这个作业要求在哪里团队作业—beta冲刺事后诸葛亮-CSDN社区这个作业的目标总结Beta冲刺团队名称熬夜会秃头团队置顶集合随笔链接熬夜会秃头——Beta冲刺置顶随笔-CSDN社区 目录 一、Beta冲刺开始前设立的任务完成…

ios 长传发布审核+safari浏览器,直接安装ipa文件

蒲公英二维码方法 个人开发者账号发布证书AD-hoc 描述文件蒲公英上传链接通过苹果safari 浏览器下载IPA包 浏览器下载方法 前置条件 1.下载 ipa 包的设备的 uuid 已加入 苹果测试设备列表如何添加到测试列表 2.web 服务, 文件服务. 3.需要AD-hoc 描述文件 添加链接描述 1.创…

微软 Power Platform 零基础 Power Pages 网页搭建教程学习实践进阶以及常见问题解答(二)

微软 Power Platform 零基础 Power Pages 网页搭建教程学习实践进阶及常见问题解答&#xff08;二&#xff09; Power Pages 学习实践进阶 微软 Power Platform 零基础 Power Pages 网页搭建教程学习实践进阶及常见问题解答&#xff08;二&#xff09;Power Pages 核心工具和组…

动态规划 | 背包问题总结

参考-代码随想录 在讲解背包问题的时候&#xff0c;我们都是按照如下五部来逐步分析&#xff0c;相信大家也体会到&#xff0c;把这五部都搞透了&#xff0c;算是对动规来理解深入了。 确定dp数组&#xff08;dp table&#xff09;以及下标的含义 确定递推公式 dp数组如何初始…