[C++核心编程-04]----C++类和对象之封装

news2024/11/28 17:44:39

目录

引言    

正文    

01-类和对象简介

02-封装简介

03-封装的意义

04-封装案例之设计学生类

05-封装的权限控制

06-struct和class的区别

07-成员属性设置为私有

08-封装案例1-设计立方体

09-封装案例2-判断点和圆的关系

总结       


引言    

        在C++中,类和对象是面向对象编程的基本概念,而封装则是面向对象编程的三大特征之一,包括封装、继承和多态。封装是指将数据和操作数据的方法封装在类中,通过类的访问控制来保护数据,只能通过类的接口进行访问和操作。

        封装的原理是通过将数据和操作数据的方法封装在类中,然后通过访问控制来保护数据的安全性和合理性。具体原理包括以下几点:

        数据隐藏:封装通过将数据成员声明为私有(private)的,外部无法直接访问这些数据成员,只能通过类的公有(public)接口来访问和修改数据,从而隐藏了类的内部实现细节。

        访问控制:通过类的访问控制修饰符(private、public、protected)来限制对数据成员和成员函数的访问权限。私有成员只能在类的内部访问,公有成员可以在类的外部访问,受保护成员可以在子类中访问。

        封装方法:类中定义的成员函数作为对数据成员的操作方法,外部用户只能通过这些成员函数来间接访问和修改数据成员,确保数据操作的合理性和安全性。

        数据保护:封装能够保护数据,防止外部的非法访问和意外修改数据,通过类的封装将用户和类的实现细节隔离开来。

        代码复用:封装将数据和方法封装在一起,提高了代码的重用性,可以在不同的地方复用类的功能,减少重复编写代码的工作

正文    

01-类和对象简介

        在C++中,类和对象是面向对象编程的基本概念,是实现封装、继承和多态的重要手段。下面详细解释一下类和对象的概念以及它们在C++中的应用:

        (1)类

        a、类是用户定义的数据类型,用来描述对象的属性和行为。通过类可以封装数据和方法,定义了对象的状态和行为。

        b、类的成员包括数据成员和成员函数,数据成员用来描述对象的属性,成员函数用来描述对象的行为。类的成员可以设置为私有(private)、公有(public)或保护(protected)。

        c、类的定义通过关键字class加上类名来声明,类体内包含类的数据成员和成员函数,通常在类定义中把数据成员设置为私有,成员函数设置为公有。

        (2)对象

        a、对象是类的具体实例,是内存中的一个区域,包含了类的数据成员的实际值。同一个类可以创建出多个不同的对象。

        b、使用对象可以调用类的成员函数来操作对象的数据,通过对象访问和修改对象的数据。

        (3)类和对象的关系

        a、类是对象的模板,描述了对象的属性和行为的定义。通过类的构造函数来创建对象。

        b、对象是类的实例,是根据类定义出来的实际存在的实例,实际使用中通过对象来操作数据和方法。

        代码示例如下:在示例代码中,定义了一个Person类,包含了姓名和年龄两个数据成员以及修改数据成员和展示信息的成员函数。通过创建Person类的对象p1,可以设置对象的姓名和年龄,并且展示对象的信息。

#include <iostream>
using namespace std;

class Person {
private:
    string name;
    int age;
    
public:
    void setName(string n) {
        name = n;
    }
    
    void setAge(int a) {
        age = a;
    }
    
    void displayInfo() {
        cout << "Name: " << name << endl;
        cout << "Age: " << age << endl;
    }
};

int main() {
    Person p1;
    p1.setName("Alice");
    p1.setAge(25);
    
    p1.displayInfo();
    
    return 0;
}

02-封装简介

         在C++中,封装是面向对象编程的三大特征之一,包括封装、继承和多态。

        封装的作用

        a、数据隐藏: 封装可以隐藏类的内部实现细节,用户只需了解如何使用类的公共接口而不需要了解其内部实现。

        b、数据保护: 通过访问控制对数据进行限制,只能通过类的方法去访问和修改数据,保证数据的安全合理。

        c、代码复用: 封装将数据和操作数据的方法组合在一起,提高了代码的重用性,可以在不同的地方复用类的功能。

        d、降低耦合度: 封装使得类之间的耦合度降低,各个类之间通过接口进行通信,方便维护和扩展。

        示例代码如下:在代码中,定义了一个Car类,其中brand和year是私有的数据成员,通过公有的成员函数进行访问和修改。这样就实现了数据的封装,外部用户只能通过类的接口来操作数据,无法直接访问或修改私有数据成员。这保证了数据安全性和代码的健壮性。

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

class Car {
private:
    string brand;
    int year;
    
public:
    Car(string b, int y) {
        brand = b;
        year = y;
    }
    
    string getBrand() {
        return brand;
    }
    
    void setBrand(string b) {
        brand = b;
    }
    
    int getYear() {
        return year;
    }
    
    void setYear(int y) {
        year = y;
    }
};

int main() {
    Car myCar("Toyota", 2020);
    cout << "Brand: " << myCar.getBrand() << endl;
    cout << "Year: " << myCar.getYear() << endl;
    
    myCar.setBrand("Honda");
    myCar.setYear(2022);
    
    cout << "New Brand: " << myCar.getBrand() << endl;
    cout << "New Year: " << myCar.getYear() << endl;
    
    return 0;
}

03-封装的意义

         封装作为面向对象编程的重要特征,具有以下意义:

        a、隐藏内部实现细节:封装可以隐藏类的内部实现细节,只暴露给外部用户必要的接口,无需了解类的具体实现细节,从而降低了复杂性,提高了代码的可维护性。

        b、保护数据安全:通过封装,可以将数据成员设置为私有(private),只允许通过类的公共接口来访问和修改数据,避免了外部直接对数据的误操作和破坏。

        c、封装行为和数据:封装将数据和操作数据的方法封装在一起,使得对象不仅具有数据的属性,还具备一系列操作自身数据的方法,实现了数据与行为的整合。

        d、简化复杂性:封装使得编程人员可以将复杂的问题拆解为多个小问题进行处理,降低了维护和调试的难度,提高了代码的可读性。

        e、提高代码的可重用性:封装将相关的数据和行为封装成类,可以在不同的程序中多次使用,避免了代码重复编写,提高了代码的重用性。

        下面给出一个具体的代码分析示例:在代码中,定义了一个Circle类表示圆,私有数据成员radius被封装起来,只能通过公有成员函数来访问和修改。getRadius和setRadius分别用于获取和设置圆的半径,通过setRadius方法限制了半径不能为负数。这样,封装保护了数据的合法性和安全性,避免了不合理的操作。

#include <iostream>
using namespace std;

class Circle {
private:
    double radius;
    
public:
    Circle(double r) : radius(r) {}
    
    double getRadius() {
        return radius;
    }
    
    void setRadius(double r) {
        if (r > 0) {
            radius = r;
        } else {
            cout << "Invalid radius value!" << endl;
        }
    }
    
    double getArea() {
        return 3.14159 * radius * radius;
    }
};

int main() {
    Circle c1(5.0);
    
    cout << "Radius: " << c1.getRadius() << endl;
    cout << "Area: " << c1.getArea() << endl;
    
    c1.setRadius(-2.0);  // 尝试设置负数半径,会触发错误输出
    
    return 0;
}

        下面给出具体代码进行分析函数封装的应用过程,这个示例演示了如何设计一个圆类 Circle,并使用该类计算圆的周长,代码解释如下:

        (1)类定义

        a、使用 class关键字定义了一个类 Circle,其中包含了公共部分和私有部分。

        b、公共部分包括一个整型数据成员 m_r,表示圆的半径,以及一个成员函数calculateZC(),用来计算圆的周长。

        c、私有部分目前为空,因此没有在示例中定义任何私有成员。

        (2)主函数

        a、在 main() 函数中,创建了一个名为 c1的 Circle类型的对象。

        b、将对象的半径设置为 10,然后调用 calculateZC() 成员函数计算圆的周长,并将结果输出到控制台。

        (3)作用分析

        a、这个示例展示了如何使用类来封装数据和行为,并通过对象来访问这些数据和行为。

        b、通过定义一个圆类,将计算周长的方法封装在类的内部,使得外部用户无需了解具体的计算过程,只需调用类的方法即可获取结果。

        类的封装性使得代码更加模块化和易于维护,提高了代码的可读性和可维护性。

#include<iostream>
using namespace std;

const double PI = 3.14; // 使用const修饰变量,设置PI的常量

// 设计一个圆类,求圆的周长
// 圆求周长的公式: 2 * PI *半径

// 设计类和结构体很像,需要class关键字,class就代表一个类,类后面紧跟着的就是类的名称
class Circle   // Circle是一个自定义的名称,见名知意
{
	// 首先设定访问权限,分为公共权限和私有权限

// 公共权限
public:

	// 属性
	// 半径
	int m_r;
	// 行为
	// 获取圆的周长  这里就应该使用函数来代表
	double calculateZC()
	{
		return 2 * PI * m_r;
	}
	
// 私有权限
private:
};

int main()
{
	// 现在有了一个圆的对象,但是还没有一个具体的圆,因此,类似结构体一样,可以定义一个具体的圆的变量名(对象)
	Circle c1;
	c1.m_r = 10;
	cout << "圆的周长: " << c1.calculateZC() << endl;
	system("pause");
	return 0;
}

        示例运行结果如下图所示:

aee2baaa215341a492d1458eac459919.png

04-封装案例之设计学生类

         接下来进行一个封装案例的编写,从实战中分析封装的作用:    

        注:一些专业术语:
        类中的属性和行为,我们统一称为  成员
        属性   又被称为   成员属性或者成员变量
        行为    成员函数或者成员方法   因为行为一般定义的都是函数和方法

        下面这个示例演示了一个简单的学生类 Student的设计以及如何创建学生对象、设置学生信息和展示学生信息。代码具体分析如下:

        (1)类定义

        Student类包含了两个公共属性 m_Name(姓名)和 m_Id(学号),以及三个行为:showStudent() 用于展示学生姓名和学号,setName() 用于设置学生的姓名,setId() 用于设置学生的学号。

        (2)主函数

        在 main() 函数中,首先创建了一个学生对象 s1,通过用 setName() 和 setId() 函数为该学生对象设置姓名和学号,然后调用 showStudent() 函数展示学生信息。

        随后创建了另一个学生对象 s2,直接为该学生对象的属性赋值,不使用设置函数,然后调用 showStudent() 函数展示学生信息。

        (3)作用分析

        通过设计学生类,封装了学生的属性和展示学生信息的行为,实现了数据与行为的整合。

        通过实例化学生对象,可以为不同的学生对象设置不同的信息,实现了对学生数据的管理和操作。

        通过类的成员函数,实现了对学生信息的设置和展示,对外部用户隐藏了类的具体实现细节。

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

// 设计学生类
class Student
{
public:  // 公共权限

	// 属性
	string m_Name;  // 姓名
	int m_Id;  // 学号

	// 行为
	// 显示学生姓名和学号
	void showStudent()
	{
		cout << "姓名: " << m_Name << "  学号" << m_Id << endl;
	}

	// 当然也可以将学生姓名的也作为一个行为输入
	void setName(string name)
	{
		m_Name = name;
	}

	// 当然也可以将学生学号的也作为一个行为输入
	void setId(int Id)
	{
		m_Id = Id;
	}

};


int main()
{

	// 下面就创建一个具体的学生,也叫实例化对象

	Student s1;
//	s1.m_Name = "张三";
	s1.setName("张三");
	s1.setId(1);

	// 显示学生
	s1.showStudent();  // 直接调用类里面的这个函数即可
	
	// 也可以继续在创建一个学生
	Student s2;
	s2.m_Name = "李四";
	s2.m_Id = 2;

	s2.showStudent();


	system("pause");
	return 0;
}

         示例运行结果如下图所示:

ae5f8a07ed4a4f22911009ee9115e195.png

05-封装的权限控制

         封装的权限控制是指在面向对象编程中,通过类的访问修饰符(public、private、protected)来限定类的成员对外部的可见性和可操作性。具体解释如下:

        public权限

        a、公共权限的成员函数和成员变量可以被任何函数访问,即外部可以通过类的对象直接访问和操作这些成员。

        b、公共成员通常用于定义类的接口,提供对外的操作接口,外部用户可以通过这些接口来访问和操作类的数据。

        private权限:        

        a、私有权限的成员函数和成员变量只能在当前类的成员函数中访问和操作,外部无法直接访问。

        b、私有成员通常用于封装类的内部实现细节,保护类的数据,并确保数据的安全性和不变性。

        protected权限

        a、受保护权限的成员函数和成员变量可以被当前类和子类(派生类)的成员函数访问,但是外部无法直接访问。

        b、受保护成员通常用于实现类的继承和派生,子类可以继承和重用受保护成员。

        总结如下:// 公共权限 public       成员  类内可以访问,类外也可以访问;   // 保护权限 protected    成员  类内可以访问,类外不可以访问;   // 私有权限 private      成员  类内可以访问,类外不可以访问。 

        下面是一个具体的代码示例:在示例中,Car类中的model和 price成员变量被设置为私有(private)权限,只能通过公共成员函数进行访问和修改。外部无法直接访问和修改这些私有成员变量,保护了车辆对象的数据安全性。这样,封装通过权限控制实现了数据的保护和隐藏,确保了类的封装性和数据的完整性。

#include <iostream>
using namespace std;

class Car {
private:
    string model;  // 私有成员变量
    double price;   // 私有成员变量
  
public:
    void setModel(string m) {  // 公共成员函数
        model = m;
    }
    
    string getModel() {  // 公共成员函数
        return model;
    }
    
    void setPrice(double p) {  // 公共成员函数
        if (p > 0) {
            price = p;
        } else {
            cout << "Invalid price value!" << endl;
        }
    }
    
    double getPrice() {  // 公共成员函数
        return price;
    }
};

int main() {
    Car myCar;
    myCar.setModel("Toyota");
    myCar.setPrice(25000);
    
    cout << "Car Model: " << myCar.getModel() << endl;
    cout << "Car Price: $" << myCar.getPrice() << endl;
    
    // 以下代码会编译失败,因为 model 和 price 是私有成员
    // cout << myCar.model << endl;
    // myCar.price = 20000;
    
    return 0;
}

        下面这个示例展示了类中成员变量在不同权限修饰符下的访问控制,包括公共(public)、保护(protected)和私有(private)权限。代码具体分析如下:

        (1)类定义

   a、Person类包含了三个成员变量:m_Name(姓名)、m_Car(汽车)、m_Password(银行卡密码),分别使用了不同的权限修饰符:公共、保护和私有。

   b、func() 是一个公共成员函数,可以在外部通过类的实例化对象调用,用来演示在成员函数中访问不同权限下的成员变量。

       (2)权限控制

        a、公共权限的成员变量 m_Name可以在类外部通过对象直接访问。

        b、保护权限的成员变量 m_Car只能在当前类及其子类中访问,无法在类外部直接访问。

        c、私有权限的成员变量 m_Password只能在当前类的成员函数中访问,外部无法直接访问。

        (3)主函数

        a、在 main() 函main数中,创建了一个 Person 类的实例 p1。

        b、尝试在外部通过对象访问不同权限下的成员变量,结果显示了公共成员变量可以访问,而保护和私有成员变量无法直接访问。

        c、通过调用 func() 成员函数,实现了在类内部访问不同权限下的成员变量,因为成员函数可以访问类的所有成员。

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

class Person
{
public:    
	string m_Name;// 姓名

protected:
	// 保护权限
	string m_Car;    // 汽车

private:
	// 私有权限
	int m_Password;  // 银行卡密码
// 测试一下是否可以类内访问

public:    // 当使用public 时是可以访问到这个函数的
	
	void func()   // 此时编译是没有任何问题的,
	{
		m_Name = "张三";
		m_Car = "宝马";
		m_Password = 12345;
	}
};

int main()
{
	Person p1;
	p1.m_Name = "李四";   // 这里可以访问
// 	p1.m_Car = "奔驰";    //  无法访问 protected 成员(在“Person”类中声明)在编译时将会报错
//	p1.m_Password = 123;  //  无法访问 protected 成员(在“Person”类中声明)在编译时将会报错
	// func(); 这里直接调用函数,肯定是无法运行的,因为这个函数在Person类中,只能使用实例化对象 p1 调用 
	p1.func();
	system("pause");
	return 0;
}

06-struct和class的区别

         对于 C++ 中的struct和 class时,主要的区别在于默认的访问权限(也称为数据封装性)以及成员函数的默认继承属性。具体解释如下:

        (1)默认访问权限

        a、对于 struct,其默认访问权限为公共(public),在 struct中定义的数据成员和成员函数可以被外部直接访问和操作。

        b、对于class,其默认访问权限为私有(private),在 class中定义的成员通常是私有的,需要通过公共的成员函数来访问和操作数据。

        (2)默认继承属性

        a、在继承方面,与 struct相比,class拥有默认的私有继承(private)。这意味着从 class继承的成员在派生类中默认为私有成员,无法直接访问。

        b、而从 struct继承的成员在派生类中默认为公共成员,可以直接访问和操作。

        下面是一个具体的代码示例:在这个示例中,我们可以看到 StudentStruct使用了 struct定义,可以直接访问成员变量 name和 id;而 StudentClass使用了 class定义,需要通过公共函数来访问成员变量 name和 id。这展示了 struct和 class的默认访问权限和默认继承属性的不同之处。

#include <iostream>
using namespace std;

// 使用 struct 定义学生结构体
struct StudentStruct {
    string name;
    int id;
};

// 使用 class 定义学生类
class StudentClass {
public:
    string name;
    int id;
};

int main() {
    // struct 可直接访问成员变量
    StudentStruct s1;
    s1.name = "Alice";
    s1.id = 123;

    // class 需要通过公共函数访问成员变量
    StudentClass s2;
    s2.name = "Bob";
    s2.id = 456;

    cout << "Student using struct: " << s1.name << " " << s1.id << endl;
    cout << "Student using class: " << s2.name << " " << s2.id << endl;

    return 0;
}

        下面这个示例展示了 class和 struct在默认访问权限上的区别,代码具体分析如下:

        (1)类定义

   a、C1是一个类(class),定义了一个公共成员变量 m_A。在类中,默认的成员访问权限是私有(private),所以外部无法直接访问 m_A。

   b、C2是一个结构体(struct),同样定义了一个成员变量 m_A。在结构体中,默认的成员访问权限是公共(public),所以外部可以直接访问 m_A。

        (2)主函数

        a、在 main() 函数中,尝试实例化类 C1和结构体 C2的对象 c1和 c2。

        b、由于 C1类的默认访问权限是私有,所以在外部无法直接访问类的成员变量 m_A,导致编译错误。

        c、相反,结构体 C2的成员变量 m_A默认为公共,可以直接在外部访问和操作。

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

class C1
{
public:
	int m_A;  // 默认权限是私有的
};
struct C2
{
	int m_A;  // 默认是公有的
}; 

int main()
{
	// struct和class的区别
	// struct的默认访问权限是公有的
	// class的默认访问权限是私有的
	C1 c1;
//	c1.m_A = 100;   //这里无法执行,无法访问
	C2 c2;
	c2.m_A = 100;

	system("pause");
	return 0;
}

07-成员属性设置为私有

         当将类中的成员属性设置为私有(private)时,意味着这些属性只能在该类的成员函数中直接访问,外部代码无法直接访问或修改这些私有属性。通过将属性设置为私有,可以增强数据的封装性和安全性,防止外部代码直接操作敏感数据。只能通过类的公共函数(成员函数)来间接访问或修改私有属性,从而实现对数据的控制和保护:

        以下是使用私有成员属性的具体代码示例:

        在这个示例中,Person类中的 name和 age属性被设置为私有属性。外部代码无法直接访问 name和 age,而是通过公共函数 setName和 setAge来间接操作这些私有属性。这样可以确保数据安全和逻辑正确性。

        通过设置成员属性为私有,类的实现细节被隐藏,只有通过提供的接口(成员函数)来访问数据,从而实现更好的封装性和安全性。

#include <iostream>
using namespace std;

class Person {
private:    // 将成员变量设置为私有权限
    string name;
    int age;

public:
    // 公共函数用于访问和设置私有属性
    void setName(string n) {
        name = n;
    }

    void setAge(int a) {
        if (a >= 0)
            age = a;
        else
            cout << "Invalid age input! Age must be non-negative." << endl;
    }

    void displayInfo() {
        cout << "Name: " << name << ", Age: " << age << endl;
    }
};

int main() {
    Person p1;
    // 不能直接访问私有属性,必须通过公共函数进行操作
    // p1.name = "Alice";  // 编译错误,无法直接访问私有属性

    p1.setName("Alice");
    p1.setAge(25);
    p1.displayInfo();

    return 0;
}

        下面给出具体代码展示如何在 C++ 中定义一个 Person类并设置不同类型的成员属性权限,代码解释如下:

        注:成员属性设置为私有的两点好处:1、可以自己控制读写权限;2、对于写可以检测数据的有效性

        (1)类定义

   a、Person类包含了三个成员属性:m_Name, m_Age, 和 m_Lover

   b、m_Name在类中被设置为私有属性,但提供了公共的 setName和 getName函数来实现对姓名的读写操作;

   c、m_Age也是私有属性,但提供了只读的 getAge函数和可写的 setAge函数,用于实现对年龄的读取和设置,并限定年龄在范围 (0~150) 内;

   d、m_Lover是只写属性,无法通过公共函数直接读取,只有 setLover函数可用于设置情人的名字。

        (2)主函数

         a、在 main() 函数中,实例化了一个 Person类对象 p

   b、由于成员属性被设置为私有,在外部无法直接访问 m_Name, m_Age, 或 m_Lover属性。通过调用公共函数 setName和 getName可以完成对姓名的读写操作;

   c、setAge和 getAge可以完成对年龄的操作;setLover可以设置情人的名字,但无法直接读取。

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

// 设置人的一个类
class Person
{

public:
	//设置姓名
	void setName(string name)
	{
		m_Name = name;
	}

	// 获取姓名
	string getName()   // 这里直接返回一个string类型的字符串姓名
	{
		return m_Name;
	}
	
	// 获取年龄   改成可读可写,年龄范围(0~150之间)
	int getAge()
	{
	//	m_Age = 0;
		return m_Age;
	}
	// 设置年龄
	void setAge(int age)
	{
		if (age<0||age>150)
		{
			m_Age = 0;
			cout << "输入错误" << endl;
			return;
		}
		m_Age = age;
	}
	// 设置情人,只写,不读
	void setLover(string name)
	{
		m_Lover = name;
	}

private:
	// 姓名    可读可写
	string m_Name;
	// 年龄    只读
	int m_Age; 
	// 情人    只写
	string m_Lover;

};

int main()
{

	Person p;
// 	p.m_Name = "张三";   //现在所有的成员属性都在私有中,无法进行访问
	p.setName("张三");
	cout << "姓名为: " << p.getName() << endl;  // 现在姓名就可以完成读写操作
	
	p.setAge(10);
	cout << "年龄为: " << p.getAge() << endl;   //年龄是一只读的参数,因此无法设置年龄

	p.setLover("无语");  // 这里是无法显示的,因为这个名字只能写入,而不能读取

	system("pause");
	return 0;
}

08-封装案例1-设计立方体

         接下来进行封装案例代码的编写,从实战中分析封装的作用:

        分为三个步骤完成:a、设计立方体类(Cube);b、求出立方体的面积和体积;c、分别用全局函数和成员函数判断两个立方体是否相等。

        下面这个示例演示了一个名为 Cube 的类,该类表示立方体,并提供了一些方法来设置其长、宽、高,计算表面积和体积,并且提供了两种方法来判断两个立方体是否相等。下面对代码进行详细解释如下:

        (1)类定义

   a、Cube类包含了三个私有成员属性 m_L、m_W 和 m_H,分别表示立方体的长、宽和高。

         b、提供了公共的成员函数 setL, getL, setW, getW, setH, getH 来设置和获取长、宽和高。

         c、提供了公共的成员函数 calculateS 和 calculateV 来计算立方体的表面积和体积。

         d、提供了一个成员函数 isSameByClass 用于判断当前立方体对象和另一个立方体对象是否相等。

        (2)全局函数

         a、提供了一个全局函数 isSame,接受两个立方体对象作为参数,用于判断这两个立方体对象是否相等。

        (3)主函数

         a、在 main() 函数中,首先创建了两个 Cube 类的对象 C1 和 C2,分别表示两个立方体,并设置它们的长、宽和高。

         b、调用了成员函数 calculateS 和 calculateV 分别计算了 C1 的表面积和体积,并将结果输出。        

         c、利用全局函数 isSame 判断了 C1 和 C2 是否相等,并输出结果。

         d、利用成员函数 isSameByClass 也判断了 C1 和 C2是否相等,并输出结果。

#include <iostream>
using namespace std;

class Cube
{
public:
	//设置长
	void setL(int L)
	{
		m_L = L;
	}
	//获取长
	int getL()
	{
		return m_L;
	}
	//设置宽
	void setW(int W)
	{
		m_W = W;
	}
	//获取宽
	int getW()
	{
		return m_W;
	}
	//设置高
	void setH(int H)
	{
		m_H = H;
	}
	//获取高
	int getH()
	{
		return m_H;
	}

	//获取立方体面积
	int calculateS()
	{
		return 2 * (m_L*m_W + m_L*m_H + m_W*m_H);
	}
	// 获取立方体体积
	int calculateV()
	{
		return m_L*m_W*m_H;
	}

	//利用成员函数判断C1和C2是否相等
	bool isSameByClass(Cube &C)  // 这里只需要一个参数即可,因为默认的已经有一个成员了,再加入一个去比较
	{
		if (m_L == C.getL() && m_W == C.getW() && m_H == C.getH())
		{
			return true;
		}
		return false;

	}
	  
private:    //设置为私有的,方便使用
	//长
	int m_L;
	//宽
	int m_W;
	//高
	int m_H;
};

// 利用全局函数判断两个立方体是否相等,可以返回bool值,真和假
bool isSame(Cube &C1,Cube &C2)
{
	if (C1.getL()==C2.getL()&&C1.getW() == C2.getW()&&C1.getH()==C2.getH())
	{
		return true;
	}
	return false;

}

int main()
{
	Cube C1;   
	C1.setL(10);
	C1.setW(10);
	C1.setH(10);

	cout << "C1的面积为: " << C1.calculateS() << endl;
	cout << "C1的体积为: " << C1.calculateV() << endl;

	// 创建第二个立方体
	Cube C2;
	C2.setL(10);
	C2.setW(10);
	C2.setH(11);
	//利用全局函数判断
	bool ret = isSame(C1, C2);
	if (ret)
	{
		cout << "全局函数判断: C1和C2相等" << endl;
	} 
	else
	{
		cout << "全局函数判断: C1和C2不相等" << endl;
	}

	// 利用成员函数判断
	ret = C1.isSameByClass(C2);
	if (ret)
	{
		cout << "成员函数判断: C1和C2相等" << endl;
	}
	else
	{
		cout << "成员函数判断: C1和C2不相等" << endl;
	}
	system("pause");
	return 0;
}

09-封装案例2-判断点和圆的关系

         接下来进行封装案例代码的编写,从实战中分析封装的作用:

        分为三个文件完成:a、圆的.cpp和.h文件;b、点的.cpp和.h文件;c、主函数文件。

        Circle.cpp文件代码如下:

#include "Circle.h"

// 设置半径
void Circle::setR(int r)
{
	m_R = r;
}
// 获取半径
int Circle::getR()
{
	return m_R;
}

// 设置圆心
void Circle::setCenter(Point center)
{
	m_Center = center;
	
}
// 获取圆心
Point Circle::getCenter()
{
	return m_Center;
}

        Circle.h文件代码如下:

#pragma once    // 这句代码是防止头文件重复包含使用的
#include <iostream>
using namespace std;
#include "point.h"

// 设计圆的类
class Circle
{
public:
	// 设置半径
	void setR(int r);
	
	// 获取半径
	int getR();


	// 设置圆心
	void setCenter(Point center);
	
	// 获取圆心
	Point getCenter();

private:
	// 圆的半径
	int m_R;
	// 圆的坐标中心   这里可以再建立一个关于点的类
	Point m_Center;   // 在类中又建立了另一个类,在本类中作为一个成员存在
};

         point.cpp文件代码如下:

#include "point.h"

void Point::setX(int x)
{
	m_X = x;
}
// 获取X坐标
int Point::getX()
{
	return m_X;
}
// 设置Y坐标
void Point::setY(int y)
{
	m_Y = y;
}
// 获取X坐标
int Point::getY()
{
	return m_Y;
}

         point.h文件代码如下:

#pragma once
#include <iostream>
using namespace std;

class Point
{
public:
	//设置成员函数
	// 设置X坐标
	void setX(int x);
	
	// 获取X坐标
	int getX();

	// 设置Y坐标
	void setY(int y);
	
	// 获取X坐标
	int getY();

private:
	// X坐标
	int m_X;
	// Y坐标
	int m_Y;
};

        main主函数文件代码如下:

#include <iostream>
using namespace std;
#include "point.h"
#include "Circle.h"

// 判断点和圆的关系,最好写成全局函数
void isInCircle(Circle &c, Point &p)
{
	// 计算两点之间的距离
	int distance =
		(c.getCenter().getX() - p.getX())*(c.getCenter().getX() - p.getX()) +
		(c.getCenter().getY() - p.getY())*(c.getCenter().getY() - p.getY());
	int rDistance = c.getR()*c.getR();
	if (distance  == rDistance)
	{
		cout << "点在圆上" << endl;
	}
	else if(distance>rDistance)
	{
		cout << "点在圆外" << endl;
	}
	else
	{
		cout << "点在圆内" << endl;
	}
}

int main()
{

	Circle c1;
	c1.setR(10);
	// 创建圆心
	Point center;
	center.setX(10);
	center.setY(0);
	c1.setCenter(center);

	// 创建点
	Point p;
	p.setX(10);
	p.setY(11);

	isInCircle(c1, p);

	system("pause");
	return 0;
}

        案例运行结果如下图所示:

ca4a8e224dcc48eb91416eee20322d26.png

总结    

        在C++中,类和对象是面向对象编程的基本概念,是实现封装、继承和多态的重要手段。封装是面向对象编程中的一项重要特性,它将数据和操作封装在一个单元(类)中,隐藏了内部的细节,只向外部提供必要的接口来访问和操作数据。封装通过访问控制(public、private、protected)来实现数据的保护和安全性,同时实现数据隐私性和信息隐藏。

        封装是面向对象编程的重要特性,它通过将数据和行为整合在一起,实现了数据的隐藏和保护,是实现面向对象编程的核心概念之一。

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

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

相关文章

NVIDIA TensorRT Model Optimizer

NVIDIA TensorRT Model Optimizer NVIDIA TensorRT 模型优化器&#xff08;ModelOpt&#xff09;是一个用于优化 AI 模型的库&#xff0c;它通过量化和稀疏性技术减小模型大小并加速推理&#xff0c;同时保持模型性能。ModelOpt 支持多种量化格式和算法&#xff0c;包括 FP8、…

深度学习知识点全面总结

ChatGPT 深度学习是一种使用神经网络来模拟人脑处理数据和创建模式的机器学习方法。下面是深度学习的一些主要知识点的总结&#xff1a; 1. 神经网络基础&#xff1a; - 神经元&#xff1a;基本的计算单元&#xff0c;模拟人脑神经元。 - 激活函数&#xff1a;用于增加神…

力扣HOT100 - 763. 划分字母区间

解题思路&#xff1a; class Solution {public List<Integer> partitionLabels(String s) {int[] last new int[26];int len s.length();for (int i 0; i < len; i) {last[s.charAt(i) - a] i;//记录字母最远的下标}List<Integer> partition new ArrayList…

大数据在IT行业的应用与发展趋势及IT行业的现状与未来

大数据在IT行业中的应用、发展趋势及IT行业的现状与未来 一、引言 随着科技的飞速发展&#xff0c;大数据已经成为IT行业的重要驱动力。从数据收集、存储、处理到分析&#xff0c;大数据技术为各行各业带来了深远的影响。本文将详细探讨大数据在IT行业中的应用、发展趋势&#…

做抖店如何提高与达人合作的几率?有效筛选+有效推品

我是王路飞。 总是有很多新手商家&#xff0c;找我吐槽&#xff0c;抖音上的达人特别不好找&#xff0c;好不容易加上了&#xff0c;要么是发消息不回复&#xff0c;要么是寄样后就没下文了。 虽然一直都说找达人带货玩法比较简单&#xff0c;但也离不开电商的基本逻辑&#…

【k8s多集群管理平台开发实践】九、client-go实现nginx-ingress读取列表、创建ingress、读取更新yaml配置

文章目录 简介 一.k8s的ingress列表1.1.controllers控制器代码1.2.models模型代码 二.创建ingress2.1.controllers控制器代码2.2.models模分代码 三.读取和更新ingress的yaml配置3.1.controllers控制器代码3.2.models模型代码 四.路由设置4.1.路由设置 五.前端代码5.1.列表部分…

低血压怎么办?低血压患者应该如何调理?

点击文末领取揿针的视频教程跟直播讲解 低血压在生活中也是一种常见的问题&#xff0c;低血压的朋友常有头晕眼黑、冒冷汗等症状&#xff0c;对工作学习产生了一定的影响。 什么是低血压呢&#xff1f; 低血压是指体循环动脉压力低于正常的状态。即血压低于正常水平。 ​一般…

LearnOpenGL(十四)之模型加载

Model类的结构&#xff1a; class Model {public:/* 函数 */Model(char *path){loadModel(path);}void Draw(Shader shader); private:/* 模型数据 */vector<Mesh> meshes;string directory;/* 函数 */void loadModel(string path);void processNode(aiNode …

初识指针(5)<C语言>

前言 在前几篇文章中&#xff0c;已经介绍了指针一些基本概念、用途和一些不同类型的指针&#xff0c;下文将介绍某些指针类型的运用。本文主要介绍函数指针数组、转移表&#xff08;函数指针的用途&#xff09;、回调函数、qsort使用举例等。 函数指针数组 函数指针数组即每个…

京东h5st4.7逆向分析

声明 本文章中所有内容仅供学习交流使用&#xff0c;不用于其他任何目的&#xff0c;不提供完整代码&#xff0c;抓包内容、敏感网址、数据接口等均已做脱敏处理&#xff0c;严禁用于商业用途和非法用途&#xff0c;否则由此产生的一切后果均与作者无关&#xff01; 本文章未…

信息量、熵、KL散度、交叉熵概念理解

信息量、熵、KL散度、交叉熵概念理解 (1) 信息量 信息量是对事件的不确定性的度量。 假设我们听到了两件事&#xff0c;分别如下&#xff1a;事件A&#xff1a;巴西队进入了世界杯决赛圈。 事件B&#xff1a;中国队进入了世界杯决赛圈。仅凭直觉来说&#xff0c;显而易见事件…

SpringAMQP-消息转换器

这边发送消息接收消息默认是jdk的序列化方式&#xff0c;发送到服务器是以字节码的形式&#xff0c;我们看不懂也很占内存&#xff0c;所以我们要手动设置一下 我这边设置成json的序列化方式&#xff0c;注意发送方和接收方的序列化方式要保持一致 不然回报错。 引入依赖&#…

STM32_HAL_TIM_1介绍

1.F1的定时器类型&#xff08;高的拥有低级的全部功能&#xff09; 高级定时器&#xff08;TIM1和TIM8&#xff09;&#xff1a; 16位自动重装载计数器。支持多种工作模式&#xff0c;包括中心对齐模式、边沿对齐模式等。可以产生7个独立的通道&#xff0c;用于PWM、输出比较、…

Cosmo Bunny Girl

可爱的宇宙兔女郎的3D模型。用额外的骨骼装配到Humanoid上,Apple混合了形状。完全模块化,包括不带衣服的身体。 技术细节 内置,包括URP和HDRP PDF。还包括关于如何启用URP和HDRP的说明。 LOD 0:面:40076,tris 76694,verts 44783 装配了Humanoid。添加到Humanoid中的其他…

测试用例编写规范

1.1目的 统一测试用例编写的规范&#xff0c;为测试设计人员提供测试用例编写的指导&#xff0c;提高编写的测试用例的可读性&#xff0c;可执行性、合理性。为测试执行人员更好执行测试&#xff0c;提高测试效率&#xff0c;最终提高公司整个产品的质量。 1.2使用范围 适用…

数字人实训室助推元宇宙人才培养

如今&#xff0c;全身动作捕捉设备已经大量应用在影视、动画、游戏领域&#xff0c;在热门的元宇宙内容领域中&#xff0c;全身动作捕捉设备逐步发挥着重要的作用&#xff0c;在包括体育训练、数字娱乐虚拟偶像、虚拟主持人、非物质文化遗产保护等等场景&#xff0c;数字人实训…

第5章 处理GET请求参数

1 什么是GET请求参数 表单GET请求参数是指在HTML表单中通过GET方法提交表单数据时所附带的参数信息。在HTML表单中&#xff0c;可以通过表单元素的name属性来指定表单字段的名称&#xff0c;通过表单元素的value属性来指定表单字段的值。当用户提交表单时&#xff0c;浏览器会将…

【数据结构】有关栈和队列相互转换问题

文章目录 用队列实现栈思路实现 用栈实现队列思路实现 用队列实现栈 Leetcode-225 用队列实现栈 思路 建立队列的基本结构并实现队列的基本操作 这部分这里就不多说了&#xff0c;需要的可以看笔者的另一篇博客 【数据结构】队列详解(Queue) 就简单带过一下需要实现的功能 …

金融业开源软件应用 评估规范

金融业开源软件应用 评估规范 1 范围 本文件规定了金融机构在应用开源软件时的评估要求&#xff0c;对开源软件的引入、维护和退出提出了实现 要求、评估方法和判定准则。 本文件适用于金融机构对应用的开源软件进行评估。 2 规范性引用文件 下列文件中的内容通过文中的规范…

数据科学:使用Optuna进行特征选择

大家好&#xff0c;特征选择是机器学习流程中的关键步骤&#xff0c;在实践中通常有大量的变量可用作模型的预测变量&#xff0c;但其中只有少数与目标相关。特征选择包括找到这些特征的子集&#xff0c;主要用于改善泛化能力、助力推断预测、提高训练效率。有许多技术可用于执…