字符串函数和内存函数

🍕博客主页:️自信不孤单

🍬文章专栏:C语言

🍚代码仓库:破浪晓梦

🍭欢迎关注:欢迎大家点赞收藏+关注

字符串函数和内存函数

文章目录

  • 字符串函数和内存函数
  • 前言
  • 1. 字符串函数介绍
    • 1.1 strlen函数
    • 1.2 strcpy函数
    • 1.3 strcat函数
    • 1.4 strcmp函数
    • 1.5 strncpy函数
    • 1.6 strncat函数
    • 1.7 strncmp函数
    • 1.8 strstr函数
    • 1.9 strtok函数
    • 1.10 strerror函数
    • 1.11 字符分类函数
  • 2. 内存函数的介绍
    • 2.1 memcpy
    • 2.2 memmove
    • 2.3 memcmp
    • 2.4 memset
  • 3. 模拟库函数的实现
    • 3.1 模拟strlen函数
    • 3.2 模拟strcpy函数
    • 3.3 模拟strcat函数
    • 3.4 模拟strcmp函数
    • 3.5 模拟strstr函数
    • 3.6 模拟memcpy函数
    • 3.7 模拟memmove函数


前言

C语言中对字符和字符串的处理很是频繁,但是C语言本身是没有字符串类型的,字符串通常放在常量字符串中或者字符数组中。其中字符串常量适用于那些对它不做修改的字符串函数。

1. 字符串函数介绍

字符串函数需要包含头文件

#include <string.h>

1.1 strlen函数

size_t strlen(const char* str);

strlen函数(string length)的作用是计算返回字符串中结束标识符'\0'之前出现的的字符个数,因此,strlen函数所处理的字符串必须是以结束标识符'\0'结尾的字符串。其返回值类型为size_t 类型,该类型为无符号类型。

strlen函数的应用:

#include <stdio.h>
int main()
{
	const char* str1 = "abcdef";
	const char* str2 = "bbb";
	if (strlen(str2) - strlen(str1) > 0)
	{
		printf("str2>str1\n");
	}
	else
	{
		printf("srt1>str2\n");
	}
	return 0;
}

输出结果为:

str1>str2

这里strlen(str2)的返回值为6,strlen(str1)的返回值为3,但为什么输出的是str1>str2呢?
由于strlen()的返回值是无符号类型,故两个无符号数相减也是无符号数,相减后的无符号数大于0,所以输出str1>str2。

要想使用相减来比较字符串长度的大小,可以强制类型转换为有符号整型。
如:

#include<stdio.h>
#include<string.h> 
int main()
{
	const char* str1 = "abcdef";
	const char* str2 = "bbb";

	if ((int)strlen(str2) - (int)strlen(str1) > 0)
	{
		printf("str2 > str1\n");
	}
	else
	{
		printf("srt1 < str2\n");
	}
	return 0;
}

1.2 strcpy函数

char* strcpy(char* destination, const char* source);

strcpy函数(string copy)的作用是,可以将字符串从源地址复制至目的地址,并返回目的空间的起始地址,并且它会将源地址内的结束标识符'\0'一并拷贝过去,因此源地址必须以'\0'结尾,且目的地址也将以'\0'结尾,通俗来讲就是用来实现字符串的复制和拷贝。并且,因为其作用为拷贝字符串,因此目标地址内的空间必须足够大,要有足够的空间容纳下源地址内的字符串,同时目的地址的空间必须是可修改的。

strcpy函数的应用:

#include <stdio.h>
#include <string.h>
int main()
{
	char arr1[20];
	char arr2[15] = "hello world";
	printf("%s", strcpy(arr1, arr2));
	return 0;
}

输出结果为:

hello world

这里要特别注意的是strcpy函数返回的是目标空间的起始地址,该函数设置返回值类型的目的是为了实现链式访问。

1.3 strcat函数

char* strcat(char* destination, const char* source);

strcat函数(string catenate)的作用是,将源地址的字符串追加补充至目的地址处,并返回目的空间的首地址。与字符串拷贝函数相同,它在进行补充追加时是从目的地址的结束标识符处'\0'开始追加的,追加至源地址的结束标识符处停止。且它同样要求目标地址内的空间必须足够大,要有足够的空间容纳下源地址内的字符串,同时目的地址的空间必须是可修改的。

strcat函数的应用:

#include <stdio.h>
#include <string.h>
int main()
{
	char arr1[20] = "hello";
	char arr2[10] = " world";
	printf("%s\n", strcat(arr1, arr2));
	return 0;
}

输出结果为:

hello world

1.4 strcmp函数

int strcmp(const char* str1, const char* str2);

strcmp函数(string compare)的作用为按照顺序依次比较两字符串对应位置字符的ASCII码值(注意不是比较两字符串的长度),直到结束标识符'\0'或对应位置的字符不同。若比较至结束标识符都没有不同则两字符串相等,若两字符串对应位置字符有不同,对应位置上字符ASCII码值小的字符串小于另一个字符串。

标准规定:
第一个字符串大于第二个字符串,则返回大于0的数字
第一个字符串等于第二个字符串,则返回0
第一个字符串小于第二个字符串,则返回小于0的数字

strcmp函数的应用:

#include<stdio.h>
#include<string.h>
int main()
{
	const char arr1[] = "abcd";
	const char arr2[] = "abz";
	int ret = strcmp(arr1, arr2);
	if (ret > 0)
	{
		printf("arr1 > arr2\n");
	}
	else if (ret == 0)
	{
		printf("arr1 = arr2\n");
	}
	else
	{
		printf("arr1 < arr2\n");
	}
	return 0;
}

输出结果为:

arr1 < arr2

1.5 strncpy函数

char* strncpy(char* destination, const char* source, size_t num);

strncpy函数(string number copy)的作用为将指定长度的字符串复制到字符数组中,即表示把源地址中字符串开始的前n个字符拷贝到目的地址中,并返回目的空间的首地址。与strcpy相同,它同样会将源地址内的结束标识符'\0'一并拷贝过去,因此源地址必须以'\0'结尾,且目的地址也将以'\0'结尾。并且,因为其作用为拷贝字符串,因此目标地址内的空间必须足够大,要有足够的空间容纳下源地址内的字符串,同时目的地址的空间必须是可修改的。

strncpy函数的应用:

#include <stdio.h>
#include <string.h>
int main()
{
	char arr1[20] = "xxxxxxxxxxxxxx";
	char arr2[12] = "hello world";
	printf("%s", strncpy(arr1, arr2, 10));
	return 0;
}

输出结果为:

hello worlxxxx

1.6 strncat函数

char* strncat(char* destination, const char* source, size_t num);

strncat函数(string num catenate)的作用为从源地址处将指定长度的字符串追加补充到目的地址中与strcat函数类似,它在进行补充追加时也是从目的地址的结束标识符处'\0'开始追加的,不同的是追加至参数(单位是字节)限制的字符数处停止。但它同样要求目标地址内的空间必须足够大,要有足够的空间容纳下补充的字符串 ,同时目的地址的空间必须是可修改的。

返回值为目的空间的首地址

strncat函数的应用:

#include <stdio.h>
#include <string.h>
int main()
{
	char arr1[20] = "hello";
	char arr2[10] = " world";
	printf("%s\n", strncat(arr1, arr2, 4));
	return 0;
}

输出结果为:

hello wor

1.7 strncmp函数

int strncmp(const char* str1, const char* str2, size_t num);

strncmp函数(string number compare)的作用为有限制(单位是字节)的按照顺序依次比较两字符串对应位置字符的ASCII码值(注意不是比较两字符串的长度),直到参数限制位数位置上全部比较结束或对应位置的字符不同。若参数限制位数位置上的字符都比较结束且都没有不同则两字符串相等,若两字符串对应位置字符有不同,对应位置上字符ASCII码值小的字符串小于另一个字符串。

标准规定:
第一个字符串大于第二个字符串,则返回大于0的数字
第一个字符串等于第二个字符串,则返回0
第一个字符串小于第二个字符串,则返回小于0的数字

strncmp函数的应用:

#include<stdio.h>
#include<string.h>
int main()
{
	const char arr1[] = "abcd";
	const char arr2[] = "abz";
	int ret = strncmp(arr1, arr2, 2);
	if (ret > 0)
	{
		printf("arr1 > arr2\n");
	}
	else if (ret == 0)
	{
		printf("arr1 = arr2\n");
	}
	else
	{
		printf("arr1 < arr2\n");
	}
	return 0;
}

输出结果为:

arr1 = arr2

1.8 strstr函数

char* strstr(const char* str1, const char* str2);

strstr函数(string string)的作用为从一个字符串中寻找其字串,通俗来讲就是从一个字符串中寻找另一个字符串。若找到目标字串则返回指向目标字串的指针,若没有找到目标字串则返回空指针。

strstr函数的应用:

#include<stdio.h>
#include<string.h>
int main()
{
	const char arr1[] = "abcdefg";
	const char arr2[] = "cde";
	char* ret = strstr(arr1, arr2);
	//从ar1中寻找arr2
	if (ret == NULL)
	{
		printf("找不到该字符串!\n");
	}
	else
	{
		printf("成功找到该字符串\"%s\"!\n", ret);
	}
	return 0;
}

输出结果为:

成功找到该字符串"cdefg"!

1.9 strtok函数

char* strtok(char* str, const char* sep);

strtok函数(string token)的作用为将字符串分解为一组字符串。该函数有两个数组作为参数,它的实际作用便是将其中一个数组为分割数组,在另一个数组中寻找这些分割符,并在分割符处将这个数组内的字符串加上结束标识符'\0',将其分割成一组(多个)字符串。若第一个参数不为NULL,将找到字符数组中的第一个标记并保存它在字符串中的位置;若第一个参数为NULL,将在同一个字符串中被保存的位置开始,查找下一个标记。

返回值:该函数返回被分解的第一个子字符串,如果没有可检索的字符串,则返回一个空指针。

strtok函数的应用:

#include<stdio.h>
#include<string.h>
int main()
{
	char arr1[] = "1592272237@qq.com";
	char arr2[30] = { 0 };
	strcpy(arr2, arr1);
	const char* arr3 = "@.";
	printf("账号:%s\n", strtok(arr2, arr3));
	//找到第一个标记停止
	printf("域名前缀:%s\n", strtok(NULL, arr3));
	//从保存好的位置开始往后找
	printf("域名后缀:%s\n", strtok(NULL, arr3));
	//从保存好的位置开始往后找
	return 0;
}

输出结果为:

账号:1592272237
域名前缀:qq
域名后缀:com

  1. strtok函数是会对数组本身进行操作的,所以我们为了保护原始数据,在定义并初始化好字符数组之后,又定义了一个新的数组并将原始数据拷贝过去,作为临时拷贝供我们进行操作。
  2. 接着我们定义并初始化了分割符数组,函数将根据分割符数组arr3对临时拷贝arr2进行分割。第一次执行函数时之前没有标记,于是直接进行操作找到第一个标记并分割打印。
  3. 此时就已经存在标记了,再连续两次找到前一次执行作下的标记按照分割符将数组分割完毕并打印。

我们可以将上述代码优化为:

#include<stdio.h>
#include<string.h>
int main()
{
	char arr1[] = "1592272237@qq.com";
	char arr2[30] = { 0 };
	strcpy(arr2, arr1);
	const char* arr3 = "@.";
	char* str = NULL;
	for (str = strtok(arr2, arr3); str != NULL; str = strtok(NULL, arr3))
	{
		printf("%s\n", str);
	}
	return 0;
}

1.10 strerror函数

char* strerror(int errnum);

strerror函数(string error)的作用为返回错误码对应的信息。即根据接收到的错误码(错误码errno为全局变量),返回错误原因的详细信息。

strerror函数的应用:

#include<stdio.h>
#include<string.h>
int main()
{
	int i = 0;
	for (i = 0; i <= 4; i++)
	{
		printf("错误原因为:%s\n", strerror(i));
	}
	return 0;
}

输出结果为:

在这里插入图片描述

1.11 字符分类函数

函数如果他的参数符合下列条件就返回真
iscntrl任何控制字符
isspace空白字符:空格' ',换页'\f',换行'\n',回车'\r',制表符'\t'或者垂直制表符'\v'
isdigit十进制数字0~9
isxdigi十六进制数字,包括所有十进制数字,小写字母a~f,大写字母A~F
islower小写字母a~z
isupper大写字母A~Z
isalpha字母a~zA~Z
isalnum字母或者数字,a~zA~Z0~9
ispunct标点符号,任何不属于数字或者字母的图形字符(可打印)
isgraph任何图形字符
isprint任何可打印字符,包括图形字符和空白字符

字符转换:

int tolower(int c);
int toupper(int c);

举个栗子:

此代码的作用是将大写字符转换为小写字符

/* isupper example */
#include <stdio.h>
#include <ctype.h>
int main()
{
	int i = 0;
	char str[] = "Test String.\n";
	char c;
	while (str[i])
	{
		c = str[i];
		if (isupper(c))
			c = tolower(c);
		putchar(c);
		i++;
	}
	return 0;
}

输出结果为:

test string.

2. 内存函数的介绍

2.1 memcpy

void* memcpy(void* destination, const void* source, size_t num);

memcpy函数(memory copy)的作用为从源内存空间向目的内存空间拷贝限制数量(单位是字节)的数据。它与strcpy函数类似,作用均为拷贝数据,不同的是strcpy仅仅只操作字符串故会在结束标识符'\0'处停止,而memcpy函数操作的是内存,内存中的数据是相邻的,故不会在结束标识符处停止。
返回值:该函数返回一个指向目标存储区destination的指针。

memcpy函数的应用:

#include <stdio.h>
#include <string.h>
struct {
	char name[40];
	int age;
} person, person_copy;
int main()
{
	char myname[] = "Pierre de Fermat";
	/* using memcpy to copy string: */
	memcpy(person.name, myname, strlen(myname) + 1);
	person.age = 46;
	/* using memcpy to copy structure: */
	memcpy(&person_copy, &person, sizeof(person));
	printf("person_copy: %s, %d \n", person_copy.name, person_copy.age);
	return 0;
}

输出结果为:

在这里插入图片描述

注:如果源内存空间和目标内存空间有重叠时,memcpy是无法保证内存复制的正确性的。

2.2 memmove

void* memmove(void* destination, const void* source, size_t num);

memmove函数(memory move)和memcpy函数都是内存复制函数,他们在用法上也一样。但是如果目标区域和源区域有重叠的话,memmove是比memcpy更安全的方法,memmove能够保证源串在被覆盖之前将重叠区域的字节拷贝到目标区域中,复制后源区域的内容会被更改。如果目标区域与源区域没有重叠,则和memcpy函数功能是完全相同的。

memmove函数的应用:

#include <stdio.h>
#include <string.h>
int main()
{
	char str[] = "memmove can be very useful......";
	memmove(str + 20, str + 15, 11);
	puts(str);
	return 0;
}

输出结果为:

在这里插入图片描述

2.3 memcmp

int memcmp(const void* ptr1,
	const void* ptr2,
	size_t num);

memcmp函数(memory compare)的作用与 strcmp 函数的作用类似,不过 memcmp 函数是从内存的角度以字节为单位进行处理,故 memcmp 函数同样需要第三个参数进行限制,而不会在结束标识符 ’ \0 ’ 处停止比较。

memcmp函数的应用:

#include <stdio.h>
#include <string.h>
int main()
{
	char buffer1[] = "DWgaOtP12df0";
	char buffer2[] = "DWGAOTP12DF0";
	int n;
	n = memcmp(buffer1, buffer2, sizeof(buffer1));
	if (n > 0) printf("'%s' is greater than '%s'.\n", buffer1, buffer2);
	else if (n < 0) printf("'%s' is less than '%s'.\n", buffer1, buffer2);
	else printf("'%s' is the same as '%s'.\n", buffer1, buffer2);
	return 0;
}

输出结果为:

在这里插入图片描述

2.4 memset

void* memset(void* dest, int c, size_t count);

memset函数(memory set)的作用是复制字符c(一个无符号字符)到参数dest所指向的字符串的前n个字符。

返回值是:该值返回一个指向存储区dest的指针。

memset函数的应用:

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

int main()
{
	char str[50];
	strcpy(str, "This is string.h library function");
	puts(str);
	memset(str, '$', 7);
	puts(str);
	return(0);
}

输出结果为:

在这里插入图片描述

3. 模拟库函数的实现

3.1 模拟strlen函数

//1.计数器
size_t my_strlen1(const char* str)
{
	assert(str);
	int count = 0;
	while (*str)
	{
		count++;
		str++;
	}
	return count;
}

//2.递归调用
size_t my_strlen2(const char* str)
{
	assert(str);
	if (*str)
		return sim_strlen2(str + 1) + 1;
	else
		return 0;
}

//3.指针相减
size_t my_strlen3(const char* str)
{
	assert(str);
	const char* p = str;
	while (*p++)
	{
		;
	}
	return (p - str - 1);
}

3.2 模拟strcpy函数

char* my_strcpy(char* destination, const char* source)
{
	char* ret = destination;
	assert(destination && source);
	while (*destination++ = *source++)
	{
		;
	}
	return ret;
}

3.3 模拟strcat函数

char* my_strcat(char* destination, const char* source)
{
	char* ret = destination;
	while (*destination)
	{
		destination++;
	}
	while (*destination++ = *source++)
	{
		;
	}
	return ret;
}

3.4 模拟strcmp函数

int my_strcmp(const char* s1, const char* s2)
{
	assert(s1 && s2);

	while (*s1 == *s2 && *s1 != '\0' && *s2 != '\0')
	{
		s1++;
		s2++;
	}
	return *s1 - *s2;
}

3.5 模拟strstr函数

char* my_strstr(const char* s1, const char* s2)
{
	assert(s1 && s2);
	if (*s2 == '\0')
	{
		return (char*)s1;
	}
	const char* p1 = s1;
	const char* p2 = s2;
	const char* start = s1; //用于记录开始比较的位置
	while (*p1)
	{	//以防主串可能小于子串,循环条件不能只为*p1 == *p2
		while (*p1 == *p2 && *p1 != '\0' && *p2 != '\0') //单字符匹配成功则匹配下一个字符
		{
			p1++;
			p2++;
		}
		if (*p2 == '\0') //匹配成功,返回开始比较的位置
			return (char*)start;
		//单趟匹配失败,子串退回到第一个字符,进行从主串的下一个字符开始与主串匹配
		p1 = start + 1;
		start = p1;
		p2 = s2;
	}
	return NULL; //跳出外层循环说明匹配失败
}

3.6 模拟memcpy函数

void* my_memcpy(void* dest, const void* src, size_t nums)
{
	assert(src && dest);
	int i = 0;
	for (i = 0; i < nums; i++)
	{
		*((char*)dest + i) = *((char*)src + i);
	}
	return (char*)dest;
}

3.7 模拟memmove函数

void* my_memmove(void* dest, const void* src, size_t nums)
{
	assert(src && dest);
	void* ret = dest;
	int i = 0;
	if (dest < src)//从前向后
	{
		for (i = 0; i < nums; i++)
		{
			*((char*)dest + i) = *((char*)src + i);
		}
	}
	else//从后向前
	{
		for (i = nums - 1; i >= 0; i--)
		{
			*((char*)dest + i) = *((char*)src + i);
		}
	}
	return ret;
}

到此,关于《字符串函数和内存函数》的内容就结束了
感谢大家的观看
创作不易,如果对您有帮助,请一键三连
您的支持就是我创作最大的动力!!!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【MySQL】MySQL的优化(一)

目录 查看SQL执行频率 定位低效率执行SQL 定位低效率执行SQL-慢查询日志 定位低效率执行SQL-show processlist 查看SQL执行频率 MySQL 客户端连接成功后&#xff0c;通过 show [session|global] status 命令可以查看服务器状态信息。通 过查看状态信息可以查看对当…

jvm类与类加载

1.类加载过程&#xff1a; 首先要加载某个类一定是出于某种目的&#xff0c;比如要运行java程序&#xff0c;那么久必须加载主类才能运行其中的方法&#xff0c;所以一般在这些情况下&#xff0c;如果类没有被加载&#xff0c;就会自动被加载&#xff1a; 1.使用new创建对象时 …

MyBatis开发环境搭建

1.创建工程 2.引入相关的依赖 pom.xml <dependencies><!--1.引入mybatis包--><dependency><groupId>org.mybatis</groupId><artifactId>mybatis</artifactId><version>3.4.6</version></dependency><!--2.单元…

FPGA和IC设计怎么选?哪个发展更好?

很多人纠结FPGA和IC设计怎么选&#xff0c;其实往小了说&#xff0c;要看你选择的具体是哪个方向岗位。往大了说&#xff0c;将来你要是走更远&#xff0c;要成为大佬&#xff0c;那基本各个方向的都要有涉及的。 不同方向就有不同的发展&#xff0c;目前在薪资上IC设计要比FP…

Qt之高仿QQ系统设置界面

QQ或360安全卫士的设置界面都是非常有特点的,所有的配置项都在一个垂直的ScrollArea中,但是又能通过左侧的导航栏点击定位。这样做的好处是既方便查看指定配置项,又方便查看所有配置项。 一.效果 下面左边是当前最新版QQ的系统设置界面,右边是我的高仿版本,几乎一毛一样…

【Linux】进程的程序替换

文章目录1. 程序替换1.创建子进程的目的是什么&#xff1f;2.了解程序是如何进行替换的3. 程序替换的基本原理当创建进程的时候&#xff0c;先有进程数据结构&#xff0c;还是先加载代码和数据&#xff1f;程序替换是整体替换&#xff0c;不是局部替换execl 返回值4. 替换函数1…

【三维几何学习】从零开始网格上的深度学习-2:卷积网络CNN篇(Pytorch)

本文参加新星计划人工智能(Pytorch)赛道&#xff1a;https://bbs.csdn.net/topics/613989052 从零开始网格上的深度学习-2:卷积网络CNN篇引言一、概述1.1 卷积操作简述1.2 网格上的面卷积二、核心代码2.1 面卷积2.2 网络框架三、基于CNN的网格分类3.1 分类结果3.2 全部代码引言…

FPGA之时钟规划图解

目录 一、前言 二、时钟规划概念 三、时钟规划的模块 四、时钟规划之时钟单元布局 4.1 BUFG 4.2 BUFH 4.3 BUFR 4.4 BUFIO 五、时钟规划之时钟单元走线 5.1 BUFG->BUFH 5.2 BUFR->FF 5.3 BUFIO->FF 一、前言 对于vivado这类使用verilog语言的进…

《Netty》从零开始学netty源码(七)之NioEventLoop.selectStrategy

NioEventLoop是一个事件轮询器&#xff0c;在它的run方法中其实是一个for死循环&#xff0c;不断重复三个过程&#xff1a;1. 获取IO事件&#xff0c;2. 处理IO事件&#xff0c;3. 处理任务队列中的task&#xff0c;而SelectStractegy就是用于第一步获取IO事件&#xff0c;它的…

css:使用filter和backdrop-filter实现高斯模糊效果

背景 今天接到一个需求是&#xff0c;使用高斯模糊的效果对一个页面进行模糊处理&#xff0c;正好借这个机会来整理一下 css3 中高斯模糊的两个 API API介绍 filter 说明&#xff1a; 该 API 是一个过滤器&#xff0c;不仅能实现高斯模糊&#xff0c;还有很多比如颜色偏移、…

接口文档包含哪些内容?怎么才能写好接口文档?十年测试老司机来告诉你

目录 接口文档结构 参数说明 示例 错误码说明 语言基调通俗易懂 及时更新与维护 总结 那么我们该如何写好一份优秀的接口文档呢&#xff1f; 接口文档结构 首先我们要知道文档结构是什么样子的。接口文档应该有清晰明确的结构&#xff0c;以便开发人员能快速定位自己需…

经典文献阅读之--Dynamic-VINS(动态点滤除VINS)

0. 简介 现在的SLAM算法在静态环境中表现良好&#xff0c;但在动态环境中很容易失败。最近的工作将基于深度学习的语义信息引入到SLAM系统以减轻动态对象的影响。然而&#xff0c;在资源受限的机器人的动态环境中应用鲁棒定位仍然具有挑战性。所以《RGB-D Inertial Odometry f…

ES+Redis+MySQL,这个高可用架构设计太顶了!

一、背景 会员系统是一种基础系统&#xff0c;跟公司所有业务线的下单主流程密切相关。如果会员系统出故障&#xff0c;会导致用户无法下单&#xff0c;影响范围是全公司所有业务线。所以&#xff0c;会员系统必须保证高性能、高可用&#xff0c;提供稳定、高效的基础服务。 …

vue笔记

第一个Vue应用 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><meta http-equiv"X-UA-Compatible" content"IEedge" /><meta name"viewport" content"widthdevice-…

【零基础入门前端系列】—动画和弹性盒模型(二十四)

【零基础入门前端系列】—动画和弹性盒模型&#xff08;二十四&#xff09; 一、概念 动画是使元素从一种样式逐渐变化为另一种样式&#xff0c;你可以改变任意多的样式任意多的次数。 请用百分比来规定变化发生的时间&#xff0c;或用关键词from和to&#xff0c;等同0%和10…

购物清单(蓝桥杯C/C++省赛)

目录 1 问题描述 2 文件的读取格式 3 代码实现 1 问题描述 小明刚刚找到工作&#xff0c;老板人很好&#xff0c;只是老板夫人很爱购物。老板忙的时候经常让小明帮忙到商场代为购物。小明很厌烦&#xff0c;但又不好推辞。 这不&#xff0c;XX大促销又来了&#xff01;老板…

项目实战典型案例26——nacos的命名空间名称和id不一致带来的思考

nacos的命名空间名称和id不一致带来的思考一&#xff1a;背景介绍Nacos命名空间相关知识点思考总结一&#xff1a;背景介绍 项目用的naocs做的配置中心和服务发现。由于开发环境和本地环境使用的都是同一个命名空间&#xff0c;我们多个服务相互调用的时候&#xff0c;由于开发…

若依分离版下拉框动态加载

最近在学习使用若依分离版框架&#xff0c;想要实现下拉框动态加载另一张表的数据&#xff0c;于是参考【字典数据-字典名称】的实现方式&#xff0c;成功试下下拉框动态加载&#xff0c;做下记录 涉及表格&#xff1a;his_user&#xff08;用户表&#xff09;-- 用户管理&…

【linux】:进程概念

文章目录 冯诺依曼体系结构一&#xff1a;操作系统二: 进程总结冯诺依曼体系结构 我们常见的计算机&#xff0c;如笔记本。我们不常见的计算机&#xff0c;如服务器&#xff0c;大部分都遵守冯诺依曼体系。 冯诺依曼体系如下图&#xff1a; 那么输入设备有哪些呢&#xff1f…

常见的Web安全漏洞:SYN攻击/CSRF/XSS

一、SYN攻击&#xff08;属于DOS攻击&#xff09; 什么情况下被动方出现SYN_RCVD状态?(flood攻击服务) 客户伪造 ip 端口&#xff0c; 向服务端发送SYN请求。完成2次握手&#xff0c;第三次服务端 等待客户端ACK确认&#xff0c;但由于客户不存在服务端一直未收到确认&#…