C++引用、内联函数、auto关键字介绍以及C++中无法使用NULL的原因

news2024/11/28 12:43:22

文章目录

    • 一、引用
      • 1.1 引用概念
      • 1.2 引用特性
      • 1.3 常引用
      • 1.4 使用场景
        • 1.4.1 做参数
        • 1.4.2做返回值
      • 1.5 引用和指针的区别
      • 1.6 小结一下
    • 二、内联函数
      • 2.1 内联的概念
      • 2.2 内联的特性
      • 2.3 【面试题】
    • 三、auto关键字(C++11)
      • 3.1 类型别名思考
      • 3.2 auto简介
    • 四、auto的使用细则
      • 4.1 基于范围的for循环(C++11)
      • 4.2 范围for的使用条件
    • 五、指针空值nullptr(C++11)

一、引用

1.1 引用概念

C++是C语言的继承,它可进行过程化程序设计,又可以进行以抽象数据类型为特点的基于对象的程序设计,还可以进行以继承和多态为特点的面向对象的程序设计。引用(reference)就是C++对C语言的重要扩充。引用就是某一变量(目标)的一个别名,对引用的操作与对变量直接操作完全一样。引用的声明方法:类型标识符 &引用名=目标变量名; -->百度百科

  • 这个引用就相当于是别名
void TestRef()
{
	int a = 10;
	int& ra = a;//<====定义引用类型
	printf("%p\n", &a);
	printf("%p\n", &ra);
}
  • 类型& 引用变量名(对象名) = 引用实体;

在这里插入图片描述

  • 我们通过调试来看一下:

在这里插入图片描述

1.2 引用特性

  1. 引用在定义时必须初始化
  2. 一个变量可以有多个引用
  3. 引用一旦引用一个实体再不能引用其他实体
void main()
{
	int a = 10;
	// int& ra; // 该条语句编译时会出错,必须要初始化
	int& ra = a;
	int& rra = a;

	printf("%p %p %p\n", &a, &ra, &rra);
}

在这里插入图片描述

1.3 常引用

  • 取别名不能放大权限
int main()
{
	int a = 0;
	// 权限的缩小
	const int& c = a;
	
	const int x = 10;
	// 权限的放大
	int& y = x;
	return 0;
}

在这里插入图片描述

  • 可以这样写,相等的就可以
const int x = 10;
const int y = x; 
  • a+x的结果是一个临时变量,临时变量具有常性,const引用就可以
  • int& n = a + x;的返回值是临时变量,临时对象具有常性,是一个权限放大
int a = 0;
const int x = 10;
const int& z = 10;
const int& m = a + x; //这样写也可以
int& n = a + x; // 这样写不可以
  • 这里const加上就可以将不同类型取别名
  • 类型转换的时候会出现一个临时变量**,临时变量具有常性,所以就可以~**
double d = 1.1;
int i = d; // 强制类型转换
int& ri = d; // 无法赋予,类型不同
const int& ri = d; // 加上const就可以了
  • 被引用的实体不能是常量
  • 引用的类型必须相同

1.4 使用场景

引用有两个场景分别是做参数和做返回值

1.4.1 做参数
  • 做参数【在我们C语言阶段的时候使用函数交换两个变量的值就需要传地址过去,否则的话,形参只是实参的一份临时拷贝】
void Swap(int* a, int* b)
{
	int tmp = *a;
	*a = *b;
	*b = tmp;
}
int main()
{
	int x = 0, y = 1;
	
	Swap(&x, &y);
	return 0;
}
  • 而我们学了引用,这个时候就可以把指针替换下去了,使用引用作为参数
void Swap(int& a, int& b)
{
	int tmp = a;
	a = b;
	b = tmp;
}

int main()
{
	int x = 0, y = 1;
	
	Swap(x, y);
	return 0;
}

在这里插入图片描述

在这里插入图片描述

  • 那么引用可以代替指针吗?【不可以!】

指针和引用的功能是类似的,有重叠的
C++的引用,对指针使用比较复杂的场景进行一些替换,让代码更简单易懂,但是不能完全替代指

  • 针引用不能完全替代指针原因:引用定义后,不能改变指向

就比如说在数据结构中学的链表,指针需要改变指向,引用不能改变指向,这就是引用不能代替指针的原因

  • 那java和pythone等其他语言有没有指针?–>没有
  • 那他们的链表是怎么实现的呢?–>引用
  • 本质上就是引用可以改变指向

  • 我们再来看一个案例,在我们学数据结构的时候单链表学习阶段
  • 有这这么一段代码,这里的pphead必须要传二级指针,有点不好理解呀
void PushBack(struct Node** pphead, int x)
{
	 *pphead = newnode;
}
int main()
{
	struct Node* plist = NULL;
	return 0;
}
  • 而我们学了引用就可以这样写了,加上一个引用,也就是phead是plist的一份临时拷贝
void PushBack(struct Node*& phead, int x)
{
	phead = newnode;
}

int main()
{
	struct Node* plist = NULL;

	return 0;
}

传值、传引用效率比较

  • 我们这里可以测试一下性能,对比一下传值、传引用
#include <time.h>

struct A { int a[10000]; };
void TestFunc1(A a) {}
void TestFunc2(A& a) {}

void main()
{
	A a;
	// 以值作为函数参数
	size_t begin1 = clock();
	for (size_t i = 0; i < 10000; ++i)
		TestFunc1(a);
	size_t end1 = clock();

	// 以引用作为函数参数
	size_t begin2 = clock();
	for (size_t i = 0; i < 10000; ++i)
		TestFunc2(a);
	size_t end2 = clock();

	// 分别计算两个函数运行结束后的时间
	cout << "TestFunc1(A)-time:" << end1 - begin1 << endl;
	cout << "TestFunc2(A&)-time:" << end2 - begin2 << endl;
}

  • 可以看到传值需要8毫秒,而传引用小于0毫秒,小于0毫秒这里显示不出来,所以显示的0

在这里插入图片描述

  • 那我再次测试一下以引用作为函数的返回值类型以值作为函数的返回值类型的性能
#include <time.h>
struct A { int a[10000]; };
A a;
// 值返回
A TestFunc1() { return a; }
// 引用返回
A& TestFunc2() { return a; }
void main()
{
	// 以值作为函数的返回值类型
	size_t begin1 = clock();
	for (size_t i = 0; i < 100000; ++i)
		TestFunc1();
	size_t end1 = clock();
	// 以引用作为函数的返回值类型
	size_t begin2 = clock();
	for (size_t i = 0; i < 100000; ++i)
		TestFunc2();
	size_t end2 = clock();
	// 计算两个函数运算完成之后的时间
	cout << "TestFunc1 time:" << end1 - begin1 << endl;
	cout << "TestFunc2 time:" << end2 - begin2 << endl;
}
  • 这就不用我说了吧~~

在这里插入图片描述

  • 以值作为参数或者返回值类型,在传参和返回期间,函数不会直接传递实参或者将变量本身直接返回,而是传递实参或者返回变量的一份临时的拷贝,因此用值作为参数或者返回值类型,效率是非常低下的,尤其是当参数或者返回值类型非常大时,效率就更低

1.4.2做返回值
  • 首先来看这段代码,这段代码是将a的值返回,然后ret接收,没有什么问题
int func()
{
	int a = 0;
	return a;
}

int main()
{
	int ret = func();
	cout << ret << endl;

	return 0;
}
  • 学了引用后,我们是不是可以这样
int& func()
{
	int a = 0;
	return a;
}

int main()
{
	int ret = func();
	cout << ret << endl;

	return 0;
}
  • 这里是报了一个警告返回局部变量或临时变量的地址,那么这个程序的结果是什么?

在这里插入图片描述

  • 我们以前说指针有野指针,那么引用也有野引用

  • 上面的代码在func函数里是将a的别名返回了,函数调用完会销毁,这里与函数的栈帧的创建与销毁有关

  • 在调用完函数后,那块空间会被销毁,然后再访问被销毁的地址,会造成野引用

  • 栈帧销毁的时候可能被清理,结果可能是随机值,但是在vs上是不清理的

  • 我们可以证明一下

int& func()
{
	int a = 0;
	return a;
}

int& fx()
{
	int b = 1;
	return b;
}

int main()
{
	int& ret = func();
	cout << ret << endl;

	fx();
	cout << ret << endl;

	return 0;
}
  • 在调用完一次后再次调用,栈帧大小是一样的,会复用前面的空间

在这里插入图片描述

结论:返回变量出了函数作用域生命周期就销毁了,不能用引用返回

  • 那什么情况下可以用引用返回呢?

全局变量/静态变量/堆上的变量等就可以用引用返回


那么我这里举例用一个现实中的场景:

  • 首先来看一下
  • 这里已经c和c++混着写了
#include<assert.h>
// 升级成类了,直接就可以使用名字,不用typedef了
struct SeqList
{
	int* a;
	int size;
	int capacity;
};

void SLInit(SeqList& sl)
{
	sl.a = (int*)malloc(sizeof(int) * 4);
	// ..
	sl.size = 0;
	sl.capacity = 4;
}

void SLPushBack(SeqList& sl, int x)
{
	//...扩容
	sl.a[sl.size++] = x;
}

// 修改
void SLModity(SeqList& sl, int pos, int x)
{
	assert(pos >= 0);
	assert(pos < sl.size);

	sl.a[pos] = x;
}

int SLGet(SeqList& sl, int pos)
{
	assert(pos >= 0);
	assert(pos < sl.size);

	return sl.a[pos];
}

int main()
{
	SeqList s;
	SLInit(s);
	// 这里接收的是引用,所以我们就不需要取地址了
	SLPushBack(s, 1);
	SLPushBack(s, 2);
	SLPushBack(s, 3);
	SLPushBack(s, 4);

	for (int i = 0; i < s.size; i++)
	{
		cout << SLGet(s, i) << " ";
	}
	cout << endl;
	
	// 获取每偶数进行*2
	for (int i = 0; i < s.size; i++)
	{
		int val = SLGet(s, i);
		if (val % 2 == 0)
		{
			SLModity(s, i, val * 2);
		}
	}
	cout << endl;
	
	for (int i = 0; i < s.size; i++)
	{
		cout << SLGet(s, i) << " ";
	}
	cout << endl;
	return 0;
}
  • 首先这里C++写有变化
struct SeqList
{
	// 成员变量
	int* a;
	int size;
	int capacity;

	// 成员函数
	void Init()
	{
		a = (int*)malloc(sizeof(int) * 4);
		// ...
		size = 0;
		capacity = 4;
	}

	void PushBack(int x)
	{
		// ... 扩容
		a[size++] = x;
	}
	// 读写返回变量
	// 临时变量具有常性
	// 所以必须返回引用,引用中间没有产生临时变量,是一个别名
	int& Get(int pos)
	{
		assert(pos >= 0);
		assert(pos < size);
	
		return a[pos];
	}
};

int main()
{
	SeqList s;
	s.Init();
	s.PushBack(1);
	s.PushBack(2);
	s.PushBack(3);
	s.PushBack(4);

	for (int i = 0; i < s.size; i++)
	{
		cout << s.Get(i)<< " ";
	}
	cout << endl;

	for (int i = 0; i < s.size; i++)
	{
		if (s.Get(i) % 2 == 0)
		{
			s.Get(i) *= 2;
		}
	}
	cout << endl;

	for (int i = 0; i < s.size; i++)
	{
		cout << s.Get(i) << " ";
	}
	cout << endl;

	return 0;
}
  • 上面的代码C++首先将C语言的结构体升级成了类了,然后函数可以定义到类里面了,在使用的时候获取值就要注意一个点:必须返回引用,之前不是说不能用引用返回码,这里就不一样了,这个malloc出来的空间,是在堆上的,所以可以返回,返回别名后,修改,修改别名也就是修改原来的地址…end

1.5 引用和指针的区别

  • 对比区别的话,要从两个维度来对比,一个是语法,一个是底层,这两个不要混在一起了

语法层面理解:

  • 来看下面的这段代码:
int main()
{
	int a = 10;
	int& ra = a;  // 语法不开空间,
	ra = 20;

	int* pa = &a; // 语法上开空间
	*pa = 20;

	return 0;
}
  • 引用是别名,不开空间,指针是地址,需要开空间
  • 这里我们上面都已经知道了~

底层理解:

  • 我们来看一下汇编代码

在这里插入图片描述

  • 看到在底层是开空间的,引用底层是指针实现的
  • 语法含义和底层实现是背离的

1.6 小结一下

语法层面上:

  1. 引用是别名,不开空间,指针是地址,需要开空间
  2. 引用必须初始化,指针可以初始化也可以不初始化
  3. 引用不能改变指向,指针可以
  4. 引用相对更安全,没有空引用,但是有空指针,容易出现野指针,但是不容易出现野引用
  5. sizeof 、++、解用访问等方面的区别

底层层面上:【汇编】
汇编层面上,没有引用,都是指针,引用编译后也转换成指针了

引用…End

二、内联函数

2.1 内联的概念

以inline修饰的函数叫做内联函数,编译时C++编译器会在调用内联函数的地方展开,没有函数调用建立栈帧的开销,内联函数提升程序运行的效率。

  • 假设我要频繁调用100w次,建立100w个栈帧
int Add(int a, int b)
{
	return a + b;
}
  • c语言如何解决这个问题的?宏函数
#define ADD(a, b) ((a)+(b))
  • 核心点:宏是预处理阶段进行替换

宏的缺点:
1、语法复杂,坑很多,不容易控制
2、不能调试
3、没有类型安全的检查

  • 这个时候C++就引入了一个概念inline,就是把函数的运算逻辑放到里面来,不进行建立栈帧
inline int Add(int a, int b)
{
	return a + b;
}

int main()
{
	int ret1 = Add(1, 2) * 3;

	int x = 1, y = 2;
	int ret2 = Add(x | y, x & y);

	return 0;
}
  • 如果在上述函数前增加inline关键字将其改成内联函数,在编译期间编译器会用函数体替换函数的调用

查看方式:

  1. 在release模式下,查看编译器生成的汇编代码中是否存在call Add
  2. 在debug模式下,需要对编译器进行设置,否则不会展开(因为debug模式下,编译器默认不会对代码进行优化,下面是vs2022的设置)

在这里插入图片描述

在这里插入图片描述

  • 在不加内联的情况下

在这里插入图片描述

  • 添加了内联后,可以看到有很大的特别

在这里插入图片描述

  • 我们再来看一个场景
  • 就是分文件定义的时候,我就想在.h文件中定义一个函数

在这里插入图片描述

  • 然而我在使用的时候两个文件都包含了这个头文件,在编译阶段链接的时候就会报链接错误,冲突的原因就是两个文件都会生成符号表,进行链接,有同名,会冲突因为是同一个函数,这咋办啊~

在这里插入图片描述

我们有三种解决方案:

第一种就是声明和定义分离

  • 这个不多说,基本都会,之前我们用的都是这样的方法

第二种方式就是static修饰函数,链接属性,只在当前文件可见

  • 在C语言阶段详细大家知道滴~~

在这里插入图片描述

第三种方式就是加入内联函数

  • 这里的内联函数相当于也就是static

在这里插入图片描述

  • 这里要注意,内联修饰的函数比较大,他是不会展开的,小函数就会展开,所以小函数使用内联,大函数使用静态

2.2 内联的特性

  1. inline是一种以空间换时间的做法,如果编译器将函数当成内联函数处理,在编译阶段,会用函数体替换函数调用,缺陷:可能会使目标文件变大,优势:少了调用开销,提高程序运行效率。
  2. inline对于编译器而言只是一个建议,不同编译器关于inline实现机制可能不同,一般建议:将函数规模较小(即函数不是很长,具体没有准确的说法,取决于编译器内部实现)、不是递归、且频繁调用的函数采用inline修饰,否则编译器会忽略inline特性。下图为《C++prime》第五版关于inline的建议:
    在这里插入图片描述
  3. inline 不建议声明和定义分离,分离会导致链接错误。因为inline被展开,就没有函数地址了,链接就会找不到。

2.3 【面试题】

宏的优缺点?

优点:

  1. 增强代码的复用性。
  2. 提高性能。

缺点:

  1. 不方便调试宏。(因为预编译阶段进行了替换)
  2. 导致代码可读性差,可维护性差,容易误用。
  3. 没有类型安全的检查 。

C++有哪些技术替代宏?

  1. 常量定义 换用const enum
  2. 短小函数定义 换用内联函数

三、auto关键字(C++11)

3.1 类型别名思考

  • 随着程序越来越复杂,程序中用到的类型也越来越复杂,经常体现在:
  1. 类型难于拼写
  2. 含义不明确导致容易出错
int main()
{
	std::map<std::string, std::string> m{ { "apple", "苹果" }, { "orange","橙子" },{"pear","梨"} };
	std::map<std::string, std::string>::iterator it = m.begin();
	while (it != m.end())
	{
		//....
	}
	return 0;
}
  • 大家来上面的一个代码,有可能看不懂,但是我们只需要知道std::map<std::string, std::string>::iterator是一个类型但是该类型太长了,特别容易写错。聪明的同学可能已经想到:可以通过typedef给类型取别名,比如:
typedef std::map<std::string, std::string> Map;

使用typedef给类型取别名确实可以简化代码,但是typedef有会遇到新的难题:

typedef char* pstring;

int main()
{
	const pstring p1; 
	const pstring* p2; 
	return 0;
}

在这里插入图片描述

在编程时,常常需要把表达式的值赋值给变量,这就要求在声明变量的时候清楚地知道表达式的
类型。然而有时候要做到这点并非那么容易,因此C++11给auto赋予了新的含义。

  • 这个时候auto就有作用了

3.2 auto简介

  • 在早期C/C++中auto的含义是:使用auto修饰的变量,是具有自动存储器的局部变量,但遗憾的是一直没有人去使用它,大家可思考下为什么?

  • C++11中,标准委员会赋予了auto全新的含义即:auto不再是一个存储类型指示符,而是作为一个新的类型指示符来指示编译器,auto声明的变量必须由编译器在编译时期推导而得。

如下所示:

int TestAuto()
{
	return 10;
}
int main()
{
	int a = 10;
	auto b = a; // 整形
	auto c = 'a'; // 字符类型
	auto f = &a; //指针类型
	auto d = TestAuto(); // 函数指针类型
	// auto e;  error 必须要对其进行初始化

	cout << typeid(b).name() << endl;
	cout << typeid(c).name() << endl;
	cout << typeid(d).name() << endl;
	cout << typeid(f).name() << endl;
	return 0;
}
  • 这里的typeid(函数名).name(),就是打印类型

在这里插入图片描述

  • 就刚刚上面的代码就可以这样写了
#include <string>
#include <map>
int main()
{
	std::map<std::string, std::string> m{ { "apple", "苹果" }, { "orange","橙子" },{"pear","梨"} };
	auto it = m.begin();
	while (it != m.end())
	{
		//....
	}
	return 0;
}
  • 类型就可以写成auto自动识别了~~

在这里插入图片描述

【注意】

使用auto定义变量时必须对其进行初始化,在编译阶段编译器需要根据初始化表达式来推导auto的实际类型。因此auto并非是一种“类型”的声明,而是一个类型声明时的“占位符”,编译器在编译期会将auto替换为变量实际的类型。

四、auto的使用细则

  1. auto与指针和引用结合起来使用

用auto声明指针类型时,用autoauto*没有任何区别,但用 auto 声明引用类型时则必须加 &

int main()
{
	int x = 10;
	auto a = &x;
	auto* b = &x;
	auto& c = x;

	cout << typeid(a).name() << endl;
	cout << typeid(b).name() << endl;
	cout << typeid(c).name() << endl;
	
	*a = 20;
	*b = 30;
	c = 40;
	return 0;
}

在这里插入图片描述

在这里插入图片描述

  1. 在同一行定义多个变量

当在同一行声明多个变量时,这些变量必须是相同的类型,否则编译器将会报错,因为编译器实际只对第一个类型进行推导,然后用推导出来的类型定义其他变量。

void TestAuto()
{
	auto a = 1, b = 2;
	auto c = 3, d = 4.0; 
}

在这里插入图片描述

  1. auto不能推导的场景
  • auto不能作为函数的参数
// 此处代码编译失败,auto不能作为形参类型,因为编译器无法对a的实际类型进行推导
void TestAuto(auto a)
{}
  • auto不能直接用来声明数组
void TestAuto()
{
	int a[] = {1,2,3};
	auto b[] = {456};
}
  • 为了避免与C++98中的auto发生混淆,C++11只保留了auto作为类型指示符的用法
  • auto在实际中最常见的优势用法就是跟以后会讲到的C++11提供的新式for循环,还有
    lambda表达式等进行配合使用。

4.1 基于范围的for循环(C++11)

范围for的语法

  • 在C++98中如果要遍历一个数组,可以按照以下方式进行:
void TestFor()
{
	int array[] = { 1, 2, 3, 4, 5 };
	
	for (int i = 0; i < sizeof(array) / sizeof(array[0]); ++i)
		array[i] *= 2;
		
	for (int* p = array; p < array + sizeof(array)/ sizeof(array[0]); ++p)
		cout << *p << endl;
}
  • 对于一个有范围的集合而言,由程序员来说明循环的范围是多余的,有时候还会容易犯错误。因此C++11中引入了基于范围的for循环。for循环后的括号由冒号“ :”分为两部分:第一部分是范围内用于迭代的变量,第二部分则表示被迭代的范围。
void TestFor()
{
	int array[] = { 1, 2, 3, 4, 5 };
	for (auto& e : array)
		e *= 2;
	for (auto e : array)
		cout << e << " ";
}
  • 注意:与普通循环类似,可以用continue来结束本次循环,也可以用break来跳出整个循环。9.2 范围for的使用条件

4.2 范围for的使用条件

  • for循环迭代的范围必须是确定的

  • 对于数组而言,就是数组中第一个元素和最后一个元素的范围;对于类而言,应该提供begin和end的方法,begin和end就是for循环迭代的范围。

注意:以下代码就有问题,因为for的范围不确定

void TestFor(int array[])
{
	for(auto& e : array)
	cout<< e <<endl;
}
  • 迭代的对象要实现++和==的操作。(关于迭代器这个问题,以后会讲,现在提一下,没办法讲清楚,现在大家了解一下就可以了)

五、指针空值nullptr(C++11)

  • 在良好的C/C++编程习惯中,声明一个变量时最好给该变量一个合适的初始值,否则可能会出现不可预料的错误,比如未初始化的指针。如果一个指针没有合法的指向,我们基本都是按照如下方式对其进行初始化:
void TestPtr()
{
	int* p1 = NULL;
	int* p2 = 0;
	// ……
}
  • NULL实际是一个宏,在传统的C头文件(stddef.h)中,可以看到如下代码:
#ifndef NULL
#ifdef __cplusplus
#define NULL 0
#else
#define NULL ((void *)0)
#endif
#endif
  • 或者我们右键转到定义就可以看到

在这里插入图片描述

在这里插入图片描述

  • 可以看到,NULL可能被定义为字面常量0,或者被定义为无类型指针(void*)的常量。不论采取何种定义,在使用空值的指针时,都不可避免的会遇到一些麻烦,比如:
void f(int)
{
	cout << "f(int)" << endl;
}
void f(int*)
{
	cout << "f(int*)" << endl;
}
int main()
{
	f(0);
	f(NULL);
	f((int*)NULL);
	return 0;
}

在这里插入图片描述

  • 程序本意是想通过f(NULL)调用指针版本的f(int*)函数,但是由于NULL被定义成0,因此与程序的初衷相悖。

  • 在C++98中,字面常量0既可以是一个整形数字,也可以是无类型的指针(void*)常量,但是编译器默认情况下将其看成是一个整形常量,如果要将其按照指针方式来使用,必须对其进行强转(void*)0。

注意:

  1. 在使用nullptr表示指针空值时,不需要包含头文件,因为nullptr是C++11作为新关键字引入的。
  2. 在C++11中,sizeof(nullptr) 与 sizeof((void*)0)所占的字节数相同。
  3. 为了提高代码的健壮性,在后续表示指针空值时建议最好使用nullptr。

本文章重点介绍了引用,内联函数的注意事项以及使用,提了一下auto关键字和空指针问题,能看完的烙铁相信已经学会了,最后请多多指教,如有疑问请在评论区或私信交流~~

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

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

相关文章

Aloudata 近期荣誉盘点!接连斩获技术创新、案例实践、投资价值等权威认可

近期&#xff0c;Aloudata 凭借持续的技术积累、丰富的产品与解决方案以及多样场景下的最佳实践案例&#xff0c;在数据智能技术创新、案例实践、投资价值等领域全面开花&#xff0c;接连荣获&#xff1a; 2023 金猿榜「大数据产业年度最具投资价值」企业&#xff0c;并携手首…

APP开发者对接穿山甲广告联盟,有哪些特点?收益如何?

穿山甲平台作为巨量引擎旗下的第三方广告变现平台&#xff0c;在行业内始终处于领先地位&#xff0c;是不少开发者首选的对接平台。 通过穿山甲广告GroMore的Bidding竞价能力&#xff0c;不断提升自身的变现效率&#xff0c;新手可直接上手&#xff0c;避免繁琐调优流程&#…

PVE报错处理:kvm [2205]: vcpu0 ignored RDMSR: 0x1b8

PVE使用过程中如果遇到&#xff1a;kvm [2205]: vcpu0 ignored RDMSR: 0x1b8 报错信息处理方法 vim /etc/modprobe.d/kvm.conf "options kvm ignore_msrsY"&#xff0c;这里在msrsY后面加一个空格&#xff0c;然后粘贴report_ignored_msrsN&#xff0c;使其变成 op…

如何在微信搭建私域流量池?

A: ①给客户打标签 添加标签&#xff0c;多维度构建用户画像&#xff0c;精准发送消息。 ②群发信息 选择自定义时间&#xff0c;上传内容 (支持文字&#xff0c;图片) &#xff0c;一键群发 。 ③建立专属素材库 将常用的话术、图片与文件录入至素材库&#xff0c;员工可随…

微信小程序(基本操作)

概念&#xff1a; 小程序&#xff1a;就是小程序&#xff0c;mini program。现在市面上有微信小程序&#xff0c;百度智能小程序等等。 微信小程序&#xff0c;简称小程序&#xff0c;英文名Mini Program&#xff0c;是一种不需要下载安装即可使用的应用&#xff0c;它实现了…

【算法】不懂数学原理也能看得懂的KMP算法

一.KMP算法的作用 举个例子&#xff0c;excel表格大家都用过吧&#xff0c;在表格内按下“CtrlF”可以弹出“查找和替换”功能&#xff0c;输入我们想要查找的关键字&#xff0c;系统就会帮我们定位到具体的位置&#xff0c;没有找到就上报具体的错误信息&#xff0c;KMP算法的…

解释性人工智能(XAI)

引言 解释性人工智能&#xff08;XAI&#xff09;是指一类旨在使人能够理解和解释机器学习模型的方法和技术。XAI的目标是提高AI系统的透明度和可理解性&#xff0c;让人们能够理解机器学习模型的决策过程、推理方式和结果。这对于社会应用和用户信任非常重要&#xff0c;因为A…

Axure 怎么用?一篇文章告诉你

Axure RP 9 该软件是一个非常实用的原型设计工具&#xff0c;了解 Axure、学会使用 Axure&#xff0c;作为产品经理&#xff0c;UI、界面规划等岗位的基本技能。特别是对于产品经理来说&#xff0c;画出优秀的原型可以更好地表达产品需求&#xff0c;提高沟通效率。如何快速入门…

Leetcode第382场周赛

Leetcode第382场周赛 本人水平有限&#xff0c;只做前三道。 一、按键变更的次数 给你一个下标从 0 开始的字符串 s &#xff0c;该字符串由用户输入。按键变更的定义是&#xff1a;使用与上次使用的按键不同的键。例如 s “ab” 表示按键变更一次&#xff0c;而 s “bBBb”…

网络工程师学习笔记——HDLCPPP

继续学习计算机网络技术——HDLC&PPP 一、HDLC HDLC&#xff08; High-Level Data Link Control &#xff09;&#xff1a;高级数据链路控制 HDLC是一种面向比特的链路层协议。 HDLC的作用&#xff1a;接口地址借用&#xff0c;节省IP地址&#xff0c;使地址更加稳定 …

Javaweb实现的学生宿舍管理系统

Javaweb实现的学生宿舍管理系统 文章目录 Javaweb实现的学生宿舍管理系统系统介绍技术选型成果展示源码获取账号地址及其他说明 系统介绍 Javaweb实现的学生宿舍管理系统采用jspservlet技术实现了如下功能模块&#xff0c;分别是宿舍管理员管理、学生管理、宿舍楼管理、缺勤记…

SwiftUI 动画入门之一:路径动画(Path Animations)

概览 在 SwiftUI 的开发中,我们往往需要使用千姿百态的动画把我们的界面元素妆点的更加鲜活灵动。 如上图所示,我们使用路径动画使折线图更加生动了!这是怎么做到的呢? 在本篇博文中,您将学到以下内容: 概览1. 路径与形状(Path and Shape)2. 路径动画的原理3. 让路径…

LeetCode刷题:使用栈解决150. 逆波兰表达式求值

给你一个字符串数组 tokens &#xff0c;表示一个根据 逆波兰表示法 表示的算术表达式。 请你计算该表达式。返回一个表示表达式值的整数。 注意&#xff1a; 有效的算符为 、-、* 和 / 。每个操作数&#xff08;运算对象&#xff09;都可以是一个整数或者另一个表达式。两个…

程序的内存模型

师从黑马程序员 内存分区模型 内存大方向划分为4个区域 1、代码区&#xff1a;存放函数体的二进制代码&#xff0c;由操作系统管理的 2、全局区&#xff1a;存放全局变量环和静态变量以及常量 3、栈区&#xff1a;由编译器自动分配释放&#xff0c;存放函数的参数值&#…

【笔试常见易错选择题01】else、表达式、二维数组、%m.ns、%m.nf、常量指针和指针常量、宏定义、传参、数组越界、位段

1. 下列main()函数执行后的结果为&#xff08;&#xff09; int func(){ int i, j, k 0; for(i 0, j -1;j 0;i, j){ k; } return k; } int main(){cout << (func());return 0; }A. -1 B. 0 C. 1 D. 2 判断为赋值语句&#xff0c;j等于0 0为假不进循环 选B. 2. 下面程…

市场复盘总结 20240201

仅用于记录当天的市场情况&#xff0c;用于统计交易策略的适用情况&#xff0c;以便程序回测 短线核心&#xff1a;不参与任何级别的调整&#xff0c;采用龙空龙模式 一支股票 10%的时候可以操作&#xff0c; 90%的时间适合空仓等待 昨日主题投资 连板进级率 6/27 22.2% 二进…

LeetCode刷题:100.相同的树

题目&#xff1a; 解题思路&#xff1a;1.首先要判断根节点的情况&#xff0c;&#xff08;1&#xff09;首先是p为空&#xff0c;q不为空&#xff0c;或者p不为空&#xff0c;q为空则要返回false&#xff08;2&#xff09;p和q都为空则返回true&#xff0c;2.然后判断根节点的…

五、CPU针脚

电脑的两大厂商是Intel、AMD&#xff0c;区分他们两家的CPU&#xff0c;最简单的就是看CPU的针脚。 从形状上也大致能够看出来&#xff0c;AMD的logo占满了CPU的框框&#xff0c;Intel的是没有占满的。 CPU针脚针脚是什么呢&#xff1f; 如下&#xff0c;这一个个和小针一样的东…

【数据结构】 归并排序超详解

1.基本思想 归并排序&#xff08;MERGE-SORT&#xff09;是建立在归并操作上的一种有效的排序算法,该算法是采用分治法&#xff08;Divide andConquer&#xff09;的一个非常典型的应用。 将已有序的子序列合并&#xff0c;得到完全有序的序列&#xff0c;即先使每个子序列有序…

【操作系统·考研】文件系统

1.概述 文件系统(File System)提供高效和便捷的磁盘访问&#xff0c;以便允许存储、定位、提取数据。 严格来说&#xff0c;VFS并不是一种实际的FS&#xff0c;它只存在于内存中&#xff0c;不存在与任何外存空间中。 VFS在系统启动时建立&#xff0c;在系统关闭时消亡。 2.结…