0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

ThreadLocal的作用以及應(yīng)用場景

倩倩 ? 來源:juejin.cn ? 作者:芋道源碼 ? 2022-09-19 10:56 ? 次閱讀


ThreadLocal的作用以及應(yīng)用場景

ThreadLocal算是一種并發(fā)容器吧,因為他的內(nèi)部是有ThreadLocalMap組成,ThreadLocal是為了解決多線程情況下變量不能被共享的問題,也就是多線程共享變量的問題。

ThreadLocalLock以及Synchronized的區(qū)別是:ThreadLocal是給每個線程分配一個變量(對象),各個線程都存有變量的副本,這樣每個線程都是使用自己(變量)對象實例,使線程與線程之間進行隔離;而LockSynchronized的方式是使線程有順序的執(zhí)行。

舉一個簡單的例子:目前有100個學生等待簽字,但是老師只有一個筆,那老師只能按順序的分給每個學生,等待A學生簽字完成然后將筆交給B學生,這就類似Lock,Synchronized的方式。而ThreadLocal是,老師直接拿出一百個筆給每個學生;再效率提高的同事也要付出一個內(nèi)存消耗;也就是以空間換時間的概念

基于 Spring Boot + MyBatis Plus + Vue & Element 實現(xiàn)的后臺管理系統(tǒng) + 用戶小程序,支持 RBAC 動態(tài)權(quán)限、多租戶、數(shù)據(jù)權(quán)限、工作流、三方登錄、支付、短信、商城等功能

  • 項目地址:https://gitee.com/zhijiantianya/ruoyi-vue-pro
  • 視頻教程:https://doc.iocoder.cn/video/

使用場景

Spring的事務(wù)隔離就是使用ThreadLocal和AOP來解決的;主要是TransactionSynchronizationManager這個類;

解決SimpleDateFormat線程不安全問題;

當我們使用SimpleDateFormatparse()方法的時候,parse()方法會先調(diào)用Calendar.clear()方法,然后調(diào)用Calendar.add()方法,如果一個線程先調(diào)用了add()方法,然后另一個線程調(diào)用了clear()方法;這時候parse()方法就會出現(xiàn)解析錯誤;如果不信我們可以來個例子:

publicclassSimpleDateFormatTest{

privatestaticSimpleDateFormatsimpleDateFormat=newSimpleDateFormat("yyyy-MM-dd");

publicstaticvoidmain(String[]args){
for(inti=0;i50;i++){
Threadthread=newThread(newRunnable(){
@Override
publicvoidrun(){
dateFormat();
}
});
thread.start();
}
}

/**
*字符串轉(zhuǎn)成日期類型
*/
publicstaticvoiddateFormat(){
try{
simpleDateFormat.parse("2021-5-27");
}catch(ParseExceptione){
e.printStackTrace();
}
}
}

這里我們只啟動了50個線程問題就會出現(xiàn),其實看巧不巧,有時候只有10個線程的情況就會出錯:

Exceptioninthread"Thread-40"java.lang.NumberFormatException:Forinputstring:""
atjava.lang.NumberFormatException.forInputString(NumberFormatException.java:65)
atjava.lang.Long.parseLong(Long.java:601)
atjava.lang.Long.parseLong(Long.java:631)
atjava.text.DigitList.getLong(DigitList.java:195)
atjava.text.DecimalFormat.parse(DecimalFormat.java:2084)
atjava.text.SimpleDateFormat.subParse(SimpleDateFormat.java:1869)
atjava.text.SimpleDateFormat.parse(SimpleDateFormat.java:1514)
atjava.text.DateFormat.parse(DateFormat.java:364)
atcn.haoxy.use.lock.sdf.SimpleDateFormatTest.dateFormat(SimpleDateFormatTest.java:36)
atcn.haoxy.use.lock.sdf.SimpleDateFormatTest$1.run(SimpleDateFormatTest.java:23)
atjava.lang.Thread.run(Thread.java:748)
Exceptioninthread"Thread-43"java.lang.NumberFormatException:multiplepoints
atsun.misc.FloatingDecimal.readJavaFormatString(FloatingDecimal.java:1890)
atsun.misc.FloatingDecimal.parseDouble(FloatingDecimal.java:110)
atjava.lang.Double.parseDouble(Double.java:538)
atjava.text.DigitList.getDouble(DigitList.java:169)
atjava.text.DecimalFormat.parse(DecimalFormat.java:2089)
atjava.text.SimpleDateFormat.subParse(SimpleDateFormat.java:1869)
atjava.text.SimpleDateFormat.parse(SimpleDateFormat.java:1514)
atjava.text.DateFormat.parse(DateFormat.java:364)
at.............

其實解決這個問題很簡單,讓每個線程new一個自己的SimpleDateFormat,但是如果100個線程都要new100個SimpleDateFormat嗎?

當然我們不能這么做,我們可以借助線程池加上ThreadLocal來解決這個問題:

publicclassSimpleDateFormatTest{

privatestaticThreadLocallocal=newThreadLocal(){
@Override
//初始化線程本地變量
protectedSimpleDateFormatinitialValue(){
returnnewSimpleDateFormat("yyyy-MM-dd");
}
};

publicstaticvoidmain(String[]args){
ExecutorServicees=Executors.newCachedThreadPool();
for(inti=0;i500;i++){
es.execute(()->{
//調(diào)用字符串轉(zhuǎn)成日期方法
dateFormat();
});
}
es.shutdown();
}
/**
*字符串轉(zhuǎn)成日期類型
*/
publicstaticvoiddateFormat(){
try{
//ThreadLocal中的get()方法
local.get().parse("2021-5-27");
}catch(ParseExceptione){
e.printStackTrace();
}
}
}

這樣就優(yōu)雅的解決了線程安全問題;

解決過度傳參問題;例如一個方法中要調(diào)用好多個方法,每個方法都需要傳遞參數(shù);例如下面示例:

voidwork(Useruser){
getInfo(user);
checkInfo(user);
setSomeThing(user);
log(user);
}

用了ThreadLocal之后:

publicclassThreadLocalStu{

privatestaticThreadLocaluserThreadLocal=newThreadLocal<>();

voidwork(Useruser){
try{
userThreadLocal.set(user);
getInfo();
checkInfo();
someThing();
}finally{
userThreadLocal.remove();
}
}

voidsetInfo(){
Useru=userThreadLocal.get();
//.....
}

voidcheckInfo(){
Useru=userThreadLocal.get();
//....
}

voidsomeThing(){
Useru=userThreadLocal.get();
//....
}
}

每個線程內(nèi)需要保存全局變量(比如在登錄成功后將用戶信息存到ThreadLocal里,然后當前線程操作的業(yè)務(wù)邏輯直接get取就完事了,有效的避免的參數(shù)來回傳遞的麻煩之處),一定層級上減少代碼耦合度。

  • 比如存儲 交易id等信息。每個線程私有。
  • 比如aop里記錄日志需要before記錄請求id,end拿出請求id,這也可以。
  • 比如jdbc連接池(很典型的一個ThreadLocal用法)
  • ....等等....

基于 Spring Cloud Alibaba + Gateway + Nacos + RocketMQ + Vue & Element 實現(xiàn)的后臺管理系統(tǒng) + 用戶小程序,支持 RBAC 動態(tài)權(quán)限、多租戶、數(shù)據(jù)權(quán)限、工作流、三方登錄、支付、短信、商城等功能

  • 項目地址:https://gitee.com/zhijiantianya/yudao-cloud
  • 視頻教程:https://doc.iocoder.cn/video/

原理分析

上面我們基本上知道了ThreadLocal的使用方式以及應(yīng)用場景,當然應(yīng)用場景不止這些這只是工作中常用到的場景;下面我們對它的原理進行分析;

我們先看一下它的set()方法;

publicvoidset(Tvalue){
Threadt=Thread.currentThread();
ThreadLocalMapmap=getMap(t);
if(map!=null)
map.set(this,value);
else
createMap(t,value);
}

是不是特別簡單,首先獲取當前線程,用當前線程作為key,去獲取ThreadLocalMap,然后判斷map是否為空,不為空就將當前線程作為key,傳入的value作為map的value值;如果為空就創(chuàng)建一個ThreadLocalMap,然后將key和value方進去;從這里可以看出value值是存放到ThreadLocalMap中;

然后我們看看ThreadLocalMap是怎么來的?先看下getMap()方法:

//在Thread類中維護了threadLocals變量,注意是Thread類
ThreadLocal.ThreadLocalMapthreadLocals=null;

//在ThreadLocal類中的getMap()方法
ThreadLocalMapgetMap(Threadt){
returnt.threadLocals;
}

這就能解釋每個線程中都有一個ThreadLocalMap,因為ThreadLocalMap的引用在Thread中維護;這就確保了線程間的隔離;

我們繼續(xù)回到set()方法,看到當map等于空的時候createMap(t, value);

voidcreateMap(Threadt,TfirstValue){
t.threadLocals=newThreadLocalMap(this,firstValue);
}

這里就是new了一個ThreadLocalMap然后賦值給threadLocals成員變量;ThreadLocalMap構(gòu)造方法:

ThreadLocalMap(ThreadLocalfirstKey,ObjectfirstValue){
//初始化一個Entry
table=newEntry[INITIAL_CAPACITY];
//計算key應(yīng)該存放的位置
inti=firstKey.threadLocalHashCode&(INITIAL_CAPACITY-1);
//將Entry放到指定位置
table[i]=newEntry(firstKey,firstValue);
size=1;
//設(shè)置數(shù)組的大小16*2/3=10,類似HashMap中的0.75*16=12
setThreshold(INITIAL_CAPACITY);
}

這里寫有個大概的印象,后面對ThreadLocalMap內(nèi)部結(jié)構(gòu)還會進行詳細的講解;

下面我們再去看一下get()方法:

publicTget(){
Threadt=Thread.currentThread();
//用當前線程作為key去獲取ThreadLocalMap
ThreadLocalMapmap=getMap(t);
if(map!=null){
//map不為空,然后獲取map中的Entry
ThreadLocalMap.Entrye=map.getEntry(this);
if(e!=null){
@SuppressWarnings("unchecked")
//如果Entry不為空就獲取對應(yīng)的value值
Tresult=(T)e.value;
returnresult;
}
}
//如果map為空或者entry為空的話通過該方法初始化,并返回該方法的value
returnsetInitialValue();
}

get()方法和set()都比較容易理解,如果map等于空的時候或者entry等于空的時候我們看看setInitialValue()方法做了什么事:

privateTsetInitialValue(){
//初始化變量值由子類去實現(xiàn)并初始化變量
Tvalue=initialValue();
Threadt=Thread.currentThread();
//這里再次getMap();
ThreadLocalMapmap=getMap(t);
if(map!=null)
map.set(this,value);
else
//和set()方法中的
createMap(t,value);
returnvalue;
}

下面我們再去看一下ThreadLocal中的initialValue()方法:

protectedTinitialValue(){
returnnull;
}

設(shè)置初始值,由子類去實現(xiàn);就例如我們上面的例子,重寫ThreadLocal類中的initialValue()方法:

privatestaticThreadLocallocal=newThreadLocal(){
@Override
//初始化線程本地變量
protectedSimpleDateFormatinitialValue(){
returnnewSimpleDateFormat("yyyy-MM-dd");
}
};

createMap()方法和上面set()方法中createMap()方法同一個,就不過多的敘述了;剩下還有一個removve()方法

publicvoidremove(){
ThreadLocalMapm=getMap(Thread.currentThread());
if(m!=null)
//2.從map中刪除以當前threadLocal實例為key的鍵值對
m.remove(this);
}

源碼的講解就到這里,也都比較好理解,下面我們看看ThreadLocalMap的底層結(jié)構(gòu)

ThreadLocalMap的底層結(jié)構(gòu)

上面我們已經(jīng)了解了ThreadLocal的使用場景以及它比較重要的幾個方法;下面我們再去它的內(nèi)部結(jié)構(gòu);經(jīng)過上的源碼分析我們可以看到數(shù)據(jù)其實都是存放到了ThreadLocal中的內(nèi)部類ThreadLocalMap中;而ThreadLocalMap中又維護了一個Entry對象,也就說數(shù)據(jù)最終是存放到Entry對象中的;

staticclassThreadLocalMap{

staticclassEntryextendsWeakReference<ThreadLocal>{
/**ThevalueassociatedwiththisThreadLocal.*/
Objectvalue;

Entry(ThreadLocalk,Objectv){
super(k);
value=v;
}

}
ThreadLocalMap(ThreadLocalfirstKey,ObjectfirstValue){
table=newEntry[INITIAL_CAPACITY];
inti=firstKey.threadLocalHashCode&(INITIAL_CAPACITY-1);
table[i]=newEntry(firstKey,firstValue);
size=1;
setThreshold(INITIAL_CAPACITY);
}
//....................
}

Entry的構(gòu)造方法是以當前線程為key,變量值Object為value進行存儲的;在上面的源碼中ThreadLocalMap的構(gòu)造方法中也涉及到了Entry;看到Entry是一個數(shù)組;初始化長度為INITIAL_CAPACITY = 16;因為 Entry 繼承了 WeakReference,在 Entry 的構(gòu)造方法中,調(diào)用了 super(k)方法就會將 threadLocal 實例包裝成一個 WeakReferenece。這也是ThreadLocal會產(chǎn)生內(nèi)存泄露的原因;

內(nèi)存泄露產(chǎn)生的原因

a6f698ac-37c4-11ed-ba43-dac502259ad0.png

如圖所示存在一條引用鏈: Thread Ref->Thread->ThreadLocalMap->Entry->Key:Value,經(jīng)過上面的講解我們知道ThreadLocal作為Key,但是被設(shè)置成了弱引用,弱引用在JVM垃圾回收時是優(yōu)先回收的,就是說無論內(nèi)存是否足夠弱引用對象都會被回收;弱引用的生命周期比較短;當發(fā)生一次GC的時候就會變成如下:

a724c54c-37c4-11ed-ba43-dac502259ad0.png

TreadLocalMap中出現(xiàn)了Key為null的Entry,就沒有辦法訪問這些key為null的Entry的value,如果線程遲遲不結(jié)束(也就是說這條引用鏈無意義的一直存在)就會造成value永遠無法回收造成內(nèi)存泄露;如果當前線程運行結(jié)束Thread,ThreadLocalMap,Entry之間沒有了引用鏈,在垃圾回收的時候就會被回收;但是在開發(fā)中我們都是使用線程池的方式,線程池的復用不會主動結(jié)束;所以還是會存在內(nèi)存泄露問題;

解決方法也很簡單,就是在使用完之后主動調(diào)用remove()方法釋放掉;

解決Hash沖突

記得在大學學習數(shù)據(jù)結(jié)構(gòu)的時候?qū)W習了很多種解決hash沖突的方法;例如:

線性探測法(開放地址法的一種): 計算出的散列地址如果已被占用,則按順序找下一個空位。如果找到末尾還沒有找到空位置就從頭重新開始找;

a7660520-37c4-11ed-ba43-dac502259ad0.png

二次探測法(開放地址法的一種)

a7c2781e-37c4-11ed-ba43-dac502259ad0.png

鏈地址法:鏈地址是對每一個同義詞都建一個單鏈表來解決沖突,HashMap采用的是這種方法;

aa7fe488-37c4-11ed-ba43-dac502259ad0.png

多重Hash法: 在key沖突的情況下多重hash,直到不沖突為止,這種方式不易產(chǎn)生堆積但是計算量太大;

公共溢出區(qū)法: 這種方式需要兩個表,一個存基礎(chǔ)數(shù)據(jù),另一個存放沖突數(shù)據(jù)稱為溢出表;

上面的圖片都是在網(wǎng)上找到的一些資料,和大學時學習時的差不多我就直接拿來用了;也當自己復習了一遍;

介紹了那么多解決Hash沖突的方法,那ThreadLocalMap使用的哪一種方法呢?我們可以看一下源碼:

privatevoidset(ThreadLocalkey,Objectvalue){
Entry[]tab=table;
intlen=tab.length;
//根據(jù)HashCode&數(shù)組長度計算出數(shù)組該存放的位置
inti=key.threadLocalHashCode&(len-1);
//遍歷Entry數(shù)組中的元素
for(Entrye=tab[i];
e!=null;
e=tab[i=nextIndex(i,len)]){
ThreadLocalk=e.get();
//如果這個Entry對象的key正好是即將設(shè)置的key,那么就刷新Entry中的value;
if(k==key){
e.value=value;
return;
}
//entry!=null,key==null時,說明threadLcoal這key已經(jīng)被GC了,這里就是上面說到
//會有內(nèi)存泄露的地方,當然作者也知道這種情況的存在,所以這里做了一個判斷進行解決臟的
//entry(數(shù)組中不想存有過時的entry),但是也不能解決泄露問題,因為舊value還存在沒有消失
if(k==null){
//用當前插入的值代替掉這個key為null的“臟”entry
replaceStaleEntry(key,value,i);
return;
}
}
//新建entry并插入table中i處
tab[i]=newEntry(key,value);
intsz=++size;
if(!cleanSomeSlots(i,sz)&&sz>=threshold)
rehash();
}

從這里我們可以看出使用的是線性探測的方式來解決hash沖突!

源碼中通過nextIndex(i, len)方法解決 hash 沖突的問題,該方法為((i + 1 < len) ? i + 1 : 0);,也就是不斷往后線性探測,直到找到一個空的位置,當?shù)焦1砟┪驳臅r候還沒有找到空位置再從 0 開始找,成環(huán)形!

使用ThreadLocal時對象存在哪里?

在java中,棧內(nèi)存歸屬于單個線程,每個線程都會有一個棧內(nèi)存,其存儲的變量只能在其所屬線程中可見,即棧內(nèi)存可以理解成線程的私有變量,而堆內(nèi)存中的變量對所有線程可見,可以被所有線程訪問!

那么ThreadLocal的實例以及它的值是不是存放在棧上呢?其實不是的,因為ThreadLocal的實例實際上也是被其創(chuàng)建的類持有,(更頂端應(yīng)該是被線程持有),而ThreadLocal的值其實也是被線程實例持有,它們都是位于堆上,只是通過一些技巧將可見性修改成了線程可見。

審核編輯 :李倩


聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 代碼
    +關(guān)注

    關(guān)注

    30

    文章

    4798

    瀏覽量

    68725
  • spring
    +關(guān)注

    關(guān)注

    0

    文章

    340

    瀏覽量

    14353
  • 線程
    +關(guān)注

    關(guān)注

    0

    文章

    505

    瀏覽量

    19705

原文標題:ThreadLocal 你真的用不上嗎?

文章出處:【微信號:芋道源碼,微信公眾號:芋道源碼】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    測試接收機的技術(shù)原理和應(yīng)用場景

    、數(shù)字信號、射頻信號等,并進行相應(yīng)的測量和分析。此外,它還具有多種優(yōu)秀性能,如高靈敏度、低噪聲、寬頻帶等,以及本振抑制、鏡像抑制等能力,能夠有效抑制干擾信號。二、應(yīng)用場景測試接收機在多個領(lǐng)域都有廣泛
    發(fā)表于 12-03 14:23

    倍頻器的技術(shù)原理和應(yīng)用場景

    倍頻器是一種用于將輸入信號的頻率倍增的電子設(shè)備,以下是關(guān)于倍頻器的技術(shù)原理和應(yīng)用場景的詳細解釋:技術(shù)原理倍頻器的技術(shù)原理主要基于非線性元件(如二極管、晶體管等)的特性和頻率變換技術(shù)。 非線性元件
    發(fā)表于 11-29 14:49

    模塊化儀器的技術(shù)原理和應(yīng)用場景

    。 二、應(yīng)用場景 自動化測試系統(tǒng):模塊化儀器在自動化測試系統(tǒng)中發(fā)揮著重要作用,可以適應(yīng)多種測試需求,提高測試效率和準確性。它們通常由軟件驅(qū)動,便于集成到測試系統(tǒng)中,實現(xiàn)自動化測量和報告。 通信與衛(wèi)星領(lǐng)域
    發(fā)表于 11-28 15:09

    系統(tǒng)放大器的技術(shù)原理和應(yīng)用場景

    系統(tǒng)放大器是一種重要的電子設(shè)備,其技術(shù)原理和應(yīng)用場景都具有一定的專業(yè)性和廣泛性。以下是對系統(tǒng)放大器的技術(shù)原理和應(yīng)用場景的詳細介紹:一、技術(shù)原理系統(tǒng)放大器的工作原理基于電子器件的非線性特性,通過控制
    發(fā)表于 11-18 14:46

    寬帶放大器的技術(shù)原理和應(yīng)用場景

    的均衡性和穩(wěn)定性。這要求寬帶放大器在設(shè)計時,需要考慮多種因素,如帶寬、增益、輸出功率以及在各種幅度和相位條件下失配的行為特性、線性度、諧波特性、功率效率等,并進行綜合考慮和優(yōu)化。二、應(yīng)用場景寬帶放大器因其
    發(fā)表于 11-13 14:35

    移動終端測試儀的技術(shù)原理和應(yīng)用場景

    ,確保設(shè)備在導航服務(wù)中的準確性和可靠性。 應(yīng)用場景移動終端測試儀的應(yīng)用場景廣泛,涵蓋了從研發(fā)到生產(chǎn)、從維護到監(jiān)管的多個環(huán)節(jié): 移動維修服務(wù):維修技術(shù)人員可以使用便攜的綜測儀快速對手機進行全面檢測
    發(fā)表于 11-04 16:01

    實時示波器的技術(shù)原理和應(yīng)用場景

    有頻譜分析功能,可以將時域信號轉(zhuǎn)換為頻域信號,從而顯示信號的頻譜特性。綜上所述,實時示波器憑借其獨特的技術(shù)原理和廣泛的應(yīng)用場景,在電子工程和通信技術(shù)領(lǐng)域發(fā)揮著不可替代的作用。
    發(fā)表于 10-23 14:22

    源測量單元設(shè)備的技術(shù)原理和應(yīng)用場景

    ,SMU的功能和應(yīng)用領(lǐng)域也在不斷擴展,其在電子測試中的重要性不言而喻。綜上所述,SMU設(shè)備憑借其獨特的技術(shù)原理和廣泛的應(yīng)用場景,在現(xiàn)代電子測試中發(fā)揮著不可或缺的作用。對于工程師來說,深入了解和靈活應(yīng)用SMU將是提升電路板測試效果的關(guān)鍵所在。
    發(fā)表于 10-22 11:10

    參數(shù)分析儀的技術(shù)原理和應(yīng)用場景

    參數(shù)分析儀的技術(shù)原理和應(yīng)用場景因其具體類型和用途的不同而有所差異。以下是對參數(shù)分析儀技術(shù)原理和應(yīng)用場景的詳細歸納: 技術(shù)原理 基于物理性質(zhì)的測量: 某些參數(shù)分析儀通過測量樣品的物理性質(zhì)(如電阻
    發(fā)表于 10-17 14:42

    智能IC卡測試設(shè)備的技術(shù)原理和應(yīng)用場景

    應(yīng)用場景中的可靠性和安全性。 綜上所述,智能IC卡測試設(shè)備在保障IC卡質(zhì)量和性能、提高系統(tǒng)安全性和穩(wěn)定性等方面發(fā)揮著重要作用。隨著技術(shù)的不斷發(fā)展和應(yīng)用場景的不斷拓展,智能IC卡測試設(shè)備的應(yīng)用前景將更加廣闊。
    發(fā)表于 09-26 14:27

    脈沖式線圈測試儀的技術(shù)原理和應(yīng)用場景

    測試儀憑借其高效、非破壞性的測試原理以及廣泛的應(yīng)用場景,在電氣制造和檢測領(lǐng)域發(fā)揮著重要作用。通過使用該測試儀,企業(yè)可以及時發(fā)現(xiàn)并解決線圈質(zhì)量問題,提高產(chǎn)品的可靠性和安全性。
    發(fā)表于 09-18 14:29

    AIOT是什么意思?AIOT的應(yīng)用場景作用

    準的決策和更自動化的操作。 AIoT的核心理念是將AI技術(shù)應(yīng)用于物聯(lián)網(wǎng)設(shè)備和系統(tǒng)中,使其具備感知、學習、推理和決策等能力,從而實現(xiàn)更加智能化的服務(wù)和應(yīng)用場景。 AIoT的應(yīng)用場景 1. 智能家居:通過AIoT技術(shù),家中的各種設(shè)備可以相互連接并進行智能控制。例如,智能音箱可
    的頭像 發(fā)表于 07-12 15:48 ?1061次閱讀
    AIOT是什么意思?AIOT的應(yīng)<b class='flag-5'>用場景</b>和<b class='flag-5'>作用</b>

    STM32待機模式適合用于那些應(yīng)用場景?

    ,再不濟寫入flash也行??! 我能想到的就是用rtc喚醒,達到定時開機作用了 因此實在是想不到待機模式的其他應(yīng)用場景,求大神給予提示!!還有,待機模式喚醒后,有程序執(zhí)行入口么?
    發(fā)表于 05-07 07:46

    NanoEdge AI的技術(shù)原理、應(yīng)用場景及優(yōu)勢

    能耗并提高數(shù)據(jù)安全性。本文將對 NanoEdge AI 的技術(shù)原理、應(yīng)用場景以及優(yōu)勢進行綜述。 1、技術(shù)原理 NanoEdge AI 的核心技術(shù)包括邊緣計算、神經(jīng)網(wǎng)絡(luò)壓縮和低功耗硬件設(shè)計。邊緣計算
    發(fā)表于 03-12 08:09

    AG32VF-MIPI應(yīng)用場景

    to 1.2Gbps MIPI D-PHY以及DSI硬核 AG32VF-MIPI的應(yīng)用場景,包括了基本的MIPI屏幕驅(qū)動,以及各種顯示橋接場合,如下圖所示。 AG32VF-MIPI系列產(chǎn)品即將正式發(fā)布。
    發(fā)表于 01-22 08:56