主頁 > 作業系統 > 從0開始學FreeRTOS-(串列&串列項)-6

從0開始學FreeRTOS-(串列&串列項)-6

2020-09-14 09:38:11 作業系統

FreeRTOS串列&串列項的原始碼解讀

第一次看串列與串列項的時候,感覺很像是鏈表,雖然我自己的鏈表也不太會,但是就是感覺很像,

在FreeRTOS中,串列與串列項使用得非常多,是FreeRTOS的一個資料結構,學習過資料結構的同學都知道,資料結構能使我們處理資料更加方便快速,能快速找到資料,在FreeRTOS中,這種串列與串列項更是必不可少的,能讓我們的系統跑起來更加流暢迅速,

言歸正傳,FreeRTOS中使用了大量的串列(List)與串列項(Listitem),在FreeRTOS調度器中,就是用到這些來跟著任務,了解任務的狀態,處于掛起、阻塞態、還是就緒態亦或者是運行態,這些資訊都會在各自任務的串列中得到,

看任務控制塊(tskTaskControlBlock)中的兩個串列項:

ListItem_t xStateListItem; /* <任務的狀態串列專案參考的串列表示該任務的狀態(就緒,已阻止,暫停),*/
ListItem_t xEventListItem; /* <用于從事件串列中參考任務,*/

一個是狀態的串列項,一個是事件串列項,他們在創建任務就會被初始化,串列項的初始化是根據實際需要來初始化的,下面會說,

FreeRTOS串列&串列項的結構體

既然知道串列與串列項的重要性,那么我們來解讀FreeRTOS中的list.c與list.h的原始碼吧,從頭檔案lsit.h開始,看到定義了一些結構體:

 struct xLIST_ITEM
{
    listFIRST_LIST_ITEM_INTEGRITY_CHECK_VALUE /* <如果configUSE_LIST_DATA_INTEGRITY_CHECK_BYTES設定為1,則設定為已知值,*/
    configLIST_VOLATILE TickType_t xItemValue; /* <正在列出的值,在大多數情況下,這用于按降序對串列進行排序, */
    struct xLIST_ITEM * configLIST_VOLATILE pxNext; /* <指向串列中下一個ListItem_t的指標, */
    struct xLIST_ITEM * configLIST_VOLATILE pxPrevious; /* <指向串列中前一個ListItem_t的指標, */
    void * pvOwner; /* <指向包含串列專案的物件(通常是TCB)的指標,因此,包含串列專案的物件與串列專案本身之間存在雙向鏈接, */
    void * configLIST_VOLATILE pvContainer; /* <指向此串列專案所在串列的指標(如果有), */
    listSECOND_LIST_ITEM_INTEGRITY_CHECK_VALUE /* <如果configUSE_LIST_DATA_INTEGRITY_CHECK_BYTES設定為1,則設定為已知值,*/
};
typedef struct xLIST_ITEM ListItem_t; /* 由于某種原因,lint希望將其作為兩個單獨的定義, */

串列項結構體的一些注意的地方:

xItemValue 用于串列項的排序,類似1—2—3—4

pxNext 指向下一個串列項的指標
pxPrevious 指向上(前)一個串列項的指標

這兩個指標實作了類似雙向鏈表的功能

pvOwner 指向包含串列專案的物件(通常是任務控制塊TCB)的指標,因此,包含串列專案的物件與串列專案本身之間存在雙向鏈接,

pvContainer 記錄了該串列項屬于哪個串列,說白點就是這個兒子是誰生的,,,

同時定義了一個MINI的串列項的結構體,MINI串列項是刪減版的串列項,因為很多時候不需要完全版的串列項,就不用浪費那么多記憶體空間了,這或許就是FreeRTOS是輕量級作業系統的原因吧,能省一點是一點,MINI串列項:

struct xMINI_LIST_ITEM
{
    listFIRST_LIST_ITEM_INTEGRITY_CHECK_VALUE           /*< Set to a known value if configUSE_LIST_DATA_INTEGRITY_CHECK_BYTES is set to 1. */
    configLIST_VOLATILE TickType_t xItemValue;
    struct xLIST_ITEM * configLIST_VOLATILE pxNext;
    struct xLIST_ITEM * configLIST_VOLATILE pxPrevious;
};
typedef struct xMINI_LIST_ITEM MiniListItem_t;

再定義了一個串列的結構體,可能看到這里,一些同學已經蒙了,串列與串列項是啥關系啊,按照杰杰的理解,是類似父子關系的,一個串列中,包含多個串列項,就像一個父親,生了好多孩子,而串列就是父親,串列項就是孩子,

typedef struct xLIST
{
    listFIRST_LIST_INTEGRITY_CHECK_VALUE /* <如果configUSE_LIST_DATA_INTEGRITY_CHECK_BYTES設定為1,則設定為已知值,*/
    configLIST_VOLATILE UBaseType_t uxNumberOfItems;
    ListItem_t * configLIST_VOLATILE pxIndex; /* <用于遍歷串列, 指向由listGET_OWNER_OF_NEXT_ENTRY()呼叫回傳的后一個串列項,*/
    MiniListItem_t xListEnd; /* <List item包含最大可能的專案值,這意味著它始終在串列的末尾,因此用作標記,*/
    listSECOND_LIST_INTEGRITY_CHECK_VALUE /* <如果configUSE_LIST_DATA_INTEGRITY_CHECK_BYTES設定為1,則設定為已知值,*/
} List_t;

串列的結構體中值得注意的是:
uxNumberOfItems 是用來記錄串列中串列項的數量的,就是記錄父親有多少個兒子,當然女兒也行~,

pxIndex 是索引編號,用來遍歷串列的,呼叫宏listGET_OWNER_OF_NEXT_ENTRY()之后索引就會指向回傳當前串列項的下一個串列項,

xListEnd 指向的是最后一個串列項,并且這個串列項是MiniListItem屬性的,是一個迷你串列項,

串列的初始化

函式:

void vListInitialise( List_t * const pxList )
{
     pxList->pxIndex = ( ListItem_t * ) &( pxList->xListEnd );           /*lint The mini list structure is used as the list end to save RAM.  This is checked and valid. */
     pxList->xListEnd.xItemValue = https://www.cnblogs.com/iot-dev/p/portMAX_DELAY;
     pxList->xListEnd.pxNext = ( ListItem_t * ) &( pxList->xListEnd );   /*lint The mini list structure is used as the list end to save RAM.  This is checked and valid. */
     pxList->xListEnd.pxPrevious = ( ListItem_t * ) &( pxList->xListEnd );/*lint The mini list structure is used as the list end to save RAM.  This is checked and valid. */
     pxList->uxNumberOfItems = ( UBaseType_t ) 0U;
     listSET_LIST_INTEGRITY_CHECK_1_VALUE( pxList );
     listSET_LIST_INTEGRITY_CHECK_2_VALUE( pxList );
}

將串列的索引指向串列中的xListEnd,也就是末尾的串列項(迷你串列項)

串列項的xItemValue數值為portMAX_DELAY,也就是0xffffffffUL,如果在16位處理器中則為0xffff,

串列項的pxNext與pxPrevious這兩個指標都指向自己本身xListEnd,

初始化完成的時候串列項的數目為0個,因為還沒添加串列項嘛~,

串列項的初始化

函式:

void vListInitialiseItem( ListItem_t * const pxItem )
{
    /* Make sure the list item is not recorded as being on a list. */
    pxItem->pvContainer = NULL;
    /* Write known values into the list item if
    configUSE_LIST_DATA_INTEGRITY_CHECK_BYTES is set to 1. */
    listSET_FIRST_LIST_ITEM_INTEGRITY_CHECK_VALUE( pxItem );
    listSET_SECOND_LIST_ITEM_INTEGRITY_CHECK_VALUE( pxItem );
}

只需要讓串列項的pvContainer指標指向NULL即可,這樣子就使得串列項不屬于任何一個串列,因為串列項的初始化是要根據實際的情況來進行初始化的,

例如任務創建時用到的一些串列項初始化:

pxNewTCB->pcTaskName[ configMAX_TASK_NAME_LEN - 1 ] = '\0';
pxNewTCB->uxPriority = uxPriority;
pxNewTCB->uxBasePriority = uxPriority;
pxNewTCB->uxMutexesHeld = 0;

vListInitialiseItem( &( pxNewTCB->xStateListItem ) );
vListInitialiseItem( &( pxNewTCB->xEventListItem ) );

或者是在定時器相關的初始化中:

pxNewTimer->pcTimerName = pcTimerName;
pxNewTimer->xTimerPeriodInTicks = xTimerPeriodInTicks;
pxNewTimer->uxAutoReload = uxAutoReload;
pxNewTimer->pvTimerID = pvTimerID;
pxNewTimer->pxCallbackFunction = pxCallbackFunction;

vListInitialiseItem( &( pxNewTimer->xTimerListItem ) );

串列項的末尾插入

函式:

void vListInsertEnd( List_t * const pxList, ListItem_t * const pxNewListItem )
{
    ListItem_t * const pxIndex = pxList->pxIndex;
    listTEST_LIST_INTEGRITY( pxList );
    listTEST_LIST_ITEM_INTEGRITY( pxNewListItem );
    listGET_OWNER_OF_NEXT_ENTRY(). */
    pxNewListItem->pxNext = pxIndex;    //  1 
    pxNewListItem->pxPrevious = pxIndex->pxPrevious;    //  2
     /* Only used during decision coverage testing. */
    mtCOVERAGE_TEST_DELAY();
    pxIndex->pxPrevious->pxNext = pxNewListItem;        //  3 
    pxIndex->pxPrevious = pxNewListItem;                //  4
    /* Remember which list the item is in. */
    pxNewListItem->pvContainer = ( void * ) pxList;
    ( pxList->uxNumberOfItems )++;
}

傳入的引數:

  • pxList:串列項要插入的串列,

  • pxNewListItem:要插入的串列項是什么,

從末尾插入,那就要先知道哪里是頭咯,我們在串列中的成員pxIndex就是用來遍歷串列項的啊,那它指向的地方就是串列項的頭,那么既然FreeRTOS中的串列很像資料結構中的雙向鏈表,那么,我們可以把它看成一個環,是首尾相連的,那么函式中說的末尾,就是串列項頭的前一個,很顯然其結構圖應該是下圖這樣子的(初始化結束后pxIndex指向了xListEnd):

為什么是這樣子的呢,一句句代碼來解釋:

一開始:

ListItem_t * const pxIndex = pxList->pxIndex;

保存了一開始的索引串列項(xListEnd)的指向,

pxNewListItem->pxNext = pxIndex;         //  1

新串列項的下一個指向為索引串列項,也就是綠色的箭頭,

pxNewListItem->pxPrevious = pxIndex->pxPrevious;      //  2

剛開始我們初始化完成的時候pxIndex->pxPrevious的指向為自己xListEnd,那么xNewListItem->pxPrevious的指向為xListEnd,如2紫色的箭頭,

pxIndex->pxPrevious->pxNext = pxNewListItem;             //  3

索引串列項(xListEnd)的上一個串列項還是自己,那么自己的下一個串列項指向就是指向了pxNewListItem,

pxIndex->pxPrevious = pxNewListItem;                              //  4

這句就很容易理解啦,如圖的4橙色的箭頭,

插入完畢的時候標記一下新的串列項插入了哪個串列,并且將uxNumberOfItems進行加一,以表示多了一個串列項,

為什么原始碼要這樣子寫呢?因為這只是兩個串列項,一個串列含有多個串列項,那么這段代碼的通用性就很強了,無論原本串列中有多少個串列項,也無論pxIndex指向哪個串列項!

看看是不是按照原始碼中那樣插入呢?

串列項的插入

原始碼:

void vListInsert( List_t * const pxList, ListItem_t * const pxNewListItem )
{
ListItem_t *pxIterator;
const TickType_t xValueOfInsertion = pxNewListItem->xItemValue;
    listTEST_LIST_INTEGRITY( pxList );
    listTEST_LIST_ITEM_INTEGRITY( pxNewListItem );
    if( xValueOfInsertion == portMAX_DELAY )
    {
        pxIterator = pxList->xListEnd.pxPrevious;
    }
    else
    {
        for( pxIterator = ( ListItem_t * ) &( pxList->xListEnd ); pxIterator->pxNext->xItemValue <= xValueOfInsertion; pxIterator = pxIterator->pxNext ) /*lint !e826 !e740 The mini list structure is used as the list end to save RAM.  This is checked and valid. */
        {
            /* There is nothing to do here, just iterating to the wanted
            insertion position. */
        }
    }
    pxNewListItem->pxNext = pxIterator->pxNext;
    pxNewListItem->pxNext->pxPrevious = pxNewListItem;
    pxNewListItem->pxPrevious = pxIterator;
    pxIterator->pxNext = pxNewListItem;
    /* Remember which list the item is in.  This allows fast removal of the
    item later. */
    pxNewListItem->pvContainer = ( void * ) pxList;
    ( pxList->uxNumberOfItems )++;
}

傳入的引數:

  • pxList:串列項要插入的串列,

  • pxNewListItem:要插入的串列項是什么,

pxList決定了插入哪個串列,pxNewListItem中的xItemValue值決定了串列項插入串列的位置,

ListItem_t *pxIterator;  
const TickType_t xValueOfInsertion = pxNewListItem->xItemValue;

定義一個輔助的串列項pxIterator,用來迭代找出插入新串列項的位置,并且保存獲取要插入的串列項pxNewListItem的xItemValue,

如果打開了串列項完整性檢查,就要用戶實作configASSERT(),原始碼中有說明,

既然是要插入串列項,那么肯定是要知道串列項的位置了,如果新插入串列項的xItemValue是最大的話(portMAX_DELAY),就直接插入串列項的末尾,否則就需要比較串列中各個串列項的xItemValue的大小來進行排列,然后得出新串列項插入的位置,

for( pxIterator = ( ListItem_t * ) &( pxList->xListEnd ); pxIterator->pxNext->xItemValue <= xValueOfInsertion; pxIterator = pxIterator->pxNext )

上面原始碼就是實作比較的程序,

與上面的從串列項末尾插入的原始碼一樣,FreeRTOS的代碼通用性很強,邏輯思維也很強,

如果串列中串列項的數量為0,那么插入的串列項就是在初始化串列項的后面,如下圖所示:

程序分析:
新串列項的pxNext指向pxIterator->pxNext,也就是指向了xListEnd(pxIterator),

pxNewListItem->pxNext = pxIterator->pxNext;

而xListEnd(pxIterator)的pxPrevious指向則為pxNewListItem,

pxNewListItem->pxNext->pxPrevious = pxNewListItem;

新串列項的(pxPrevious)指標指向xListEnd(pxIterator)

pxIterator 的 pxNext 指向了新串列項

pxNewListItem->pxPrevious = pxIterator;
pxIterator->pxNext = pxNewListItem;

與從末尾插入串列項其實是一樣的,前提是當前串列中串列項的數目為0,

假如串列項中已經有了元素呢,程序又是不一樣的了,原來的串列是下圖這樣子的:


假設插入的串列項的xItemValue是2,而原有的串列項的xItemValue值是3,那么,按照原始碼,我們插入的串列項是在中間了,而pxIterator則是①號串列項,

插入后的效果:

分析一下插入的程序:

新的串列項的pxNext指向的是pxIterator->pxNext,也就是③號串列項,因為一開始pxIterator->pxNext=指向的就是③號串列項!!

pxNewListItem->pxNext = pxIterator->pxNext;

而pxNewListItem->pxNext 即③號串列項的指向上一個串列項指標(pxPrevious)的則指向新插入的串列項,也就是②號串列項了,

pxNewListItem->pxNext->pxPrevious = pxNewListItem;

新插入串列項的指向上一個串列項的指標pxNewListItem->pxPrevious指向了輔助串列項pxIterator,很顯然要連接起來嘛!

pxNewListItem->pxPrevious = pxIterator;   

同理,pxIterator串列項的指向下一個串列項的指標則指向新插入的串列項了pxNewListItem,

pxIterator->pxNext = pxNewListItem;

而其他沒改變指向的地方不需改動,(圖中的兩條直線做的連接線是不需要改動的)
當插入完成的時候,記錄一下新插入的串列項屬于哪個串列,并且讓該串列下的串列項數目加一,

pxNewListItem->pvContainer = ( void * ) pxList;
         ( pxList->uxNumberOfItems )++;

洗掉串列項

原始碼:

UBaseType_t uxListRemove( ListItem_t * const pxItemToRemove )
{
/* The list item knows which list it is in.  Obtain the list from the list
item. */
List_t * const pxList = ( List_t * ) pxItemToRemove->pvContainer;
    pxItemToRemove->pxNext->pxPrevious = pxItemToRemove->pxPrevious;
    pxItemToRemove->pxPrevious->pxNext = pxItemToRemove->pxNext;
    /* Only used during decision coverage testing. */
    mtCOVERAGE_TEST_DELAY();
    /* Make sure the index is left pointing to a valid item. */
    if( pxList->pxIndex == pxItemToRemove )
    {
        pxList->pxIndex = pxItemToRemove->pxPrevious;
    }
    else
    {
        mtCOVERAGE_TEST_MARKER();
    }
    pxItemToRemove->pvContainer = NULL;
    ( pxList->uxNumberOfItems )--;
    return pxList->uxNumberOfItems;
}

其實洗掉是很簡單的,不用想都知道,要洗掉串列項,那肯定要知道該串列項是屬于哪個串列吧,pvContainer就是記錄串列項是屬于哪個串列的,

洗掉就是把串列中的串列項從串列中去掉,其本質其實就是把他們的連接關系洗掉掉,然后讓洗掉的串列項的前后兩個串列連接起來就行了,假如是只有一個串列項,那么洗掉之后,串列就回到了初始化的狀態了,

pxItemToRemove->pxNext->pxPrevious = pxItemToRemove->pxPrevious;
pxItemToRemove->pxPrevious->pxNext = pxItemToRemove->pxNext;

這兩句代碼就實作了將洗掉串列項的前后兩個串列項連接起來,

按照上面的講解可以理解這兩句簡單的代碼啦,

假如洗掉的串列項是當前索引的串列項,那么在洗掉之后,串列中的pxIndex就要指向洗掉串列項的上一個串列項了,

if( pxList->pxIndex == pxItemToRemove )
{
      pxList->pxIndex = pxItemToRemove->pxPrevious;
}

當然還要把當前洗掉的串列項的pvContainer指向NULL,讓它不屬于任何一個串列,因為,洗掉的本質是洗掉的僅僅是串列項的連接關系,其記憶體是沒有釋放掉的,假如是動態記憶體分配的話,
并且要把當前串列中串列項的數目回傳一下,

至此,串列的原始碼基本講解完畢,

最后

大家還可以了解一下遍歷串列的宏,它在list.h檔案中:

define listGET_OWNER_OF_NEXT_ENTRY( pxTCB, pxList )                                        \
{                                                                                            \
List_t * const pxConstList = ( pxList );                                                    \
    /* Increment the index to the next item and return the item, ensuring */                \
    /* we don't return the marker used at the end of the list.  */                          \
    ( pxConstList )->pxIndex = ( pxConstList )->pxIndex->pxNext;                            \
    if( ( void * ) ( pxConstList )->pxIndex == ( void * ) &( ( pxConstList )->xListEnd ) )  \
    {                                                                                       \
        ( pxConstList )->pxIndex = ( pxConstList )->pxIndex->pxNext;                        \
    }                                                                                       \
    ( pxTCB ) = ( pxConstList )->pxIndex->pvOwner;                                          \
}

這是一個宏,用于串列的遍歷,回傳的是串列中串列項的pxOwner成員,每次呼叫這個宏(函式)的時候,其pxIndex索引會指向當前回傳串列項的下一個串列項,

關注我

歡迎關注我公眾號

更多資料歡迎關注“物聯網IoT開發”公眾號!

轉載請註明出處,本文鏈接:https://www.uj5u.com/caozuo/33395.html

標籤:嵌入式

上一篇:從0開始學FreeRTOS-(創建任務)-2

下一篇:從0開始學FreeRTOS-(訊息佇列)-5

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • CA和證書

    1、在 CentOS7 中使用 gpg 創建 RSA 非對稱密鑰對 gpg --gen-key #Centos上生成公鑰/密鑰對(存放在家目錄.gnupg/) 2、將 CentOS7 匯出的公鑰,拷貝到 CentOS8 中,在 CentOS8 中使用 CentOS7 的公鑰加密一個檔案 gpg -a ......

    uj5u.com 2020-09-10 00:09:53 more
  • Kubernetes K8S之資源控制器Job和CronJob詳解

    Kubernetes的資源控制器Job和CronJob詳解與示例 ......

    uj5u.com 2020-09-10 00:10:45 more
  • VMware下安裝CentOS

    VMware下安裝CentOS 一、軟硬體準備 1 Centos鏡像準備 1.1 CentOS鏡像下載地址 下載地址 1.2 CentOS鏡像下載程序 點擊下載地址進入如下圖的網站,選擇需要下載的版本,這里選擇的是Centos8,點擊如圖所示。 決定選擇Centos8后,選擇想要的鏡像源進行下載,此 ......

    uj5u.com 2020-09-10 00:12:10 more
  • 如何使用Grep命令查找多個字串

    如何使用Grep 命令查找多個字串 大家好,我是良許! 今天向大家介紹一個非常有用的技巧,那就是使用 grep 命令查找多個字串。 簡單介紹一下,grep 命令可以理解為是一個功能強大的命令列工具,可以用它在一個或多個輸入檔案中搜索與正則運算式相匹配的文本,然后再將每個匹配的文本用標準輸出的格式 ......

    uj5u.com 2020-09-10 00:12:28 more
  • git配置http代理

    git配置http代理 經常遇到克隆 github 慢的問題,這里記錄一下幾種配置 git 代理的方法,解決 clone github 過慢。 目錄 git配置代理 git單獨配置github代理 git配置全域代理 配置終端環境變數 git配置代理 主要使用 git config 命令 git單獨 ......

    uj5u.com 2020-09-10 00:12:33 more
  • Linux npm install 裝包時提示Error EACCES permission denied解

    npm install 裝包時提示Error EACCES permission denied解決辦法 ......

    uj5u.com 2020-09-10 00:12:53 more
  • Centos 7下安裝nginx,使用yum install nginx,提示沒有可用的軟體包

    Centos 7下安裝nginx,使用yum install nginx,提示沒有可用的軟體包。 18 (flaskApi) [root@67 flaskDemo]# yum -y install nginx 19 已加載插件:fastestmirror, langpacks 20 Loading ......

    uj5u.com 2020-09-10 00:13:13 more
  • Linux查看服務器暴力破解ssh IP

    在公網的服務器上經常遇到別人爆破你服務器的22埠,用來挖礦或者干其他嘿嘿嘿的事情~ 這種情況下正確的做法是: 修改默認ssh的22埠 使用設定密鑰登錄或者白名單ip登錄 建議服務器密碼為復雜密碼 創建普通用戶登錄服務器(root權限過大) 建立堡壘機,實作統一管理服務器 統計爆破IP [root ......

    uj5u.com 2020-09-10 00:13:17 more
  • CentOS 7系統常見快捷鍵操作方式

    Linux系統中一些常見的快捷方式,可有效提高操作效率,在某些時刻也能避免操作失誤帶來的問題。 ......

    uj5u.com 2020-09-10 00:13:31 more
  • CentOS 7作業系統目錄結構介紹

    作業系統存在著大量的資料檔案資訊,相應檔案資訊會存在于系統相應目錄中,為了更好的管理資料資訊,會將系統進行一些目錄規劃,不同目錄存放不同的資源。 ......

    uj5u.com 2020-09-10 00:13:35 more
最新发布
  • vim的常用命令

    Vim的6種基本模式 1. 普通模式在普通模式中,用的編輯器命令,比如移動游標,洗掉文本等等。這也是Vim啟動后的默認模式。這正好和許多新用戶期待的操作方式相反(大多數編輯器默認模式為插入模式)。 2. 插入模式在這個模式中,大多數按鍵都會向文本緩沖中插入文本。大多數新用戶希望文本編輯器編輯程序中一 ......

    uj5u.com 2023-04-20 08:43:21 more
  • vim的常用命令

    Vim的6種基本模式 1. 普通模式在普通模式中,用的編輯器命令,比如移動游標,洗掉文本等等。這也是Vim啟動后的默認模式。這正好和許多新用戶期待的操作方式相反(大多數編輯器默認模式為插入模式)。 2. 插入模式在這個模式中,大多數按鍵都會向文本緩沖中插入文本。大多數新用戶希望文本編輯器編輯程序中一 ......

    uj5u.com 2023-04-20 08:42:36 more
  • docker學習

    ###Docker概述 真實專案部署環境可能非常復雜,傳統發布專案一個只需要一個jar包,運行環境需要單獨部署。而通過Docker可將jar包和相關環境(如jdk,redis,Hadoop...)等打包到docker鏡像里,將鏡像發布到Docker倉庫,部署時下載發布的鏡像,直接運行發布的鏡像即可。 ......

    uj5u.com 2023-04-19 09:26:53 more
  • 設定Windows主機的瀏覽器為wls2的默認瀏覽器

    這里以Chrome為例。 1. 準備作業 wsl是可以使用Windows主機上安裝的exe程式,出于安全考慮,默認情況下改功能是無法使用。要使用的話,終端需要以管理員權限啟動。 我這里以Windows Terminal為例,介紹如何默認使用管理員權限打開終端,具體操作如下圖所示: 2. 操作 wsl ......

    uj5u.com 2023-04-19 09:25:49 more
  • docker學習

    ###Docker概述 真實專案部署環境可能非常復雜,傳統發布專案一個只需要一個jar包,運行環境需要單獨部署。而通過Docker可將jar包和相關環境(如jdk,redis,Hadoop...)等打包到docker鏡像里,將鏡像發布到Docker倉庫,部署時下載發布的鏡像,直接運行發布的鏡像即可。 ......

    uj5u.com 2023-04-19 09:19:04 more
  • Linux學習筆記

    IP地址和主機名 IP地址 ifconfig可以用來查詢本機的IP地址,如果不能使用,可以通過install net-tools安裝。 Centos系統下ens33表示主網卡;inet后表示IP地址;lo表示本地回環網卡; 127.0.0.1表示代指本機;0.0.0.0可以用于代指本機,同時在放行設 ......

    uj5u.com 2023-04-18 06:52:01 more
  • 解決linux系統的kdump服務無法啟動的問題

    問題:專案麒麟系統服務器的kdump服務無法啟動,沒有相關日志無法定位問題。 1、查看服務狀態是關閉的,重啟系統也無法啟動 systemctl status kdump 2、修改grub引數,修改“crashkernel”為“512M(有的機器數值太大太小都會導致報錯,建議從128M開始試,或者加個 ......

    uj5u.com 2023-04-12 09:59:50 more
  • 解決linux系統的kdump服務無法啟動的問題

    問題:專案麒麟系統服務器的kdump服務無法啟動,沒有相關日志無法定位問題。 1、查看服務狀態是關閉的,重啟系統也無法啟動 systemctl status kdump 2、修改grub引數,修改“crashkernel”為“512M(有的機器數值太大太小都會導致報錯,建議從128M開始試,或者加個 ......

    uj5u.com 2023-04-12 09:59:01 more
  • 你是不是暴露了?

    作者:袁首京 原創文章,轉載時請保留此宣告,并給出原文連接。 如果您是計算機相關從業人員,那么應該經歷不止一次網路安全專項檢查了,你肯定是收到過資訊系統技術檢測報告,要求你加強風險監測,確保你提供的系統服務堅實可靠了。 沒檢測到問題還好,檢測到問題的話,有些處理起來還是挺麻煩的,尤其是線上正在運行的 ......

    uj5u.com 2023-04-05 16:52:56 more
  • 細節拉滿,80 張圖帶你一步一步推演 slab 記憶體池的設計與實作

    1. 前文回顧 在之前的幾篇記憶體管理系列文章中,筆者帶大家從宏觀角度完整地梳理了一遍 Linux 記憶體分配的整個鏈路,本文的主題依然是記憶體分配,這一次我們會從微觀的角度來探秘一下 Linux 內核中用于零散小記憶體塊分配的記憶體池 —— slab 分配器。 在本小節中,筆者還是按照以往的風格先帶大家簡單 ......

    uj5u.com 2023-04-05 16:44:11 more