第三节-类与对象(中)

news2024/11/18 7:41:03

1.类的6个默认成员函数

如果一个类中什么成员都没有,简称为空类(空类大小为1)

空类中真的什么都没有吗?并不是,任何类在什么都不写时,编译器会自动生成以下6个默认成员函数。

默认成员函数:用户没有显式实现,编译器会生成的成员函数称为默认成员函数。

class Date {};

 


 

2. 构造函数


构造函数是用于初始化对象的特殊成员函数。虽然名称为“构造”但它的主要任务是初始化对象的成员变量,而不是为对象分配内存。构造函数的使用对于确保对象在创建时处于有效状态至关重要。

2.1 函数名与类名相同


构造函数的名字必须与类名相同。这是C++的语法要求

解释:构造函数的名字与类名相同,使得编译器能够识别它是用于初始化对象的函数,而不是普通的成员函数。
示例:
class MyClass {
public:
    MyClass() { /* 构造函数体 */ }
}

 2.2 无返回值


构造函数没有返回值,甚至不能声明void。这是C++语言规定的,构造函数的唯一目的是初始化对象,因此不需要返回任何值。

解释:构造函数的任务是初始化对象,而不是返回数据。返回值的存在会违背构造函数的设计目的。
示例:

class MyClass {
public:
    MyClass() { /* 构造函数体,无返回值 */ }
};

2.3 对象实例化时系统会自动调用


构造函数在对象实例化时自动调用。开发者不需要显式调用构造函数,编译器会在对象创建时自动执行它。

解释:构造函数的自动调用确保了对象在创建时立即处于有效状态。无论对象是作为局部变量、全局变量还是动态分配的变量,构造函数都会在创建时运行。

示例:
MyClass obj; // 调用构造函数

2.4 构造函数可以重载


构造函数可以重载,即同一个类中可以有多个构造函数,它们的参数列表必须不同。这允许对象在创建时根据不同的需求进行不同的初始化。

解释:通过构造函数的重载,可以灵活地初始化对象。例如,一个类可以有无参构造函数和带参构造函数,以满足不同的初始化需求。
示例:
class MyClass {
public:
    MyClass() { /* 无参构造函数 */ }
    MyClass(int x) { /* 带参构造函数 */ }
};


2.5 默认构造函数的生成规则


如果类中没有显式定义构造函数,编译器会自动生成一个无参的默认构造函数。但一旦用户定义了任何构造函数,编译器就不再生成默认构造函数。

解释:默认构造函数提供了一个基本的初始化方式。如果用户定义了其他形式的构造函数(如带参数的),编译器认为用户不再需要默认构造函数,因此不会自动生成


示例:
class MyClass {
    // 没有显式定义构造函数,编译器会生成默认的无参构造函数
};
MyClass obj; // 调用默认构造函数


2.6 无参构造函数与全缺省构造函数的关系


无参构造函数、全缺省构造函数、默认生成的构造函数不能同时存在。如果定义了无参构造函数或全缺省构造函数,编译器将不会再生成默认构造函数。

解释:这三种构造函数提供了相似的功能,即初始化对象而不需要显式提供参数。为了避免冲突,它们只能存在其中之一。


示例:
class MyClass {
public:
    MyClass() { /* 无参构造函数 */ }
    // MyClass(int x = 0) { /* 全缺省构造函数,不能与无参构造函数同时存在 */ }
};


注意:无参构造函数、全缺省构造函数、编译器自动默认生成的构造函数全都叫默认构造函数!!!总结来说,可以不传参的就是默认构造函数,这三个不能同时存在,但是默认构造函数可以和带参的构造函数同时存在(即上文所说的函数重载),例如半缺省、没有缺省值等的普通构造函数。

#include <iostream>
using namespace std;

class MyClass {
public:
    // 全缺省构造函数,即默认构造函数
    MyClass(int x = 10, int y = 20) {
        cout << "Called MyClass(int x = 10, int y = 20)" << endl;
        cout << "x = " << x << ", y = " << y << endl;
    }

    // 普通带参构造函数(没有缺省值)
    MyClass(double z) {
        cout << "Called MyClass(double z)" << endl;
        cout << "z = " << z << endl;
    }
};
int main() {
    MyClass obj1;           // 调用全缺省构造函数
    MyClass obj2(100);      // 调用全缺省构造函数,因为100是int类型
    MyClass obj3(3.14);     // 调用普通的带参构造函数
    return 0;
}

 2.7 内置类型与自定义类型成员变量的初始化


如果是编译器自动生成的默认构造函数对内置类型成员变量的初始化没有要求,其值不确定。对于自定义类型的成员变量,编译器会调用它们的默认构造函数进行初始化。

解释:内置类型(如int、char)的成员变量如果没有显式初始化,其值可能是未定义的。自定义类型的成员变量则必须通过其默认构造函数初始化。

示例:(这里是初始化列表,在这个系列的之后博客会讲到)
class MyClass {
public:
    MyClass() : _value(0) { /* 初始化内置类型成员变量 */ }
private:
    int _value;
    std::string _name;// 自动调用std::string的默认构造函数
}:
示例代码梳理
以下是展示上述特点的详细代码示例:

这里只是为了方便写的示例哈,如前文所述,无参和全缺省的默认构造函数只能存在一种

#include<iostream>
using namespace std;

class Date {
public:
    // 1. 无参构造函数
    Date() {
        _year = 1;
        _month = 1;
        _day = 1;
    }

    // 2. 带参构造函数
    Date(int year, int month, int day) {
        _year = year;
        _month = month;
        _day = day;
    }

    // 3. 全缺省构造函数
    Date(int year = 1, int month = 1, int day = 1) {
        _year = year;
        _month = month;
        _day = day;
    }

    void Print() {
        cout << _year << "/" << _month << "/" << _day << endl;
    }

private:
    int _year;
    int _month;
    int _day;
};

int main() {
    // 调用无参构造函数
    Date d1;

    // 调用带参构造函数
    Date d2(2025, 1, 1);

    // 调用全缺省构造函数
    Date d3;

    d1.Print();
    d2.Print();

    return 0;
}


通过这个详细的解析和示例代码,我们可以清晰地理解C++类的默认成员函数和构造函数的特点及其作用。这样,开发者可以根据具体需求灵活地使用和自定义这些函数,以便更好地控制对象的生命周期和资源管理。

3. 析构函数


析构函数是与构造函数功能相反的一个函数,它用于在对象生命周期结束时释放资源。C++中规定,析构函数会在对象销毁时自动调用,以完成对象中资源的清理工作。这一特性使得C++能够有效地管理内存和其他资源,防止资源泄漏。

3.1 析构函数名


析构函数的名称是类名的前面加上一个“~”符号,这是C++语法中的规定。它用于明确表示该函数是析构函数。

解释:析构函数的名字与类名相同,但在前面加上“~”符号,这使得编译器能够识别这是一个析构函数.

示例:
class MyClass {
public:
    ~MyClass() {
        // 析构函数体
    }
};


3.2无参数无返回值


析构函数不接受任何参数,也没有返回值。它的唯一任务是清理对象的资源。

解释:由于析构函数是系统自动调用的,因此它不能有参数,也不需要返回任何值.

示例:
class MyClass {
public:
    ~MyClass() {
        // 无参数,无返回值
    }
};


3.3一个类只能有一个析构函数


每个类只能定义一个析构函数。如果类中没有显式定义析构函数,系统会自动生成一个默认的析构函数。

解释:C++规定,一个类只能有一个析构函数,因为一个对象只能在生命周期结束时被销毁一次。

示例:
class MyClass {
public:
    ~MyClass() {
        // 只能有一个析构函数
    }
};


3.4 自动调用析构函数


当一个对象的生命周期结束(如对象超出作用域或显式删除对象)时,系统会自动调用析构函数来清理资源。

解释:析构函数的自动调用确保了对象在被销毁时可以正确地释放资源,防止资源泄漏。

示例:
class MyClass {
public:
    ~MyClass() {
        cout << "Object is being destroyed" << endl;
    }
};

int main() {
    MyClass obj; // 当obj超出作用域时,系统会自动调用析构函数
    return 0;
}


3.5 析构函数对内置类型成员


如果类中没有显式定义析构函数,编译器会自动生成一个默认析构函数。这个默认析构函数对内置类型的成员变量不做任何处理。

解释:对于内置类型(如int、char等),默认析构函数不需要释放资源。但对于自定义类型的成员,编译器生成的析构函数会调用这些成员的析构函数。

示例:
class MyClass {
private:
    int _value;
public:
    // 编译器自动生成的析构函数对内置类型不做处理
};


3.6 显式写析构函数情况


如果显式定义了析构函数,对于自定义类型的成员变量,它们的析构函数也会被自动调用。

解释:当显式定义析构函数时,C++确保所有自定义类型的成员都会在对象销毁时调用其析构函数,正确地释放资源。

示例:
class MyClass {
private:
    std::string _name; // 自定义类型成员
public:
    ~MyClass() {
        // 自定义类型的成员变量会自动调用其析构函数
    }
};


3.7 析构函数可以不写的情况


如果类中没有动态分配的资源或其他需要手动释放的资源,可以不显式定义析构函数,使用编译器生成的默认析构函数。

解释:对于没有动态资源的类,编译器生成的析构函数已经足够使用,不需要额外的析构逻辑。

示例:
class MyClass {
private:
    int _value; // 没有动态资源,编译器生成的析构函数已足够
};


3.8局部析构顺序


在一个局部作用域内定义的多个对象,C++规定后定义的对象会先调用析构函数。

解释:这一规则确保了对象按照“后进先出”的顺序销毁,符合栈的逻辑。

示例:
class MyClass {
public:
    ~MyClass() {
        cout << "Destructor called" << endl;
    }
};

int main() {
    MyClass obj1;
    MyClass obj2;
    // obj2会在obj1之前被销毁
    return 0;
}


示例代码梳理
以下是完整展示上述析构函数特点的详细代码示例:

#include<iostream>
using namespace std;

typedef int STDataType;

class Stack {
public:
    Stack(int n = 4) {
        _a = (STDataType*)malloc(sizeof(STDataType) * n);
        if (_a == nullptr) {
            perror("malloc申请空间失败");
            return;
        }
        _capacity = n;
        _top = 0;
    }

    ~Stack() { // 自定义析构函数,释放动态分配的内存
        cout << "~Stack()" << endl;
        free(_a);
        _a = nullptr;
        _top = _capacity = 0;
    }

private:
    STDataType* _a;
    size_t _capacity;
    size_t _top;
};

// 两个Stack实现队列
class MyQueue {
public:
    MyQueue() : pushst(), popst() {}

    // 默认析构函数自动调用两个Stack成员的析构函数
    // 显式定义的析构函数,也会自动调用Stack成员的析构函数
    /*~MyQueue() {}*/

private:
    Stack pushst;
    Stack popst;
};

int main() {
    Stack st;

    MyQueue mq; // MyQueue的析构函数会自动调用pushst和popst的析构函数

    return 0;
}

 

4. 拷贝构造函数


拷贝构造函数是一种特殊的构造函数,它用于通过已有对象来创建一个新的对象。在C++中,如果构造函数的第一个参数是自身类类型的引用,并且任何额外的参数都有默认值,那么这个构造函数就是拷贝构造函数。

4.1拷贝构造函数是构造函数的一个重载


拷贝构造函数实际上是构造函数的一种重载形式,它与普通构造函数的区别在于其参数类型和目的。解释:拷贝构造函数的定义方式与普通构造函数类似,但它的第一个参数必须是同类对象的引用,用于创建新对象时进行对象的复制。

示例:
class MyClass {
public:
    MyClass(int value) {
_value = value;
} { }  // 普通构造函数

    MyClass(const MyClass& other) {        // 拷贝构造函数
        _value = other._value;
    }

private:
    int _value;
};


4.2拷贝构造函数的第一个参数必须是类类型对象的引用


拷贝构造函数的第一个参数必须是类类型的引用,不能是传值,因为传值会导致编译器不断调用拷贝构造函数,最终引发无限递归,导致编译错误。解释:通过引用传递对象可以避免不必要的拷贝操作,并且能够直接访问原对象的内容。这种设计是为了防止调用拷贝构造函数时再次触发拷贝,从而引发无限递归。

示例:
class MyClass {
public:
    MyClass(const MyClass& other) {  // 正确:使用引用作为参数
        _value = other._value;
    }

    // MyClass(MyClass other) {       // 错误:传值会导致无限递归
    //     _value = other._value;
    // }

private:
    int _value;
};


4.3. 调用拷贝构造函数


在C++中,当自定义类型对象需要被拷贝时(如传值传参或返回对象时),系统会自动调用拷贝构造函数。这是C++管理对象生命周期的一个基本机制。

解释:无论是通过值传递一个对象,还是从函数中返回一个对象,C++都会调用拷贝构造函数来创建新的对象副本。这确保了对象能够被正确地拷贝和初始化。

示例:
void Func(MyClass obj) {
    // 传值调用,自动调用拷贝构造函数
}

MyClass ReturnObject() {
    MyClass temp(10);
    return temp;  // 返回对象时,自动调用拷贝构造函数
}

int main() {
    MyClass obj1(10);
    MyClass obj2 = obj1;  // 调用拷贝构造函数
    Func(obj1);           // 调用拷贝构造函数
    MyClass obj3 = ReturnObject();  // 调用拷贝构造函数
    return 0;
}


4.4. 编译器会自动生成拷贝构造函数


如果类中没有显式定义拷贝构造函数,编译器会自动生成一个默认的拷贝构造函数。这个默认的拷贝构造函数会对内置类型成员变量进行浅拷贝,对自定义类型成员变量调用它们的拷贝构造函数。

解释:编译器生成的默认拷贝构造函数能够满足大部分情况下的需求,尤其是对于没有指针成员或动态资源的类。然而,对于涉及动态分配的资源,浅拷贝不合适,需要自定义拷贝构造函数来实现深拷贝。(如下会讲)

示例:
class SimpleClass {
public:
    int _value;

    // 未显式定义拷贝构造函数,编译器会生成默认的拷贝构造函数
};

int main() {
    SimpleClass obj1;
    obj1._value = 42;
    SimpleClass obj2 = obj1;  // 自动生成的拷贝构造函数
    return 0;
}


4.5. 编译器自动生成的拷贝构造函数


如果类成员全部是内置类型(如int、char),编译器自动生成的拷贝构造函数可以完成所需的拷贝,无需显式定义。然而,如果类成员包含指针或动态资源,编译器生成的浅拷贝可能不合适,需要自定义实现深拷贝。

解释:浅拷贝只会复制指针的地址,而不会复制指针所指向的数据。这在动态内存管理中可能导致多个对象共享同一块内存,从而引发资源释放时的冲突。因此,对于涉及动态内存的类,通常需要自定义深拷贝构造函数。
 

示例:
class Stack {
public:
    Stack(int size) {
        _data = new int[size];
        _size = size;
    }

    // 自定义拷贝构造函数,实现深拷贝
    Stack(const Stack& other) {
        _data = new int[other._size];//之后在内存管理会讲到
        _size = other._size;
        for (int i = 0; i < _size; ++i) {
            _data[i] = other._data[i];
        }
    }

    ~Stack() {
        delete[] _data;  // 析构函数释放资源
    }

private:
    int* _data;
    int _size;
};


4.6 拷贝构造函数在传值返回时的行为


当通过传值返回一个对象时,会产生一个临时对象,系统会调用拷贝构造函数来完成对象的复制。然而,传引用返回不会调用拷贝构造函数,而是返回对象的引用。

解释:在C++中,通过值返回对象时,编译器会调用拷贝构造函数来创建返回值的副本。如果通过引用返回对象,则没有拷贝发生。然而,引用返回需要确保返回的对象在函数结束后仍然存在,否则会导致悬空引用。
 

示例:
MyClass ReturnByValue() {
    MyClass temp(10);
    return temp;  // 调用拷贝构造函数,返回对象副本
}

MyClass& ReturnByReference() {
    static MyClass temp(10);  // 使用static,确保返回的引用有效
    return temp;  // 返回引用,不调用拷贝构造函数
}

int main() {
    MyClass obj1 = ReturnByValue();    // 调用拷贝构造函数
    MyClass& obj2 = ReturnByReference();  // 不调用拷贝构造函数
    return 0;
}
示例代码梳理
以下是展示上述拷贝构造函数特点的详细代码示例:

#include<iostream>
using namespace std;

class Date {
public:
    Date(int year = 1, int month = 1, int day = 1)
        : _year(year), _month(month), _day(day) {}

    // 自定义拷贝构造函数
    Date(const Date& d) {
        _year = d._year;
        _month = d._month;
        _day = d._day;
    }

    void Print() const {
        cout << _year << "-" << _month << "-" << _day << endl;
    }

private:
    int _year;
    int _month;
    int _day;
};

void Func1(Date d) {
    d.Print();
}

Date Func2() {
    Date tmp(2024, 7, 5);
    return tmp;  // 返回值传递,调用拷贝构造函数
}

int main() {
    Date d1(2024, 7, 5);

    Func1(d1);  // 传值传参,调用拷贝构造函数

    Date d2(d1);  // 显式调用拷贝构造函数
    d2.Print();

    Date d3 = d1;  // 另一种形式的拷贝构造
    d3.Print();

    Date ret = Func2();  // 调用拷贝构造函数
    ret.Print();

    return 0;
}


通过这些代码示例和解释,我们可以深入理解C++中拷贝构造函数的特性及其应用场景。这些知识点对编写高效、安全的C++代码至关重要,特别是在处理自定义类型和动态资源时,掌握拷贝构造函数的用法可以有效防止潜在的错误和资源泄漏。

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

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

相关文章

828华为云征文|华为云弹性云服务器FlexusX实例下的Nginx性能测试

本文写的是华为云弹性云服务器FlexusX实例下的Nginx性能测试 目录 一、华为云弹性云服务器FlexusX实例简介二、测试环境三、测试工具四、测试方法五、测试结果 下面是华为云弹性云服务器FlexusX实例下的Nginx性能测试。 一、华为云弹性云服务器FlexusX实例简介 华为云弹性云服…

Woocommerce怎么分类显示产品?如何将Shopify的产品导入到Woocommerce?

WooCommerce作为WordPress的一个电子商务插件&#xff0c;功能强大、使用简洁&#xff0c;能够轻松集成到WordPress网站中&#xff0c;为用户提供了一个完整的在线商店解决方案&#xff0c;在国外还是挺受欢迎的。 Woocommerce怎么分类显示产品&#xff1f; 在Woocommerce中&a…

数据资产新范式,URP城市焕新平台东博会首发!

城市数据资产蕴藏着巨大的宝藏。今年1月&#xff0c;国家数据局印发《“数据要素”三年行动计划&#xff08;2024—2026年&#xff09;》&#xff0c;将“数据要素智慧城市”上升为“数据要素”计划的重要部分&#xff0c;加速释放城市数据资产价值。 高质量发展以数据要素驱动…

聚星文社最新风格图库角色

聚星文社最新风格图库角色涵盖了各种不同的风格和类型。以下是一些可能的角色风格&#xff1a; Docs聚星文社https://iimenvrieak.feishu.cn/docx/ZhRNdEWT6oGdCwxdhOPcdds7nof 现代都市风格角色&#xff1a;这种角色通常穿着时尚的衣服&#xff0c;有时尚的发型和化妆。他们可…

【Python】Jet Bridge:快速构建内部工具和管理面板的高效解决方案

Jet Bridge 是一个开源的后台管理工具构建框架&#xff0c;专门用于帮助开发者快速创建内部工具、管理面板和仪表板。它允许用户通过现有的数据库结构快速生成强大的 CRUD&#xff08;创建、读取、更新、删除&#xff09;接口&#xff0c;并提供了直观的可视化界面。Jet Bridge…

反思式思维链大模型 o1 有啥用?

&#xff08;注&#xff1a;本文为小报童精选文章。已订阅小报童或加入知识星球「玉树芝兰」用户请勿重复付费&#xff09; 失望 OpenAI o1 刚出来的时候&#xff0c;我其实对这种 reflection 模型有点儿免疫了。因为刚刚被 reflection 70B 模型诳过一回。 第一时间&#xff0c…

漏洞挖掘 | 某系统中少见的前端登录校验

0 前言 我也是第一次碰到前端登录校验的站点&#xff0c;那所谓前端校验&#xff0c;就是不走后端&#xff0c;这种情况大概率会在前端存着登录的账号和密码&#xff0c;除此之外&#xff0c;一些验证码也可能会在前端校验。 1 测试 如下图&#xff0c;点普通的功能点均显示…

Deep Learning for Video Anomaly Detection: A Review 深度学习视频异常检测综述阅读

Deep Learning for Video Anomaly Detection: A Review 深度学习视频异常检测综述阅读 AbstractI. INTRODUCTIONII. BACKGROUNDA. Notation and TaxonomyB. Datasets and Metrics III. SEMI-SUPERVISED VIDEO ANOMALY DETECTIONA. Model InputB. MethodologyC. Network Archite…

栏目一:使用echarts绘制简单图形

栏目一&#xff1a;使用echarts绘制简单图形 前言1. 在线编辑图形1.1 折线图1.2 柱状图1.3 扇形图 2. 本地绘制图表2.1 下载echarts.min.js2.2 创建一个简单的图形 前言 Echarts是一款基于JavaScript的可视化图表库。它提供了丰富的图表类型和交互功能&#xff0c;可以用于在网…

Golang | Leetcode Golang题解之第445题两数相加II

题目&#xff1a; 题解&#xff1a; func reverseList(head *ListNode) *ListNode {if head nil || head.Next nil {return head}newHead : reverseList(head.Next)head.Next.Next head // 把下一个节点指向自己head.Next nil // 断开指向下一个节点的连接&#xff0c;保证…

Study-Oracle-10-ORALCE19C-RAC集群搭建(一)

一、硬件信息及配套软件 1、硬件设置 RAC集群虚拟机:CPU:2C、内存:10G、操作系统:50G Openfile数据存储:200G (10G*2) 2、网络设置 主机名公有地址私有地址VIP共享存储(SAN)rac1192.168.49.13110.10.10.20192.168.49.141192.168.49.130rac2192.168.49.13210.10.10.3…

使用dockerfile来构建一个包含Jdk17的centos7镜像(构建镜像:centos7-jdk17)

文章目录 1、dockerfile简介2、入门案例2.1、创建目录 /opt/dockerfilejdk172.2、上传 jdk-17_linux-x64_bin.tar.gz 到 /opt/dockerfilejdk172.3、在/opt/dockerfilejdk17目录下创建dockerfile文件2.4、执行命令构建镜像 centos7-jdk17 : 不要忘了后面的那个 .2.5、查看镜像是…

Mixture-of-Experts (MoE): 条件计算的诞生与崛起【上篇】

大型语言模型&#xff08;LLM&#xff09;的现代进步主要是缩放定律的产物[6]。 假设模型是在足够大的数据集上训练出来的&#xff0c;那么随着底层模型规模的增加&#xff0c;我们会看到性能的平滑提升。 这种扩展规律最终促使我们创建了 GPT-3 以及随后的其他&#xff08;更强…

力扣高频 SQL 50 题(基础版)|分析、题解

注意一些语法 1、group by出现在having前面&#xff0c;但是having中所使用的聚合必须是select中的 2、date类型之间的比较&#xff1a;datediff&#xff08;&#xff09; 差的绝对值 or 用字符框起来比较边界 3、算日期长度需要相减之后加一 4、round(, n)n默认是0&#x…

【Java】内存分析 —— 栈内存、堆内存与垃圾对象的形成

图1 内存分析 从图1可以看出&#xff0c;在创建Person对象时&#xff0c;程序会占用两块内存区域&#xff0c;分别是栈内存和堆内存。其中Person类型的变量p被存放在栈内存中&#xff0c;它是一个引用&#xff0c;会指向真正的对象&#xff1b;通过new Person()创建的对象则放…

UDP校验和计算及网络中的校验和机制

UDP (User Datagram Protocol) 是一种无连接的传输层协议&#xff0c;它不像 TCP 那样提供可靠的传输保证。虽然 UDP 不保证数据可靠性&#xff0c;但它仍然提供了一个可选的校验和机制来检测数据在传输过程中出现的错误。 理解UDP校验和的计算过程和其在网络中的作用至关重要。…

学习C语言(21)

整理今天的学习内容 1.结构体实现位段 &#xff08;1&#xff09;位段的声明 位段的成员必须是 int、unsigned int 或signed int &#xff0c;在C99中位段成员的类型也可以选择其他类型 例&#xff1a; &#xff08;2&#xff09;位段的内存分配 位段的空间上是按照需要以…

【 Java 】工具类 —— Collections 与 Arrays 的实用操作全解析

Collections工具类 在Java中&#xff0c;针对集合的操作非常频繁&#xff0c;例如对集合中的元素排序、从集合中查找某个元素等。针对这些常见操作&#xff0c;Java提供了一个工具类专门用来操作集合&#xff0c;这个类就是Collections&#xff0c;它位于java.util包中。Colle…

揭开量子计算和加密未来的秘密

加密保护您的数据 您是否想知道如何保证您的在线数据安全&#xff1f;这就是加密的作用所在。加密是一种使用秘密代码更改数据的过程。这些更改只能由拥有正确密钥的接收者解码和读取。 加密是保护敏感和个人信息安全的重要工具。使用加密的一些示例包括信用卡详细信息、消息…

嵌入式linux系统中Sysfs设备驱动管理方法

大家好,今天主要给大家分享一下,如何使用linux系统里面的Sysfs进行设备管理,希望对大家有所收获。 第一:Sysfs设备驱动管理简介 sysfs 是非持久性虚拟文件系统,它提供系统的全局视图,并通过它们的 kobiect 显示内核对象的层次结构(拓扑)。每个 kobiect 显示为目录和目录…