【玩转c++】c++ :string类讲解(万字详解)

news2025/1/11 18:35:14

目录

🍁1. 为什么要学习string类

🍁2. 标准库中的string类

🍁3. string类各种接口

默认成员函数

Iterators迭代器  

capacity容量

Element access:元素访问

Modifiers:修改

字符串操作

 成员变量

 非成员函数

🍁4. 扩展阅读

  •  本期主题:c++string类的介绍
  • 博客主页:小峰同学
  • 分享小编的在Linux中学习到的知识和遇到的问题
  • 小编的能力有限,出现错误希望大家不吝赐
  • 做为程序员不会有人还没女朋友吧?

 


  • 🍁1. 为什么要学习string类

  C语言中的字符串  

C语言中,字符串是以'\0'结尾的一些字符的集合,为了操作方便,C标准库中提供了一些str系列的库函数, 但是这些库函数与字符串是分离开的,不太符合OOP的思想,而且底层空间需要用户自己管理,稍不留神可 能还会越界访问。

  两个 oj 题  

ll字符串转整形数字

字符串相加

在OJ中,有关字符串的题目基本以string类的形式出现,而且在常规工作中,为了简单、方便、快捷,基本都使用string类,很少有人去使用C库中的字符串操作函数。


  • 🍁2. 标准库中的string类

字符串是表示字符序列的类
string类是使用char(即作为它的字符类型,使用它的默认char_traits和分配器类型(关于模板的更多信息,请参阅basic_string)

string类是basic_string模板类的一个实例,它使用char来实例化basic_string模板类,并用char_traits 和allocator作为basic_string的默认参数(关于更多的模板信息请参考basic_string)。

注意,这个类独立于所使用的编码来处理字节:如果用来处理多字节或变长字符(如UTF-8)的序列,这个类的所有成员(如长度或大小)以及它的迭代器,将仍然按照字节(而不是实际编码的字符)来操作。还有 一些别的编码类型的字符串类,比如支持utf-16的u16string,支持utf-32的u32string。

总结:

  • 1. string是表示字符串的字符串类
  • 2. 该类的接口与常规容器的接口基本相同,再添加了一些专门用来操作string的常规操作。 比特就业课
  • 3. string在底层实际是:basic_string模板类的别名,typedef basic_string<char> string;
  • 这里的是typedef basic_string<char> string ,而不是 typedef basic_string string,因为basic_string 是模板名不是类型,basic_string<char>才是类型。
  • 4. 不能操作多字节或者变长字符的序列。
  • 多字节的需要用到 u16string,或者 u32string ,或者 wstring 类型。
  • 5.在使用string类时,必须包含#include<string>头文件 以及using namespace std;

  • 🍁3. string类各种接口

由于我们主要用的是utf-8 编码的字符串,所以本文章,主要讲解string类。

默认成员函数

   (cinstructor)构造    

int main()
{
	//string();
	string s;
	cout << s<<endl;//空字符
	


	//string(const string & str);
	string s1("zhang");
	cout << s1 << endl;//zhang
	const char* p = "zhang";
	string s2(p);
	cout << s2 << endl;//zhang
	char arr[] = "zhang";
	string s3(arr);
	cout << s3 << endl;//zhang

	//拷贝构造
	//string (const string& str, size_t pos, size_t len = npos);
	//npos 是 -1 size_t类型的一个很大的数。
	string s4(s2, 2, 2);//从下标为2的位置向后两个字符。
	cout << s4 << endl;//an
	string s5(s2,2);//从下标为2的位置开始知道最后,npos是一个很大的数。
	cout << s5 << endl;//ang


	//string (const char* s);
	//这里没有加 explicit 支持单参数或者第一个参数无默认值得构造其余均有默认值得构造函数,还具有类型转换的作用。
	//string s6("zhang");//一样的
	string s6 = "zhang";//这里会先那"zhang"去拷贝构造一个零时对象,然后拷贝构造给s6,(vs这里会优化,直接拿"zhang"去构造S6.
	cout<< s6 << endl;//zhang

	//string(const char* s, size_t n);
	string s7(p, 2);//取出p字符串的前两个。
	cout << s7 << endl;//zh


	//string(size_t n, char c);
	string s8(5, 'z');//5个'z'组成的字符串。
	cout << s8 << endl;//zzzzz	
	


	//template <class InputIterator>
	//string(InputIterator first, InputIterator last);
	string::iterator first = s1.begin();//s1 = "zhang"
	string::iterator last  = s1.end();
	string s9(first,last);
	cout << s9 << endl;//zhang


	return 0;
}

    destructor销毁   

     operator=    赋值重载

int main()
{
	string s1("zhang");
	string s2;
	s2 = s1;
	cout << s2 << endl;

	string s3;
	s3 = "zhang";
	cout << s3 << endl;

	string s4;
	s4 = 'z';
	cout << s4 << endl;


	return 0;
}

   三种遍历方式: 

  • 1.下标
  • 2.范围for
  • 3.迭代器


int main()
{
	string s1("123456");

	//下标
	for (int i = 0; i < s1.size(); i++)
	{
		s1[i]++;
		//本质生是调用 operator[];
	}
	cout << s1 << endl;//234567

	//范围for
	for (auto& i : s1)
	{
		i++;
	}
	cout << s1 << endl;//345678

	//迭代器 
	string::iterator first = s1.begin();
	while (first != s1.end())
	{
		(*first)++;
		first++;
	}
	cout << s1 << endl;//456789

	return 0;
}

Iterators迭代器  

  •  begin迭代器开端,重载了两个分别返回,iterator 和 const_iterator (参数是常量的,返回常量迭代器)

  • end()迭代器的结束,重载了两个分别返回,iterator 和 const_iterator (参数是常量的,返回常量迭代器)

int main()
{
	string s1("zhang");
	string::iterator first = s1.begin();
	string::iterator last = s1.end();
	while (first != last){
		cout << (*first);//这里可读可写
		first++;
	}	cout << endl;

	const string s2("zhang");//s2 是const 类型的
	//string::iterator rfirst = s2.begin();//这个会报错
	//string::iterator rlast = s2.end();//权限提升报错
	string::const_iterator cfirst = s2.begin();
	string::const_iterator clast = s2.end();
	while (cfirst != clast){
		cout << (*cfirst);//这里只是可读
		cfirst++;
	}	cout << endl;

	return 0;
}
  •  rbegin迭代器开端,重载了两个分别返回,reverse_iterator 和 const_reverse_iterator (参数是常量的,返回常量迭代器)

  • rend()迭代器的结束,重载了两个分别返回,reverse_iterator 和 const_reverse_iterator (参数是常量的,返回常量迭代器)


int main()
{
	string s2("123456");

	string::reverse_iterator rfirst = s2.rbegin();
	string::reverse_iterator rlast = s2.rend();
	while (rfirst != rlast){
		cout << (*rfirst);//这里可读可写
		rfirst++;//注意这里是++ ,不是--
	}//654321
	cout << endl;

	const string s3(s2);
	string::const_reverse_iterator  crfirst = s3.rbegin();
	string::const_reverse_iterator  crlast = s3.rend();
	while (crfirst != crlast) {
		cout << (*crfirst);//这里可读不可写
		//(*crfirst)++;不可写
		crfirst++;
	}//654321

	return 0;
}

capacity容量

int main()
{
	string s1("123456");
	//size()有效字符的长度
	cout << s1.size() << endl;
	//length()长度
	cout << s1.length() << endl;
	//max_size()最长长度(无意义)任何string类型都是一样的:4294967294
	cout << s1.max_size() << endl;
	//capacity() //容量空间的大小
	cout << s1.capacity() << endl;	
	//clear//清空数据,到底清理空间与否,不知道(VS没有清理空间)
	s1.clear();
	cout << s1.capacity() << endl;
	cout << s1.size() << endl;
	s1.reserve(200);
	//shrink_to_fit() //减小容量,让字符串减小其容量以适合其大小,可能不是和size相同
	s1.shrink_to_fit();
	cout << s1.capacity() << endl;
	cout << s1.size() << endl;

	string s("123456");
	string s2(s);
	//reserve()//提前开辟n个空间。
	//提前看开辟空间,减小扩容次数,提高效率。
	s2.reserve(100);//不会改变size
	cout << s2.capacity() << endl;
	cout << s2.size() << endl;


	string s3(s);
	string s4(s);
	string s5(s);
	//resize() //把string的长度改变到n
	//n<size, 删除数据到n个长度。
	//size<n<capacity, 插入数据,没有给定值默认是'\0'。
	//n>capacity,插入数据顺便扩容,没有给定值默认是'\0'。
	cout << s.capacity() << endl;//15
	cout << s.size() << endl;//6
	s3.resize(4);
	cout << s3.size() << endl;
	cout << s3.capacity() << endl;
	cout << s3 << endl;
	s4.resize(10,'x');
	cout << s4.size() << endl;
	cout << s4.capacity() << endl;
	cout << s4 << endl;
	s5.resize(20,'x');
	cout << s5.size() << endl;
	cout << s5.capacity() << endl;
	cout << s5 << endl;


	return 0;
}

注意:

  • 1. size()与length()方法底层实现原理完全相同,引入size()的原因是为了与其他容器的接口保持一 致,一般情况下基本都是用size()。
  • 2. clear()只是将string中有效字符清空,一般不改变底层空间大小。
  • 3. resize(size_t n) 与 resize(size_t n, char c)都是将字符串中有效字符个数改变到n个,不同的是当字 符个数增多时:resize(n)用0来填充多出的元素空间,resize(size_t n, char c)用字符c来填充多出的 元素空间。注意:resize在改变元素个数时,如果是将元素个数增多,可能会改变底层容量的大 小,如果是将元素个数减少,底层空间总大小不变。
  • 4. reserve(size_t res_arg=0):为string预留空间,不改变有效元素个数,当reserve的参数小于 string的底层空间总大小时,reserver不会改变容量大小

Element access:元素访问

Modifiers:修改

int main()
{
	string s1 = "zhang";
	string s2 = "xue";
	s1 += s2;
	cout << s1 << endl;//zhangxue
	const char* pc = "feng";
	s1 += pc;
	cout << s1 << endl;//zhangxuefeng
	char p = 'z';
	s1 += p;
	cout << s1 << endl;//zhangxuefengz
	return 0;
}

int main()
{
	string s1 = "zhang";
	string s2 = "xue";
	s1.append(s2);
	cout << s1 << endl;//zhangxue
	string s3 = "feng";
	s1.append(s3, 0,2);
	cout << s1 << endl;//zhangxuefe
	char p1[] = "ng";
	s1.append(p1);
	cout << s1 << endl;//zhangxuefeng
	char p2[] = "gao";
	s1.append(p2,2);
	cout << s1 << endl;//zhangxuefengga
	s1.append(2, 'z');
	cout << s1 << endl;//zhangxuefenggazz
	return 0;
}

int main()
{
	string s1;
	s1.push_back('z');
	s1.push_back('h');
	s1.push_back('a');
	s1.push_back('n');
	s1.push_back('g');
	cout << s1 << endl;//zhang

	return 0;
}

#include<iostream>
#include<string>
using namespace std;

int main()
{
	string s1("gaowenya");
	string s2 = "zhangxuefeng";
	s1.assign(s2);
	cout << s1 << endl;//zhangxuefeng

	//s1.clear();
	s1.assign("zhangxuefeng");
	cout << s1 << endl;//zhangxuefeng


	s1.clear();//这里是否手动清理与否都一样,assign会自己清理。
	s1.assign(s2,5,3);
	cout << s1 << endl;//xue

	s1.clear();
	s1.assign("zhangxuefeng", 5);
	cout << s1 << endl;//zhang

	s1.clear();
	s1.assign(5, '5');
	cout << s1 << endl;//55555

	string::const_iterator first = s2.begin();
	string::const_iterator back = s2.end();
	first++;
	back--;
	s1.assign(first, back);
	cout << s1 << endl;//hangxuefen

	return 0;
}

#include<iostream>
#include<string>
using namespace std;


int main()
{
	string s1;
	s1 = "0123456789";
	const char* s2 = "zhangxuefeng";
	string s3 = "gaowenya";

	s1.insert(5, s3);
	cout << s1<< endl;//01234gaowenya56789

	s1 = "0123456789";
	s1.insert(5, s3,3,3 );
	cout << s1 << endl;//01234wen56789

	s1 = "0123456789";
	s1.insert(5, s2,5);
	cout << s1 << endl;//01234zhang56789

	s1 = "0123456789";
	s1.insert(5, 5, 'z');
	cout << s1 << endl;//01234zzzzz56789



	string s5 = "0123456789";
	//string::const_iterator p = s5.begin();
	string::iterator p = s5.begin();

	s5.insert(p+5, 5, 'z');
	cout << s5 << endl;//01234zzzzz56789

	s5.insert(p+1, 'z');
	cout << s5 << endl;//0z1234zzzzz56789


	string s4 = "zhangxuefeng";
	string::iterator first = s4.begin();
	string::iterator last = s4.end();
	
	string s6 = "0123456789";
	string::iterator p2 = s6.begin();
	s6.insert(p2 + 5, first, last);
	cout << s6 << endl;//01234zhangxuefeng56789


	return 0;
}

#include<iostream>
#include<string>
using namespace std;

int main()
{
	string s1 = "zhangxuefeng";
	s1.erase(5, 3);
	cout << s1 << endl;//zhangfeng

	string s2 = "zhangxuefeng";
	s2.erase(5);
	cout << s2 << endl;//zhang


	string s3 = "zhangxuefeng";
	string::iterator first = s3.begin();
	s3.erase(first + 5);
	cout << s3 << endl;//zhanguefeng

	string s4 = "zhangxuefeng";
	string::iterator first4 = s4.begin();
	s4.erase(first4 + 5, first4 + 8);
	cout << s4 << endl;//zhangfeng



	return 0;
}

int main()
{
	string s2 = "zhangxuefeng";
	s2.replace(5, 3, "zhang");//xue 用 zhang  替换掉
	cout << s2 << endl;

	return 0;
}

 


int main()
{
	string s1("zhang");
	string s2("gao");
	cout << "交换前" << endl;
	cout << s1 << endl;
	cout << s2 << endl;
	s1.swap(s2);
	cout << "交换后" << endl;
	cout << s1 << endl;
	cout << s2 << endl;
	return 0;
}

注意:

  • 1. 在string尾部追加字符时,s.push_back(c) / s.append(1, c) / s += 'c'三种的实现方式差不多,一般 情况下string类的+=操作用的比较多,+=操作不仅可以连接单个字符,还可以连接字符串。
  • 2. 对string操作时,如果能够大概预估到放多少字符,可以先通过reserve把空间预留好。
  • 3.replace和assign区别:assign是先清理再替换,replace是直接替换不会清理。

字符串操作

#include<iostream>
#include<string>
#include<string.h>
using namespace std;

int main()
{
	string str("Please split this sentence into tokens");

	char* cstr = new char[str.length() + 1];
	//strcpy(cstr, str.c_str());
	std::strcpy(cstr, str.data());

	// cstr now contains a c-string copy of str

	char* p = strtok(cstr, " ");
	while (p != 0)
	{
		cout << p << endl;
		p = strtok(NULL, " ");
	}
	//Please
	//split
	//this
	//sentence
	//into
	//tokens

	delete[] cstr;
	return 0;
}

#include<iostream>
#include<string>
using namespace std;
int main()
{
	char buffer[20];
	string s1("zhang xue feng");
	size_t len = s1.copy(buffer, 3, 6);//注意3  6  的意义。
	buffer[len] = '\0';
	//注意要手动加上'\0',copy不会自动加'\0'。
	cout << buffer << endl;//xue
	return 0;
}

#include<iostream>
#include<string>
using namespace std;

int main()
{
	string str("There are two needles in this haystack with needles.");
	string str2("needle");

	// different member versions of find in the same order as above:
	size_t found = str.find(str2);
	if (found != string::npos)
		cout << "first 'needle' found at: " << found << '\n';

	found = str.find("needles are small", found + 1, 6);//这里不给出6就默认屁匹配别的函数重载
	if (found != string::npos)
		cout << "second 'needle' found at: " << found << '\n';

	found = str.find("haystack");
	if (found != string::npos)
		cout << "'haystack' also found at: " << found << '\n';

	found = str.find('.');
	if (found != string::npos)
		cout << "Period found at: " << found << '\n';

	// let's replace the first needle:
	str.replace(str.find(str2), str2.length(), "preposition");//把s2 对象内容换成preposition。
	cout << str << '\n';

	return 0;
}

#include<iostream>
#include<string>
using namespace std;
int main()
{
	string s1("test.cpp.tar");
	size_t len1 = s1.find('.');
	size_t len2 = s1.rfind('.');
	cout << "find: " << s1.substr(len1) << endl;
	cout << "rfind: " << s1.substr(len2) << endl;
	return 0;
}

#include<iostream>
#include<string>
using namespace std;

int main()
{
	string str("Please, replace the vowels in this sentence by asterisks.");
	size_t found = str.find_first_of("aeiou");
	int i = 5;
	while (i--)
	{
		str[found] = '*';
		found = str.find_first_of("aeiou", found + 1);
	}
	cout << str << '\n';
	//Pl**s*, r*pl*c* the vowels in this sentence by asterisks.
	return 0;
}
#include<iostream>
#include<string>
using namespace std;
int main()
{
	string str("Please, replace the vowels in this sentence by asterisks.");
	size_t found = str.find_last_of("aeiou");
	int i = 5;
	while (i--)
	{
		str[found] = '*';
		found = str.find_last_of("aeiou", found + 1);
	}
	cout << str << '\n';
	//Please, replace the vowels in this sent*nc* by *st*r*sks.

	return 0;
}

 成员变量

 非成员函数

#include<iostream>
#include<string>
using namespace std;
int main()
{
	string s1("zhang");
	const char* s2 = "xue";
	string s3("feng");
	string s4 = s1 + s2 + s3;
	cout << s4 << endl;
	//zhangxuefeng
	return 0;
}

#include<iostream>
#include<string>
using namespace std;
int main()
{
	string s1("zhang");
	string s2("gao");
	s1.swap(s2);//这个是内置函数
	swap(s1, s2);//这个是非成员函数,和全局内置的swap构成函数重载
	return 0;
}

int main()
{
	string str1;
	getline(cin, str1);
	//这样写遇见空格不会结束读取,只有遇到回车才会结束读取。
	cout << str1 << endl;

	string str2;
	getline(cin, str2,'.');//只有遇到  '.'  才会结束读取。
    //此时'\n'只是一个普通的字符串,用于刷新缓冲区。
	cout << str2 << endl;

	string str3;
	cin >> str3;//遇见空格和回车会结束读取。
	cout << str3 << endl;

	return 0;
}


🍁4. 扩展阅读

vs和Linuxg++下string类的大小。

注意:下述结构是在32位平台下进行验证,32位平台下指针占4个字节。

  • vs下string的结构 string总共占28个字节,内部结构稍微复杂一点,先是有一个联合体,
  • 联合体用来定义string中字 符串的存储空间:
  • 当字符串长度小于16时,使用内部固定的字符数组来存放
  • 当字符串长度大于等于16时,从堆上开辟空间
  • 这种设计也是有一定道理的,大多数情况下字符串的长度都小于16,那string对象创建好之后,内部已经有了16个字符数组的固定空间,不需要通过堆创建,效率高。
  • 其次:还有一个size_t字段保存字符串长度,一个size_t字段保存从堆上开辟空间总的容量
  • 最后:还有一个指针做一些其他事情。 故总共占16+4+4+4=28个字节。

 注意:下述结构是在64位平台下进行验证,64位平台下指针占8个字节

  • G++下,string是通过写时拷贝实现的,string对象总共占8个字节,
  • 内部只包含了一个指针,该指 针将来指向一块堆空间,
  • 内部包含了如下字段: 空间总大小 字符串有效长度 引用计数

 扩展好文

C++面试中STRING类的一种正确写法

stl中string怎么了?

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

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

相关文章

stm32f407VET6 系统学习 day07 通用定时器, OLED 屏幕使用 PWM 的使用

1. 通用定时器的知识 1.STM32共有14个定时器&#xff0c;其中12个16位定时器&#xff0c;2个32 位定时器 2. 通用定时器特点 1. 16/32位向上、向下、向上/向下(中心对齐)计数模式&#xff0c;自动装载计数器&#xff08;TIMXCNT) 。 2. 16位可编程预分频器(TIMx_PSC)&…

-bash: lsof: command not found解决办法

简言 centos系统&#xff0c;检测端口时使用lsof命令发现lsof功能未开启&#xff0c;如下图 [rootiZwz9501p9hnysn92hpx27Z tnt_game]# lsof -bash: lsof: command not found 安装lsof centos系统下可以直接使用yum安装lsof功能&#xff0c;如下图 yum可自动完成安装lsof ls…

gitlab-ci.yml关键字(一)image、variables、include

image 这是一个全局关键字&#xff0c;如果流水线的执行器是使用docker来运行的话&#xff0c;那可以指定docker中的docker镜像。如果执行器是shell的话&#xff0c;那该关键字是无用的&#xff0c;即便机器中已近安装了docker的环境&#xff0c;该关键字可以在全局或者某一个…

NeurIPS2021 | ViTAE+: vision transformer中的归纳偏置探索

参考资料&#xff1a;NeurIPS 2021 | ViTAE: vision transformer中的归纳偏置探索 - 知乎 paper地址&#xff1a;https://openreview.net/pdf?id_RnHyIeu5Y5 论文标题&#xff1a;ViTAE: Vision Transformer Advanced by Exploring Intrinsic Inductive Bias code&#xff…

假设检验之卡方检验

之前我对卡方检验的了解都是一知半解的&#xff0c;即知道作用是对离散变量分布差异的比较&#xff0c;根据期望频数和观察频数的差异计算出来一个卡方值&#xff0c;之后根据自由度和显著性水平查卡方分布对应的临界值&#xff0c;比较大小得出有无明显差异的结论。 一般我们都…

基于FPGA平台实现 ARM Cortex-M0 SOC(一)简介

本系列笔记为基于FPGA平台实现 ARM Cortex-M0 SOC 集创赛作品复盘 Platform&#xff1a; ARM Cortex-M0 Design Srart AT510 XLINX FPGA ARM MDK 5 CM0-Design start 是ARM公司放出的一个免费的ARM 内核学习版本&#xff0c;它比M3还要简单&#xff0c;并且官方把整块代码模糊化…

TFN CK1840B 喇叭天线 定向 18GHz~40GHz

TFN CK1840B 喇叭天线 定向 18GHz~40GHz 产品概述 TFN CK1840B喇叭天线工作频率为 18GHz~40GHz。具有频带宽&#xff0c; 性能可靠&#xff0c; 增益高等优 点&#xff0c; 是理想的 EMC 测试、电子对抗等领域的定向接收、发射天线。 应用领域 ● 电子对抗领域 ● EMC 测试…

基于python多光谱遥感数据处理、图像分类、定量评估及机器学习方法应用

普通数码相机记录了红、绿、蓝三种波长的光&#xff0c;多光谱成像技术除了记录这三种波长光之外&#xff0c;还可以记录其他波长&#xff08;例如&#xff1a;近红外、热红外等&#xff09;光的信息。与昂贵、不易获取的高光谱、高空间分辨率卫星数据相比&#xff0c;中等分辨…

Gateway

Gateway—SpringCloud微服务网关组件 一、Spring Cloud Gateway简介 1.为什么要用Gateway&#xff1f; 在微服务架构中&#xff0c;通常一个系统会被拆分为多个微服务&#xff0c;微服务之间的调用可以用OpenFeign&#xff0c;但面对这么多微服务客户端调用会遇到哪些问题呢…

Hudi(3):Hudi之基本概念

目录 0. 相关文章链接 1. 时间轴&#xff08;TimeLine&#xff09; 1.1. Instant action&#xff1a;在表上执行的操作类型 1.2. Instant time 1.3. State 1.4. 两个时间概念 2. 文件布局&#xff08;File Layout&#xff09; 2.1. Hudi表的文件结构 2.2. Hudi存储的两…

Cocos 引擎生态部负责人李阳:己之所欲,可施于人,希望通过生态促进国内引擎技术发展

前言 “小小的身体&#xff0c;大大的能量&#xff0c;这个应该是我对大表姐最直接的感觉&#xff0c;在她娇小的身躯里蕴含了无限的精力和潜力&#xff0c;很像漫威里的神奇女侠&#xff0c;作为一个具备冒险精神的非典型程序员&#xff0c;大表姐热爱的体育活动都是很具挑战…

大数据系列——什么是ClickHouse?ClickHouse有什么用途?

目录 一、什么是ClickHouse 二、ClickHouse有什么用途 三、ClickHouse的不足 四、适用场景 五、ClickHouse特点 六、ClickHouse VS MySQL 七、类SQL 语句 八、核心概念 一、什么是ClickHouse clickHouse是俄罗斯的 Yandex 公司于 2016 年开源的列式存储数据库&#x…

win11系统用户名称为中文导致文件夹出现繁体字文件夹、系统路径配置错误修改教程(博主亲测,基于win11,系统文件保留)

写在前面&#xff1a;很多人在拿到新电脑激活那会&#xff0c;命名就是简单的中文&#xff0c;但是中文命名电脑系统名称&#xff0c;会导致系统用户文件夹自动命名为中文&#xff0c;在后期使用中会导致c盘系统用户文件夹下面出现不知名繁体字文件夹&#xff0c;甚至有的朋友会…

终难逃一阳

阳了&#xff0c;抗原试剂显示我阳了。每天都带口罩的我还是未能逃过此劫。真是覆巢之下&#xff0c;焉有完卵。 ​ 1.背景 12月初国家逐步放开防疫&#xff0c;随之而来的就是奥秘克戎肆虐全国。身边同事和朋友一个接着一个倒下&#xff0c;朋友圈里更是哀嚎一片。好在专家…

《CSAPP》笔记——链接、异常控制流、虚拟内存

文章目录传送门链接基础链接器的意义编译器驱动程序静态链接ELF目标文件格式可重定位目标文件符号和符号表链接过程符号解析解析规则静态链接库带有静态链接库的解析过程重定位重定位条目重定位节重定位符号引用重定位相对引用重定位绝对引用加载可执行目标文件动态链接共享库库…

Kafka 消费者组开发

Kafka consumer - 消费者组 上一篇文章学习到kafka消费者、消费者组之间处理消息的差异&#xff0c;总结起来就是&#xff1a; 同一个消费组的不同消费实例 共同消费topiic的消息, 一个消息只会消费一次&#xff1b; 也叫做集群消费同一个消息被不同的消费组同时消费&#xf…

机器学习基石1(ML基本概念和VC dimension)

文章目录一、什么是机器学习?二、什么时候可以使用机器学习?三、感知机perceptron四、机器学习的输入形式五、机器真的可以学习吗&#xff1f;六、vc dimension一、什么是机器学习? 其实第一个问题和第二个问题是穿插到一块儿回答的&#xff0c;首先机器学习要解决的是常规的…

RedisTemplate操作redis

目录 Redis Repositories方式 a、启用 Repository 功能 b、注解需要缓存的实体 c、创建一个 Repository 接口 d、测试类中测试 Redis Repositories方式 Spring Data Redis 从 1.7 开始提供 Redis Repositories &#xff0c;可以无缝的转换并存储 domain objects&#xff0…

TOPSIS法(熵权法)(模型+MATLAB代码)

TOPSIS可翻译为逼近理想解排序法&#xff0c;国内简称为优劣解距离法 TOPSIS法是一种常用的综合评价方法&#xff0c;其能充分利用原始数据的信息&#xff0c;其结果能精确地反映各评价方案之间的距离 一、模型介绍 极大型指标&#xff08;效益型指标&#xff09; &#xff…

OR-Tools工具介绍以及实战(从入门到超神Python版)

目录前言0、安装一、什么是优化问题&#xff1f;1-1、优化问题介绍1-2、举例1-2-1、导入所需要的库1-2-2、声明求解器1-2-3、创建变量1-2-4、定义约束条件1-2-5、定义目标函数1-2-6、调用求解器&#xff0c;并且显示结果。二、python有关于各种优化问题示例2-1、简单的线性编程…