当前位置: 首页 > news >正文

串---链串实现

链串详解

本文档将详细介绍链串的基本概念、实现原理及其在 C 语言中的具体应用。通过本指南,读者将了解如何使用链串进行各种字符串操作。

1. 什么是链串?

链串是一种用于存储字符串的数据结构,它使用一组动态分配的节点来保存字符串中的字符序列。链串通常由两部分组成:

  • 一个指向链表头结点的指针 s
  • 一系列的节点,每个节点包含一个字符 data 和指向下一个节点的指针 next

在本程序中,我们定义了一个名为 LinkStrNode 的结构体来表示链串中的单个节点,并使用一个指针 s 来表示整个链串。

typedef struct StringNode {char data;struct StringNode *next;
} LinkStrNode;

2. 基本操作

2.1 生成串 (StrAssign)

此函数用于将一个 C 语言字符串转换为链串。

void StrAssign(LinkStrNode *&s, const char str[]) {LinkStrNode *r;LinkStrNode *p;s = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 创建头结点r = s;for (int i = 0; str[i] != '\0'; i++) {p = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 分配新节点p->data = str[i]; // 设置数据r->next = p; // 将新节点链接到链表r = p; // 移动指针到新节点}r->next = NULL; // 设置尾节点的 next 为 NULL
}

参数:

  • s: 目标链串的头结点指针。
  • str: 源 C 语言字符串。

功能:

  • s 分配一个头结点。
  • str 中的每个字符创建一个新的节点,并将其添加到链表中。

2.2 销毁串 (DestroyStr)

此函数用于释放链串所占用的内存。

void DestroyStr(LinkStrNode *&s) {LinkStrNode *pre = s;LinkStrNode *p = s->next;while (p != NULL) {free(pre); // 释放前一个节点pre = p; // 移动 pre 到当前节点p = pre->next; // 移动 p 到下一个节点}free(pre); // 释放最后一个节点s = NULL; // 清空串指针
}

参数:

  • s: 要销毁的链串的头结点指针。

功能:

  • 依次释放链串中的每个节点,并清空头结点指针。

2.3 判断是否为空串 (StrEmpty)

此函数用于检查一个链串是否为空。

bool StrEmpty(LinkStrNode *s){return s->next == NULL;
} 

参数:

  • s: 要检查的链串的头结点指针。

返回值:

  • 如果链串为空(即头结点的 next 指向 NULL),则返回 true;否则返回 false

2.4 串的复制 (StrCopy)

此函数用于将一个链串复制到另一个链串。

void StrCopy(LinkStrNode *&s, LinkStrNode *t) {LinkStrNode *p = t->next;LinkStrNode *q;LinkStrNode *r;s = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 创建新的头结点r = s;while (p != NULL) {q = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 分配新节点q->data = p->data; // 设置数据r->next = q; // 将新节点链接到链表r = q; // 移动指针到新节点p = p->next; // 移动原串指针}r->next = NULL; // 设置尾节点的 next 为 NULL
}

参数:

  • s: 目标链串的头结点指针。
  • t: 源链串的头结点指针。

功能:

  • s 分配一个头结点。
  • 复制 t 中的所有节点到 s 的链表中。

2.5 判断串相等 (StrEqual)

此函数用于检查两个链串是否相等。

bool StrEqual(LinkStrNode *s, LinkStrNode *t) {LinkStrNode *p = s->next;LinkStrNode *q = t->next;while (p != NULL && q != NULL && p->data == q->data) {p = p->next;q = q->next;}return (p == NULL && q == NULL);
}

参数:

  • s: 第一个链串的头结点指针。
  • t: 第二个链串的头结点指针。

返回值:

  • 如果两个链串具有相同的长度且所有字符都相等,则返回 true;否则返回 false

2.6 获得串的长度 (StrLength)

此函数用于获取一个链串的长度。

int StrLength(LinkStrNode *s) {int i = 0;LinkStrNode *p = s->next;while (p != NULL) {i++;p = p->next;}return i;
}

参数:

  • s: 要获取长度的链串的头结点指针。

返回值:

  • 返回链串的长度。

2.7 串的连接 (Concat)

此函数用于将两个链串连接成一个新的链串。

LinkStrNode *Concat(LinkStrNode *s, LinkStrNode *t) {LinkStrNode *str;LinkStrNode *p = s->next;LinkStrNode *q;LinkStrNode *r;str = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 创建新的头结点r = str;while (p != NULL) {q = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 分配新节点q->data = p->data; // 设置数据r->next = q; // 将新节点链接到链表r = q; // 移动指针到新节点p = p->next; // 移动原串指针}p = t->next;while (p != NULL) {q = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 分配新节点q->data = p->data; // 设置数据r->next = q; // 将新节点链接到链表r = q; // 移动指针到新节点p = p->next; // 移动原串指针}r->next = NULL; // 设置尾节点的 next 为 NULLreturn str;
}

参数:

  • s: 第一个链串的头结点指针。
  • t: 第二个链串的头结点指针。

返回值:

  • 返回一个新的链串,该串包含 st 的所有节点。

2.8 获得子串 (SubStr)

此函数用于从一个链串中获取指定位置的子串。

LinkStrNode *SubStr(LinkStrNode *s, int i, int j) {int k;LinkStrNode *str;LinkStrNode *p = s->next;LinkStrNode *q;LinkStrNode *r;str = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 创建新的头结点str->next = NULL;r = str;if (i <= 0 || i > StrLength(s) || j < 0 || i + j - 1 > StrLength(s)) {return str;}for (k = 1; k < i; k++) { // 移动到起始位置p = p->next;}for (k = 1; k <= j; k++) { // 复制子串q = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 分配新节点q->data = p->data; // 设置数据r->next = q; // 将新节点链接到链表r = q; // 移动指针到新节点p = p->next; // 移动原串指针}r->next = NULL; // 设置尾节点的 next 为 NULLreturn str;
}

参数:

  • s: 原始链串的头结点指针。
  • i: 子串的起始位置(1-indexed)。
  • j: 子串的长度。

返回值:

  • 返回一个新的链串,该串包含 s 从位置 i 开始的长度为 j 的子串。

2.9 子串的插入 (InsertStr)

此函数用于在一个链串的指定位置插入另一个链串。

LinkStrNode *InsertStr(LinkStrNode *s, int i, LinkStrNode *t) {int k;LinkStrNode *str;LinkStrNode *p1 = s->next;LinkStrNode *p2 = t->next;LinkStrNode *q;LinkStrNode *r;str = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 创建新的头结点str->next = NULL;r = str;if (i <= 0 || i > StrLength(s) + 1) {return str;}for (k = 1; k < i; k++) { // 移动到插入位置q = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 分配新节点q->data = p1->data; // 设置数据r->next = q; // 将新节点链接到链表r = q; // 移动指针到新节点p1 = p1->next; // 移动原串指针}while (p2 != NULL) { // 插入子串q = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 分配新节点q->data = p2->data; // 设置数据r->next = q; // 将新节点链接到链表r = q; // 移动指针到新节点p2 = p2->next; // 移动原串指针}while (p1 != NULL) { // 继续复制剩余部分q = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 分配新节点q->data = p1->data; // 设置数据r->next = q; // 将新节点链接到链表r = q; // 移动指针到新节点p1 = p1->next; // 移动原串指针}r->next = NULL; // 设置尾节点的 next 为 NULLreturn str;
}

参数:

  • s: 原始链串的头结点指针。
  • i: 插入位置(1-indexed)。
  • t: 要插入的链串的头结点指针。

返回值:

  • 返回一个新的链串,该串包含 s 和在位置 i 插入的 t

2.10 子串的删除 (DelStr)

此函数用于从一个链串中删除指定位置的子串。

LinkStrNode *DelStr(LinkStrNode *s, int i, int j) {int k;LinkStrNode *str;LinkStrNode *p = s->next;LinkStrNode *q;LinkStrNode *r;str = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 创建新的头结点str->next = NULL;r = str;if (i <= 0 || i > StrLength(s) || j < 0 || i + j - 1 > StrLength(s)) {return str;}for (k = 1; k < i; k++) { // 移动到删除位置前q = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 分配新节点q->data = p->data; // 设置数据r->next = q; // 将新节点链接到链表r = q; // 移动指针到新节点p = p->next; // 移动原串指针}for (k = 0; k < j; k++) { // 跳过要删除的部分p = p->next; // 移动原串指针}while (p != NULL) { // 复制剩余部分q = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 分配新节点q->data = p->data; // 设置数据r->next = q; // 将新节点链接到链表r = q; // 移动指针到新节点p = p->next; // 移动原串指针}r->next = NULL; // 设置尾节点的 next 为 NULLreturn str;
}

参数:

  • s: 原始链串的头结点指针。
  • i: 子串的起始位置(1-indexed)。
  • j: 子串的长度。

返回值:

  • 返回一个新的链串,该串包含 s 除位置 i 开始的长度为 j 的子串外的所有节点。

2.11 子串的替换 (ReplaceSubStr)

此函数用于在一个链串的指定位置替换一个子串。

LinkStrNode *ReplaceSubStr(LinkStrNode *s, int i, int j, LinkStrNode *t) {LinkStrNode *newStr = InsertStr(s, i, t); // 先执行插入操作LinkStrNode *delStr = DelStr(newStr, i, j); // 再执行删除操作DestroyStr(newStr); // 销毁中间串return delStr; // 返回替换后的串
}

参数:

  • s: 原始链串的头结点指针。
  • i: 要替换子串的起始位置(1-indexed)。
  • j: 要替换子串的长度。
  • t: 新的子串的头结点指针。

返回值:

  • 返回一个新的链串,该串包含 s 除位置 i 开始的长度为 j 的子串被 t 替换外的所有节点。

2.12 串的比较 (StrCompare)

此函数用于比较两个链串。

int StrCompare(LinkStrNode *s, LinkStrNode *t) {LinkStrNode *p = s->next, *q = t->next;while (p != NULL && q != NULL) {if (p->data != q->data) {return p->data > q->data ? 1 : -1;}p = p->next;q = q->next;}if (p == NULL && q == NULL) {return 0;} else if (p == NULL) {return -1;} else {return 1;}
}

参数:

  • s: 第一个链串的头结点指针。
  • t: 第二个链串的头结点指针。

返回值:

  • 如果 st 相等,返回 0;
  • 如果 s 小于 t,返回负数;
  • 如果 s 大于 t,返回正数。

2.13 打印链串 (PrintStr)

此函数用于显示一个链串的内容。

void PrintStr(LinkStrNode *s) {LinkStrNode *p = s->next;while (p != NULL) {printf("%c", p->data);p = p->next;}printf("\n");
}

参数:

  • s: 要显示的链串的头结点指针。

3. 示例程序

下面是一个示例程序,演示了如何使用上述定义的功能。

#include <stdio.h>
#include <stdlib.h>
#include <string.h> // 用于使用 strlen 函数
#define MaxSize 100// 链串的结构体
typedef struct StringNode {char data;struct StringNode *next;
} LinkStrNode;// 生成串
void StrAssign(LinkStrNode *&s, const char str[]) {LinkStrNode *r;LinkStrNode *p;s = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 创建头结点r = s;for (int i = 0; str[i] != '\0'; i++) {p = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 分配新节点p->data = str[i]; // 设置数据r->next = p; // 将新节点链接到链表r = p; // 移动指针到新节点}r->next = NULL; // 设置尾节点的 next 为 NULL
}// 销毁串
void DestroyStr(LinkStrNode *&s) {LinkStrNode *pre = s;LinkStrNode *p = s->next;while (p != NULL) {free(pre); // 释放前一个节点pre = p; // 移动 pre 到当前节点p = pre->next; // 移动 p 到下一个节点}free(pre); // 释放最后一个节点s = NULL; // 清空串指针
}//判断是否为空串
bool StrEmpty(LinkStrNode *s){return s->next == NULL;
} // 串的复制
void StrCopy(LinkStrNode *&s, LinkStrNode *t) {LinkStrNode *p = t->next;LinkStrNode *q;LinkStrNode *r;s = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 创建新的头结点r = s;while (p != NULL) {q = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 分配新节点q->data = p->data; // 设置数据r->next = q; // 将新节点链接到链表r = q; // 移动指针到新节点p = p->next; // 移动原串指针}r->next = NULL; // 设置尾节点的 next 为 NULL
}// 判断串相等
bool StrEqual(LinkStrNode *s, LinkStrNode *t) {LinkStrNode *p = s->next;LinkStrNode *q = t->next;while (p != NULL && q != NULL && p->data == q->data) {p = p->next;q = q->next;}return (p == NULL && q == NULL);
}// 获得串的长度
int StrLength(LinkStrNode *s) {int i = 0;LinkStrNode *p = s->next;while (p != NULL) {i++;p = p->next;}return i;
}// 串的连接
LinkStrNode *Concat(LinkStrNode *s, LinkStrNode *t) {LinkStrNode *str;LinkStrNode *p = s->next;LinkStrNode *q;LinkStrNode *r;str = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 创建新的头结点r = str;while (p != NULL) {q = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 分配新节点q->data = p->data; // 设置数据r->next = q; // 将新节点链接到链表r = q; // 移动指针到新节点p = p->next; // 移动原串指针}p = t->next;while (p != NULL) {q = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 分配新节点q->data = p->data; // 设置数据r->next = q; // 将新节点链接到链表r = q; // 移动指针到新节点p = p->next; // 移动原串指针}r->next = NULL; // 设置尾节点的 next 为 NULLreturn str;
}// 获得子串
LinkStrNode *SubStr(LinkStrNode *s, int i, int j) {int k;LinkStrNode *str;LinkStrNode *p = s->next;LinkStrNode *q;LinkStrNode *r;str = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 创建新的头结点str->next = NULL;r = str;if (i <= 0 || i > StrLength(s) || j < 0 || i + j - 1 > StrLength(s)) {return str;}for (k = 1; k < i; k++) { // 移动到起始位置p = p->next;}for (k = 1; k <= j; k++) { // 复制子串q = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 分配新节点q->data = p->data; // 设置数据r->next = q; // 将新节点链接到链表r = q; // 移动指针到新节点p = p->next; // 移动原串指针}r->next = NULL; // 设置尾节点的 next 为 NULLreturn str;
}// 子串的插入
LinkStrNode *InsertStr(LinkStrNode *s, int i, LinkStrNode *t) {int k;LinkStrNode *str;LinkStrNode *p1 = s->next;LinkStrNode *p2 = t->next;LinkStrNode *q;LinkStrNode *r;str = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 创建新的头结点str->next = NULL;r = str;if (i <= 0 || i > StrLength(s) + 1) {return str;}for (k = 1; k < i; k++) { // 移动到插入位置q = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 分配新节点q->data = p1->data; // 设置数据r->next = q; // 将新节点链接到链表r = q; // 移动指针到新节点p1 = p1->next; // 移动原串指针}while (p2 != NULL) { // 插入子串q = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 分配新节点q->data = p2->data; // 设置数据r->next = q; // 将新节点链接到链表r = q; // 移动指针到新节点p2 = p2->next; // 移动原串指针}while (p1 != NULL) { // 继续复制剩余部分q = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 分配新节点q->data = p1->data; // 设置数据r->next = q; // 将新节点链接到链表r = q; // 移动指针到新节点p1 = p1->next; // 移动原串指针}r->next = NULL; // 设置尾节点的 next 为 NULLreturn str;
}// 子串的删除
LinkStrNode *DelStr(LinkStrNode *s, int i, int j) {int k;LinkStrNode *str;LinkStrNode *p = s->next;LinkStrNode *q;LinkStrNode *r;str = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 创建新的头结点str->next = NULL;r = str;if (i <= 0 || i > StrLength(s) || j < 0 || i + j - 1 > StrLength(s)) {return str;}for (k = 1; k < i; k++) { // 移动到删除位置前q = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 分配新节点q->data = p->data; // 设置数据r->next = q; // 将新节点链接到链表r = q; // 移动指针到新节点p = p->next; // 移动原串指针}for (k = 0; k < j; k++) { // 跳过要删除的部分p = p->next; // 移动原串指针}while (p != NULL) { // 复制剩余部分q = (LinkStrNode *)malloc(sizeof(LinkStrNode)); // 分配新节点q->data = p->data; // 设置数据r->next = q; // 将新节点链接到链表r = q; // 移动指针到新节点p = p->next; // 移动原串指针}r->next = NULL; // 设置尾节点的 next 为 NULLreturn str;
}// 子串的替换
LinkStrNode *ReplaceSubStr(LinkStrNode *s, int i, int j, LinkStrNode *t) {LinkStrNode *newStr = InsertStr(s, i, t); // 先执行插入操作LinkStrNode *delStr = DelStr(newStr, i, j); // 再执行删除操作DestroyStr(newStr); // 销毁中间串return delStr; // 返回替换后的串
}int StrCompare(LinkStrNode *s, LinkStrNode *t) {LinkStrNode *p = s->next, *q = t->next;while (p != NULL && q != NULL) {if (p->data != q->data) {return p->data > q->data ? 1 : -1;}p = p->next;q = q->next;}if (p == NULL && q == NULL) {return 0;} else if (p == NULL) {return -1;} else {return 1;}
}// 打印链串
void PrintStr(LinkStrNode *s) {LinkStrNode *p = s->next;while (p != NULL) {printf("%c", p->data);p = p->next;}printf("\n");
}// 主函数
int main() {LinkStrNode *s, *t;char input[MaxSize];// 用户输入第一个串printf("请输入第一个字符串: ");scanf("%s", input);StrAssign(s, input);// 用户输入第二个串printf("请输入第二个字符串: ");scanf("%s", input);StrAssign(t, input);// 显示两个串printf("第一个字符串: ");PrintStr(s);printf("第二个字符串: ");PrintStr(t);// 检查是否为空串if (StrEmpty(s)) {printf("第一个串为空串。\n");} else {printf("第一个串不为空串。\n");}if (StrEmpty(t)) {printf("第二个串为空串。\n");} else {printf("第二个串不为空串。\n");}// 比较两个串int result = StrCompare(s, t);if (result == 0) {printf("两个串相等。\n");} else if (result < 0) {printf("第一个串小于第二个串。\n");} else {printf("第一个串大于第二个串。\n");}// 计算两个串的长度printf("第一个串的长度为: %d\n", StrLength(s));printf("第二个串的长度为: %d\n", StrLength(t));// 连接两个串LinkStrNode *concatenated = Concat(s, t);printf("连接后的字符串: ");PrintStr(concatenated);// 获取子串int start, length;printf("请输入子串起始位置 (1-%d): ", StrLength(s));scanf("%d", &start);printf("请输入子串长度: ");scanf("%d", &length);LinkStrNode *substring = SubStr(s, start, length);printf("子串为: ");PrintStr(substring);// 插入子串printf("请输入插入位置 (1-%d): ", StrLength(s) + 1);scanf("%d", &start);LinkStrNode *inserted = InsertStr(s, start,t );printf("插入后的字符串: ");PrintStr(inserted);// 删除子串printf("请输入删除起始位置 (1-%d): ", StrLength(s));scanf("%d", &start);printf("请输入删除长度: ");scanf("%d", &length);LinkStrNode *deleted = DelStr(s, start, length);printf("删除后的字符串: ");PrintStr(deleted);// 替换子串printf("请输入替换起始位置 (1-%d): ", StrLength(s));scanf("%d", &start);printf("请输入替换长度: ");scanf("%d", &length);LinkStrNode *replaced = ReplaceSubStr(s, start, length, t);printf("替换后的字符串: ");PrintStr(replaced);return 0;
}

4. 使用说明

  1. 编译并运行上述示例程序。
  2. 根据提示输入两个字符串。
  3. 观察并理解各个操作的结果。

5. 总结

通过本文档,您可以了解到链串的基本概念和常用操作。使用这些操作,您可以轻松地处理字符串数据,进行各种字符串相关的任务。希望这份指南能够帮助您更好地理解和使用链串。

相关文章:

串---链串实现

链串详解 本文档将详细介绍链串的基本概念、实现原理及其在 C 语言中的具体应用。通过本指南&#xff0c;读者将了解如何使用链串进行各种字符串操作。 1. 什么是链串&#xff1f; 链串是一种用于存储字符串的数据结构&#xff0c;它使用一组动态分配的节点来保存字符串中的…...

科技赋能生活——便携气象站

传统气象站往往庞大而复杂&#xff0c;需要专业人员维护&#xff0c;它小巧玲珑&#xff0c;设计精致&#xff0c;可以轻松放入背包或口袋&#xff0c;随身携带&#xff0c;不占空间。无论是城市白领穿梭于高楼大厦间&#xff0c;还是户外爱好者深入山林湖海&#xff0c;都能随…...

Golang——GC原理

1.垃圾回收的目的 将未被引用到的对象销毁&#xff0c;回收其所占的内存空间。 2.根对象是什么 全局变量&#xff1a;在编译器就能确定的存在于程序整个生命周期的变量。 执行栈&#xff1a;每个goroutine都包含自己的执行栈&#xff0c;这些执行栈上包含栈上的变量及指向分配…...

OpenStack概述

一、初识OpenStack OpenStack Docs: 概况 一&#xff09;OpenStack架构简述 1、理解OpenStack OpenStack既是一个社区&#xff0c;也是一个项目和一个开源软件&#xff0c;提供开放源码软件&#xff0c;建立公共和私有云&#xff0c;它提供了一个部署云的操作平台或工具集&…...

机器学习练手(三):基于决策树的iris 多分类和波士顿房价预测

总结&#xff1a;本文为和鲸python 可视化探索训练营资料整理而来&#xff0c;加入了自己的理解&#xff08;by GPT4o&#xff09; 原活动链接 原作者&#xff1a;vgbhfive&#xff0c;多年风控引擎研发及金融模型开发经验&#xff0c;现任某公司风控研发工程师&#xff0c;对…...

PS 2024 百种常用插件下载安装教程【免费使用,先到先得】

文章目录 软件介绍软件下载安装步骤 专栏推荐&#xff1a; 超多精品软件&#xff08;持续更新中…&#xff09; 软件推荐&#xff1a; PS 2024 PR 2024 软件介绍 PS常用插件 此软件整合了市面近百款ps处理插件&#xff0c;可实现&#xff1a;一键制作背景&#xff0c;一键抠图…...

逻辑推理之lora微调

逻辑推理微调 比赛介绍准备内容lora微调lora微调介绍lora优势代码内容 start_vllm相关介绍调用 运行主函数提交结果总结相应连接 比赛介绍 本比赛旨在测试参与者的逻辑推理和问题解决能力。参与者将面对一系列复杂的逻辑谜题&#xff0c;涵盖多个领域的推理挑战。 比赛的连接:…...

前端-防抖代码

//防抖debounce(fn, time 1000) {let timer null;return function (...args) {if (timer) clearTimeout(timer);timer setTimeout(() > {fn.apply(this, args);}, time);};},// 输入变化处理函数async inputChange(value) {if (!this.debouncedInputChange) {this.deboun…...

langchain 入门指南 - 让 LLM 自动选择不同的 Prompt

前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到网站。 让 LLM 自动选择不同的 Prompt 在上一篇文章中&#xff0c;我们学会了如何让 langchain 来自动选择不同的 LLM Chain&#xff0c;以便回…...

web浏览器播放rtsp视频流,海康监控API

概述 这里记录一下如何让前端播放rtsp协议的视频流 ​ 项目中调用海康API&#xff0c;生成的视频流(hls、ws、rtmp等)通过PotPlayer播放器都无法播放&#xff0c;说明视频流有问题&#xff0c;唯独rtsp视频流可以播放。 但是浏览器本身是无法播放rtsp视频的&#xff0c;即使…...

操作系统原理:程序、进程、线程的概念

文章目录 程序、进程、线程的概念程序&#xff08;Program&#xff09;进程&#xff08;Process&#xff09;线程&#xff08;Thread&#xff09;关系总结 在日常对操作系统的使用中&#xff0c;大家肯定对程序、进程和线程多少有所耳闻。作为操作系统的重要一部分&#xff0c;…...

Golang是如何实现动态数组功能的?Slice切片原理解析

Hi 亲爱的朋友们&#xff0c;我是 k 哥。今天&#xff0c;咱们聊一聊Golang 切片。 当我们需要使用数组&#xff0c;但是又不能提前定义数组大小时&#xff0c;可以使用golang的动态数组结构&#xff0c;slice切片。在 Go 语言的众多特性里&#xff0c;slice 是我们经常用到的数…...

SQL注入 报错注入+附加拓展知识,一篇文章带你轻松入门

第5关--------------------------------------------> 前端直接不会显示账号密码的打印&#xff1b;但是在接收前端的数据的那部分后端那里&#xff0c;会看前端传递过来的值是否正确&#xff0c;如果不正确&#xff0c;后端接收值那里就会当MySQL语句执行错误&#xff0c;…...

springboot项目里的包spring-boot-dependencies依赖介绍

springboot项目里的包’spring-boot-dependencies‘依赖 我们一般是在项目的pom dependencyManagement标签里引入spring-boot-dependencies&#xff0c;或者根spring-boot-starter-parent里也是继承了它,也正是因为继承了这个依赖&#xff0c;所以我们在写依赖时才不需要写版本…...

C# 下的限定符运算详解(全部,任意,包含)与示例

文章目录 1.限定符概述2. 全部限定符运算&#xff08;All&#xff09;3. 任意限定符运算&#xff08;Any&#xff09;4. 包含限定符运算&#xff08;Contains&#xff09;总结 当我们在C#编程中需要进行条件判断或集合操作时&#xff0c;限定符&#xff08;qualifiers&#xff…...

消息队列RabbitMQ部分知识

1.简述RabbitMQ的架构设计 RabbitMQ 是一个开源的消息代理&#xff0c;采用了高级消息队列协议&#xff08;AMQP&#xff09;&#xff0c;其架构设计主要包括以下几个关键组件和概念&#xff1a; 1.消息生产者&#xff08; Producer&#xff09;&#xff1a; 负责发送消息到…...

看门狗应用编程-I.MX6U嵌入式Linux C应用编程学习笔记基于正点原子阿尔法开发板

看门狗应用编程 看门狗应用编程介绍 看门狗定时器的基本概念 看门狗是一个可以在一定时间内被复位/重置的计数器 如果在规定时间内没有复位&#xff0c;看门狗计时器溢出会对CPU产生复位信号使系统重启 有些看门狗可以只产生中断信号而不会使系统复位 I.MX6UL/I.MX6ULL So…...

Bug 解决 | 本地项目上线后出现错误

目录 一、前言 二、原因分析 1、本地代码误发线上 2、环境差异 3、配置差异 4、资源路径差异 5、API 接口差异 6、用量差异 一、前言 大家好&#xff0c;我是小洪爱分享。在开发上线项目的过程中&#xff0c;我们经常会遇到一种让人头疼的情况。那就是开发好的项目功能…...

为什么我工作 10 年后转行当程序员?逆袭翻盘!

今天文章的主人公暂且称他为 A 君。不过 A 君有点特别&#xff0c;非科班&#xff0c;工作 10 年后才转行 iOS 程序员。今年 36 岁&#xff0c;目前在某行业头部企业任职前端负责人&#xff0c;管理 40 人的前端团队。 废话不多说&#xff0c;我们开始 A 君&#xff08;为了描…...

见证中国数据库的崛起:从追赶到引领的壮丽征程《四》

见证中国数据库的崛起&#xff1a;从追赶到引领的壮丽征程《四》 四、未来展望&#xff1a;中国数据库的机遇与挑战新技术带来的机遇全球化竞争的挑战数据安全与隐私保护的挑战人才培养的持续挑战 【纪录片】中国数据库前世今生 在数字化潮流席卷全球的今天&#xff0c;数据库作…...

Docker 离线安装指南

参考文章 1、确认操作系统类型及内核版本 Docker依赖于Linux内核的一些特性&#xff0c;不同版本的Docker对内核版本有不同要求。例如&#xff0c;Docker 17.06及之后的版本通常需要Linux内核3.10及以上版本&#xff0c;Docker17.09及更高版本对应Linux内核4.9.x及更高版本。…...

【Python】 -- 趣味代码 - 小恐龙游戏

文章目录 文章目录 00 小恐龙游戏程序设计框架代码结构和功能游戏流程总结01 小恐龙游戏程序设计02 百度网盘地址00 小恐龙游戏程序设计框架 这段代码是一个基于 Pygame 的简易跑酷游戏的完整实现,玩家控制一个角色(龙)躲避障碍物(仙人掌和乌鸦)。以下是代码的详细介绍:…...

【第二十一章 SDIO接口(SDIO)】

第二十一章 SDIO接口 目录 第二十一章 SDIO接口(SDIO) 1 SDIO 主要功能 2 SDIO 总线拓扑 3 SDIO 功能描述 3.1 SDIO 适配器 3.2 SDIOAHB 接口 4 卡功能描述 4.1 卡识别模式 4.2 卡复位 4.3 操作电压范围确认 4.4 卡识别过程 4.5 写数据块 4.6 读数据块 4.7 数据流…...

Golang dig框架与GraphQL的完美结合

将 Go 的 Dig 依赖注入框架与 GraphQL 结合使用&#xff0c;可以显著提升应用程序的可维护性、可测试性以及灵活性。 Dig 是一个强大的依赖注入容器&#xff0c;能够帮助开发者更好地管理复杂的依赖关系&#xff0c;而 GraphQL 则是一种用于 API 的查询语言&#xff0c;能够提…...

屋顶变身“发电站” ,中天合创屋面分布式光伏发电项目顺利并网!

5月28日&#xff0c;中天合创屋面分布式光伏发电项目顺利并网发电&#xff0c;该项目位于内蒙古自治区鄂尔多斯市乌审旗&#xff0c;项目利用中天合创聚乙烯、聚丙烯仓库屋面作为场地建设光伏电站&#xff0c;总装机容量为9.96MWp。 项目投运后&#xff0c;每年可节约标煤3670…...

CocosCreator 之 JavaScript/TypeScript和Java的相互交互

引擎版本&#xff1a; 3.8.1 语言&#xff1a; JavaScript/TypeScript、C、Java 环境&#xff1a;Window 参考&#xff1a;Java原生反射机制 您好&#xff0c;我是鹤九日&#xff01; 回顾 在上篇文章中&#xff1a;CocosCreator Android项目接入UnityAds 广告SDK。 我们简单讲…...

vue3 定时器-定义全局方法 vue+ts

1.创建ts文件 路径&#xff1a;src/utils/timer.ts 完整代码&#xff1a; import { onUnmounted } from vuetype TimerCallback (...args: any[]) > voidexport function useGlobalTimer() {const timers: Map<number, NodeJS.Timeout> new Map()// 创建定时器con…...

C++.OpenGL (10/64)基础光照(Basic Lighting)

基础光照(Basic Lighting) 冯氏光照模型(Phong Lighting Model) #mermaid-svg-GLdskXwWINxNGHso {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-GLdskXwWINxNGHso .error-icon{fill:#552222;}#mermaid-svg-GLd…...

Axios请求超时重发机制

Axios 超时重新请求实现方案 在 Axios 中实现超时重新请求可以通过以下几种方式&#xff1a; 1. 使用拦截器实现自动重试 import axios from axios;// 创建axios实例 const instance axios.create();// 设置超时时间 instance.defaults.timeout 5000;// 最大重试次数 cons…...

【开发技术】.Net使用FFmpeg视频特定帧上绘制内容

目录 一、目的 二、解决方案 2.1 什么是FFmpeg 2.2 FFmpeg主要功能 2.3 使用Xabe.FFmpeg调用FFmpeg功能 2.4 使用 FFmpeg 的 drawbox 滤镜来绘制 ROI 三、总结 一、目的 当前市场上有很多目标检测智能识别的相关算法&#xff0c;当前调用一个医疗行业的AI识别算法后返回…...