C++学习——类和对象(二)

news2025/1/13 10:23:14

  紧接着我们上一部分类和对象的讲解之后,我们再来学习一下类当中的几大特点,以及使用方法。

  一:实例化对象赋初值

  首先我们需要学习的就是该如何为我们实例化出的对象赋初值。

  1.外部赋值

  对于对象赋初值我们有很多的形式,我们甚至可以像结构体一样,对于我们类当中的数据进行一个一个赋值,(但是可能会 麻烦很多)所示代码如下:

#include<iostream>

//设置一个学生类,并对其在外部赋初值
class Stu
{
public:
	void print()
	{
		std::cout << name << std::endl;
		std::cout << age << std::endl;
	}
	int age;
	char name[20];
};

int main()
{
	Stu s1;
	s1.age = 12;
	strcpy(s1.name, "张三");
	s1.print();
	return 0;
}

   我们需要在主函数当中实例化一个类的对象,之后通过对象使用 . 操作符进行进一步的对指定的属性进行赋初值。可能age看起来很正常,但是对于我们类中的数组,他们会自动退化成为一个指针,所以我们需要使用字符串拷贝函数对其进行赋初值,或者我们一个字节一个字节的手动赋值也是可以的,但是没有太大的必要,在类的外部进行赋初值还有一个弊端那就是:我们的成员变量的权限只能是public的时候才可以对其进行在外部赋值,否则就会报错。我们来检查以下程序执行的效果:

  2.使用默认参数进行赋值

  但是我们这样在外部一个一个赋值就会显得很麻烦,那么有没有什么更好的方法可以为对象赋值呢?我们就可以使用默认参数为我们的成员变量进行赋值,所示的代码如下:

#include<iostream>

//定义一个学生类。使用默认参数为成员变量进行赋初值
class Stu
{
public:
	void print()
	{
		std::cout << age << std::endl;
		std::cout << name << std::endl;
	}
private:
	int age=12;
	char name[20]="张三";
};

int main()
{
	Stu s1;
	s1.print();

	return 0;
}

   我们可以在类的内部在声明成员变量的时候就给一个值作为我们成员变量的默认值,只要我们不主动修改那么系统就会自动使用默认的成员变量对其进行初始化。我们先来测试一下程序的运行效果:

  3.设置对外接口进行修改(成员函数)

  同样的,我们很快就会发现一个弊端,那就是使用默认参数赋值,是不是就是说无法在函数的外部对于我们的成员变量进行修改了呢?毕竟我们的成员变量已经被设置成为了private权限,在外部无法对其进行访问了。事实上也确实如此,我们先来介绍一种很麻烦的允许我们访问并修改私有成员变量的方式——设置成员函数作为与外部的接口进行对接的形式。代码如下:

#include<iostream>

//定义一个学生类。使用默认参数为成员变量进行赋初值
class Stu
{
public:
	void print()
	{
		std::cout << _age << std::endl;
		std::cout << _name << std::endl;
	}
	//定义一系列对外接口,方便修改私有化的成员变量
	void ChangeName(const char name[20])
	{
		strcpy(_name, name);
	}
	void ChangeAge(int age)
	{
		_age = age;
	}
private:
	int _age=12;
	char _name[20]="张三";
};

int main()
{
	Stu s1;
	std::cout << "使用默认值对成员变量进行初识化" << std::endl;
	s1.print();
	std::cout << "调用成员函数对默认初始化的值进行修改" << std::endl;
	s1.ChangeAge(99);
	s1.ChangeName("李四");
	s1.print();
	return 0;
}

   就像是上面的代码所展示的那样,我们需要设置对外的函数接口之后才可以对内部的成员变量进行特定的访问,但是问题也就随之而来,假如我们的成员变量很多呢?我们需要一个一个设置接口吗?答案是当然不,事实上C++的语法虽然是支持的,但是这种形式并不常用,那么接下来就引出了在类和对象章节当中最为重要的一部分了——类的6个默认成员函数。我们先来测试一下以上代码的运行效果:

  二:默认成员函数

  在类和对象最重要的一部分就是了解认识并能够熟练使用类的六个默认成员函数。那么那么首先介绍以下什么是默认成员函数呢?

  当我们的类当中不存在所指定的六个成员函数的时候系统会自动生成的六个成员函数,这样的函数就叫做默认成员函数。默认成员函数主要有以下的六种:

  具体的六种默认成员函数如同我们上面所示。我们会依次进行解释。通常情况下默认成员函数会在我们的类当中进行定义的时候自动生成。但是当我们主动定义的时候,即使运用了运算符重载生成了其他类型的成员函数我们的编译器也不会在生成对应的默认成员函数了。我们可以从后面的例子中进行理解。

  1.构造函数

  第一个需要我们讲解的就是我们的构造函数。我们会发现每一个自定义在实例化对象使用的时候都需要进行各种初始化,要么是在外部进行各种初始化,这都有各种各样的弊端。所以我们就在想能不能创造一种函数,在创建对象的时候就可以进行指定传参,之后就生成对应的对象就好了。于是便有了构造函数。我们的构造函数就是用来方便我们进行对象初始化的。通过一段代码进行理解:

#include<iostream>

//定义一个学生类,使用构造函数进行特定的初始化
class Stu
{
public:
	Stu(int age = 12, const char name[20] = "张三")
	{
		_age = age;
		strcpy(_name, name);
	}
	void print()
	{
		std::cout << _name<<" " << _age << std::endl;
	}
private:
	int _age;
	char _name[20];
};
int main()
{
	Stu s1;
	s1.print();
	Stu s2(17, "李四");
	s2.print();
	return 0;
}

    构造函数要求函数名必须和我们的类名相同,并且没有返回值,我们可以对于构造函数设置默认参数方便我们初始化。在定义完成构造函数之后我们不需要主动调用构造函数,在每次使用类生成指定的对象的时候就会默认调用我们的构造函数。程序运行的结果如下:

  但是我们的默认构造函数并不是如此,因为我们上面使用了默认参数所以在实例化无参对象的时候依旧可以正常的运行,但是假如我们把默认参数去掉,那么就会产生找不到构造函数的报错。

#include<iostream>

//定义一个学生类,使用构造函数进行特定的初始化
class Stu
{
public:
	Stu(int age, const char name[20])
	{
		_age = age;
		strcpy(_name, name);
	}
	void print()
	{
		std::cout << _name<<" " << _age << std::endl;
	}
private:
	int _age;
	char _name[20];
};
int main()
{
	Stu s1;
	s1.print();
	Stu s2(17, "李四");
	s2.print();
	return 0;
}

   我们会发现编译器给了我们详细的报错:没有合适的构造函数可用。这是因为我们已经自动生成了一个构造函数,编译器就不会在生成默认构造了。可以联想到编译器生成的默认构造函数并没有参数。因为我们在自己创建构造函数之前就可以无参的使用类实例化对象。但是我们的默认构造函数又会对我们的成员变量进行怎样的处理呢?我们可以通过测试得出结果:

  通过打印出的效果我们很容易联想到:生成的不会是随机值吧?没错,确实是随机值。但是不是对象当中所有的内容都会生成随机值。那么我们直接先看结论:我们的默认构造函数不会对内置类型进行赋值,但是对于我们的自定义类型会调用相应的构造函数。(内置类型指的是我们系统自定义的类型,如:int,char,指针等)

  简单的来说我们的编译器只会对自定义类型产生相应的反应,也就是调用相应的构造函数,我们可以通过一组代码进行测试:

#include<iostream>

//重新定义一个类,作为Stu类的私有变量,检测构造函数的调用
class Teacher
{
public:
	Teacher(int age=23,const char name[20]="王老师")
	{
		_age = age;
		strcpy(_name, name);
	}
	void print()
	{
		std::cout << _name << std::endl;
		std::cout << _age << std::endl;
	}
private:
	int _age;
	char _name[20];

};

//定义一个学生类,使用构造函数进行特定的初始化
class Stu
{
public:
	/*Stu(int age, const char name[20])
	{
		_age = age;
		strcpy(_name, name);
	}*/
	void print()
	{
		std::cout << _name << std::endl;
		std::cout<< _age << std::endl;
		std::cout << "***************************" << std::endl;
		t1.print();
	}
private:
	int _age;
	char _name[20];
	Teacher t1;

};
int main()
{
	Stu s1;
	s1.print();
	return 0;
}

   我们重新定义一个新的类,将其实例化后的对象作为Stu类的成员变量。之后可以验证我们的结论:默认构造函数不会对类当中的内置类型进行初始化,但是会调用自定义类型的构造函数,进而初始化自定义类型。程序验证的结果如下:

   总结一下:构造函数的好处就是可以在生成对象的时候自动调用并匹配,进而达到初始化对象的作用。

2.析构函数

  接下来就来介绍一下和构造函数相对应的析构函数。很容易可以想到有自动调用初始化对象的函数,那是不是就会有自动调用的销毁对象的函数呢?确实存在,也就是我们的析构函数。但是析构函数的作用实际上是针对动态开辟的空间的,也就是自动释放我们在堆区开辟的内存空间。但是与其说是自动释放不如说是在程序结束的时候会自动调用我们析构函数当中所编写好的代码,进而执行相应的操作而已。举一个简单的例子:还记得我们在数据结构章节当中编写过的栈的结构吗?当时是不是需要很多开辟内存的操作?所以在程序运行结束之后为了养成良好的编程习惯我们需要将我们申请的内存还回去。但是会有很多出口,很容易忘记,但是只要有了析构函数就不需要再考虑这个问题了。只要程序一结束那么就会自动的释放内存的空间。示例如下:

typedef int DataType;
class Stack
{
public:
	Stack(size_t capacity = 3)
	{
		_array = (DataType*)malloc(sizeof(DataType) * capacity);
		if (NULL == _array)
		{
			perror("malloc申请空间失败!!!");
			return;
		}
		_capacity = capacity;
		_size = 0;
	}
	void Push(DataType data)
	{
	    CheckCapacity();
		_array[_size] = data;
		_size++;
	}
	~Stack()
	{
		if (_array)
		{
			free(_array);
			_array = NULL;
			_capacity = 0;
			_size = 0;
		}
	}
private:
	DataType* _array;
	int _capacity;
	int _size;
};

 

  这就是我们的析构函数,析构函数和我们的构造函数很相同,函数名都必须和类名相同。作为析构函数的标志是前面很明显的~号。有了析构函数之后我们就不需要再仔细考虑什么时候应该释放空间,以及空间重复释放的问题了。

  作为6大默认构造之一,析构函数同样具有构造函数的特点。如果我们没有自动定义析构函数的话,系统就会自动生成一个默认的析构函数。默认生成的析构函数同样不会对内置类型进行处理,但是会调用我们其他类的析构函数,进行内存的释放操作。

  需要注意的是:析构函数实质上对于内置类型并不会产生太大的影响,所以用的最多的地方就是当我们对内存空间进行动态的申请的时候,可以定义一个析构函数帮助我们释放申请的空间,其他情况下析构函数不定义也没有关系。

  3.拷贝构造

  接着就来认识一下拷贝构造函数:拷贝构造根据定义来说就是使用一个已经初始化好的对象,进行初始化另一个没有初始化的对象。这个过程就好像将我们的对象拷贝了一份一样,顾名思义拷贝构造。同样的我们通过代码进行理解拷贝构造函数:

#include<iostream>

//定义一个类,用于验证我们的拷贝构造函数
class Stu
{
public:
	//定义一个默认构造,用于初始化对象
	Stu(int age = 12, const char name[20] = "张三")
	{
		_age = age;
		strcpy(_name, name);
	}
	//定义一个拷贝构造函数,用于拷贝对象并初始化
	Stu(const Stu& s1)
	{
		_age = s1._age;
		strcpy(_name, s1._name);
	}
	void print()
	{
		std::cout << _age << std::endl;
		std::cout << _name << std::endl;
	}
	//由于我们没有动态开辟内存所以不需要定义析构函数
private:
	int _age;
	char _name[20];
};

int main()
{
	Stu s1(14, "小明");
	s1.print();
	std::cout << "*****调用拷贝构造所产生的新对象*****" << std::endl;
	Stu s2(s1);
	s1.print();
	return 0;
}

  实质上拷贝构造其实还是一个构造函数,所以在定义的要求上和构造函数完全相同,都是函数名必须和类名完全相同,不需要返回值。我们可以认为拷贝构造是构造函数的重载。我们的形参必须是一个自定义类型。我们可以在自定义类型当中进行进一步的变量修改。

  需要注意的是在定义拷贝构造的时候我们的形参必须使用自定义类型的引用,否则就会产生报错,这是因为我们在函数传参的时候会生成一个临时变量,在这个时候就相当于需要调用我们的拷贝构造,但是调用拷贝构造的时候有需要传参,那么就有出现了临时变量的拷贝,就有需要调用拷贝构造函数,进而形成死递归的现象。所以在定义的时候必须使用引用的形式传参。

  同样的,作为默认成员函数。在没有定义的情况下系统同样会自动生成一个拷贝构造供我们使用。但是默认生成的拷贝构造只能进行浅拷贝。也就是对我们内置类型的值进行拷贝。而对于我们自定义类型的拷贝依旧会调用我们相应的拷贝构造。我们可以根据如下代码进行验证:
 

#include<iostream>

//定义一个类,用于验证我们的拷贝构造函数
class Stu
{
public:
	//定义一个默认构造,用于初始化对象
	Stu(int age = 12, const char name[20] = "张三")
	{
		_age = age;
		strcpy(_name, name);
	}
	定义一个拷贝构造函数,用于拷贝对象并初始化
	//Stu(const Stu& s1)                         
	//{                                    //因为成员变量均为内置类型
	//	_age = s1._age;                    //所以可以省去直接调用默认的拷贝构造
	//	strcpy(_name, s1._name);
	//}
	void print()
	{
		std::cout << _age << std::endl;
		std::cout << _name << std::endl;
	}
	//由于我们没有动态开辟内存所以不需要定义析构函数
private:
	int _age;
	char _name[20];
};

int main()
{
	Stu s1(14, "小明");
	s1.print();
	std::cout << "*****调用拷贝构造所产生的新对象*****" << std::endl;
	Stu s2(s1);
	s1.print();
	return 0;
}

  我们上面的代码同样是可以达到我们拷贝的要求的。程序运行的结果如下:

  4.赋值运算符重载函数

   在介绍这一个默认成员函数的时候就不得不向大家提起一个新的概念了:那就是运算符重载函数。我们大家都知道自定义类型,比如数字是可以进行加减以及赋值等运算的。但是我们的自定义类型又该如何进行这一部分的操作呢?想要让我们的自定义类型也可以完成加减以及赋值的运算那么就得编写运算符重载函数了。那么接下来就先来认识一下运算符重载函数,我们照常通过一段代码进行理解:
 

#include<iostream>

//定义一个类,用于编写运算符重载函数
class Stu
{
	//将在全局范围内定义的运算符重载函数定义成为Stu类的友元函数
	//即该函数在全局的范围内也可以使用Stu的私有成员变量
	friend int operator-(Stu s1, Stu s2);
public:
	Stu(int age=12,const char name[20]="张三")
	{
		_age = age;
		strcpy(_name, name);
	}
	void print()
	{
		std::cout << _age << _name<<std::endl;
	}
private:
	int _age;
	char _name[20];
};

//定义一个运算符重载函数
int operator-(Stu s1, Stu s2)
{
	return abs(s1._age - s2._age);
}

int main()
{
	Stu s1(18, "李四");
	Stu s2(12,"王五");
	s1.print();
	s2.print();
	std::cout << "使用运算符重载函数计算两个学生的年龄差" << std::endl;
	std::cout << (s1-s2) << std::endl;
	return 0;
}

  运算符重载函数以operator为标志,后面紧跟着我们想要重载的运算符即可。之后我们在函数体当中编写我们想要进行的操作即可。需要我们注意的是:假如我们的运算符重载函数定义在全局的范围内但是我们又想访问类当中的私有的成员变量的时候我们就可以将该函数设置为友元函数。设置为友元函数之后我们就可以访问指定类当中的私有变量了。(友元函数我们会在下一篇博客当中详细讲解)假如我们不想设置友元函数我们还可以将运算符重载函数设置在类里面进行操作。这样我们就可以通过隐藏的this指针进行访问私有成员变量了。示例代码如下:

#include<iostream>

//定义一个类,用于编写运算符重载函数
class Stu
{
public:
	Stu(int age=12,const char name[20]="张三")
	{
		_age = age;
		strcpy(_name, name);
	}
	void print()
	{
		std::cout << _age << _name<<std::endl;
	}
	int operator-(Stu& s2)
	{
		return abs(_age - s2._age);
	}
private:
	int _age;
	char _name[20];
};

int main()
{
	Stu s1(18, "李四");
	Stu s2(12,"王五");
	s1.print();
	s2.print();
	std::cout << "使用运算符重载函数计算两个学生的年龄差" << std::endl;
	std::cout << (s1-s2) << std::endl;
	return 0;
}

  需要注意的是,将我们的运算符重载函数定义在类当中的时候因为编译器会自动传入一个this指针作为第一个类对象的参数,所以我们只需要将第二个需要比较的对象作为函数的参数传入即可。也就是我们上面的代码所示。程序运行的结果如下:

  在了解过运算符重载函数之后想要理解赋值运算符重载函数也就很简单了。赋值运算符重载函数其实就是运算符重载函数的一种。也就是我们的赋值符号即 = 的重载。赋值运算符的重载写法和我们正常的运算符重载形式完全相同。所示代码如下:

#include<iostream>

//定义一个类,用于编写运算符重载函数
class Stu
{
public:
	Stu(int age=12,const char name[20]="张三")
	{
		_age = age;
		strcpy(_name, name);
	}
	void print()
	{
		std::cout << _age << _name<<std::endl;
	}
	void operator=(Stu& s2)
	{
		_age = s2._age;
		strcpy(_name, s2._name);
	}
private:
	int _age;
	char _name[20];
};

int main()
{
	Stu s1(18, "李四");
	std::cout << "未赋值之前的s1" << std::endl;
	s1.print();
	Stu s2(12,"王五");
	std::cout << "赋值之后的s1" << std::endl;
	s1 = s2;
	s1.print();
	return 0;
}

  使用相同的方法,将我们的赋值运算进行重载,之后就可以为自定义类型进行赋值了。代码所运行的效果如下: 

   赋值重载函数在没有主动定义的时候,其实也可以发挥作用,但是和其他的默认成员函数一样只会对其中的内置类型发挥作用,对于我们的自定义类型会默认调用其中定义好的赋值重载函数。将我们定义的运算符重载函数屏蔽掉再来测试代码的运行:

#include<iostream>

//定义一个类,用于编写运算符重载函数
class Stu
{
public:
	Stu(int age=12,const char name[20]="张三")
	{
		_age = age;
		strcpy(_name, name);
	}
	void print()
	{
		std::cout << _age << _name<<std::endl;
	}
	/*void operator=(Stu& s2)
	{
		_age = s2._age;
		strcpy(_name, s2._name);
	}*/
private:
	int _age;
	char _name[20];
};

int main()
{
	Stu s1(18, "李四");
	std::cout << "未赋值之前的s1" << std::endl;
	s1.print();
	Stu s2(12,"王五");
	std::cout << "赋值之后的s1" << std::endl;
	s1 = s2;
	s1.print();
	return 0;
}

   通过结果我们会发现,即使没有自动定义赋值运算符重载函数程序也可以正常的运行,程序所运行的结果如下:

   5&6普通对象取地址和const对象取地址

  最后这两个默认成员函数其实不需要特别进行注意,不需要特定的进行定义。因为普通的对象取地址得到的是首地址依靠编译器给出的默认的成员函数使用即可。const对象也是相同的。所以我们只需要测试一下,针对自定义类型生成的对象可以正常的取地址并输出即可。测试结果如下:

  那么我们的默认成员函数也就讲解完毕了,我们的类的重难点也就克服了大半,接下来剩下的关于类和对象的其他的使用的细节以及用法我们会在下一篇博客当中继续为大家讲解。 

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

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

相关文章

【lesson1】Linux发展史

文章目录 推动技术进步的基本模式为什么科技一直在进步&#xff1f;科技进步动力是什么&#xff1f; 理解操作系统的发展计算机的发明操作系统的发展Linux的发展 开源为什么有人愿意参加开源项目呢&#xff1f;开源的好处 Linux的应用为什么Windows更好用&#xff1f;OS(操作系…

CGT Asia嘉年华|2023第四届亚洲细胞与基因治疗 创新峰会(广州站)10月升级启航

近年来&#xff0c;全球CGT发展突飞猛进&#xff0c;为遗传罕见病、难治性慢性病和肿瘤患者带来了新的希望&#xff0c;也成为整个国际领域科技竞争的未来焦点。国家发改委发布的《“十四五”生物经济发展规划》明确指出要重点发展基因诊疗、干细胞治疗、免疫细胞治疗等新技术&…

ROS学习——通信机制(常用命令)

2.4 常用命令 Autolabor-ROS机器人入门课程《ROS理论与实践》零基础教程088常用命令简介_Chapter2-ROS通信机制_哔哩哔哩_bilibili 机器人系统中启动的节点少则几个&#xff0c;多则十几个、几十个&#xff0c;不同的节点名称各异&#xff0c;通信时使用话题、服务、消息、参…

Leetcode刷题4

⼆叉树、BFS、堆、Top K、⼆叉搜索树、模拟、图算法 一、二叉树 二叉树的前序中序后序 二叉树节点定义 为了方便演示&#xff0c;我们先定义一个二叉树节点类。 class TreeNode:def __init__(self, val0, leftNone, rightNone):self.val valself.left leftself.right r…

keil5汇编实现1-100累加

.text 用于声明下面的代码都存放在.text文本段 .globl _start 声明_start代码内容属于全局内容_start: 汇编语言的标签&#xff0c;类似于c语言中的函数mov r0,#0 总和mov r1,#1 比较累加的值&#xff0c;次数bl fun 跳转&#xff0c;下一个指令地址保存到lr中cmp r1,#…

LLM微调 | LoRA: Low-Rank Adaptation of Large Language Models

&#x1f525; 发表于论文&#xff1a;(2021) LoRA: Low-Rank Adaptation of Large Language Models &#x1f604; 目的&#xff1a;大模型预训练微调范式&#xff0c;微调成本高。LoRA只微调新增的小部分参数。 文章目录 1、背景2、动机3、LoRA原理4、总结 1、背景 adapter…

【UE4 塔防游戏系列】11-多种类型敌人

目录 效果 前言 步骤 一、创建多种不同类型敌人 二、创建波次 &#xff08;修改游戏模式&#xff09; 效果 前言 我们之前使用的敌人都是同一种敌人&#xff0c;都是名为“Crossbowman”敌人 这里我们根据&#xff08;【UE4 塔防游戏系列】03-创建第一个敌人&#xff09…

学会在重装系统前如何备份软件,再也不怕失去珍贵的应用!

​Windows系统是电脑的重要组成部分&#xff0c;它不仅提供了友好的用户界面&#xff0c;还承担着许多关键的功能和任务&#xff0c;为我们提供了一个稳定、安全和效率的工作环境&#xff0c;使我们能够充分发挥电脑的潜力&#xff0c;优化工作效率和生活品质。 随着系统使…

软件测试面试及笔试题

1、什么是软件测试&#xff1f; 【要点】 在规定条件下对程序进行操作&#xff0c;以发现错误&#xff0c;对软件质量进行评估&#xff0c;包括对软件形成过程的文档、数据以及程序进行测试。 【详解】 软件测试就是在软件投入运行前对软件需求分析、软件设计规格说明书和软…

网络安全/黑客技术—学习笔记

一、什么是网络安全 网络安全可以基于攻击和防御视角来分类&#xff0c;我们经常听到的 “红队”、“渗透测试” 等就是研究攻击技术&#xff0c;而“蓝队”、“安全运营”、“安全运维”则研究防御技术。 无论网络、Web、移动、桌面、云等哪个领域&#xff0c;都有攻与防两面…

【产品经理】小型团队通用工作流程SOP方案

&#xff1a;所谓SOP&#xff0c;即标准作业程序&#xff0c;指将某一事件的标准操作步骤和要求以统一的格式描述出来&#xff0c;用于指导和规范日常的工作。实际执行过程中sop核心是符合本企业并可执行&#xff0c;不流于形式。 一、跨部门工作流程 跨部门流程及职能如下图展…

2.6Checkbutton勾选项

2.6Checkbutton勾选项 运行之后的效果将会像下面的图片一样&#xff0c;此时不作任何操作. 如果只选中第一个选项&#xff0c;即图中的python, 效果就会如下. 如果只选中第二个选项&#xff0c;即图中的c, 效果就会如下. 如果两个选项都选中, 效果就会如下. Checkbutton部…

2023年7月22日(星期六):骑行海囗

2023年7月22日(星期六)&#xff1a;骑行海囗&#xff0c;早8:30到9:00&#xff0c; 大观公园门囗集合&#xff0c;9:30点准时出发 【因迟到者&#xff0c;骑行速度快者&#xff0c;可自行追赶偶遇。】 偶遇地点: 大观公园门囗集合&#xff0c;家住南&#xff0c;东&#xff0…

【第11天】面向对象程序设计_对象的创建,使用,继承

对象的创建及应用 对象的创建 对象可以认为是在一类事物中抽象出某一个特定通过这个特例来处理这类事物出现的问题。在程序语言中通过new关键字来创建对象。前文在讲解构造函数时介绍过每实例化一个对象就会自动调用一次构造函数&#xff0c;实质上这个过程就是创建对象的过程…

实例018 类似windows xp的程序界面

实例说明 在Windows XP环境下打开控制面板&#xff0c;会发现左侧的导航界面很实用。双击展开按钮&#xff0c;导航栏功能显示出来&#xff0c;双击收缩按钮&#xff0c;导航按钮收缩。下面通过实例介绍此种主窗体的设计方法。运行本例&#xff0c;效果如图1.18所示。 ​编辑…

C++ vector容器注意事项

容量&#xff08;capacity&#xff09;和大小&#xff08;size&#xff09;的区别 vector 容器的容量&#xff08;用 capacity 表示&#xff09;&#xff0c;指的是在不分配更多内存的情况下&#xff0c;容器可以保存的最多元素个数&#xff1b;而 vector 容器的大小&#xff…

如何提升环境、生态、水文、土地、土壤、农业、大气等领域的数据分析能力

专题一、空间数据获取与制图 1.1 软件安装与应用讲解 1.2 空间数据介绍 1.3海量空间数据下载 1.4 ArcGIS软件快速入门 1.5 Geodatabase地理数据库 专题二、ArcGIS专题地图制作 2.1专题地图制作规范 2.2 空间数据的准备与处理 2.3 空间数据可视化&#xff1a;地图符号与…

zabbix监控linux主机、监控windows10主机

目录 一、环境准备 1、关闭防火墙 2、准备三台服务器、添加主机声明 3、修改主机名 4、此篇接着上一篇zabbix监控自己的环境下操作&#xff0c;server&#xff08;192.168.147.135&#xff09;已经配置好 二、源码安装zabbix 1、下载包、安装依赖包、联网同步清华时间 2…

vue3组件中使用live2d看板娘(仰望星空的sun)

文章目录 前言思路和实现步骤以及遇到的问题原文章地址My备份Git 相关问题問題解決解決index.html中正常放入canvas問題 前言 因为上个官方包版的canvas总是不能按理想状态更好的控制&#xff0c;渲染在vue中&#xff0c;所以我依然想参考“仰望星空的sun”大佬的笔记以及git开…

Echarts 修改背景颜色、全屏自适应屏幕

修改背景色&#xff1a; 全屏自适应屏幕 首先拿到外面的div的高度 通过DOM获取clientHeight即为无论全屏与否都是DIV的整个高度 在通过高度去做自适应就好了