浅析C++智能指针和enable_shared_from_this 机制

C语言与CPP编程 2020-12-27 00:00

大家好,我是小牛,今天跟聊一下 BAT 面试 C++ 开发工程师必问的一个考点:智能指针。

小艾:你昨晚面 C++ 去了?

小牛:对啊,不是这个厂主要技术栈都是 C++ 嘛,我就面去了。

小艾:问了点啥啊?

小牛:BAT 这 C++ 问的都差不多,又问智能指针了。

小艾:那来讲讲呗。

小牛:来。

智能指针的引入

大家都知道,指针是 C++ 中非常重要的一部分,大家在初期学习 C++ 的时候一定学过类似这样的指针方式。

int  *ptr;

这种指针也被称为裸指针。但是使用裸指针会存在一些不足

  1. 如果使用裸指针分配内存后,忘记手动释放资源,会出现内存泄漏。

  2. 如果使用多个裸指针指向同一资源,其中一个指针对资源进行释放,其它指针成为空悬指针,如果再次释放会存在不可预测的错误。上图中当 sp1 把资源释放后,sp2 成了空悬指针。空悬指针指的是指针所指向的对象已经释放的时候自身却没有被置为 nullptr。sp1 通过 free/delete 释放资源的内存时,内存不会立刻被系统回收,而是状态改变为可被其它地方申请的状态。这时当再次操作 sp2,这块内存可能被其它地方申请了,而具体被谁申请了是不确定的,因此可能导致的错误也是不可预测的。

  3. 如果程序异常退出时,裸指针的释放资源的代码未能执行,也会造成内存泄漏。

为了改善裸指针的不足,确保资源的分配和释放是配对的,开发者提出了智能指针。智能指针主要是对裸指针进行了一次面向对象的封装,在构造函数中初始化资源地址,在析构函数中释放资源。 当资源应该被释放时,指向它的智能指针可以确保自动地释放它。

C++ 库中,为智能指针提供了不带引用计数和带引用计数的两种方案。

引用计数用于表示有多少智能指针引用同一资源。不带引用计数的智能指针采用独占资源的方式,而带引用计数的智能指针则可以同时多个指向同一资源。下面介绍一下它们的主要特点区别

智能指针的分类

不带引用计数的智能指针

不带引用计数的智能指针包括 auto_ptrscoped_ptrunique_ptr 三种指针。

不带引用计数的智能指针

1. auto_ptr:

我们先来看个例子:

#include<memory>
int main()
{
 auto_ptr<int>  ptr(new int(6));//定义auto_ptr指针ptr
 auto_ptr<int>  ptr1(ptr);  //拷贝构造ptr定义ptr1
 *ptr=8;//对空指针ptr赋值会产生不可预料的错误
 return 0;
}

开始时 ptr 指向资源,一个整型数字6,当用 ptr1 拷贝构造 ptr 时,ptr1 指向资源,而 ptr 则指向 nullptr。下一行程序中如果对空指针 ptr 赋值 8,将会产生不可预料的错误。

下图表示 auto_ptr 指针对资源的指向过程。

auto_ptr

使用拷贝构造时,如果只有最后一个 auto_ptr 持有资源,其余 auto_ptr 持有的资源会被置为 nullptr

因此需要注意,不能在容器中使用 auto_ptr,当容器发生拷贝时,原容器中 auto_ptr 持有的资源会置 nullptr

下面我们再来看一下 auto_ptr 的部分源码和部分解析:

template<class _Ty>
class auto_ptr
{
 
public:
 typedef _Ty element_type;

 explicit auto_ptr(_Ty * _Ptr=nullptr) noexcept
     : _Myptr(_Ptr)//初始化列表
     
//构造函数
     }

 auto_ptr(auto_ptr& _Right) noexcept
  : _Myptr(_Right.release())
  { //拷贝构造函数,会调用release()函数
  }
  
 _Ty * release() noexcept
  
{
            /*使用拷贝构造时,最后一个auto_ptr持有资源,
   其余被置为nullptr*/
 
  _Ty * _Tmp = _Myptr;
  _Myptr = nullptr;
  return (_Tmp);
  }
private:
 _Ty * _Myptr;//指向资源 
};

当试图调用 auto_ptr 的拷贝构造函数时,在初始化列表中调用了 release() 函数,release() 函数用一个 _Tmp 指针保存资源并返回用于初始化当前的 auto_ptr 的类成员 _Myptr,而 _Right 对应的 _Myptr 被置为 nullptr

2. scoped_ptr

scoped_ptrauto_ptr 有些不同,它私有化了拷贝构造函数和赋值函数,资源的所有权无法进行转移,也无法在容器中使用。

下面使用一段代码表现 scoped_ptr 的特性,如果不规范使用会发生错误。

正确用法:

scoped_ptr<intsp1(new int(6));//初始化sp1指针

错误用法:

scoped_ptr<intsp2(sp1);//错误,无法拷贝构造

这种方法是错误的,因为scoped_ptr私有化了拷贝构造函数,无法显式调用。

 scoped_ptr<intsp3(new int(5))//初始化sp2指针
 sp1
=sp3;//错误,无法赋值

这种方法是错误的,因为scoped_ptr私有化了赋值构造函数,无法显式调用。

有时候面试官会问到,scoped_ptr 是如何保证资源的所有权的?

这时候就可以按照上面的讲解来回答:

scoped_ptr 私有化了拷贝构造函数和赋值函数,资源的所有权无法进行转移,所以保证了资源的所有权。

然后再来看一下 scoped_ptr 的部分源码和部分解析:

template<class T
class scoped_ptr
{

private:
    T * px; 
    scoped_ptr(scoped_ptr const &);//拷贝构造函数
    scoped_ptr & operator=(scoped_ptr const &);//赋值构造函数
 
public:
    typedef T element_type;
    explicit scoped_ptr( T * p = nullptr )px( p )
    
{
    }
    ~scoped_ptr() //析构函数
};

scoped_ptr 通过私有化拷贝构造函数和赋值构造函数来拒绝浅拷贝的发生。

值得注意的是,auto_ptr 是通过将除最后一个以外的其它 auto_ptrnullptr 来避免浅拷贝的发生,它的资源所有权是可以转移的。

scoped_ptr 是直接禁止了拷贝与赋值,资源所有权无法转移。

3. unique_ptr

unique_ptr 删除了拷贝构造函数和赋值函数,因此不支持普通的拷贝或赋值操作。如下所示:

unique_ptr<intp1(new int(6));//正确写法
unique_ptr<intp2(p1)//这么写是错误的:
// unique_ptr不支持拷贝
unique_ptr<int> p3;
p3=p2;//这么写是错误的:unique_ptr不支持赋值

再来看一下 unique_ptr 的部分源码和部分解析:

template<class _Ty,class _Dx>
class unique_ptr:
 public _Unique_ptr_base<_Ty, _Dx>

public:
 typedef _Unique_ptr_base<_Ty, _Dx> _Mybase;
 typedef typename _Mybase::pointer pointer;
 typedef _Ty element_type;
 typedef _Dx deleter_type;

 unique_ptr(unique_ptr&& _Right) noexcept
  : _Mybase(_Right.release(),
   _STD forward<_Dx>(_Right.get_deleter()))
  { // 右值引用的拷贝构造函数
  }

 unique_ptroperator=(unique_ptr&& _Right) noexcept
  { //提供了右值引用的operator=赋值构造函数
  if (this != _STD addressof(_Right))
   { 
   reset(_Right.release());
   this->get_deleter() = _STD forward<_Dx>
   (_Right.get_deleter());
   }
  return (*this);
  }
 /*
 删除了unique_ptr的拷贝构造和赋值函数,拒绝浅拷贝
 */

 unique_ptr(const unique_ptr&) = delete;
 unique_ptroperator=(const unique_ptr&) = delete;
 };

unique_ptrscoped_ptr一样禁止了拷贝构造和赋值构造,引入了带右值引用的拷贝构造和赋值。可以把 unique_ptr 作为函数的返回值。

不带引用计数的智能指针总结:

相同点:最终只有一个智能指针持有资源。

不同点:

  1. auto_ptr 进行拷贝构造时,会对之前的auto_ptr的资源置nullptr操作;
  2. scoped_ptr 通过私有化了拷贝构造和赋值函数杜绝浅拷贝;
  3. unique_ptr 通过删除了拷贝构造和赋值函数函数杜绝浅拷贝,但引入了带右值引用的拷贝构造和赋值函数。

带引用计数的智能指针

当需要多个智能指针指向同一个资源时,使用带引用计数的智能指针。

每增加一个智能指针指向同一资源,资源引用计数加一,反之减一。当引用计数为零时,由最后一个指向资源的智能指针将资源进行释放。

下图表示带引用计数智能指针的工作过程。sp1 对象和 sp2 对象通过指针指向同一资源,引用计数器记录了引用资源的对象个数。

智能指针的工作过程

当 sp1 对象发生析构时,引用计数器的值减 1,由于引用计数不等于 0,资源并未释放,如下图所示:

sp1 对象发生析构

当 sp2 对象也发生析构,引用计数减为 0,资源释放,如下图所示:

sp2 对象也发生析构

即引用计数可以保证多个智能指针指向资源时资源在所有智能对其取消引用再释放,避免过早释放产生空悬指针。带引用计数的智能指针包括 shared_ptrweak_ptr

资源释放

1. shared_ptr

shared_ptr 一般称为强智能指针,一个 shared_ptr 对资源进行引用时,资源的引用计数会增加一,通常用于管理对象的生命周期。只要有一个指向对象的 shared_ptr 存在,该对象就不会析构。

上图中引用计数的工作过程就使用了 shared_ptr

2. weak_ptr

weak_ptr 一般被称为弱智能指针,其对资源的引用不会引起资源的引用计数的变化,通常作为观察者,用于判断资源是否存在,并根据不同情况做出相应的操作。

比如使用 weak_ptr 对资源进行弱引用,当调用 weak_ptrlock() 方法时,若返回 nullptr,则说明资源已经不存在,放弃对资源继续操作。否则,将返回一个 shared_ptr 对象,可以继续操作资源。

另外,一旦最后一个指向对象的 shared_ptr 被销毁,对象就会被释放。即使有 weak_ptr 指向对象,对象也还是会被释放。


小艾问:既然它这引用都不算数,那它有什么用呢?

小牛答:别急,我们来慢慢讲。


enable_shared_from_this 机制

小牛:考虑下面这样一个场景:

在多线程环境中,假设有一个对象池类 ObjectPool 和一个对象类 Object。ObjectPool 类主要实现通过不同的 key 返回对应 Object 对象。

要求同一程序中由 Object 类实例出的不同对象只有一个,即当多处用到同一个对象,Object 对象应该被共享。同时当对象不再需要时应该被析构,并删除对应的 key。

多线程应用场景

小艾说:这还不简单,看我的。代码刷的一下就写完了。

//场景代码
#include<memory>
class ObjectPool:boost::noncopyable
{
public:
 shared_ptr<Object> get(const string& key)
 
{
 shared_ptr<Object> shObject;
    MutexLockGuard lock(mutex);
 weak_ptr<Object>& wkObject=object[key];
 shObject=wkObject.lock();
 //对象存在,提升成功并返回
 if(!shObject){
  /*对象不存在,提升失败,shOject重新
  指向新创建的Object对象,
  并绑定回调函数,让对象Oject需要析构时
  调用OjectPool对象的成员函数*/

       shObject.reset(new Object(key),
                    boost::bind(&
        ObjectPool::deleteObject,this,
        _1));
       wkObject=shObject;
 }
 return shObject;
 }
 private:
 void deleteObject(Object* obj)
 
{   /*回调函数,在对象需要析构时调用,从map中
 删除对象和对应的key*/

  if(obj){
   MutexLockGuard lock(mutex);
   object.erase(obj->key());
  }
  delete obj;
 }
 mutable MutexLock mutex;
 std::map<string,weak_ptr<Object>> object;
 /*map中不能使用shared_ptr,这会导致Oject对象永远不会被销
 毁*/

};

小牛说:你这有问题啊?

小艾答:有什么问题?为了实现 Object 类析构时调用 ObjectPool 的回调函数,代码中把 ObjectPool 的 this 指针保存在了 boost::function 处。

小牛说:那线程安全问题就来了。如果 ObjectPool 先于 Object 对象析构,就会发生 core dump。因为 ObjectPool 对象已经不存在了,也就没有办法调用其成员方法。

小艾问:那怎么解决呢?

小牛说:简单啊,只需将 this 指针替换成指向当前对象的 shared_ptr,从而保证在 Object 对象需要调用 ObjectPool::deleteObjectObjectPool 还活着。你要不试试实现一下?

小艾说:那我写一个吧。

shared_ptr<A> getSharedPtr() 

   return shared_ptr<A>(this); 
}

小牛答:问题来了,在多线程环境中,在需要返回 this 对象时是无法得知对象的生存情况的。因此不能直接返回 this 对象

给你普及个解决方法吧,你可以通过继承 enable_shared_from_this 模板对象,然后调用从基类继承而来的 shared_from_this 方法来安全返回指向同一资源对象的 shared_ptr

小艾:为什么继承 enable_shared_from_this 模板对象就可以安全返回?

小牛:在回答你的问题前,我们先来讲讲 shared_ptr构造函数拷贝构造函数对资源和引用计数影响的区别。

下面从 shared_ptr实现原理来看:

shared_ptr_Ptr_base 继承了 element_type_Ref_count_base 类型的两个成员变量。

template<class _Ty>
class _Ptr_base
{
 
private:
 element_type * _Ptr{nullptr}; // 指向资源的指针
 _Ref_count_base * _Rep{nullptr}; // 指向资源引用计数的指针
};

_Ref_count_base 中定义了原子类型的变量 _Uses_Weaks,它们分别记录资源的引用个数和资源观察者的个数。

class __declspec(novtable) _Ref_count_base
{
 
private:
 _Atomic_counter_t _Uses;//记录资源引用个数
 _Atomic_counter_t _Weaks;//记录观察者个数
}

当要使用 shared_ptr 管理同一资源,调用 shared_ptr 的构造函数和拷贝构造函数是不一样的,它们虽然使得不同 shared_ptr 指向同一资源,但管理引用计数资源的方式却不一样。

下面给出两个 shared_ptr 管理同一资源(A对象)使用不同构造函数对引用计数对象的影响。

方式1:调用构造函数

class A
{

  public:
  A(){}
  ~A(){}
};
A *p = new A(); 
shared_ptr<A> ptr1(p);//调用构造函数
shared_ptr<A> ptr2(p);//调用构造函数
方式1:调用构造函数

如上图所示,方式1中 ptr1 和 ptr2 都调用了 shared_ptr 的构造函数,该构造方式使得 ptr1ptr2 都开辟了自已的引用资源对象 _Ref_count_base,即 _Ref_count_base 有两个,都记录了 A 对象的引用计数为 1,析构时 ptr1ptr2 的引用计数各自减为 1,导致 A 对象析构两次,出现逻辑错误。

方式2:调用拷贝构造函数

class A
{

public:
   A(){}
   ~A(){}
}
A *p = new A(); 
shared_ptr<A> ptr1(p);//调用构造函数
shared_ptr<A> ptr2(ptr1);//调用拷贝构造函数
方式2:调用拷贝构造函数

如上图所示,方式2中由于 ptr2 拷贝构造 ptr1,它们引用的 _Ref_count_base 是同一个,因此引用计数为 2,析构的时候 A 对象只析构一次,正常运行。

在明白了 shared_ptr 构造函数和拷贝构造函数的做的事情不同后,就能理解当需要返回一个需要 shared_ptr 管理的对象为什么不能写成 return shared_ptr< A >(this) 了。

小艾:说的没错,因为这样会调用 shared_ptr 的构造函数,对于 this 对象再创建一个新的引用计数对象,从而导致对象多次析构而出现逻辑错误。

小牛:再给你深入讲讲 enable_shared_from_this 的实现机制。

如下所示,enable_shared_from_this 类中包含一个作为观察者的成员变量。

template<class _Ty>
class enable_shared_from_this
{
 
public:
   mutable weak_ptr<_Ty> _Wptr;//指向资源
};

当一个类继承了 enable_shared_from_this 类,就继承了 _Wptr 这个成员变量。

当使用 shared_ptr< A >(new A()) 第一次构造智能指针对象时,就会初始化一个作为观察者的弱智能指针 _Wptr 指向A对象资源。

再通过 shared_from_this() 方法代替 shared_ptr 的普通构造函数来返回一个 shared_ptr 对象,从而避免产生额外的引用计数对象。

shared_ptr<A> getSharedPtr() 

   return shared_from_this(); 
}

shared_from_this 函数中,主要尝试将弱智能指针提升为强智能指针来返回一个 shared_ptr 对象。

这样还能在多线程环境中判断对象是否存活,存活即提升成功,安全返回。如果对象已经析构,则放弃提升,即起到了保证线程安全的作用。

小牛:了解了enable_shared_from_this,要不再试试改代码?

小艾:那我来改一下之前的代码。

第一处修改:

class ObjectPool:boost::noncopyable
//为
class ObjectPool:public boost::enable_shared_from_this<ObjectPool>,
                boost::noncopyable
{/*...*/};

第二处修改:

//改变
shared_ptr<Object> get(const string& key)
{
  /*...*/
  shObject.reset(new Object(key),
                     boost::bind(&ObjectPool::deleteObject,this,_1));
  /*...*/   
}
//为
shared_ptr<Object> get(const string& key)
{
  /*...*/
  shObject.reset(new Object(key),
                     boost::bind(&ObjectPool::deleteObject,shared_from_this(),_1));
  /*...*/   
}

完整代码:

#include<memory>
class ObjectPool:public boost::enable_shared_from_this<ObjectPool>,
                boost::noncopyable
{
public:
shared_ptr<Object> get(const string& key)
{
   shared_ptr<Object> shObject;
   MutexLockGuard lock(mutex);
   weak_ptr<Object>& wkObject=object[key];
   shObject=wkObject.lock();//对象存在,提升成功并返回
   if(!shObject){
    /*对象不存在,提升失败,shOject重新指向新创建的
    Object对象,并绑定回调函数,让对象Oject需要析构时
    调用OjectPool对象的成员函数*/

      shObject.reset(new Object(key),
                      boost::bind(&
          ObjectPool::deleteObject,shared_from_this(),
          _1));
      wkObject=shObject;
   }
   return shObject;
}
private:
void deleteObject(Object* obj)
{   /*回调函数,在对象需要析构时调用,从map中删除对象和对
应的key*/

    if(obj){
     MutexLockGuard lock(mutex);
     object.erase(obj->key());
    }
    delete obj;
}
mutable MutexLock mutex;
std::map<string,weak_ptr<Object>> object;
/*map中不能使用shared_ptr,这会导致Oject对象永远不会被销
毁*/

};

小牛:不错不错,这下懂了 shared_ptrweak_ptr 结合的用法了吧。

带引用计数智能指针总结:

  1. shared_ptr 会增加资源的引用计数,常用于管理对象的生命周期。
  2. weak_ptr 不会增加资源的引用计数,常作为观察者用来判断对象是否存活。
  3. 使用 shared_ptr 的普通拷贝构造函数会产生额外的引用计数对象,可能导致对象多次析构。使用 shared_ptr 的拷贝构造函数则只影响同一资源的同一引用计数的增减。
  4. 当需要返回指向当前对象的 shared_ptr 时,优先使用 enable_shared_from_this 机制。

总结

今天我们了解了面试中常常会问到的C++ 智能指针的相关知识点,结合源码和示例理清各种智能指针的特点。

并且结合一个实际的多线程应用场景,讲解了enable_shared_from_this 机制,希望能对大家的学习有所帮助。

总结

参考

  1. https://blog.csdn.net/qiangweiyuan/article/details/88562935
  2. 《Linux多线程服务端编程使用muduo C++ 网络库》
  3. 《C++ Primer》
  4. 《More Effective C++》
C语言与CPP编程 C语言/C++开发,C语言/C++基础知识,C语言/C++学习路线,C语言/C++进阶,数据结构;算法;python;计算机基础等
评论
  • 在当前人工智能(AI)与物联网(IoT)的快速发展趋势下,各行各业的数字转型与自动化进程正以惊人的速度持续进行。如今企业在设计与营运技术系统时所面临的挑战不仅是技术本身,更包含硬件设施、第三方软件及配件等复杂的外部因素。然而这些系统往往讲究更精密的设计与高稳定性,哪怕是任何一个小小的问题,都可能对整体业务运作造成严重影响。 POS应用环境与客户需求以本次分享的客户个案为例,该客户是一家全球领先的信息技术服务与数字解决方案提供商,遭遇到一个由他们所开发的POS机(Point of Sal
    百佳泰测试实验室 2025-01-09 17:35 78浏览
  • 根据环洋市场咨询(Global Info Research)项目团队最新调研,预计2030年全球中空长航时无人机产值达到9009百万美元,2024-2030年期间年复合增长率CAGR为8.0%。 环洋市场咨询机构出版了的【全球中空长航时无人机行业总体规模、主要厂商及IPO上市调研报告,2025-2031】研究全球中空长航时无人机总体规模,包括产量、产值、消费量、主要生产地区、主要生产商及市场份额,同时分析中空长航时无人机市场主要驱动因素、阻碍因素、市场机遇、挑战、新产品发布等。报告从中空长航时
    GIRtina 2025-01-09 10:35 74浏览
  • 在过去十年中,自动驾驶和高级驾驶辅助系统(AD/ADAS)软件与硬件的快速发展对多传感器数据采集的设计需求提出了更高的要求。然而,目前仍缺乏能够高质量集成多传感器数据采集的解决方案。康谋ADTF正是应运而生,它提供了一个广受认可和广泛引用的软件框架,包含模块化的标准化应用程序和工具,旨在为ADAS功能的开发提供一站式体验。一、ADTF的关键之处!无论是奥迪、大众、宝马还是梅赛德斯-奔驰:他们都依赖我们不断发展的ADTF来开发智能驾驶辅助解决方案,直至实现自动驾驶的目标。从新功能的最初构思到批量生
    康谋 2025-01-09 10:04 75浏览
  • HDMI 2.2 规格将至,开启视听新境界2025年1月6日,HDMI Forum, Inc. 宣布即将发布HDMI规范2.2版本。新HDMI规范为规模庞大的 HDMI 生态系统带来更多选择,为创建、分发和体验理想的终端用户效果提供更先进的解决方案。新技术为电视、电影和游戏工作室等内容制作商在当前和未来提供更高质量的选择,同时实现多种分发平台。96Gbps的更高带宽和新一代 HDMI 固定比率速率传输(Fixed Rate Link)技术为各种设备应用提供更优质的音频和视频。终端用户显示器能以最
    百佳泰测试实验室 2025-01-09 17:33 84浏览
  • 故障现象一辆2017款东风风神AX7车,搭载DFMA14T发动机,累计行驶里程约为13.7万km。该车冷起动后怠速运转正常,热机后怠速运转不稳,组合仪表上的发动机转速表指针上下轻微抖动。 故障诊断 用故障检测仪检测,发动机控制单元中无故障代码存储;读取发动机数据流,发现进气歧管绝对压力波动明显,有时能达到69 kPa,明显偏高,推断可能的原因有:进气系统漏气;进气歧管绝对压力传感器信号失真;发动机机械故障。首先从节气门处打烟雾,没有发现进气管周围有漏气的地方;接着拔下进气管上的两个真空
    虹科Pico汽车示波器 2025-01-08 16:51 111浏览
  • 一个真正的质量工程师(QE)必须将一件产品设计的“意图”与系统的可制造性、可服务性以及资源在现实中实现设计和产品的能力结合起来。所以,可以说,这确实是一种工程学科。我们常开玩笑说,质量工程师是工程领域里的「侦探」、「警察」或「律师」,守护神是"墨菲”,信奉的哲学就是「墨菲定律」。(注:墨菲定律是一种启发性原则,常被表述为:任何可能出错的事情最终都会出错。)做质量工程师的,有时会不受欢迎,也会被忽视,甚至可能遭遇主动或被动的阻碍,而一旦出了问题,责任往往就落在质量工程师的头上。虽然质量工程师并不负
    优思学院 2025-01-09 11:48 97浏览
  •  在全球能源结构加速向清洁、可再生方向转型的今天,风力发电作为一种绿色能源,已成为各国新能源发展的重要组成部分。然而,风力发电系统在复杂的环境中长时间运行,对系统的安全性、稳定性和抗干扰能力提出了极高要求。光耦(光电耦合器)作为一种电气隔离与信号传输器件,凭借其优秀的隔离保护性能和信号传输能力,已成为风力发电系统中不可或缺的关键组件。 风力发电系统对隔离与控制的需求风力发电系统中,包括发电机、变流器、变压器和控制系统等多个部分,通常工作在高压、大功率的环境中。光耦在这里扮演了
    晶台光耦 2025-01-08 16:03 87浏览
  • 光伏逆变器是一种高效的能量转换设备,它能够将光伏太阳能板(PV)产生的不稳定的直流电压转换成与市电频率同步的交流电。这种转换后的电能不仅可以回馈至商用输电网络,还能供独立电网系统使用。光伏逆变器在商业光伏储能电站和家庭独立储能系统等应用领域中得到了广泛的应用。光耦合器,以其高速信号传输、出色的共模抑制比以及单向信号传输和光电隔离的特性,在光伏逆变器中扮演着至关重要的角色。它确保了系统的安全隔离、干扰的有效隔离以及通信信号的精准传输。光耦合器的使用不仅提高了系统的稳定性和安全性,而且由于其低功耗的
    晶台光耦 2025-01-09 09:58 62浏览
  • 职场是人生的重要战场,既是谋生之地,也是实现个人价值的平台。然而,有些思维方式却会悄无声息地拖住你的后腿,让你原地踏步甚至退步。今天,我们就来聊聊职场中最忌讳的五种思维方式,看看自己有没有中招。1. 固步自封的思维在职场中,最可怕的事情莫过于自满于现状,拒绝学习和改变。世界在不断变化,行业的趋势、技术的革新都在要求我们与时俱进。如果你总觉得自己的方法最优,或者害怕尝试新事物,那就很容易被淘汰。与其等待机会找上门,不如主动出击,保持学习和探索的心态。加入优思学院,可以帮助你快速提升自己,与行业前沿
    优思学院 2025-01-09 15:48 69浏览
  • 在智能网联汽车中,各种通信技术如2G/3G/4G/5G、GNSS(全球导航卫星系统)、V2X(车联网通信)等在行业内被广泛使用。这些技术让汽车能够实现紧急呼叫、在线娱乐、导航等多种功能。EMC测试就是为了确保在复杂电磁环境下,汽车的通信系统仍然可以正常工作,保护驾乘者的安全。参考《QCT-基于LTE-V2X直连通信的车载信息交互系统技术要求及试验方法-1》标准10.5电磁兼容试验方法,下面将会从整车功能层面为大家解读V2X整车电磁兼容试验的过程。测试过程揭秘1. 设备准备为了进行电磁兼容试验,技
    北汇信息 2025-01-09 11:24 80浏览
  • 1月7日-10日,2025年国际消费电子产品展览会(CES 2025)盛大举行,广和通发布Fibocom AI Stack,赋智千行百业端侧应用。Fibocom AI Stack提供集高性能模组、AI工具链、高性能推理引擎、海量模型、支持与服务一体化的端侧AI解决方案,帮助智能设备快速实现AI能力商用。为适应不同端侧场景的应用,AI Stack具备海量端侧AI模型及行业端侧模型,基于不同等级算力的芯片平台或模组,Fibocom AI Stack可将TensorFlow、PyTorch、ONNX、
    物吾悟小通 2025-01-08 18:17 72浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦