主頁 >  其他 > 《嵌入式 - Lwip開發指南》第5章 LWIP測速

《嵌入式 - Lwip開發指南》第5章 LWIP測速

2021-10-18 14:12:44 其他

最近有個網友在詢問關于LWIP的速度,本文就LWIP網速做個簡單測驗,為了對比,本文將使用無系統和有系統兩種環境

5.1網路測速工具介紹

不過在測速之前,需要介紹下測速的工具,這里有兩個軟體:iPerf與jperf

iPerf 是一個跨平臺的網路性能測驗工具,它支持Win/Linux/Mac/Android/iOS 等平臺,iPerf 可以測驗TCP 和UDP(我們一般不對UDP 進行測速)帶寬質量,iPerf 可以測量最大TCP 帶寬,可以具有多種引數進行測驗,同時iPerf 還可以報告帶寬,延遲抖動和資料包丟失的情況,我們可以利用iPerf的這些特性來測驗一些網路設備如路由器,防火墻,交換機等的性能,

在這里插入圖片描述

iPerf下載地址

雖然iPerf 很好用,但是它卻是命令列格式的軟體,對使用測驗的人員并不友好,使用者需要記下他繁瑣的命令,不過它還有一個圖形界面程式叫做JPerf,使用JPerf 程式能簡化了復雜命令列引數的構造,而且 它還保存測驗結果,并且將測驗結果實時圖形化出來,更加一目了然,當然,JPerf 也肯定擁有iPerf 的所有功能,本質執行的iperf的功能,因此本文使用JPerf測速,

關于JPerf軟體請自行在后文指引下獲取,下載JPerf后,解壓,

在這里插入圖片描述

然后單擊jperf.bat即可打開軟體,

在這里插入圖片描述

值得注意的是,運行該軟體還需要Java環境,請自行配置Java環境,

5.2無系統測速(RAW API)

要想使用JPerf測速,必須先實作TCP服務器或客戶端,關于TCP理論這里就不在贅述了,網上的資料很多,這里只講解如何使用RAW API實作TCP服務器,

5.2.1 TCP相關的RAW API

在開始實作TCP服務器之前,我們首先來看一看LwIP中與TCP相關的RAW API函式有哪些,并簡單的了解一下其功能,

  • 建立TCP連接的API函式
函式描述
tcp_new()創建TCP的PCB控制塊
tcp_bind()系結服務器的IP和埠號
tcp_listen()監聽TCP的PCB控制塊
tcp_accepted()通知 LWIP 協議堆疊一個 TCP 連接被接受了
tcp_conect()連接遠端主機,客戶端使用
  • 發送TCP資料的API函式
函式描述
tcp_write()構造一個報文并放到控制塊的發送緩沖佇列中
tcp_sent()控制塊 sent 欄位注冊的回呼函式,資料發送成功后被回呼
tcp_output()將發送緩沖佇列中的資料發送出去
  • 接收 TCP 資料
函式描述
tcp_recv()控制塊 recv 欄位注冊的回呼函式,當接收到新資料時被呼叫
tcp_recved()當程式處理完資料后一定要呼叫這個函式,通知內核更新接收視窗
  • 輪詢函式
函式描述
tcp_poll()控制塊 poll 欄位注冊的回呼函式,該函式周期性呼叫
  • 關閉和中止連接
函式描述
tcp_close()關閉一個 TCP 連接
tcp_err()控制塊 err 欄位注冊的回呼函式,遇到錯誤時被呼叫
tcp_abort()中斷 TCP 連接

在具體實作TCP服務器之前,先配合著下LWIP,關于如何移植LWIP可以參看筆者以前的文章,

移植LWIP(無系統)

筆者這里使用靜態IP,并開啟TCP模塊,

在這里插入圖片描述

5.2.2 TCP服務器實作流程

前面了解了TCP所涉及到的API函式,也通過STM32CubeMX打開了相關配置,那么使用這些函式怎么實作一個TCP服務器呢?我們先簡單說明一下其基本的流程,

1.新建控制塊
使用tcp_new()函式建立一個TCP控制塊,

2.系結控制塊
對于服務器來說,新建一個控制快后,需要在控制塊上系結本地IP和埠,以方便客戶端的連接,

3.控制塊偵聽
使用tcp_listen函式,對于服務器來說,需要顯性呼叫tcp_listen函式以使控制塊進入監聽狀態,等待客戶端的連接請求,

4.建立連接
在tcp_listen函式進入服務器監聽狀態后,需要馬上使用tcp_accept函式來注冊一個接收處理函式,因為一旦有客戶端連接請求被成功建立后,服務器就會呼叫這個處理函式,

5.接受并處理資料
一旦連接成功,accept回呼函式會呼叫tcp_recv函式注冊一個接收完成的處理函式,對于服務器來說,接收到了客戶端的資料或操作要求,就會呼叫這一回呼函式進行處理,這其實是一個復雜的程序:接收到資料后,首先通知更新接受視窗(使用tcp_recved函式),處理并發送資料(使用tcp_write函式),資料發送成功則清除已發送的資料(使用tcp_sent函式),最后關閉連接(使用函式tcp_close),

整個流程圖所示如下:

在這里插入圖片描述

5.2.3 TCP服務器代碼實作

前面分析了TCP服務器的實作流程,接下來就是通過前面介紹的API來實作,

首先是TCP服務器的初始化,其實作代碼如下:

/**
  * @brief  TCP服務器初始化
  * @param  None
  * @retval res
  */
uint8_t tcp_server_init(void)
{
	uint8_t res = 0;		
	err_t err;  
  
	struct tcp_pcb *tcppcbnew;  	//定義一個TCP服務器控制塊
	struct tcp_pcb *tcppcbconn;  	//定義一個TCP服務器控制塊
		
  /* 為tcp服務器分配一個tcp_pcb結構體 */
  tcppcbnew = tcp_new();

	if(tcppcbnew)			//創建成功
	{ 
		//將本地IP與指定的埠號系結在一起,IP_ADDR_ANY為系結本地所有的IP地址
		err = tcp_bind(tcppcbnew,IP_ADDR_ANY,TCP_SERVER_PORT);	
		if(err==ERR_OK)	//系結完成
		{
			tcppcbconn=tcp_listen(tcppcbnew); 			//設定tcppcb進入監聽狀態
			
			//初始化LWIP的tcp_accept的回呼函式
			tcp_accept(tcppcbconn,tcp_server_accept); 	
		}
		else 
		{
			res=1;
		}			
	}
	else 
	{
		res=1;
  }
	
	return res;
}

可以看到tcp_accept()函式注冊了一個回呼函式,實作代碼如下:

/**
  * @brief  lwIP tcp_accept()的回呼函式
  * @param  arg,newpcb, err
  * @retval ret_err
  */
err_t tcp_server_accept(void *arg, struct tcp_pcb *newpcb,err_t err)
{
	err_t ret_err;
	struct tcp_server_struct *es; 
 	LWIP_UNUSED_ARG(arg);
	LWIP_UNUSED_ARG(err);
	tcp_setprio(newpcb,TCP_PRIO_MIN);//設定新創建的pcb優先級
	
	es=(struct tcp_server_struct*)mem_malloc(sizeof(struct tcp_server_struct)); //分配記憶體
 	
	if(es!=NULL) //記憶體分配成功
	{
		es->state = ES_TCPSERVER_ACCEPTED;  	//接收連接
		es->pcb = newpcb;
		es->p = NULL;
		
		tcp_arg(newpcb, es);
		tcp_recv(newpcb, tcp_server_recv);	//初始化tcp_recv()的回呼函式
		tcp_err(newpcb, tcp_server_error); 	//初始化tcp_err()回呼函式
		tcp_poll(newpcb, tcp_server_poll,1);	//初始化tcp_poll回呼函式
		tcp_sent(newpcb, tcp_server_sent);  	//初始化發送回呼函式
		  
		tcp_server_flag |= 1<<5;				//標記有客戶端連上了

		ret_err=ERR_OK;
	}
	else 
	{
		ret_err=ERR_MEM;
	}
	
	return ret_err;
}

這個函式中用于與客戶端進行資料互動,函式中有注冊了接收發送等函式,本文最重要的就是需要接收函式,代碼如下:

/**
  * @brief  lwIP tcp_recv()函式的回呼函式
  * @param  arg,tpcb, p, err
  * @retval ret_err
  */
err_t tcp_server_recv(void *arg, struct tcp_pcb *tpcb, struct pbuf *p, err_t err)
{
	err_t ret_err;
	uint32_t data_len = 0;
	struct pbuf *q;
  struct tcp_server_struct *es;
	LWIP_ASSERT("arg != NULL",arg != NULL);
	es=(struct tcp_server_struct *)arg;
	
	if(p == NULL) //從客戶端接收到空資料
	{
		es->state = ES_TCPSERVER_CLOSING;//需要關閉TCP 連接了
		es->p = p; 
		ret_err = ERR_OK;
	}
	else if(err != ERR_OK)	//從客戶端接收到一個非空資料,但是由于某種原因err!=ERR_OK
	{
		if(p)
		{
			pbuf_free(p);	//釋放接收pbuf
		}
		ret_err = err;
	}
	else if(es->state == ES_TCPSERVER_ACCEPTED) 	//處于連接狀態
	{
		if(p != NULL)  //當處于連接狀態并且接收到的資料不為空時將其列印出來
		{
			memset(tcp_server_recvbuf, 0, TCP_SERVER_RX_BUFSIZE);  //資料接識訓沖區清零
			for(q = p; q != NULL; q = q->next)  //遍歷完整個pbuf鏈表
			{
				//判斷要拷貝到TCP_SERVER_RX_BUFSIZE中的資料是否大于TCP_SERVER_RX_BUFSIZE的剩余空間,如果大于
				//的話就只拷貝TCP_SERVER_RX_BUFSIZE中剩余長度的資料,否則的話就拷貝所有的資料
				if(q->len > (TCP_SERVER_RX_BUFSIZE-data_len)) 
				{
					memcpy(tcp_server_recvbuf+data_len,q->payload,(TCP_SERVER_RX_BUFSIZE-data_len));//拷貝資料
				}
				else 
				{
					memcpy(tcp_server_recvbuf+data_len,q->payload,q->len);
				}
				data_len += q->len;  	
				if(data_len > TCP_SERVER_RX_BUFSIZE) 
				{
					break; //超出TCP客戶端接收陣列,跳出
				}					
			}
			tcp_server_flag |= 1<<6;	//標記接收到資料了

			tcp_recved(tpcb,p->tot_len);//用于獲取接收資料,通知LWIP可以獲取更多資料
			pbuf_free(p);  	//釋放記憶體
			ret_err=ERR_OK;
		}
	}
	else//服務器關閉了
	{
		tcp_recved(tpcb,p->tot_len);//用于獲取接收資料,通知LWIP可以獲取更多資料
		es->p = NULL;
		pbuf_free(p); //釋放記憶體
		ret_err = ERR_OK;
	}
	return ret_err;
}

可以看到,以上函式都是一層一層的呼叫,都是使用的回呼函式,其他相關函式請自行參看原始碼,這里就不細講了,

最后再main()函式初始化TCP服務器即可,

/**
  * @brief  The application entry point.
  * @retval int
  */
int main(void)
{
  /* USER CODE BEGIN 1 */

  /* USER CODE END 1 */

  /* Enable I-Cache---------------------------------------------------------*/
  SCB_EnableICache();

  /* Enable D-Cache---------------------------------------------------------*/
  SCB_EnableDCache();

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_USART3_UART_Init();
  MX_LWIP_Init();
  
	/* USER CODE BEGIN 2 */
	tcp_server_init();

  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
    /* USER CODE END WHILE */
		MX_LWIP_Process();					//LWIP輪詢任務
    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */
}

然后編譯工程,下載到板子中,打開jperf軟體,配合好相應引數,其結果如下:

在這里插入圖片描述

從上圖可以看出,傳輸速度大約為1-2M左右,還是有點慢的,

那么要想提高LwIP網路傳輸速度的方法,就要對LwIP的配置進行合適的調整,主要增加記憶體的Heap Size、記憶體池大小、TCP報文段數量、最大TCP報文段、TCP發送緩沖區佇列的最大長度等,

關于以上引數的修改可通過STM32CubeMX配置,也就是如下選項:

在這里插入圖片描述
其對應的檔案是lwipopts.h和opt.h,主要的配置引數在opt.h中,

在這里插入圖片描述

筆者直接在檔案中修改的,修改的引數如下:

//記憶體堆 heap 大小
#define MEM_SIZE (24*1024)
 
/* memp 結構的 pbuf 數量,如果應用從 ROM 或者靜態存盤區發送大量資料時這個值應該設定大一點 */
#define MEMP_NUM_PBUF 24
 
/* 最多同時在 TCP 緩沖佇列中的報文段數量 */
#define MEMP_NUM_TCP_SEG 150
 
/* 記憶體池大小 */
#define PBUF_POOL_SIZE 64
 
/* 最大 TCP 報文段, TCP_MSS = (MTU - IP 報頭大小 - TCP 報頭大小 */
#define TCP_MSS (1500 - 40)
 
/* TCP 發送緩沖區大小(位元組) */
#define TCP_SND_BUF (11*TCP_MSS)
 
/* TCP 接收視窗大小 */
#define TCP_WND (11*TCP_MSS)

修改后再進行編譯,測驗結果如下:

在這里插入圖片描述

對比前文使用的默認引數,可以發現,速度增加明顯,大約快了4-5倍,

關于LWIP的性能優化會在后面的章節講解,本文的重點是測速,

5.3 RT-Thread系統測速(Socket API)

上一節使用RAW API來實作TCP服務器,本節將使用Socket API來實作TCP服務器,關于TCP服務器的實作可參考筆者博文,

TCP服務器實作

實作TCP的服務器代碼如下:

#include <rtthread.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netdev.h>
#include <stdio.h>
#include <string.h>

#define SERVER_PORT   8888
#define BUFF_SIZE 4096

static char recvbuff[BUFF_SIZE];

static void net_server_thread_entry(void *parameter)
{

    int sfd, cfd, maxfd, i, nready, n;

    struct sockaddr_in server_addr, client_addr;

    struct netdev *netdev = RT_NULL;

    char sendbuff[] = "Hello client!";

    socklen_t client_addr_len;
    fd_set all_set, read_set;

    //FD_SETSIZE里面包含了服務器的fd
    int clientfds[FD_SETSIZE - 1];

    // 通過名稱獲取 netdev 網卡物件
    netdev = netdev_get_by_name((char*)parameter);
    if (netdev == RT_NULL)
    {
        rt_kprintf("get network interface device(%s) failed.\n", (char*)parameter);
    }

    //創建socket
    if ((sfd = socket(AF_INET, SOCK_STREAM, 0)) < 0)
    {
        rt_kprintf("Socket create failed.\n");
    }

    server_addr.sin_family = AF_INET;
    server_addr.sin_port = htons(SERVER_PORT);
    //server_addr.sin_addr.s_addr = htonl(INADDR_ANY);

    // 獲取網卡物件中 IP 地址資訊
    server_addr.sin_addr.s_addr = netdev->ip_addr.addr;

    //系結socket
    if (bind(sfd, (struct sockaddr *)&server_addr, sizeof(struct sockaddr)) < 0)
    {
        rt_kprintf("socket bind failed.\n");
        closesocket(sfd);
    }
    rt_kprintf("socket bind network interface device(%s) success!\n", netdev->name);

    //監聽socket
    if(listen(sfd, 5) == -1)
    {
        rt_kprintf("listen error");
    }
    else
    {
        rt_kprintf("listening...\n");
    }

    client_addr_len = sizeof(client_addr);

    //初始化 maxfd 等于 sfd
    maxfd = sfd;

    //清空fdset
    FD_ZERO(&all_set);

    //把sfd檔案描述符添加到集合中
    FD_SET(sfd, &all_set);

    //初始化客戶端fd的集合
    for(i = 0; i < FD_SETSIZE -1 ; i++)
    {
        //初始化為-1
        clientfds[i] = -1;
    }
    while(1)
    {
        //每次select回傳之后,fd_set集合就會變化,再select時,就不能使用,
        //所以我們要保存設定fd_set 和 讀取的fd_set
        read_set = all_set;
        nready = select(maxfd + 1, &read_set, NULL, NULL, NULL);

        //沒有超時機制,不會回傳0
        if(nready < 0)
        {
            rt_kprintf("select error \r\n");

        }

        //判斷監聽的套接字是否有資料
        if(FD_ISSET(sfd, &read_set))
        {
            //有客戶端進行連接了
            cfd = accept(sfd, (struct sockaddr *)&client_addr, &client_addr_len);
            if(cfd < 0)
            {
                rt_kprintf("accept socket error\r\n");
                //繼續select
                continue;
            }
            rt_kprintf("new client connect fd = %d\r\n", cfd);

            //把新的cfd 添加到fd_set集合中
            FD_SET(cfd, &all_set);

            //更新要select的maxfd
            maxfd = (cfd > maxfd)?cfd:maxfd;

            //把新的cfd 保存到cfds集合中
            for(i = 0; i < FD_SETSIZE -1 ; i++)
            {
                if(clientfds[i] == -1)
                {
                    clientfds[i] = cfd;
                    //退出,不需要添加
                    break;
                }
            }

            //沒有其他套接字需要處理:這里防止重復作業,就不去執行其他任務
            if(--nready == 0)
            {
                //繼續select
                continue;
            }
        }

        //遍歷所有的客戶端檔案描述符
        for(i = 0; i < FD_SETSIZE -1 ; i++)
        {
            if(clientfds[i] == -1)
            {
                //繼續遍歷
                continue;
            }

            //判斷是否在fd_set集合里面
            if(FD_ISSET(clientfds[i], &read_set))
            {
                n = recv(clientfds[i], recvbuff, sizeof(recvbuff), 0);
                //rt_kprintf("clientfd %d:  %s \r\n",clientfds[i], recvbuff);

                if(n <= 0)
                {
                    //從集合里面清除
                    FD_CLR(clientfds[i], &all_set);
                    //當前的客戶端fd 賦值為-1
                    clientfds[i] = -1;                }
                else
                {
                    //寫回客戶端
                    n = send(clientfds[i], sendbuff, strlen(sendbuff), 0);
                    if(n < 0)
                    {
                        //從集合里面清除
                        FD_CLR(clientfds[i], &all_set);

                        //當前的客戶端fd 賦值為-1
                        clientfds[i] = -1;
                    }
                }
            }
        }
    }
}

static int server(int argc, char **argv)
{
    rt_err_t ret = RT_EOK;

    if (argc != 2)
    {
        rt_kprintf("bind_test [netdev_name]  --bind network interface device by name.\n");
        return -RT_ERROR;
    }

    /* 創建 serial 執行緒 */
    rt_thread_t thread = rt_thread_create("server",
                                          net_server_thread_entry,
                                          argv[1],
                                          2048,
                                          5,
                                          10);

    /* 創建成功則啟動執行緒 */
    if (thread != RT_NULL)
    {
        rt_thread_startup(thread);
    }
    else
    {
        ret = RT_ERROR;
    }

    return ret;
}


#ifdef FINSH_USING_MSH
#include <finsh.h>
MSH_CMD_EXPORT(server, network interface device test);
#endif /* FINSH_USING_MSH */

添加好相應代碼進行編譯,編譯后才能后下載韌體,測驗結果如下:

在這里插入圖片描述

可以看出,其傳輸速度也在1M左右,相對無系統的環境,其速度相對慢些,

要想提高速度,就配置下LWIP引數,筆者配置的引數如下:

在這里插入圖片描述

編譯,下載,測驗結果如下:

在這里插入圖片描述

可以看到其速度還是有所提升的,只是沒有無系統時提升的明顯,至于原因后面的章節將會具體分析,




資源獲取方法

1.長按下面二維碼,關注公眾號[嵌入式實驗樓]
2.在公眾號回復關鍵詞[LWIP]獲取資料
在這里插入圖片描述




歡迎訪問我的網站

BruceOu的嗶哩嗶哩
BruceOu的主頁
BruceOu的博客
BruceOu的CSDN博客
BruceOu的簡書
BruceOu的知乎

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

標籤:其他

上一篇:在Jmeter中設定全域變數,實作跨執行緒組傳參

下一篇:慕課版軟體質量保證與測驗(第五章.課后作業)

標籤雲
其他(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)

熱門瀏覽
  • 網閘典型架構簡述

    網閘架構一般分為兩種:三主機的三系統架構網閘和雙主機的2+1架構網閘。 三主機架構分別為內端機、外端機和仲裁機。三機無論從軟體和硬體上均各自獨立。首先從硬體上來看,三機都用各自獨立的主板、記憶體及存盤設備。從軟體上來看,三機有各自獨立的作業系統。這樣能達到完全的三機獨立。對于“2+1”系統,“2”分為 ......

    uj5u.com 2020-09-10 02:00:44 more
  • 如何從xshell上傳檔案到centos linux虛擬機里

    如何從xshell上傳檔案到centos linux虛擬機里及:虛擬機CentOs下執行 yum -y install lrzsz命令,出現錯誤:鏡像無法找到軟體包 前言 一、安裝lrzsz步驟 二、上傳檔案 三、遇到的問題及解決方案 總結 前言 提示:其實很簡單,往虛擬機上安裝一個上傳檔案的工具 ......

    uj5u.com 2020-09-10 02:00:47 more
  • 一、SQLMAP入門

    一、SQLMAP入門 1、判斷是否存在注入 sqlmap.py -u 網址/id=1 id=1不可缺少。當注入點后面的引數大于兩個時。需要加雙引號, sqlmap.py -u "網址/id=1&uid=1" 2、判斷文本中的請求是否存在注入 從文本中加載http請求,SQLMAP可以從一個文本檔案中 ......

    uj5u.com 2020-09-10 02:00:50 more
  • Metasploit 簡單使用教程

    metasploit 簡單使用教程 浩先生, 2020-08-28 16:18:25 分類專欄: kail 網路安全 linux 文章標簽: linux資訊安全 編輯 著作權 metasploit 使用教程 前言 一、Metasploit是什么? 二、準備作業 三、具體步驟 前言 Msfconsole ......

    uj5u.com 2020-09-10 02:00:53 more
  • 游戲逆向之驅動層與用戶層通訊

    驅動層代碼: #pragma once #include <ntifs.h> #define add_code CTL_CODE(FILE_DEVICE_UNKNOWN,0x800,METHOD_BUFFERED,FILE_ANY_ACCESS) /* 更多游戲逆向視頻www.yxfzedu.com ......

    uj5u.com 2020-09-10 02:00:56 more
  • 北斗電力時鐘(北斗授時服務器)讓網路資料更精準

    北斗電力時鐘(北斗授時服務器)讓網路資料更精準 北斗電力時鐘(北斗授時服務器)讓網路資料更精準 京準電子科技官微——ahjzsz 近幾年,資訊技術的得了快速發展,互聯網在逐漸普及,其在人們生活和生產中都得到了廣泛應用,并且取得了不錯的應用效果。計算機網路資訊在電力系統中的應用,一方面使電力系統的運行 ......

    uj5u.com 2020-09-10 02:01:03 more
  • 【CTF】CTFHub 技能樹 彩蛋 writeup

    ?碎碎念 CTFHub:https://www.ctfhub.com/ 筆者入門CTF時時剛開始刷的是bugku的舊平臺,后來才有了CTFHub。 感覺不論是網頁UI設計,還是題目質量,賽事跟蹤,工具軟體都做得很不錯。 而且因為獨到的金幣制度的確讓人有一種想去刷題賺金幣的感覺。 個人還是非常喜歡這個 ......

    uj5u.com 2020-09-10 02:04:05 more
  • 02windows基礎操作

    我學到了一下幾點 Windows系統目錄結構與滲透的作用 常見Windows的服務詳解 Windows埠詳解 常用的Windows注冊表詳解 hacker DOS命令詳解(net user / type /md /rd/ dir /cd /net use copy、批處理 等) 利用dos命令制作 ......

    uj5u.com 2020-09-10 02:04:18 more
  • 03.Linux基礎操作

    我學到了以下幾點 01Linux系統介紹02系統安裝,密碼啊破解03Linux常用命令04LAMP 01LINUX windows: win03 8 12 16 19 配置不繁瑣 Linux:redhat,centos(紅帽社區版),Ubuntu server,suse unix:金融機構,證券,銀 ......

    uj5u.com 2020-09-10 02:04:30 more
  • 05HTML

    01HTML介紹 02頭部標簽講解03基礎標簽講解04表單標簽講解 HTML前段語言 js1.了解代碼2.根據代碼 懂得挖掘漏洞 (POST注入/XSS漏洞上傳)3.黑帽seo 白帽seo 客戶網站被黑帽植入劫持代碼如何處理4.熟悉html表單 <html><head><title>TDK標題,描述 ......

    uj5u.com 2020-09-10 02:04:36 more
最新发布
  • 2023年最新微信小程式抓包教程

    01 開門見山 隔一個月發一篇文章,不過分。 首先回顧一下《微信系結手機號資料庫被脫庫事件》,我也是第一時間得知了這個訊息,然后跟蹤了整件事情的經過。下面是這起事件的相關截圖以及近日流出的一萬條資料樣本: 個人認為這件事也沒什么,還不如關注一下之前45億快遞資料查詢渠道疑似在近日復活的訊息。 訊息是 ......

    uj5u.com 2023-04-20 08:48:24 more
  • web3 產品介紹:metamask 錢包 使用最多的瀏覽器插件錢包

    Metamask錢包是一種基于區塊鏈技術的數字貨幣錢包,它允許用戶在安全、便捷的環境下管理自己的加密資產。Metamask錢包是以太坊生態系統中最流行的錢包之一,它具有易于使用、安全性高和功能強大等優點。 本文將詳細介紹Metamask錢包的功能和使用方法。 一、 Metamask錢包的功能 數字資 ......

    uj5u.com 2023-04-20 08:47:46 more
  • vulnhub_Earth

    前言 靶機地址->>>vulnhub_Earth 攻擊機ip:192.168.20.121 靶機ip:192.168.20.122 參考文章 https://www.cnblogs.com/Jing-X/archive/2022/04/03/16097695.html https://www.cnb ......

    uj5u.com 2023-04-20 07:46:20 more
  • 從4k到42k,軟體測驗工程師的漲薪史,給我看哭了

    清明節一過,盲猜大家已經無心上班,在數著日子準備過五一,但一想到銀行卡里的余額……瞬間心情就不美麗了。最近,2023年高校畢業生就業調查顯示,本科畢業月平均起薪為5825元。調查一出,便有很多同學表示自己又被平均了。看著這一資料,不免讓人想到前不久中國青年報的一項調查:近六成大學生認為畢業10年內會 ......

    uj5u.com 2023-04-20 07:44:00 more
  • 最新版本 Stable Diffusion 開源 AI 繪畫工具之中文自動提詞篇

    🎈 標簽生成器 由于輸入正向提示詞 prompt 和反向提示詞 negative prompt 都是使用英文,所以對學習母語的我們非常不友好 使用網址:https://tinygeeker.github.io/p/ai-prompt-generator 這個網址是為了讓大家在使用 AI 繪畫的時候 ......

    uj5u.com 2023-04-20 07:43:36 more
  • 漫談前端自動化測驗演進之路及測驗工具分析

    隨著前端技術的不斷發展和應用程式的日益復雜,前端自動化測驗也在不斷演進。隨著 Web 應用程式變得越來越復雜,自動化測驗的需求也越來越高。如今,自動化測驗已經成為 Web 應用程式開發程序中不可或缺的一部分,它們可以幫助開發人員更快地發現和修復錯誤,提高應用程式的性能和可靠性。 ......

    uj5u.com 2023-04-20 07:43:16 more
  • CANN開發實踐:4個DVPP記憶體問題的典型案例解讀

    摘要:由于DVPP媒體資料處理功能對存放輸入、輸出資料的記憶體有更高的要求(例如,記憶體首地址128位元組對齊),因此需呼叫專用的記憶體申請介面,那么本期就分享幾個關于DVPP記憶體問題的典型案例,并給出原因分析及解決方法。 本文分享自華為云社區《FAQ_DVPP記憶體問題案例》,作者:昇騰CANN。 DVPP ......

    uj5u.com 2023-04-20 07:43:03 more
  • msf學習

    msf學習 以kali自帶的msf為例 一、msf核心模塊與功能 msf模塊都放在/usr/share/metasploit-framework/modules目錄下 1、auxiliary 輔助模塊,輔助滲透(埠掃描、登錄密碼爆破、漏洞驗證等) 2、encoders 編碼器模塊,主要包含各種編碼 ......

    uj5u.com 2023-04-20 07:42:59 more
  • Halcon軟體安裝與界面簡介

    1. 下載Halcon17版本到到本地 2. 雙擊安裝包后 3. 步驟如下 1.2 Halcon軟體安裝 界面分為四大塊 1. Halcon的五個助手 1) 影像采集助手:與相機連接,設定相機引數,采集影像 2) 標定助手:九點標定或是其它的標定,生成標定檔案及內參外參,可以將像素單位轉換為長度單位 ......

    uj5u.com 2023-04-20 07:42:17 more
  • 在MacOS下使用Unity3D開發游戲

    第一次發博客,先發一下我的游戲開發環境吧。 去年2月份買了一臺MacBookPro2021 M1pro(以下簡稱mbp),這一年來一直在用mbp開發游戲。我大致分享一下我的開發工具以及使用體驗。 1、Unity 官網鏈接: https://unity.cn/releases 我一般使用的Apple ......

    uj5u.com 2023-04-20 07:40:19 more