2023王道考研数据结构笔记第二章线性表

news2025/1/11 0:32:41

第二章 线性表

2.1 线性表的定义

2.1.1 线性表的基本概念

线性表是具有相同数据类型的n(n>0)个数据元素的有限序列,其中n为表长,当n=0时线性表是一个空表。若用L命名线性表,则其一般表示为:
L = ( a 1 , a 2 , . . . , a i , a i + 1 , . . . , a n ) L=(a_1,a_2,...,a_i,a_{i+1},...,a_n) L=(a1,a2,...,ai,ai+1,...,an)
img

几个概念:

a i a_i ai是线性表中的“第i个”元素线性表中的位序 注意:位序从1开始,数组下标从0开始

a 1 a_1 a1表头元素 a n a_n an表尾元素

除第一个元素外,每个元素有且仅有一个直接前驱;除最后一个元素外,每个元素有且仅有一个直接后继

2.1.2 线性表的基本操作

InitList(&L): 初始化表。构造一个空的线性表L,分配内存空间。——从无到有

DestroyList(&L): 销毁操作。销毁线性表,并释放线性表L所占用的内存空间。——从有到无

ListInsert(&L,i,e): 插入操作。在表L中的第i个位置上插入指定元素e。——增

ListDelete(&L,i,&e): 删除操作。删除表L中第i个位置的元素,并用e返回删除元素的值。——删

LocateElem(L,e): 按值查找操作。在表L中查找具有给定关键字值的元素。——查(改之前也要查)

GetElem(L,i): 按位查找操作。获取表L中第i个位置的元素的值。

其他常用操作:

Length(L): 求表长。返回线性表的长度,即L中数据元素的个数。

PrintList(L): 输出操作。按前后顺序输出线性表L的所有元素值。

Empty(L): 判空操作。若L为空表,则返回true,否则返回false。

Tips:

① 对数据的操作(记忆思路)——创销、增删改查

② C语言函数的定义——<返回值类型>函数名(<参数1类型>,<参数2类型>,…)

③ 实际开发中,可根据实际需求定义其他的基本操作

④ 函数名和参数的形式、命名都可改变

⑤ 什么时候要传入引用 & ?对参数的修改结果需要“带回来”

2.2 顺序表的定义

2.2.1. 顺序表的基本概念

线性表的顺序存储又称 顺序表 。它是用一组地址连续的存储单元依次存储线性表中的数据元素,从而使得**逻辑上相邻的两个元素在物理上也相邻。**顺序表的特点是 表中元素的逻辑顺序与其物理顺序相同

特点:

  • 随机访问,即可以在 O(1) 时间内找到第 i 个元素。

顺序表最大的特点是随机访问,可通过首地址和元素序号在O(1)的时间复杂度内找到指定的元素,因为顺序表是连续存放的。

  • 存储密度高,每个节点只存储数据元素。

  • 拓展容量不方便(即便采用动态分配的方式实现,拓展长度的时间复杂度也比较高,因为需要把数据复制到新的区域)。

  • 插入删除操作不方便,需移动大量元素:O(n)

在这里插入图片描述

2.2.2. 顺序表的实现

静态实现:

// 顺序表实现(静态分配)
#define MaxSize 10

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

typedef struct {
    int data[MaxSize];
    int length;
} SqList;

void InitList(SqList &L) {
    L.length = 0;
}

int main() {
    SqList L;
    InitList(L);
    for (int i = 0; i < MaxSize; i++) {
        printf("data[%d]=%d\n", i, L.data[i]);
    }
    printf("%d", L.length);
    return 0;
}

Q:如果“数组”存满了怎么办?

A:可以放弃治疗,顺序表的表长刚开始确定后就无法更改(存储空间是静态的)

动态实现:

//顺序表实现(动态分配)
#define InitSize 10     // 初始化顺序表的长度

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

typedef struct {
    int *data;      // 声明动态分配数组的指针
    int MaxSize;    // 最大长度
    int length;     // 当前长度
} SeqList;

// 初始化顺序表
void InitList(SeqList &L) {
    // 用malloc函数申请一片连续的存储空间
    L.data = (int *) malloc(sizeof(int) * InitSize);
    L.length = 0;
    L.MaxSize = InitSize;
}

// 增加动态数组的长度,本质上是将数据从旧的区域复制到新区域
void IncreaseSize(SeqList &L, int len) {
    // 使p指针和data指向同一目标
    int *p = L.data;
    L.data = (int *) malloc(sizeof(int) * (L.MaxSize + len));  // 申请一块新的连续空间用于存放新表,并将data指针指向新区域
    for (int i = 0; i < L.length; i++) {
        L.data[i] = p[i];  //将数据复制到新区域
    }
    L.MaxSize += len;
    free(p);  //释放原区域的内存空间
}

// 打印顺序表
void printList(SeqList L) {
    for (int i = 0; i < L.length; i++) {
        printf("%d, ", L.data[i]);
    }
}

int main() {
    SeqList L;
    InitList(L);
    printf("增加前顺序表的长度:%d \n", L.MaxSize);
    printList(L);
    IncreaseSize(L, 5);
    printf("增加后顺序表的长度:%d \n", L.MaxSize);
    return 0;
}

malloc() 函数的作用:会申请一片存储空间,并返回存储空间第一个位置的地址,也就是该位置的指针。

2.2.3. 顺序表的基本操作

插入

ListInsert(&L,i,e): 插入操作。在表L中的==第i个位置(位序)==上插入指定元素e。

// 将元素e插入到顺序表L的第i个位置
bool ListInsert(SqList& L, int i, int e) {
    if (i < 1 || i > L.length + 1)
        return false;
    if (L.length >= MaxSize)
        return false;
    for (int j = L.length; j >= i; j--) {  // 将第i个元素及之后的元素后移
        L.data[j] = L.data[j - 1];
    }
    L.data[i - 1] = e;  // 在位置i处插入元素e
    L.length++;
    return true;
}

最好时间复杂度:O(1)(插入在表尾)

最坏时间复杂度:O(n)(插入在表头)

平均时间复杂度:O(n)

删除

ListDelete(&L,i,&e): 删除操作。删除表L中第i个位置的元素,并用e返回删除元素的值。

bool ListDelete(SqList& L, int i, int& e) {
    if (i < 1 || i > L.length) {  //判断i的范围是否有效
        return false;
    }
    e = L.data[i - 1];
    for (int j = i; j < L.length; j++) {  //将第i个位置后的元素前移
        L.data[j - 1] = L.data[j];
    }
    L.length--;
    return true;
}

最好时间复杂度:O(1)(删除表尾元素)

最坏时间复杂度:O(n)(删除表头元素)

平均时间复杂度:O(n)

查找

  • 按位查找
  • GetElem(L,i): 按位查找操作。获取表L中第i个位置的元素的值。
// 按位查找
int getElemByLoc(SqList L, int i) {
    return L.data[i - 1];
}

时间复杂度:O(1)

由于顺序表的各个数据元素在内存中连续存放,因此可以根据起始地址和数据元素大小立即找到第i个元素——“随机存取”特性。

  • 按值查找
  • LocateElem(L,e): 按值查找操作。在表L中查找具有给定关键字值的元素。
int getElemByValue(SqList L, int value) {
    for (int i = 0; i < L.length; i++) {
        if (L.data[i] == value) {
            return i + 1;   //返回其位序
        }
    }
    return 0;
}

注意:C语言中,结构体的比较不能直接用“==”,需要依次对比各个分量来判断两个结构体是否相等

最好时间复杂度:O(1)(目标元素在表头)

最坏时间复杂度:O(n)(目标元素在表尾)

平均时间复杂度:O(n)

2.3 链表

2.3.1 单链表

在这里插入图片描述

  1. 单链表:用链式存储实现了线性结构。一个结点存储一个数据元素,各结点间的前后关系用一个指针表示。
  2. 特点:
    1. 优点:不要求大片连续空间,改变容量方便。插入和删除操作不需要移动大量元素
    2. 缺点:不可随机存取,要耗费一定空间存放指针。
  3. 两种实现方式:
    1. 带头结点,写代码更方便。头结点不存储数据,头结点指向的下一个结点才存放实际数据。(L->next= NULL)
    2. 不带头结点,麻烦。对第一个数据结点与后续数据结点的处理需要用不同的代码逻辑,对空表和非空表的处理需要用不同的代码逻辑。(L = NULL)

用代码定义一个单链表中结点类型的描述如下:

//用代码定义一个单链表
typedef struct LNode {   //定义单链表结点类型
    int data;            //每个结点存放一个数据元素
    struct LNode* next;  //由于指针域中的指针要指向的也是一个节点,因此要声明为 LNode 类型
} LNode, *LinkList;      //这里的*LinkList强调元素是一个单链表.LNode强调元素是一个节点。本质上是同一个结构体

单链表的元素离散分布在各个存储空间中,是非随机存取的存储结构,不能直接找到表中每个特定的结点。查找结点时,需要从头往后依次遍历。

2.3.2 单链表的基本操作

1、单链表的初始化
// 不带头结点
bool InitList(LinkList &L) {
    L = NULL;   // 空表,不含任何结点
    return true;
}

// 带头结点
bool InitListWithHeadNode(LinkList &L) {
    L = (LNode *) malloc(sizeof(LNode)); // 分配一个头结点
    if (L == NULL) {    // 内存不足,分配失败
        return false;
    }
    L->next = NULL; // 头结点之后暂时还没有结点
    return true;
}
2、头插法建立单链表

在这里插入图片描述

头插法的一个重要应用:单链表的逆置

// 头插法建立单链表
LinkList List_HeadInsert(LinkList &L) {
    LNode *s;
    int x;
    L = (LinkList) malloc(sizeof(LNode));
    L->next = NULL;
    cout << "请输入结点的值,输入9999结束:" << endl;
    cin >> x;
    while (x != 9999) {
        s = (LNode *) malloc(sizeof(LNode)); // 创建新结点
        s->data = x;
        s->next = L->next;
        L->next = s;
        cin >> x;
    }
    return L;
}
3、尾插法建立单链表

在这里插入图片描述

// 尾插法建立单链表
LinkList List_TailInsert(LinkList &L) {
    int x;
    L = (LinkList) malloc(sizeof(LNode));
    LNode *s;
    LNode *r = L;   // 尾指针
    cout << "请输入结点的值,输入9999结束:" << endl;
    cin >> x;
    while (x != 9999) {
        s = (LNode *) malloc(sizeof(LNode));
        s->data = x;
        r->next = s;
        r = s;
        cin >> x;
    }
    r->next = NULL;
    return L;
}

头插法、尾插法:核心就是初始化操作、指定结点的后插操作

尾插法注意设置一个指向表尾结点的指针

头插法的重要应用:链表的逆置

动手试一试:给你一个LinkList L 如何逆置

4、按位查找结点

在单链表中从第一个结点出发,顺指针next 域逐个往下搜索,直到找到第i个结点为止,否则返回最后一个结点指针域NULL。

// 从单链表中查找指定位序的结点,并返回
LNode *getElem(LinkList L, int i) {
    if (i < 0) {
        return NULL;
    }
    LNode *p;
    int j = 0;  // 定义一个j指针标记下标
    p = L;      // p指针用来标记第i个结点
    while (p != NULL && j < i) {  // p==NULL 超出表长,返回空值  j>i 超出所查询元素的下标
        p = p->next;
        j++;        // j指针后移
    }
    return p;
}
5、按值查找结点

从单链表的第一个结点开始,由前往后依次比较表中各结点数据域的值若某结点数据域的值等于给定值e,则返回该结点的指针;若整个单链表中没有这样的结点,则返回NULL。

LNode *LocateElem(LinkList L, int value){
    LNode *p = L->next;
    while(p!=NULL && p->data != value){
        p = p->next;
    }
    return p;
}
6、头插法插入元素

插入结点操作将值为x的新结点插入到单链表的第i个位置上。先检查插入位置的合法性,然后找到待插入位置的前驱结点,即第i-1个结点,再在其后插入新结点。

算法首先调用按序号查找算法GetElem(L,i-1),查找第i-1个结点。假设返回的第i-1个结点为p,然后令新结点s的指针域指向p 的后继结点,再令结点p 的指针域指向新插入的结点*s。

在这里插入图片描述

// 在第i个元素前插入元素value
bool ListInsert(LinkList &L, int i, int e) {
    if (i < 1)
        return false;
    LNode *p;           //指针p指向当前扫描到的结点
    int j = 0;            //当前p指向的是第几个结点
    p = L;              //循环找到第i-1个结点
    while (p != NULL && j < i - 1) {       //如果i>lengh,p最后会等于NULL
        p = p->next;
        j++;
    }
    //p值为NULL说明i值不合法
    if (p == NULL)
        return false;
    //在第i-1个结点后插入新结点
    LNode *s = (LNode *) malloc(sizeof(LNode));
    s->data = e;
    s->next = p->next;
    p->next = s;
    //将结点s连到p后
    return true;
}
7、删除结点

删除结点操作是将单链表的第i个结点删除。先检查删除位置的合法性,后查找表中第i-1个结点,即被删结点的前驱结点,再将其删除。

在这里插入图片描述

// 删除第i个结点并将其所保存的数据存入value
bool ListDelete(LinkList &L, int i, int &value) {
    if (i < 1)
        return false;
    LNode *p;       //指针p指向当前扫描到的结点
    int j = 0;        //当前p指向的是第几个结点
    p = L;
    //循环找到第i-1个结点
    while (p != NULL && j < i - 1) {
        //如果i>lengh,p和p的后继结点会等于NULL
        p = p->next;
        j++;
    }
    if (p == NULL)
        return false;
    if (p->next == NULL)
        return false;
    //令q暂时保存被删除的结点
    LNode *q = p->next;
    value = q->data;
    p->next = q->next;
    free(q);
    return true;
}

2.3.3 双链表

单链表结点中只有一个指向其后继的指针,使得单链表只能从头结点依次顺序地向后遍历。要访问某个节点的前驱结点(插入、删除操作)时,只能从头开始遍历,访问后继节点的时间复杂度为O(1),访问前驱结点的时间复杂度为O(n)。

在这里插入图片描述

为了解决如上问题,引入了双链表,双链表结点中有两个指针priornext,分别指向其前驱结点和后继结点。

单链表:无法逆向检索,有时候不太方便

双链表:可进可退,存储密度更低一丢丢

双链表的类型描述如下:

typedef struct DNode {
    int data;   // 数据域
    struct DNode *prior, *next;     // 前驱和后继指针
}DNode, *DLinkList;
1、双链表的初始化
bool InitDLinkList(DLinkList &L) {
    L = (DNode *) malloc(sizeof(DNode));//DLinkList强调是个双链表,DNode *强调是个结点
    if (L == NULL) {
        return false;
    }
    L->prior = NULL; // 头结点的前驱指针永远指向NULL
    L->next = NULL; // 后继指针暂时为空
    return true;
}
2、双链表的后插操作

在这里插入图片描述

// 将节点s插入到结点p之后
bool InsertNextDNode(DNode *p, DNode *s) {
    if (p == NULL || s == NULL) {  //非法参数
        return false;
    }
    s->next = p->next;  // 将p的后继赋给s的后继
    // 判断p之后是否还有前驱节点
    if (p->next != NULL) {
        p->next->prior = s;
    }
    s->prior = p;
    p->next = s;
    return true;
}

双链表的前插操作、按位序插入操作都可以转换成后插操作

3、双链表的删除操作

在这里插入图片描述

// 删除p结点的后继结点
bool DeleteNextDNode(DNode *p) {
    if (p == NULL) {
        return false;
    }
    // 找到p的后继结点q
    DNode *q = p->next;
    if (q == NULL) {
        return false;
    }
    p->next = q->next;  // 将q的后继赋给p的后继
    if (q->next != NULL) {  // 若q的后继结点不为空
        q->next->prior = p; // 将p赋给q的后继节点的前驱节点
    }
    free(q);    // 释放q
    return true;
}

// 销毁一个双链表
bool DestoryList(DLinkList &L) {
    // 循环释放各个数据结点
    while (L->next != NULL) {
        DeleteNextDNode(L);
        free(L);
        // 头指针置空
        L = NULL;
    }
}

双链表不可随机存取,按位查找、按值查找操作都只能用遍历的方式实现。时间复杂度O(n).

//后向遍历
while(p!=NULL){
    //对结点p做相应处理
    p=p->next;
}

//前向遍历
while(p!=NULL){
    //对结点p做相应处理
    p=p->prior;
}

//前向遍历(跳过头结点)
while(p->prior!=NULL){
    //对结点p做相应处理
    p=p->prior;
}

2.3.4 循环链表

循环链表是另一种形式的链式存储结构。它的特点是表中最后一个结点的指针域指向头结点,整个链表形成一个环。
循环单链表

循环双链表:
在这里插入图片描述

代码定义循环链表

//循环单链表的定义
typedef struct LNode{
    ElemType data;
    struct LNode *next;
}LNode,*LinkList;

//循环双链表的定义
typedef struct DNode{
    ElemType data;
    struct DNode *prior,*next;
}DNode,*DLinkList;
1、循环单链表的初始化
bool InitList(LinkList &L){
    L=(LNode *)malloc(sizeof(LNode)); //分配一个头结点
    if(L==NULL)
        return false;   //内存不足,分配失败
    L->next=L;       //头结点next指向头结点
    return true;
}
//判断循环单链表是否为空
if (L->next == L){
	return true;
} else {
	return false;
}
//判断结点p是否为循环单链表的表尾结点
bool isTail(LinkList L,LNode *p){
    if(p->next==L)
        return true;
    else
        return false;
}

单链表:从一个结点出发只能找到后续的各个结点

循环单链表:从一个结点出发可以找到其他任何一个结点

2、循环双链表的初始化

双链表:表头结点的prior指向NULL;表尾结点的next指向NULL

循环双链表:表头结点的prior指向表尾结点;表尾结点的next指向表头结点

//初始化空的循环双链表
bool InitDLinkList(DLinklist &L){
    L=(DNode *)malloc(sizeof(DNode));
    if(L==NULL)
        return false;
    L->prior=L;
    L->next=L;
    return true;
}
//判断循环双链表是否为空
bool Empty(DLinklist L){
    if(L->next==L)
        return true;
    else
        return false;
}
//判断结点p是否为循环双链表的表尾结点
bool isTail(DLinkList L,LNode *p){
    if(p->next==L)
        return true;
    else
        return false;
}

2.3.5 静态链表

单链表:各个结点在内存中星罗棋布、散落天涯。

静态链表:用数组的方式实现的链表。分配一整片连续的内存空间,各个结点集中安置,每个结点包括了数据元素和下一个结点的数组下标(游标)

每个数据元素4B,每个游标4B(每个结点共8B),设起始地址为addr, e 1 e_1 e1的存放地址为 addr+8*2

在这里插入图片描述

(0号结点充当”头结点“,游标充当”指针“,游标为-1表示已经到达表尾)

  • **优点:**增、删操作不需要大量移动元素。
  • **缺点:**不能随机存取,只能从头结点开始依次往后查找,容量固定不可变!
  • 适用场景:①不支持指针的低级语言;②数据元素数量固定不变

定义1:

#define MaxSize 10        //静态链表的最大长度
struct Node{              //静态链表结构类型的定义  
    ElemType data;        //存储数据元素    
    int next;             //下一个元素的数组下标
};
void testSLinkList(){
    struct Node a[MaxSize]; //数组a作为静态链表
    //后续代码...
}

定义2:

#define MaxSize 10        //静态链表的最大长度
typedef struct{           //静态链表结构类型的定义       
    ELemType data;        //存储数据元素     
    int next;             //下一个元素的数组下标
}SLinkList[MaxSize];      //用SLinkList定义一个长度为MaxSize的Node型数组

void testSLinkList(){      
    SLinkList a;
     //后续代码...
}

第一种是我们更加熟悉的写法,第二种写法则更加侧重于强调 a 是一个静态链表而非数组。

静态链表实现基本操作的注意点:

  1. 初始化静态链表时,需要把a[0]的next设为-1,并将空闲结点next设置为某个特殊值,比如-2.
  2. 按位序查找结点时,从头结点出发按个往后遍历结点,时间复杂度O(n).
  3. 按位序插入结点步骤:①找到一个空的结点,存入数据元素;②从头结点出发找到位序为i-1的结点;③修改新节点的next为-1;④修改i-1号结点的next为新结点的下标;

2.4 顺序表vs链表

顺序表链表
逻辑结构属于线性表,都是线性结构属于线性表,都是线性结构
存储结构顺序存储
优点:支持随机存取,存储密度高
缺点:大片连续空间分配不方便,改变容量不方便
链式存储
优点:离散的小空间分配方便,改变容量方便
缺点:不可随机存取,存储密度低
基本操作——创建需要预分配大片连续空间。若分配空间过小,则之后不方便拓展容量;若分配空间过大,则浪费内存资源。
静态分配:静态数组,容量不可改变。
动态分配:动态数组,容量可以改变,但是需要移动大量元素,时间代价高(使用malloc()free())。
只需要分配一个头结点或者只声明一个头指针。
基本操作——销毁修改 Length = 0
静态分配:静态数组——系统自动回收空间。
动态分配:动态数组——需要手动free()
依次删除各个结点 free()
基本操作——增删插入 / 删除元素要将后续元素后移 / 前移;
时间复杂度:O(n),时间开销主要来自于移动元素
若数据元素很大,则移动的时间代价很高
插入/删除元素只需要修改指针;
时间复杂度:O(n),时间开销主要来自查找目标元素
查找元素的时间代价更低
基本操作——查找按位查找:O(1)
按值查找:O(n),若表内元素有序,可在O(log2n) 时间内找到(二分法)
按位查找:O(n)
按值查找:O(n)

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

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

相关文章

泛微e-cology9 <0.56 存在sql注入漏洞

漏洞描述 泛微e-cology9 是一个 OA 系统。 泛微e-cology9 10.56 之前版本存在 sql 注入漏洞&#xff0c;攻击者可利用该漏洞读取或修改数据库中的敏感信息&#xff0c;进而操控管理系统。 漏洞名称泛微e-cology9 <0.56 存在sql注入漏洞漏洞类型SQL注入发现时间2023/2/23漏…

论文阅读-DISTILLING KNOWLEDGE FROM READER TORETRIEVER FOR QUESTION ANSWERING

论文链接&#xff1a;https://arxiv.org/pdf/2012.04584.pdf 目录 方法 交叉注意机制 交叉注意力得分作为段落检索的相关性度量 用于段落检索的密集双编码器 将交叉注意力分数提取到双编码器 数据集 方法 我们的系统由两个模块组成&#xff0c;即检索器和阅读器&#xf…

文献阅读 Image-to-Image Translation with Conditional Adversarial Networks

前言 本文中所使用的cGAN模型为上篇论文的main idea&#xff0c;这里再进行一下相关介绍。 题目 Image-to-Image Translation with Conditional Adversarial Networks 使用条件对抗网络实现图片到图片的翻译 摘要 我们研究了条件对抗网络作为图像到图像翻译问题的通用方案。…

【C++】inline 内联函数

文章目录&#x1f4d5; 概念&#x1f4d5; 使用前的准备&#x1f4d5; 使用&#x1f4d5; 特性&#x1f4d5; 概念 在 C 中&#xff0c;为了解决一些频繁调用的小函数大量消耗栈空间&#xff08;栈内存&#xff09;的问题&#xff0c;特别的引入了 inline 修饰符&#xff0c;表…

你需要同款“Unreal项目自动化编译、打包和部署”方案吗?

在过往几期的UWA Pipeline最佳实践案例中&#xff0c;我们分享了如何通过Pipeline实现性能优化、性能管理、游戏内容验收和云真机系统的应用&#xff08;实现批量真机设备的自动化测试&#xff0c;以及针对特效性能优化的方式&#xff09;&#xff0c;其实这些高效的方法并不局…

改进 YOLO V5 的密集行人检测算法研究(论文研读)——目标检测

改进 YOLO V5 的密集行人检测算法研究&#xff08;2021.08&#xff09;摘 要&#xff1a;1 YOLO V52 SENet 通道注意力机制3 改进的 YOLO V5 模型3.1 训练数据处理改进3.2 YOLO V5 网络改进3.3 损失函数改进3.3.1 使用 CIoU3.3.2 非极大值抑制改进4 研究方案与结果分析4.1 实验…

碰撞高校智慧,凝聚青春力量 | 因“AI”而“深”-高校开源专场顺利举办!

为聚焦AI新发展&#xff0c;探寻开源新环境&#xff0c;聆听高校声音&#xff0c;2月25日上午&#xff0c;第四届OpenI/O启智开发者大会以“因AI而深”为主题在深圳人才研修院举办高校开源专场&#xff0c;共同交流高校优质开源项目的发展经验、社区优秀开发者的心路历程及高校…

ROS2功能包Hello world(python)

文章目录环境准备Python创建工作空间、功能包及节点方法编译使用环境准备 为了便于日后复现&#xff0c;相关环境已经打包到docker中。 拉取docker镜像 docker pull 1224425503/ros2_foxy_full:latest新建容器 docker run -dit --rm --privilegedtrue --network host -e NV…

cnpm adduser 报错 409 Conflict

今天遇到一个问题&#xff0c;cnpm adduser 一直失败&#xff0c;返回 409 Conflict。 我们先来看下报错信息 409 Conflict - PUT http://registry.cnpm.xxxx.com.cn/-/user/org.couchdb.user:mingyu6 - conflict第一步 分析 http 错误码 409 Conflict&#xff1a;请求与服务…

Java函数式编程

从JDK8开始java支持函数式编程&#xff0c;函数式编程是面向数学的抽象&#xff0c;将计算描述为一种表达式求值&#xff0c;函数式程序就是一个表达式。JDK也提供了几个常用的函数式接口。 java的函数式编程还没有例如js那样的规范和深奥&#xff0c; 在函数式编程中&#xf…

蓝桥杯每日一题:不同路径数(dfs深度优先)

给定一个 nm的二维矩阵&#xff0c;其中的每个元素都是一个 [1,9] 之间的正整数。 从矩阵中的任意位置出发&#xff0c;每次可以沿上下左右四个方向前进一步&#xff0c;走过的位置可以重复走。 走了 k 次后&#xff0c;经过的元素会构成一个 (k1) 位数。 请求出一共可以走出…

浅谈功能测试和性能测试的区别

刚刚入门软件测试的宝子&#xff0c;可能经常会看到&#xff1a;功能测试、性能测试&#xff0c;这些到底是什么呀&#xff1f;那我就带大家了解一下~ 一、定义 功能测试 功能测试就是对产品的各功能进行验证&#xff0c;根据功能测试用例&#xff0c;逐项测试&#xff0…

操作指南:如何高效使用Facebook Messenger销售(二)

上一篇文章我们介绍了使用Facebook Messenger作为销售渠道的定义、好处及注意事项&#xff0c;本节我们将详细介绍怎么将Facebook Messenger销售与SaleSmartly&#xff08;ss客服&#xff09;结合&#xff0c;实现一站式管理多主页配图来源&#xff1a;SaleSmartly&#xff08;…

喀秋莎Camtasia2023最新版本电脑录屏剪辑软件

录屏软件的鼻祖是techSmith 的喀秋莎&#xff08;Techsmith Camtasia Studio&#xff09;&#xff0c;视频编辑软件Camtasia 2023发布&#xff0c;十大新功能放出!作为一个自媒体人&#xff0c;每天都要录制编辑视频&#xff0c;选择一个好的视频编辑工具就是大家首先面临的一个…

招投标管理系统-适合于招标代理、政府采购、企业采购、工程交易等业务的企业

招投标管理系统-适合于招标代理、政府采购、企业采购、工程交易等业务的企业 招投标管理系统是一个用于内部业务项目管理的应用平台。以项目为主线&#xff0c;从项目立项&#xff0c;资格预审&#xff0c;标书编制审核&#xff0c;招标公告&#xff0c;项目开标&#xff0c;项…

Android Kotlin实战之高阶使用泛型扩展协程懒加载详解

前言&#xff1a; 通过前面几篇文章&#xff0c;我们已基本掌握kotlin的基本写法与使用&#xff0c;但是在开发过程中&#xff0c;以及一些开源的API还是会出现大家模式的高阶玩法以及问题&#xff0c;如何避免&#xff0c;接下来讲解针对原来的文章进行一些扩展&#xff0c;解…

【10K功能测试】-【20K自动化测试】之间的区别

前言 按测试执行的类型来分&#xff1a;功能测试、自动化测试 1&#xff0e;功能测试 功能测试俗称点点点测试。初级测试人员的主要测试任务就是执行测试工程师所写的测试用 例&#xff0c;记录用例的执行状态及bug情况。与开发人员进行交互直到bug被修复。 功能测试理论上是…

顺丰同城测试开发一面 49min答案,全文7000字,面试总结都在这里了

今天给大家分享一份顺丰同城的测试开发一面面试真题。老规矩&#xff0c;当你看到这份面试题的时候&#xff0c;先不要着急去看答案&#xff0c;你可以想想假如你在面试现场&#xff0c;你会怎么回答&#xff1f;这个思考的过程其实也是很重要的。 全文7000字干货&#xff0c;…

ADC0832的AD模数转换原理及编程

✅作者简介&#xff1a;嵌入式领域优质创作者&#xff0c;博客专家 ✨个人主页&#xff1a;咸鱼弟 &#x1f525;系列专栏&#xff1a;单片机设计专栏 &#x1f4c3;推荐一款求职面试、刷题神器&#x1f449;注册免费刷题 目录 一、描述 二、模数转换原理&#xff1a; 三、…

Gorm -- 添加记录

文章目录添加单条记录直接添加模型对象赋予默认值方法一&#xff1a; gorm 标签赋予默认值方法二&#xff1a; 设置钩子方法&#xff08;Hooks&#xff09;指定字段插入插入时忽略某些字段插入时禁止使用钩子方法添加多条记录通过对象列表插入通过字典列表插入在字典中使用SQL内…