一篇文章带你详细了解C++智能指针

在这里插入图片描述

一篇文章带你详细了解C++智能指针

  • 为什么要有智能指针
  • 内存泄漏
    • 1.什么是内存泄漏,它的危害是什么
    • 2.内存泄漏的分类
    • 3.如何避免内存泄漏
  • 智能指针的使用及原理
    • 1.RAII
    • 2.智能指针的原理
    • 3.auto_ptr
    • 4.unique_ptr
    • 5.shared_ptr
    • 6.weak_ptr

为什么要有智能指针

C++引入智能指针的主要目的是为了解决手动管理内存的问题,提高程序的健壮性和可维护性。在C++中,内存管理由程序员手动完成,包括内存的分配和释放。手动管理内存可能导致一些常见的问题,如内存泄漏、释放已经释放的内存(二次释放)、野指针等。

智能指针是一种封装了指针的类,它可以自动管理内存的生命周期,使得内存的分配和释放更加安全和方便。

我们考虑一个简单的情景,展示为什么需要智能指针以及它是如何解决问题的。

假设有一个Person类表示一个人,该类有一个成员变量是name,并且我们在动态内存中为其分配内存:

#include <iostream>
#include <cstring>class Person {
public:Person(const char* n) {name = new char[strlen(n) + 1];strcpy(name, n);}~Person() {delete[] name;}void printName() const {std::cout << name << std::endl;}private:char* name;
};int main() {Person* personPtr = new Person("John");personPtr->printName();delete personPtr; // 忘记释放内存return 0;
}

在这个例子中,我们通过 new 在堆上创建了一个 Person 对象,但在程序结束前忘记了调用 delete 来释放内存。这会导致内存泄漏,因为自定义对象的析构函数不会被调用,从而无法释放 name 的内存。

这里我们可以使用智能指针中的unique_ptr来管理Person对象:

#include <iostream>
#include <memory>
#include <cstring>class Person {
public:Person(const char* n) {name = new char[strlen(n) + 1];strcpy(name, n);}~Person() {delete[] name;}void printName() const {std::cout << name << std::endl;}private:char* name;
};int main() {std::unique_ptr<Person> personPtr = std::make_unique<Person>("John");personPtr->printName(); // 在作用域结束时自动释放内存return 0;
}

当然这里只是常规的情况,普通的内存泄漏问题,很多人觉得只要多注意一些就可以了,但是如果是下面的情况呢?

考虑以下场景,其中 Person 类有一个成员变量 bestFriend 表示另一个 Person 对象,两个人互为最好的朋友:

#include <iostream>class Person {
public:Person(const char* n) : name(n), bestFriend(nullptr) {}~Person() {std::cout << name << " destroyed." << std::endl;}void setBestFriend(Person* friendPtr) {bestFriend = friendPtr;}private:const char* name;Person* bestFriend;
};int main() {Person* john = new Person("John");Person* mary = new Person("Mary");john->setBestFriend(mary);mary->setBestFriend(john);// delete john;// delete mary;return 0;
}

在这个例子中,JohnMary 形成了循环引用,因为它们彼此引用对方作为最好的朋友。如果我们尝试使用原始指针进行 delete,它们的析构函数将永远不会被调用,导致内存泄漏。这里我们就可以使用智能指针中的shared_ptr就可以解决这个问题,因为 std::shared_ptr 使用引用计数来跟踪对象的引用数量,当引用计数为零时,对象会被正确地销毁。然而,使用原始指针来管理这种情况会导致无法释放的内存。

**如果两个或多个对象相互引用,形成循环引用,而使用原始指针管理内存,可能导致内存泄漏,因为循环引用会导致引用计数无法归零,从而无法释放对象。**所以在对对象的管理中我们会遇到很多原始指针解决不了的问题,所以才有了智能指针的由来。

内存泄漏

1.什么是内存泄漏,它的危害是什么

内存泄漏是指在程序运行过程中,分配的内存空间在不再需要时没有被释放,导致系统中的可用内存减少。内存泄漏可能发生在各种编程语言中,包括C++、Java、C#等。内存泄漏的危害主要包括:

  1. 资源浪费: 内存泄漏导致未释放的内存无法被重新使用,导致系统中的可用内存逐渐减小。这可能导致系统性能下降、程序变得缓慢,甚至在极端情况下导致系统崩溃。
  2. 程序性能下降: 随着时间的推移,内存泄漏会导致程序使用的内存越来越多,从而增加了垃圾回收的负担,使得程序运行变得更加缓慢。这对于长时间运行的服务或应用程序来说尤其是一个严重的问题。
  3. 系统稳定性下降: 内存泄漏可能导致系统内存耗尽,最终导致系统不稳定,甚至崩溃。这对于一些关键系统、服务器或嵌入式系统来说可能是致命的。
  4. 难以调试: 内存泄漏通常不容易被发现,因为程序在运行时没有显著的错误提示。随着内存泄漏的累积,程序可能会在某一刻因为内存不足而崩溃,而这个问题的根本原因可能很难追踪。
  5. 安全隐患: 内存泄漏也可能导致安全问题。恶意攻击者可以利用未释放的内存来执行缓冲区溢出等攻击,从而破坏程序的正常执行,甚至入侵系统。

2.内存泄漏的分类

内存泄漏可以分为几种常见的类型,每种类型都有不同的原因和表现。以下是一些常见的内存泄漏分类:

  1. 堆内存泄漏(Heap Memory Leak): 堆内存泄漏是指在动态分配内存时,没有正确释放这些内存导致的泄漏。这通常发生在使用 newmalloc 分配内存后忘记使用 deletefree 进行释放。
  2. 栈内存泄漏(Stack Memory Leak): 栈内存泄漏通常是由于在函数或代码块中分配的局部变量没有在该函数或代码块结束时被正确释放。栈内存泄漏通常较为轻微,因为在函数结束时,栈上的局部变量会自动被销毁。
  3. 全局/静态内存泄漏(Global/Static Memory Leak): 全局变量和静态变量在程序的整个生命周期内存在,如果没有在程序结束时释放相关内存,就会导致全局或静态内存泄漏。
  4. 循环引用(Circular Reference): 循环引用是指两个或多个对象相互引用,形成一个循环结构,并且它们的引用计数无法归零。这种情况下,即使没有其他引用,这些对象也无法被垃圾回收。
  5. 虚拟机泄漏(Memory Leak in Managed Runtimes): 在使用托管运行时(如Java虚拟机、.NET运行时等)的环境中,有时会出现虚拟机泄漏,即运行时本身没有正确释放的内存。
  6. 资源泄漏(Resource Leak): 除了内存之外,资源泄漏还可以包括其他类型的资源,例如文件句柄、网络连接等。如果这些资源在使用完毕后没有被释放,就会导致资源泄漏。

3.如何避免内存泄漏

避免内存泄漏是编程中非常重要的任务之一。以下是一些常见的方法和最佳实践,有助于减少或避免内存泄漏:

  1. 使用智能指针: 使用C++的智能指针,如std::shared_ptrstd::unique_ptr等,可以自动管理内存的释放。这样可以减少手动释放内存的机会,防止忘记释放或重复释放的问题。

    // 使用 std::shared_ptr
    std::shared_ptr<int> smartPtr = std::make_shared<int>(42);
    
  2. RAII(资源获取即初始化)原则: 使用对象生命周期管理资源。确保在对象创建时分配资源,在对象销毁时释放资源。智能指针正是基于这个原则设计的。

  3. 避免手动管理内存: 尽量避免使用 newdelete 进行手动内存管理。使用标准容器和智能指针等抽象层级更高的工具,它们能够更安全地管理内存。

  4. 使用析构函数: 在类的析构函数中释放在构造函数中分配的资源。确保资源的释放操作被正确实现。

    class ResourceHolder {
    public:ResourceHolder() {// 分配资源resource = new Resource;}~ResourceHolder() {// 释放资源delete resource;}private:Resource* resource;
    };
    
  5. 避免循环引用: 当存在循环引用时,使用弱引用(std::weak_ptr)来打破循环引用关系,防止引用计数无法归零。

    #include <iostream>
    #include <memory>class Person;class Car {
    public:void setOwner(std::shared_ptr<Person> person) {owner = person;}private:std::shared_ptr<Person> owner;
    };class Person {
    public:void buyCar() {car = std::make_shared<Car>();car->setOwner(shared_from_this());}private:std::shared_ptr<Car> car;
    };
    
  6. 使用工具进行静态和动态分析: 使用工具如静态分析器(如Clang Static AnalyzerCppcheck)、动态分析器(如Valgrind)等,帮助发现潜在的内存泄漏问题。

  7. 使用现代C++特性: C++11及其后续版本引入了许多现代C++特性,如移动语义、智能指针、Lambda表达式等,这些特性有助于更安全和高效地管理内存。

  8. 良好的编程习惯: 养成良好的编程习惯,注重代码的规范性和清晰性,有助于及早发现潜在的问题,并减少内存泄漏的发生。

智能指针的使用及原理

1.RAII

RAII(Resource Acquisition Is Initialization)是一种C++编程范式,是一种基于对象生命周期管理资源的策略。RAII的核心思想是,在对象的构造函数中获取资源(如内存、文件句柄、网络连接等),而在对象的析构函数中释放这些资源。这样,资源的生命周期与对象的生命周期绑定在一起,从而确保资源在适当的时候被正确释放。

下面我们看一个使用RAII思想设计的SmartPtr

template<class T>
class SmartPtr
{
public:SmartPtr(T* ptr):_ptr(ptr){}~SmartPtr(){cout << "delete:" << _ptr << endl;delete _ptr;}private:T* _ptr;
};int div()
{int a, b;cin >> a >> b;if (b == 0)throw invalid_argument("除0错误");return a / b;
}void Func()
{SmartPtr<int> sp1(new int);SmartPtr<int> sp2(new int);cout << div() << endl;
}int main()
{try{Func();}catch (exception& e){cout << e.what() << endl;}return 0;
}

在这里我们可以看到,通过SmartPtr类对div类新对象的建立,我们可以在其创建新对象时自动将指针进行托管,最终不用我们手动去调用析构函数,而是在作用域将结束时,通过SmartPtr指向的div对象指针自动调用析构,从而防止了内存泄漏的问题。

2.智能指针的原理

上述的SmartPtr还不能将其称为智能指针,因为它还不具有指针的行为。指针可以解引用,也可以通过->去访问所指空间中的内容,因此:此处模板类中还得需要将* ->进行重载,才可让其像指针一样去使用 。

template<class T>
class SmartPtr
{
public:SmartPtr(T* ptr):_ptr(ptr){}~SmartPtr(){cout << "delete:" << _ptr << endl;delete _ptr;}T& operator*(){return *_ptr;}T* operator->(){return _ptr;}private:T* _ptr;
};struct Date
{int _year;int _month;int _day;
};
int main()
{SmartPtr<int> sp1(new int);*sp1 = 10;cout << *sp1 << endl;SmartPtr<int> sparray(new Date);sparray->_year = 2023;sparray->_month = 1;sparray->_day = 1;
}

需要注意的是这里应该是sparray.operator->()->_year = 2018;应该是sparray->->_year这里语法上为了可读性,省略了一个->

3.auto_ptr

auto_ptr 是 C++98 标准中引入的一种智能指针,用于管理动态分配的内存。它是第一个尝试提供自动内存管理的 C++ 标准库智能指针,然而,由于其独特的拥有权转移语义,导致了一些问题,因此在后续的 C++ 标准中被更现代的智能指针(如 std::unique_ptrstd::shared_ptr)所取代。

以下是 auto_ptr 的一些关键特点和历史:

  1. 独占所有权: auto_ptr 具有独占所有权的特性,即当一个 auto_ptr 拥有某块动态分配的内存时,其他任何 auto_ptr 都不能指向同一块内存。这种独占性导致了一些潜在的问题,特别是在涉及到复制和拷贝构造函数时。
  2. 拥有权的转移: auto_ptr 支持拥有权的转移,即一个 auto_ptr 对象的所有权可以转移到另一个对象,而被转移的对象会变成空指针。这样的语义在某些情况下可能导致程序员不经意间的错误。
  3. 不适用于容器: 由于拥有权的转移语义,auto_ptr 不能安全地用于标准库容器,因为容器的操作可能导致拷贝或复制 auto_ptr 对象,从而引发悬空指针和内存泄漏问题。
  4. 被现代智能指针替代: 由于 auto_ptr 存在的问题,C++11 引入了更为安全和灵活的智能指针,如 std::unique_ptrstd::shared_ptr。这些智能指针提供了更好的内存管理语义和更丰富的功能,取代了 auto_ptr
  5. 被标记为废弃: C++11 标准中将 auto_ptr 标记为废弃(deprecated),并建议使用更为安全的替代方案。在 C++17 标准中,auto_ptr 被完全移除。

简化模拟实现auto_ptr

namespace yulao
{template<class T>class auto_ptr {public:auto_ptr(T* ptr = nullptr): _ptr(ptr){}auto_ptr(auto_ptr<T>& ap):_ptr(ap._ptr){ap._ptr = nullptr;}auto_ptr<T>& operator=(auto_ptr<T>& ap){if (this != &ap){if (_ptr){cout << "Delete:" << _ptr << endl;delete _ptr;}_ptr = ap._ptr;ap._ptr = nullptr;}return *this;}~auto_ptr(){if (_ptr){cout << "Delete:" << _ptr << endl;delete _ptr;}}T& operator*(){return *_ptr;}T* operator->(){return _ptr;}private:T* _ptr;};
}

C++98中 auto_ptr 管理权转移,被拷贝对象的出现悬空问题,很多公司是明确的要求了不能使用它

比如下面的情况

class A
{
public:~A(){cout << "~A()" << endl;}int _a1 = 0;int _a2 = 0;
};int main()
{auto_ptr<A> ap1(new A);ap1->_a1++;ap1->_a2++;auto_ptr<A> ap2(ap1);ap1->_a1++;ap1->_a2++;return 0;
}

这里使用了 auto_ptr 来管理动态分配的对象 A 的内存。然而,需要注意的是,auto_ptr 具有拥有权转移的语义,因此在将一个 auto_ptr 赋值给另一个后,原始的 auto_ptr 将变为 nullptr 指针。这会导致后续对原始指针成员 _a1_a2 的访问可能导致未定义行为。auto_ptr 在现代C++中已经被废弃,有了更为安全的 unique_ptr 来管理动态分配的内存。unique_ptr 没有拥有权转移的问题,并提供了更好的所有权管理。

4.unique_ptr

在C++11出现之前其实已经有了智能指针(不是指auto_ptr),是在第三方的一个C++库中,名叫boost库。

Boost库是一个由C++社区开发和维护的开源库集合,提供了许多高质量、可移植且通用的C++工具和组件。在C++11引入标准智能指针之前,Boost库已经提供了类似的功能。

Boost库中的智能指针主要是boost::shared_ptrboost::scoped_ptr。以下是它们在C++98到C++11之间的历史发展:

  1. boost::scoped_ptr 在C++98时代,Boost库引入了boost::scoped_ptr。这是一个独占所有权的智能指针,其目的是在其生命周期结束时自动释放所管理的资源。然而,由于它的独占性质,boost::scoped_ptr不能共享资源。

    #include <boost/scoped_ptr.hpp>int main() {boost::scoped_ptr<int> myInt(new int);// myInt 的生命周期结束时,所管理的内存将被释放return 0;
    }
    
  2. boost::shared_ptr 随着C++98的发展,Boost库还引入了boost::shared_ptr。这是一个引用计数智能指针,它允许多个shared_ptr对象共享对同一对象的所有权,并在最后一个shared_ptr离开作用域时释放资源。

    #include <boost/shared_ptr.hpp>int main() {boost::shared_ptr<int> sharedInt(new int);// 多个 sharedInt 对象共享同一块内存资源return 0;
    }
    
  3. C++11引入标准智能指针: 随着C++11标准的发布,标准库中引入了std::unique_ptrstd::shared_ptrstd::unique_ptr,提供了更为强大和灵活的智能指针实现。C++11的智能指针取代了Boost库中的对应版本,并成为标准语言特性。

unique_ptr的实现原理:简单粗暴的防拷贝,下面简化模拟实现了一份unique_Ptr来了解它的原理

template<class T>
class unique_ptr
{
private:// 防拷贝 C++98// 只声明不实现 //unique_ptr(unique_ptr<T>& ap);//unique_ptr<T>& operator=(unique_ptr<T>& ap);
public:unique_ptr(T* ptr = nullptr): _ptr(ptr){}// 防拷贝 C++11unique_ptr(unique_ptr<T>& ap) = delete;unique_ptr<T>& operator=(unique_ptr<T>& ap) = delete;~unique_ptr(){if (_ptr){cout << "Delete:" << _ptr << endl;delete _ptr;}}T& operator*(){return *_ptr;}T* operator->(){return _ptr;}private:T* _ptr;
};

这里我们可以看到为了防止出现auto_ptr的拷贝空指针的情况,我们可以直接将拷贝构造和赋值重载直接delete,或者设为私有声明后不实现两种方式。

int main()
{yulao::unique_ptr<int> sp1(new int);yulao::unique_ptr<int> sp2(sp1);//此时拷贝会报错return 0;
}

5.shared_ptr

shared_ptr与unique_ptr最大的区别就是支持拷贝

shared_ptr的原理:是通过引用计数的方式来实现多个shared_ptr对象之间共享资源

  1. shared_ptr在其内部,给每个资源都维护了着一份计数,用来记录该份资源被几个对象共享。
  2. 在对象被销毁时(也就是析构函数调用),就说明自己不使用该资源了,对象的引用计数减一。
  3. 如果引用计数是0,就说明自己是最后一个使用该资源的对象,必须释放该资源;
  4. 如果不是0,就说明除了自己还有其他对象在使用该份资源,不能释放该资源,否则其他对象就成野指针了。

==引用计数支持多个拷贝管理同一个资源,最后一个析构对象释放资源 ==

template<class T>
class shared_ptr
{
public:shared_ptr(T* ptr = nullptr): _ptr(ptr), _pCount(new int(1)){}void Release(){if (--(*_pCount) == 0){cout << "Delete:" << _ptr << endl;delete _ptr;delete _pCount;}}~shared_ptr(){Release();}// sp1(sp2)shared_ptr(const shared_ptr<T>& sp): _ptr(sp._ptr), _pCount(sp._pCount){(*_pCount)++;}shared_ptr<T>& operator=(const shared_ptr<T>& sp){if (_ptr == sp._ptr){return *this;}Release();// 共管新资源,++计数_ptr = sp._ptr;_pCount = sp._pCount;(*_pCount)++;return *this;}T& operator*(){return *_ptr;}T* operator->(){return _ptr;}private:T* _ptr;// 引用计数int* _pCount;
};
  1. 引用计数管理: 使用 _pCount 来追踪对象的引用计数,每个 shared_ptr 指向同一个对象时,它们共享同一个 _pCount
  2. 构造函数和析构函数: 在构造函数中,你初始化 _pCount 为1,表示当前有一个 shared_ptr 指向这个对象。在析构函数中,你通过 Release() 函数来释放资源,如果引用计数为0,则删除对象和引用计数。
  3. 拷贝构造函数和拷贝赋值运算符: 当一个 shared_ptr 被拷贝构造或赋值给另一个 shared_ptr 时,引用计数会增加。这样,多个 shared_ptr 可以共享同一块内存,当最后一个 shared_ptr 被销毁时,对象才会被释放。
  4. 重载 operator *operator-> 使得 shared_ptr 的使用方式类似于原始指针。
  5. 删除资源的时机: 在你的代码中,资源的释放是在 Release() 函数中进行的,该函数在析构函数和赋值运算符中被调用。这确保了资源在最后一个引用被释放时被正确删除。
int main(){shared_ptr<A> sp1(new A);shared_ptr<A> sp2(sp1);shared_ptr<A> sp3(sp1);sp1->_a1++;sp1->_a2++;std::cout << sp2->_a1 << ":" << sp2->_a2 << std::endl;sp2->_a1++;sp2->_a2++;std::cout << sp1->_a1 << ":" << sp1->_a2 << std::endl;
}

我们通过这段代码可以看到shared_ptr可以很好的支持对对象指针的拷贝并进行管理,实现多指针管理同一个对象,但不会造成多次调用析构的情况。

但是在C++中,使用 shared_ptr 来管理资源,当形成循环引用时,可能导致对象无法正确释放。下面是一个简化的双向链表的例子:

#include <memory>
#include <iostream>template<typename T>
class Node {
public:T data;std::shared_ptr<Node<T>> next;std::shared_ptr<Node<T>> prev;Node(const T& val) : data(val), next(nullptr), prev(nullptr) {std::cout << "Node constructed with value: " << val << std::endl;}~Node() {std::cout << "Node destructed with value: " << data << std::endl;}
};int main() {// 创建一个双向链表节点1auto node1 = std::make_shared<Node<int>>(1);// 创建一个双向链表节点2auto node2 = std::make_shared<Node<int>>(2);// 形成循环引用node1->next = node2;node2->prev = node1;// 输出每个节点的引用计数std::cout << "Reference counts: node1=" << node1.use_count() << ", node2=" << node2.use_count() << std::endl;// 节点1和节点2的引用计数不为零,它们不会被释放return 0;
}

在上面的例子中,node1node2 形成了双向链表的循环引用。node1 持有 node2shared_ptr,而 node2 同时持有 node1shared_ptr。这导致两个节点的引用计数不会变为零,它们的析构函数也不会被调用。这就是循环引用的问题。

为了解决这个问题,可以使用 weak_ptr 来打破循环引用。在上面的例子中,可以将 prevnext 成员改为 std::weak_ptr 类型。这样,即使形成了循环引用,weak_ptr 不会增加引用计数,也不会影响节点的析构。

6.weak_ptr

首先我们要知道weak_ptr不是常规智能指针,没有RAII,不支持直接管理资源,weak_ptr主要用shared_ptr构造,用来解决shared_ptr循环引用问题,这里将它单独列出,只是为了更好的讲清楚这个问题

简单的模拟实现weak_ptr

// 辅助型智能指针,使命配合解决shared_ptr循环引用问题
template<class T>
class weak_ptr
{
public:weak_ptr():_ptr(nullptr){}weak_ptr(const shared_ptr<T>& sp):_ptr(sp.get()){}weak_ptr(const weak_ptr<T>& wp):_ptr(wp._ptr){}weak_ptr<T>& operator=(const shared_ptr<T>& sp){_ptr = sp.get();return *this;}T& operator*(){return *_ptr;}T* operator->(){return _ptr;}
public:T* _ptr;
};
  1. 构造函数: 默认构造函数,以及从 shared_ptr 和另一个 weak_ptr 构造的构造函数。这是 weak_ptr 常见的构造方式。
  2. 赋值运算符:shared_ptr 赋值的运算符。这允许将 shared_ptr 赋值给 weak_ptr,使得 weak_ptr 可以观察 shared_ptr 所指向的对象,但并不影响引用计数。
  3. 重载 operator *operator-> 这使得 weak_ptr 的使用方式类似于原始指针。
  4. 注意事项: 在实际使用中,weak_ptr 通常用于解决循环引用问题,而不是直接与裸指针交互。weak_ptr 不会增加对象的引用计数,因此不会影响对象的生命周期。

以下是一个简单的示例,演示了如何使用 weak_ptrshared_ptr 避免循环引用:

#include <iostream>
#include <memory>class B;  // 前向声明class A {
public:shared_ptr<B> b_ptr;A() { std::cout << "A constructed" << std::endl; }~A() { std::cout << "A destructed" << std::endl; }
};class B {
public:weak_ptr<A> a_ptr;  // 使用 weak_ptr 避免循环引用B() { std::cout << "B constructed" << std::endl; }~B() { std::cout << "B destructed" << std::endl; }
};int main() {shared_ptr<A> a = make_shared<A>();shared_ptr<B> b = make_shared<B>();a->b_ptr = b;b->a_ptr = a;return 0;
}

在这个例子中,AB 类相互引用,但通过使用 weak_ptr 避免了循环引用。

最后再提一句shared_ptr的线程安全问题,需要等到后面作者将Linux文章更新到多线程再进行讲解,希望大家能够持续关注我!!!

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

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

相关文章

WindowsServer服务器系列:定时备份 MySQL

一、编写脚本 echo 取日期、时间变量值 set yy%date:~0,4% set mm%date:~5,2% set dd%date:~8,2% if /i %time:~0,2% lss 10 set hh0%time:~1,1% if /i %time:~0,2% geq 10 set hh%time:~0,2% set mn%time:~3,2% set ss%time:~6,2% set date%yy%%mm%%dd% set time%hh%%mn%%ss…

Vue2中v-html引发的安全问题

前言&#xff1a;v-html指令 1.作用&#xff1a;向指定节点中渲染包含html结构的内容。 2.与插值语法的区别&#xff1a; (1).v-html会替换掉节点中所有的内容&#xff0c;{{xx}}则不会。 (2).v-html可以识别html结构。 3.严重注意&#xff1a;v-html有安全性问题&#xff0…

IT外包的三种模式

在当今数字化时代&#xff0c;企业为了更好地专注于核心业务&#xff0c;通常选择将IT部门或项目外包给专业的IT外包服务公司。IT外包作为一种灵活的业务模式&#xff0c;不仅能够提高效率&#xff0c;还能够降低企业的运营成本。IT外包包含着不同的业务模式&#xff0c;其中有…

外包干了2个多月,技术明显有退步了。。。。。

先说一下自己的情况&#xff0c;本科生&#xff0c;19年通过校招进入武汉某软件公司&#xff0c;干了接近4年的功能测试&#xff0c;今年国庆&#xff0c;感觉自己不能够在这样下去了&#xff0c;长时间呆在一个舒适的环境会让一个人堕落!而我已经在一个企业干了四年的功能测试…

VMware 虚拟机 电脑重启后 NAT 模式连不上网络问题修复

问题描述&#xff1a; 昨天 VMware 安装centos7虚拟机&#xff0c;网络模式配置的是NAT模式&#xff0c;配置好后&#xff0c;当时能连上外网&#xff0c;今天电脑重启后&#xff0c;发现连不上外网了 检查下各个配置&#xff0c;都没变动&#xff0c;突然就连不上了 网上查了…

搭梯子之后电脑连接WIFI打不开浏览器网页:远程计算机或者设备不接受连接

问题描述&#xff1a; 打不开网页&#xff0c;但是能正常使用微信等app windows网络诊断&#xff1a; 远程计算机或者设备不接受连接 解决办法&#xff1a; 电脑搜索【internet选项】 进入连接&#xff0c;点击局域网设置&#xff0c;将里面的代理服务器选项关掉就可以正常打开…

Diary21-全网最全的HTML讲解(含可用代码)

HTML学习 1.网页基本信息 DOCTYPE:是一种规范&#xff0c;告诉浏览器我们要使用什么规范 head标签代表网页头部 title标签代表网页标题 body标签代表网页主体 下面是创建的第一个网页的源代码(在IDEA创建一个html文件会直接生成&#xff0c;我这个其实只改了网页标题) &l…

阿里云实时数据仓库HologresFlink

1. 实时数仓Hologres特点 专注实时场景&#xff1a;数据实时写入、实时更新&#xff0c;写入即可见&#xff0c;与Flink原生集成&#xff0c;支持高吞吐、低延时、有模型的实时数仓开发&#xff0c;满足业务洞察实时性需求。亚秒级交互式分析&#xff1a;支持海量数据亚秒级交…

玩弄GPTs:人人都会的Prompt模板

角色定义 分享一个自用的Prompt模板&#xff0c;只要学会了这个模板&#xff0c;当遇到新场景时&#xff0c;直接套用就行。 简单总结&#xff1a; 角色定义(Master)背景(Background)规则(rule)技能(skill)限制(constaints)工作流(workflow) 经过实际测试发现&#xff0c;这…

从零开始学习 JavaScript APl(七):实例解析关于京东案例头部案例和放大镜效果!

大家好关于JS APl 知识点已经全部总结了&#xff0c;第七部部分全部都是案例部分呢&#xff01;&#xff01;&#xff08;素材的可以去百度网盘去下载&#xff01;&#xff01;&#xff01;&#xff09; 目录 前言 一、个人实战文档 放大镜效果 思路分析&#xff1a; 关于其它…

指针(三)

函数指针 定义&#xff1a;整型指针是指向整形的指针,数组指针式指向数组的指针,其实函数指针就是指向函数的指针。 函数指针基础&#xff1a; &#xff08;&#xff09;优先级要高于*&#xff1b;一个变量除去了变量名&#xff0c;便是它的变量类型&#xff1b;一个指针变量…

两种伦敦银缺口 如何为我们的交易服务?

我们做伦敦银也会碰到缺口&#xff0c;有的朋友会说伦敦银不是24小时交易的品种吗&#xff1f;怎么有缺口呢&#xff1f;虽说伦敦银是24小时交易的品种&#xff0c;但是在北京时间的凌晨也会停止交易一段时间&#xff0c;这是平台结算时间。在亚盘早段伦敦银重新开盘之后&#…

Android的前台服务

概述 前台服务是用户主动意识到的一种服务&#xff0c;因此在内存不足时&#xff0c;系统也不会考虑将其终止。前台服务必须为状态栏提供通知&#xff0c;将其放在运行中的标题下方。这意味着除非将服务停止或从前台移除&#xff0c;否则不能清除该通知。 在 Android 8.0&…

一对一聊天

1.创建包 .服务界面 package yiduiy;import java.awt.BorderLayout; import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; import java.io.PrintWriter; import java.net.ServerSocket; import java.net.Socket; import java.uti…

1.2 C语言简介

一、为什么要讲C语言 C语言是编程界的长青藤&#xff0c;可以查看语言排名发现&#xff0c;虽然现在语言很多&#xff0c;但是C语言一直占有一定地址 来源网站&#xff1a;https://www.tiobe.com/tiobe-index/ 在系统、嵌入式、底层驱动等领域存在一定的唯一性&#xff08;C语…

Python Opencv实践 - 简单的AR项目

这个简单的AR项目效果是&#xff0c;通过给定一张静态图片作为要视频中要替换的目标物品&#xff0c;当在视频中检测到图片中的物体时&#xff0c;通过单应矩阵做投影&#xff0c;将视频中的物体替换成一段视频播放。这个项目的所有素材来自自己的手机拍的视频。 静态图片&…

Java注册并监听全局快捷键

背景 之前在博客中分享了SWT托盘功能, 随之带来一个问题, 当程序最小化后无法快速唤醒, 按照平时使用软件的思路, 自然想到了注册全局快捷键, 本文介绍使用java方式实现全局快捷键的注册. 方案 通过google,搜到一个现成的库: jintellitype, 使用maven可以直接引用, 非常方便…

istio为什么能代替传统的SpringCloud 服务网格Istio概述

服务网格Istio概述 什么是服务网格(Service Mesh)&#xff1f;istio简介边车模式&#xff08;Sidecar&#xff09;为什么istio能代替传统SpringCloud&#xff1f;整体架构 首先奉上 istio官网 什么是服务网格(Service Mesh)&#xff1f; 服务网格详解 服务网格&#xff08;Se…

SpringMVC修炼之旅(2)基础入门

一、第一个程序 1.1环境配置 略 1.2代码实现 package com.itheima.controller;import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.ResponseBody;//定义…

【JaveSE】:认识异常

异常 一.异常的概念二.异常的体系结构1.基础结构2.异常分类 三.异常的处理1.防御式编程2.异常的抛出3.异常捕获4.try-catch捕获并处理5.finally 四.异常处理流程五.自定义异常 一.异常的概念 在Java中&#xff0c;将程序执行过程中发生的不正常行为称为异常。 算术异常 数组越界…