C++核心编程——类和对象(一)
本专栏记录C++学习过程包括C++基础以及数据结构和算法,其中第一部分计划时间一个月,主要跟着黑马视频教程,学习路线如下,不定时更新,欢迎关注。
当前章节处于:
---------第1阶段-C++基础入门
---------第2阶段实战-通讯录管理系统,
=====>第3阶段-C++核心编程,
---------第4阶段实战-基于多态的企业职工系统
---------第5阶段-C++提高编程
---------第6阶段实战-基于STL泛化编程的演讲比赛
---------第7阶段-C++实战项目机房预约管理系统
文章目录
- 一、 封装
- 1.1 封装基本概念
- 1.2 访问权限
- 1.3 成员属性设置成私有
- 二、对象特性
- 2.1 对象的初始化和清理
- 2.2 构造函数的分类与调用
- 2.3 构造函数调用规则
- 2.4 浅拷贝与深拷贝
- 2.5 初始化列表
- 2.6 类对象作为类成员
- 2.7 静态成员
- 三. C++对象模型和this指针
- 3.1 成员变量和成员函数分开存储
- 3.2 this指针概念
- 3.3 空指针访问成员函数
- 3.4 const修饰成员函数
类和对象是C++的核心,C++面向对象的三大特性:封装、继承、多态
,C++认为万事万物都是对象,对象尤其属性和行为。
一、 封装
1.1 封装基本概念
封装是C++面向对象三大特性之一,意义在于:
- 将属性和行为作为一个整体,表现生活中的事务
- 将属性和行为加以权限控制
语法:
class 类名{ 访问权限:属性/ 行为};
代码举例说明,设计一个圆类,计算圆的周长:
#include <iostream>
using namespace std;
#define PI 3.14
class Circle {// 权限
public://属性int r = 0;// 行为double getLength() {return r * r * PI;}
};int main() {// 实例化对象Circle c1;c1.r = 3;cout << "圆的周长为:" << c1.getLength() << endl;system("pause");return 0;}
圆的周长为:28.26
请按任意键继续. . .
创建类后需要实例化,才可以使用,相当于类是一个模具,实例化出来的对象才是产品。
#include <iostream>
using namespace std;
class Student {// 权限
public:// 属性string name;string cardnum;// 行为string getname() {return name;}string getcardnum() {return cardnum;}
};int main() {// 实例化对象Student stu;stu.name = "张三";stu.cardnum = "123";cout << "姓名为:" << stu.getname() << " 学号为:" << stu.getcardnum() << endl;system("pause");return 0;}
姓名为:张三 学号为:123
请按任意键继续. . .
1.2 访问权限
访问权限分为三种:
- public 公共权限:
类内可以访问,类外可以访问
- protected 保护权限
类内可以访问,类外不可以访问
- private 私有权限
类内可以访问,类外不可以访问
protected和private区别在于继承时,儿子可以继承父亲中的protected内容,不可以继承父亲中private中内容
#include <iostream>
using namespace std;
class Person {
public:string name;
protected:string car;
private:string password;
public:void test() {// 类内都可以访问name = "张三";car = "自行车";password = "12345";}
};int main() {Person p1;p1.name = "李四";// 类外可以访问// p1.car; // 类外不可访问// p1.password;// 类外不可访问system("pause");return 0;}
struct也可以定义类,但是struct默认权限是公共权限,而class默认权限是公有;
#include <iostream>
using namespace std;
class test1 {// 默认私有int a = 0;
};
struct test2 {// 默认公共int b = 0;
};int main() {test1 test_1;test_1.a = 10; // 不能访问test2 test_2;test_2.b = 20; // 可以访问system("pause");return 0;}
1.3 成员属性设置成私有
优点在于:
- 将所有成员属性设置成私有,可以自己控制读写权限
- 对于写权限,可以检测数据的有效性
通过get和set方法进行访问和赋值
#include <iostream>
using namespace std;class Student {// 权限
private:// 属性string name="张三"; // 可读可写string cardnum="123";// 只读int score=0; // 只写
public:// 可读可写void setname(string inputname) {name = inputname;}string getname() {return name;}// 只读string getcarnum() {return cardnum;}// 只写void setscore(int inputscore) {score = inputscore;}
};
int main() {Student stu;cout << stu.getname() << endl;stu.setname("李四");cout << stu.getname() << endl;system("pause");return 0;}
张三
李四
请按任意键继续. . .
可以用get和set方法进行值的合法性判断。
练习案例1:设计立方体类
- 设计立方体类(Cube)
- 求出立方体的表面积和体积
- 分别用全局函数和成员函数判断两个立方体是否相等
全局函数需要传入两个实例化对象,成员函数只要传入一个实例化对象即可
#include <iostream>
using namespace std;
class Cube {
public:double x=0.0f; // 边长double y=0.0f; // 边长double z=0.0f; // 边长void setinf(double inputx, double inputy, double inputz) {x = inputx;y = inputy;z = inputz;}int getS() {return 3 * (x * y+x*z+y*z);}int getV() {return x * y * z;}bool judge(Cube c2) {if ((c2.x == x)&&(c2.y==y)&&(c2.z==z)){return true;}else {return false;}}
};
// 定义全局函数
bool g_judge(Cube c1, Cube c2) {if ((c2.x == c1.x) && (c2.y == c1.y) && (c2.z == c1.z)) {return true;}else {return false;}
}
int main() {Cube c1,c2; // 创建c1.setinf(1, 2, 3);c2.setinf(2, 3, 4);cout <<"c1表面积:"<< c1.getS() << endl;cout <<"c1体积:"<< c1.getV() << endl;// 成员函数判断string flag = c1.judge(c2) ? "Yes" : "No";cout <<"通过成员函数判断两立方体是否相等:"<< flag << endl;// 全局函数判断string g_flag = g_judge(c1,c2) ? "Yes" : "No";cout << "通过全局函数判断两立方体是否相等:" << flag << endl;system("pause");return 0;}
c1表面积:33
c1体积:6
通过成员函数判断两立方体是否相等:No
通过全局函数判断两立方体是否相等:No
请按任意键继续. . .
练习案例2:点和圆的关系
计算一个圆形类(Circle)和一个点类(Point),计算点和圆的关系
#include <iostream>
using namespace std;
class Circle {
public:int x = 0;int y = 0;int r;void set_inf(int inputx,int inputy,int inputr) {x = inputx;y = inputy;r = inputr;}
};class Point {
public:int P_x = 0;int P_y = 0;void set_inf(int input_P_x, int input_P_y) {P_x = input_P_x;P_y = input_P_y;}
};string judge(Circle c, Point p) {int distance = sqrt((c.x - p.P_x)*(c.x - p.P_x) + (c.y - p.P_y)*(c.y - p.P_y));if (distance == c.r) {return "在圆上";}else if (distance < c.r) {return "在圆内";}else {return "在圆外";}
}
int main() {Circle c1;Point p1;c1.set_inf(1, 1, 1);p1.set_inf(1, 0);string flag = judge(c1, p1);cout << flag << endl;system("pause");return 0;}
在圆上
请按任意键继续. . .
代码比较多,可以分文件编写,分成Circle.cpp,Circle.h,Point.h,Point.cpp,cpp中放函数实现方法,.h中放函数申明
circle.cpp
#include "circle.h"
void Circle::set_inf(int inputx, int inputy, int inputr) {x = inputx;y = inputy;r = inputr;
};
circle.h
#pragma once
#include <iostream>
using namespace std;class Circle {
public:int x = 0;int y = 0;int r;void set_inf(int inputx, int inputy, int inputr);
};
point.cpp
#include "point.h"void Point::set_inf(int input_P_x, int input_P_y) {P_x = input_P_x;P_y = input_P_y;
}
point.h
#pragma once
#include <iostream>
using namespace std;class Point {
public:int P_x = 0;int P_y = 0;void set_inf(int input_P_x, int input_P_y);
};
main.cpp
#include <iostream>
using namespace std;
#include "circle.h"
#include "point.h"
//class Circle {
//public:
// int x = 0;
// int y = 0;
// int r;
// void set_inf(int inputx,int inputy,int inputr) {
// x = inputx;
// y = inputy;
// r = inputr;
// }
//};//class Point {
//public:
// int P_x = 0;
// int P_y = 0;
// void set_inf(int input_P_x, int input_P_y) {
// P_x = input_P_x;
// P_y = input_P_y;
// }
//};string judge(Circle c, Point p) {int distance = sqrt((c.x - p.P_x)*(c.x - p.P_x) + (c.y - p.P_y)*(c.y - p.P_y));if (distance == c.r) {return "在圆上";}else if (distance < c.r) {return "在圆内";}else {return "在圆外";}
}
int main() {Circle c1;Point p1;c1.set_inf(1, 1, 1);p1.set_inf(1, 0);string flag = judge(c1, p1);cout << flag << endl;system("pause");return 0;}
在圆上
请按任意键继续. . .
二、对象特性
2.1 对象的初始化和清理
C++面向对象来源于生活,每个对象都会有初始化设置以及对象销毁前的清理数据设置。对象的初始化和清理也是两个非常重要的安全问题。
- 一个对象或者变脸没有初始状态,对其使用后果是未知。
- 同样,使用完一个对象或变量,没有及时清理,也会造成一定的安全问题。
C++利用构造函数
和析构函数
解决以上问题,两个函数会被编译器自动调用,如果不提供构造和析构,编译器会提供构造函数和析构函数的空实现。
- 构造函数:创建对象时为对象的成员属性赋值;
- 析构函数:对象销毁前系统自动调用,执行一些清理工作。
构造函数语法 类名(){}
- 构造函数,没有返回值也不写void
- 函数名称与类名相同
- 构造函数可以有参数,因此可以发生重载
- 程序在调用对象时候会自动调用构造,无须手动调用,而且只会调用一次
析构函数语法~类名(){}
- 析构函数,没有返回值也不写void
- 函数名称与类名相同,在名称前加上符号~
- 析构函数不可以有参数,因此不可以发成重载
- 程序在对象销毁前会自动调用析构,无须手动调用,而且只会调用一次
#include <iostream>
using namespace std;
class Person {
public: // 加上作用域Person() {cout << "构造函数的调用!" << endl;}~Person() {cout << "析构函数的调用!" << endl;}
};void test() {Person p; // 在栈区创建实例化对象
}
int main() {test();system("pause");return 0;
}
构造函数的调用!
析构函数的调用!
请按任意键继续. . .
2.2 构造函数的分类与调用
分类方式:
- 按参数分:有参构造和无参构造
- 按类型分:普通构造和拷贝构造
调用方式:
- 括号法
- 显示法
- 隐式转化法
#include <iostream>
using namespace std;class Person {// 构造函数// 分类方式:有参构造、无参构造 普通构造和拷贝构造// 无参构造
public:Person() {cout << "调用的为无参构造!" << endl;}// 有参构造Person(int a) {age = a;cout << "调用的为有参构造!" << endl;}// 以上都为普通构造// 拷贝构造Person(const Person &p) {age = p.age;cout << "调用的为拷贝构造!" << endl;}int age;
};int main() {// 调用方式// 1. 括号法// 无参调用Person p1;// 有参调用Person p2(10);// 拷贝调用Person p3(p2);// 2.显示法Person p4 = Person(10);// Person(p4); 不可用拷贝构造 初始化匿名函数Person p5 = Person(p4); // Person(p4)为匿名对象,当前行执行结束后,会立即被析构掉// 隐式转化法Person p6 = 10; //相当于Person p6 = Person(10)system("pause");return 0;
}
调用的为无参构造!
调用的为有参构造!
调用的为拷贝构造!
调用的为有参构造!
调用的为拷贝构造!
调用的为有参构造!
请按任意键继续. . .
拷贝构造函数使用时机
- 使用一个已经创建完毕的对象来初始化一个新对象
- 值传递的方式给函数参数传值
- 值方式返回局部对象
#include <iostream>
using namespace std;
class Person {
public:Person() {cout << "无参构造" << endl;}Person(int a) {age = a;cout << "有参构造" << endl;}Person(const Person &p) {age = p.age;cout << "拷贝构造" << endl;}int age;
};void test() {Person p1(10);Person p2(p1); // 用一个已经创建好的对象来初始化一个新对象cout <<"p1的年龄为:"<< p1.age << endl;cout <<"p2的年龄为:"<< p2.age << endl;}
void test2(Person p) {
}Person test3() {static Person p;return p;
}
//void todo() {
// test3();
//}
int main() {//Person p(10);Person p = test3();system("pause");return 0;
}
无参构造
拷贝构造
请按任意键继续. . .
值传递的方式给函数参数传值和值方式返回局部对象都相当于将原来的值拷贝了一份,因此相当于把原来的拷贝了一份.
2.3 构造函数调用规则
- 如果用户定义有参构造函数,C++不再提供默认无参构造,但是会提供默认拷贝构造
- 如果用户定义拷贝构造函数,C++不会再提供其他构造函数
可以这么理解: 拷贝构造>有参构造>无参构造,如果用户提供一个构造,编译器会自动补齐比他更高级的构造,不会提供比他更低级的构造。比如如果用户定义有参构造函数,C++不再提供默认无参构造,但是会提供默认拷贝构造
#include <iostream>
using namespace std;class Person {
public:Person(int a) {m_age = a;}int m_age;
};int main() {//Person p;// 报错Person p(10); // 不报错Person p2(p); // 不报错system("pause");return 0;}
2.4 浅拷贝与深拷贝
- 浅拷贝:简单的赋值拷贝操作
- 深拷贝:在堆区重新申请空间,进行拷贝操作
#include <iostream>
using namespace std;
class Person {
public:Person() {cout << "无参构造" << endl;}Person(int a,int n) {cout << "有参构造" << endl;age = a;// num = &n;报错num = new int(n);}Person(const Person& p) {cout << "拷贝函数" << endl;age = p.age;//num = p.num; // 报错num = new int (*p.num);}// 析构函数~Person() {cout << "调用析构函数" << endl;// 将指针开辟区域释放if (num != NULL) {delete num;num = NULL;}}int age;int* num; // 指针在堆区开辟
};
void test() {int num = 123;// 有参构造Person p1(10, num);// 拷贝构造Person p2(p1); // 报错
}
int main() {test();system("pause");return 0;}
有参构造
拷贝函数
调用析构函数
调用析构函数
请按任意键继续. . .
主要原因就是浅拷贝的地址被释放之后,二次释放。深拷贝会重新开辟一个地址,这样就不会再同一个地址释放多次。
2.5 初始化列表
语法:构造函数():属性1(值1),属性2(值2)…{}
#include <iostream>
using namespace std;class Person {
public:Person() {}Person(string n,int a,string ad):name(n),age(a),address(ad){}string name;int age;string address;
};int main() {Person p1("张三", 13, "安徽");cout << "姓名:" <<p1.name<< " 年龄:" << p1.age<<" 住址:" << p1.address<<endl;//Person p2("李四", 12); //报错,少参数//cout << "姓名:" << p2.name << " 年龄:" << p2.age << " 住址:" << p2.address << endl;system("pause");return 0;}
姓名:张三 年龄:13 住址:安徽
请按任意键继续. . .
2.6 类对象作为类成员
#include <iostream>
using namespace std;class Student {
public:string name;int score;Student(string n,int s):name(n),score(s){cout << "Student有参构造" << endl;}~Student() {cout << "Student析构函数" << endl;}};class Teacher {
public:string name;Student stu;Teacher(string t_n,string s_n,int s):name(t_n),stu(s_n,s){}
};void test() {Teacher t1("李老师", "张三", 12);cout << "老师姓名:" << t1.name << " 学生姓名:" << t1.stu.name << " 学生分数:" << t1.stu.score << endl;
}
int main() {test();system("pause");return 0;}
Student有参构造
老师姓名:李老师 学生姓名:张三 学生分数:12
Student析构函数
请按任意键继续. . .
2.7 静态成员
在成员变量和成员函数前加上关键字static
,成为静态成员,分为:
-
静态成员变量
- 所有对象共享同一份数据
- 在编译阶段分配内存
- 类内声明,类外初始化
-
静态成员函数
- 所有对象共享同一个函数
- 静态成员函数只能访问静态成员变量
#include <iostream>
using namespace std;
class Person {
public:string name;static int score; // 静态成员变量,需要类内定义,类外初始化void func() {name = "张三";score = 10;cout << "调用函数func!" << endl;}static void func2() {//name = "李四"; // 报错,静态成员函数只能调用静态成员变量score = 200;cout << "调用函数func2!" << endl;}
};
int Person::score=100;int main() {Person p1;cout <<"p1分数为:"<< p1.score << endl;Person p2;p2.score = 60;cout << "p2分数为" << p2.score << endl;cout << "p1分数为" << p1.score << endl;// 直接使用类名调用静态成员变量cout << "Person::score:" << Person::score << endl; // 注意不能使用. 而是使用::// 静态成员函数p2.func();p2.func2();system("pause");return 0;}
p1分数为:100
p2分数为60
p1分数为60
Person::score:60
调用函数func!
调用函数func2!
请按任意键继续. . .
三. C++对象模型和this指针
3.1 成员变量和成员函数分开存储
C++中类内成员变量和成员函数分开存储,只有非静态成员变量才属于类的对象上。
#include <iostream>
using namespace std;
class Person {
public:int a; // 是类的对象static int b;// 不是类的对象void func() {}static void func2(){}
};
int Person::b = 0;
int main() {Person p;cout << "size of p=" << sizeof(p) << endl;system("pause");return 0;}
size of p=4
请按任意键继续. . .
3.2 this指针概念
c++提供特殊的对象指针——this指针,this指针指向被调用的成员函数所属对象。this指针本质是一个指针常量,指向对象不能发生变化,指向对象的值可以发生变化。
- this指针时隐含每一个非静态成员函数内的一种指针
- this指针不需要定义,直接使用即可
this指针的用途:
- 当形参和成员变量同名时,可用this指针来区分
- 在类的非静态成员函数中返回对象本身,可使用return *this
#include <iostream>
using namespace std;
class Person {
public:int age;Person(int age) {//age = age; // 编译器认为这两个age是一样的//1. 防止形参和成员变量同名this->age = age; // 将形参age赋值给成员变量age}//2. 返回对象本身// 如果不是引用类型的话,每一次都会创建一个新对象,而不是在原来的对象上进行累加Person& Personadd(Person p) {this->age += p.age;return *this;}
};int main() {Person p(10);Person p1(20);cout << "p年龄:" << p.age << endl;cout << "p1年龄:" << p.age << endl;p1.Personadd(p).Personadd(p); // 加两次cout << "年龄:" << p1.age << endl;system("pause");return 0;}
p年龄:10
p1年龄:10
年龄:40
请按任意键继续. . .
3.3 空指针访问成员函数
空指针可以调用成员函数,但是要注意有没有用到this指针。如果用到了this指针,需要加以判断保证代码的健壮性
#include <iostream>
using namespace std;class Person {
public:int age;void func() {cout << "func的调用" << endl;}void func2() {// 添加判断 增强代码的健壮性if (this == NULL) {return;}cout << "年龄为" << age << endl; // age相当于this.age 如果是空指针的话会报错}
};int main() {// 创建一个空指针Person* p = NULL;p->func();p->func2(); // 用到this指针会报错system("pause");return 0;}
func的调用
请按任意键继续. . .
3.4 const修饰成员函数
常函数
- 成员函数后加const后我们称这个函数为常函数
- 常函数不可以修改成员属性
- 成员属性声明时加关键字mutable后,在常函数中仍然可以修改
常对象
- 声明对象前加const称该对象为常对象
- 常对象只能调用常函数
#include <iostream>
#include <iostream>
using namespace std;
class Person{
public:string name; //成员变量mutable int age; // mutable修饰使得常函数也能进行修改// 无参构造函数Person() {}// 定义一个成员函数void test() {cout << "调用test函数" << endl;}// 定义一个常函数void func() const{//name = "李四"; // 报错,不能修改正常的成员属性age = 12; // 不报错,可以修改mutable修饰的成员变量}};int main() {// 创建一个常对象const Person p;//p.test();// 不可以调用正常的成员函数p.func(); // 可以调用常函数cout << "年龄为:" << p.age << endl;system("pause");return 0;}
年龄为:12
请按任意键继续. . .
相关文章:

C++核心编程——类和对象(一)
本专栏记录C学习过程包括C基础以及数据结构和算法,其中第一部分计划时间一个月,主要跟着黑马视频教程,学习路线如下,不定时更新,欢迎关注。 当前章节处于: ---------第1阶段-C基础入门 ---------第2阶段实战…...
脱模斜度是什么意思,为什么要有脱模斜度,没有斜度不行吗?
问题描述:脱模斜度是什么意思,为什么要有脱模斜度,没有斜度不行吗? 问题解答: 脱模斜度是指在模具中的零件在脱模(从模具中取出)过程中相对于模具开合方向的倾斜程度。在模具设计和制造中&…...

【现代密码学】笔记9-10.3-- 公钥(非对称加密)、混合加密理论《introduction to modern cryphtography》
【现代密码学】笔记9-10.3-- 公钥(非对称加密)、混合加密理论《introduction to modern cryphtography》 写在最前面8.1 公钥加密理论随机预言机模型(Random Oracle Model,ROM) 写在最前面 主要在 哈工大密码学课程 张…...

牛客-寻找第K大、LeetCode215. 数组中的第K个最大元素【中等】
文章目录 前言牛客-寻找第K大、LeetCode215. 数组中的第K个最大元素【中等】题目及类型思路思路1:大顶堆思路2:快排二分随机基准点 前言 博主所有博客文件目录索引:博客目录索引(持续更新) 牛客-寻找第K大、LeetCode215. 数组中的第K个最大元…...

MySQL的各种日志
目录 一、错误日志 二、二进制日志 1、介绍 2、作用 3、相关信息 4、日志格式 5、查看二进制文件 6、二进制日志文件删除 三、查询日志 四、慢日志 一、错误日志 记录MySQL在启动和停止时,以及服务器运行过程中发生的严重错误的相关信息,当数据库…...

rust跟我学六:虚拟机检测
图为RUST吉祥物 大家好,我是get_local_info作者带剑书生,这里用一篇文章讲解get_local_info是怎么检测是否在虚拟机里运行的。 首先,先要了解get_local_info是什么? get_local_info是一个获取linux系统信息的rust三方库,并提供一些常用功能,目前版本0.2.4。详细介绍地址:…...
测试bug分析
项目场景: 提示:这里简述项目相关背景: 例如:项目场景:示例:通过蓝牙芯片(HC-05)与手机 APP 通信,每隔 5s 传输一批传感器数据(不是很大) 问题描述 提示:这里描述项目中遇到的问题࿱…...

css-盒子等样式学习
盒子居中,继承外层盒子的宽高 兼容性(border-box)将边框收到盒子内部 初始化div 不用管box-setting content-box 还原 创建为一个类 ,让所有需要还原的类 进行继承 padding 用法表示margin上下左右边距 body 外边距&…...
数据库系统概论 第1章绪论 1.1数据库的四个基本概念
1.1.1 数据库的4个基本概念 - 数据(Data) - 数据库(Database, DB) - 数据库管理系统(DataBase Management System, DBMS) - 数据库系统(DataBase System, DMS) 1. 数据 - 数据(Data)是数据库中存储…...
使用Linux搭建svn
1.安装 Apache 和 Subversion 软件包 sudo yum install httpd subversion mod_dav_svn2.启动 Apache 服务 sudo systemctl start httpd3.设置 Apache 服务开机自启动 sudo systemctl enable httpd4.创建/svn 目录 sudo mkdir /svn5.设置 /svn 目录的权限: sudo…...

Kafka的安装、管理和配置
Kafka的安装、管理和配置 1.Kafka安装 官网: https://kafka.apache.org/downloads 下载安装包,我这里下载的是https://archive.apache.org/dist/kafka/3.3.1/kafka_2.13-3.3.1.tgz Kafka是Java生态圈下的一员,用Scala编写,运行在Java虚拟机上…...

某银行主机安全运营体系建设实践
随着商业银行业务的发展,主机规模持续增长,给安全团队运营工作带来极大挑战,传统的运营手段已经无法适应业务规模的快速发展,主要体现在主机资产数量多、类型复杂,安全团队难以对全量资产进行及时有效的梳理、管理&…...
虚拟化技术、Docker、K8s笔记总结
一、虚拟化技术 是一种将物理资源(如服务器、存储设备、网络设备等)抽象、转换和分割成多个逻辑资源的技术。通过虚拟化技术,用户可以在单个物理设备上运行多个相互独立的虚拟环境,从而提高资源的利用率、降低运维成本和提高系统…...

基于springboot+vue的在线拍卖系统(前后端分离)
博主主页:猫头鹰源码 博主简介:Java领域优质创作者、CSDN博客专家、公司架构师、全网粉丝5万、专注Java技术领域和毕业设计项目实战 主要内容:毕业设计(Javaweb项目|小程序等)、简历模板、学习资料、面试题库、技术咨询 文末联系获取 项目背景…...
【征服redis3】一文征服redis的jedis客户端
使用数据库的时候,我们可以用JDBC来实现mysql数据库与java程序之间的通信,为了提高通信效率,我们有了数据库连接池比如druid等等。而我们想通过Java程序控制redis,同样可以借助一些工具来实现,这就是redis客户端&#…...
Python如何操作RabbitMQ实现direct关键字发布订阅模式?有录播直播私教课视频教程
direct关键字发布订阅模式 基本用法 发布者 import json from rabbitmq import pika import rabbitmq# 建立连接 credentials rabbitmq.PlainCredentials(zhangdapeng,zhangdapeng520, ) # mq用户名和密码 connection_target rabbitmq.ConnectionParameters(host127.0.0.…...
如何应用数据图表了解家里的 Unifi 网络状况?
1. 前言 自从之前写了《【让 IT 更简单】使用 Ubiquiti 全家桶对朋友家进行网络改造》 《【Rethinking IT】如何结合 Unifi 和 MikroTik 设备打造家庭网络》两篇文章后,相信给各位正在用 Unifi 或者打算使用 Unifi 的朋友应该有所帮助。 那么,今天我就…...

新版K8s:v1.28拉取Harbor仓库镜像以及本地镜像(docker弃用改用containerd,纯纯踩坑)
目录 一、项目概述二、环境三、项目样式Harborkuboard运行样式 四、核心点Harbor安装config.toml文件修改(containerd)ctr、nerdctl相关命令kuboard工作负载 五、总结 一、项目概述 使用Kuboard作为k8s集群的管理平台,Harbor作为镜像仓库,拉取Harbor镜像…...

Unity URP切换品质和Feature开关的性能问题
现在对我的项目进行安卓端发布,需要切换品质和一些Feature开关。 我是这样做的。 划分品质 首先Renerer分为2个Android和PC,图中其他不用参考。 每个副本的URP Asset分为pc和android,例如图中的 hall和hall_android。 我们可以看到hall用的…...
jmeter解决返回unicode编辑
一般乱码有两种方法来解决: 1、修改配置文件jmeter.properties中默认编码格式ISO-8859-1(不支持中文),修改为utf-8 sampleresult.default.encoding utf-82、添加BeanShell PostProcessor加入 prev.setDataEncoding("utf-8")3、还有一种返回…...

大话软工笔记—需求分析概述
需求分析,就是要对需求调研收集到的资料信息逐个地进行拆分、研究,从大量的不确定“需求”中确定出哪些需求最终要转换为确定的“功能需求”。 需求分析的作用非常重要,后续设计的依据主要来自于需求分析的成果,包括: 项目的目的…...
pam_env.so模块配置解析
在PAM(Pluggable Authentication Modules)配置中, /etc/pam.d/su 文件相关配置含义如下: 配置解析 auth required pam_env.so1. 字段分解 字段值说明模块类型auth认证类模块,负责验证用户身份&am…...
Qt Http Server模块功能及架构
Qt Http Server 是 Qt 6.0 中引入的一个新模块,它提供了一个轻量级的 HTTP 服务器实现,主要用于构建基于 HTTP 的应用程序和服务。 功能介绍: 主要功能 HTTP服务器功能: 支持 HTTP/1.1 协议 简单的请求/响应处理模型 支持 GET…...

Redis数据倾斜问题解决
Redis 数据倾斜问题解析与解决方案 什么是 Redis 数据倾斜 Redis 数据倾斜指的是在 Redis 集群中,部分节点存储的数据量或访问量远高于其他节点,导致这些节点负载过高,影响整体性能。 数据倾斜的主要表现 部分节点内存使用率远高于其他节…...
JS设计模式(4):观察者模式
JS设计模式(4):观察者模式 一、引入 在开发中,我们经常会遇到这样的场景:一个对象的状态变化需要自动通知其他对象,比如: 电商平台中,商品库存变化时需要通知所有订阅该商品的用户;新闻网站中࿰…...

LabVIEW双光子成像系统技术
双光子成像技术的核心特性 双光子成像通过双低能量光子协同激发机制,展现出显著的技术优势: 深层组织穿透能力:适用于活体组织深度成像 高分辨率观测性能:满足微观结构的精细研究需求 低光毒性特点:减少对样本的损伤…...

android13 app的触摸问题定位分析流程
一、知识点 一般来说,触摸问题都是app层面出问题,我们可以在ViewRootImpl.java添加log的方式定位;如果是touchableRegion的计算问题,就会相对比较麻烦了,需要通过adb shell dumpsys input > input.log指令,且通过打印堆栈的方式,逐步定位问题,并找到修改方案。 问题…...

STM32---外部32.768K晶振(LSE)无法起振问题
晶振是否起振主要就检查两个1、晶振与MCU是否兼容;2、晶振的负载电容是否匹配 目录 一、判断晶振与MCU是否兼容 二、判断负载电容是否匹配 1. 晶振负载电容(CL)与匹配电容(CL1、CL2)的关系 2. 如何选择 CL1 和 CL…...

脑机新手指南(七):OpenBCI_GUI:从环境搭建到数据可视化(上)
一、OpenBCI_GUI 项目概述 (一)项目背景与目标 OpenBCI 是一个开源的脑电信号采集硬件平台,其配套的 OpenBCI_GUI 则是专为该硬件设计的图形化界面工具。对于研究人员、开发者和学生而言,首次接触 OpenBCI 设备时,往…...

【UE5 C++】通过文件对话框获取选择文件的路径
目录 效果 步骤 源码 效果 步骤 1. 在“xxx.Build.cs”中添加需要使用的模块 ,这里主要使用“DesktopPlatform”模块 2. 添加后闭UE编辑器,右键点击 .uproject 文件,选择 "Generate Visual Studio project files",重…...