主頁 > 後端開發 > freemodbus移植進STM32(包含HAL庫和標準庫兩種方法)

freemodbus移植進STM32(包含HAL庫和標準庫兩種方法)

2023-05-23 08:08:17 後端開發

freemodbus移植

基于freemodbus1.6
使用HAL庫
軟體:stm32cubemx stm32cubeide

后續會更新標準庫的移植,以及rtos下的移植(盡量)

下載freemodbus1.6

這個獲取方法網上到處都是,不細說了,

cubemx新建工程

新建工程只列出了與移植freemodbus相關的設定
這里我使用的是485通信,所以額外使能了一個引腳
image
使能一個定時器,這里我用的是tim2,并且開始定時器2中斷
image
其他設定如下圖,引數其實設什么無所謂,因為后面要改的,我們并不用系統的初始化函式,
image
然后使能一個串口,我這里用的串口1,引數其實設什么無所謂,因為后面要改的,
image
這里可以把串口1和定時器2的最前面的取消勾選,就不會生成他們的初始化函式,不勾也沒有太大關系,因為我們的函式在他之后,會覆寫掉系統的設定,
image

另外在中斷優先級設定中,將串口優先級設定高于定時器2,數字越小越高,
image
相關的中斷處理函式也要生成,
image

然后就可以generate code!生成代碼,

代碼修改

首先我們在我們專案的根目錄中新建一個freemodbus檔案夾,檔案夾中再建一個modbus檔案夾,一個port檔案夾,
image
把你最開始下載下來的freemodbus中modbus檔案夾中的內容復制到你剛才的modbus檔案夾中,
image
把你最開始下載下來的freemodbus中demo/bare路徑下的內容全部復制到你剛才的port檔案夾中
image

然后我們進入cubeide,右鍵專案->屬性,配置頭檔案和源檔案路徑,
image
把如圖六個頭檔案路徑添加,
image
把如圖最下面兩個源檔案路徑添加,
image

先在port.h檔案中補充這兩個宏定義,這是HAL庫的全域中斷開啟、關閉函式,
image

ok,然后我們修改portserail.c
這兩個函式前面的static標志去掉,
image
vMBPortSerialEnable函式修改如下

void
vMBPortSerialEnable( BOOL xRxEnable, BOOL xTxEnable )
{
    /* If xRXEnable enable serial receive interrupts. If xTxENable enable
     * transmitter empty interrupts.
     */
	if (xRxEnable)						//將串口收發中斷和modbus聯系起來,下面的串口改為自己使能的串口
				{
					__HAL_UART_ENABLE_IT(&huart2,UART_IT_RXNE);	//我用的是串口2,故為&huart2
					HAL_GPIO_WritePin(EN485_GPIO_Port, EN485_Pin, GPIO_PIN_RESET);//
				}
			else
				{
					__HAL_UART_DISABLE_IT(&huart2,UART_IT_RXNE);
					HAL_GPIO_WritePin(EN485_GPIO_Port, EN485_Pin, GPIO_PIN_SET);//
				}
	if (xTxEnable)
				{
				HAL_GPIO_WritePin(EN485_GPIO_Port, EN485_Pin, GPIO_PIN_SET);//
					__HAL_UART_ENABLE_IT(&huart2,UART_IT_TXE);
				}//
			else
				{
				HAL_GPIO_WritePin(EN485_GPIO_Port, EN485_Pin, GPIO_PIN_RESET);//
					__HAL_UART_DISABLE_IT(&huart2,UART_IT_TXE);
				}

}

串口初始化函式如下

BOOL
xMBPortSerialInit( UCHAR ucPORT, ULONG ulBaudRate, UCHAR ucDataBits, eMBParity eParity )
{
			huart2.Instance = USART2;
		    huart2.Init.BaudRate = ulBaudRate;
		    huart2.Init.StopBits = UART_STOPBITS_1;
		    huart2.Init.Mode = UART_MODE_TX_RX;
		    huart2.Init.HwFlowCtl = UART_HWCONTROL_NONE;
		    huart2.Init.OverSampling = UART_OVERSAMPLING_16;

		    switch(eParity)
		    {
		    // 奇校驗
		    case MB_PAR_ODD:
		        huart2.Init.Parity = UART_PARITY_ODD;
		        huart2.Init.WordLength = UART_WORDLENGTH_9B;            // 帶奇偶校驗資料位為9bits
		        break;

		    // 偶校驗
		    case MB_PAR_EVEN:
		        huart2.Init.Parity = UART_PARITY_EVEN;
		        huart2.Init.WordLength = UART_WORDLENGTH_9B;            // 帶奇偶校驗資料位為9bits
		        break;

		    // 無校驗
		    default:
		        huart2.Init.Parity = UART_PARITY_NONE;
		        huart2.Init.WordLength = UART_WORDLENGTH_8B;            // 無奇偶校驗資料位為8bits
		        break;
		    }

		    return HAL_UART_Init(&huart2) == HAL_OK ? TRUE : FALSE;
}

收發位元組函式如下

BOOL
xMBPortSerialPutByte( CHAR ucByte )
{
    /* Put a byte in the UARTs transmit buffer. This function is called
     * by the protocol stack if pxMBFrameCBTransmitterEmpty( ) has been
     * called. */
	HAL_GPIO_WritePin(EN485_GPIO_Port, EN485_Pin, GPIO_PIN_SET);//

	    if(HAL_UART_Transmit (&huart2 ,(uint8_t *)&ucByte,1,10) != HAL_OK )
	        return FALSE ;//HAL_UART_Transmit最后一位形參為最大發送時間,
	    				  //超出改時間退出發送,可能導致485發送失敗,可稍微長一點,
	    else
	        return TRUE;
}

BOOL
xMBPortSerialGetByte( CHAR * pucByte )
{
    /* Return the byte in the UARTs receive buffer. This function is called
     * by the protocol stack after pxMBFrameCBByteReceived( ) has been called.
     */
	   HAL_GPIO_WritePin(EN485_GPIO_Port, EN485_Pin, GPIO_PIN_RESET);

	    if(HAL_UART_Receive (&huart2,(uint8_t *)pucByte,1,10) != HAL_OK )
	        return FALSE ;
	    else
	        return TRUE;
}

然后我們修改porttimer.c
首先依舊去掉這個函式前的static標志,方便之后呼叫,函式宣告和函式物體前的static都要去掉
image

然后修改這幾個函式:
定時器初始化函式

BOOL
xMBPortTimersInit( USHORT usTim1Timerout50us )
{
			TIM_ClockConfigTypeDef sClockSourceConfig = {0};
		    TIM_MasterConfigTypeDef sMasterConfig = {0};

		    htim2.Instance = TIM2;
		    htim2.Init.Prescaler = 3599;								// 50us記一次數
		    htim2.Init.CounterMode = TIM_COUNTERMODE_UP;
		    htim2.Init.Period = usTim1Timerout50us-1;					// usTim1Timerout50us * 50即為定時器溢位時間
		    htim2.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1;
		    htim2.Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_DISABLE;
		    if (HAL_TIM_Base_Init(&htim2) != HAL_OK)
		    {
		        return FALSE;
		    }
		    sClockSourceConfig.ClockSource = TIM_CLOCKSOURCE_INTERNAL;
		    if (HAL_TIM_ConfigClockSource(&htim2, &sClockSourceConfig) != HAL_OK)
		    {
		        return FALSE;
		    }
		    sMasterConfig.MasterOutputTrigger = TIM_TRGO_UPDATE;
		    sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE;
		    if (HAL_TIMEx_MasterConfigSynchronization(&htim2, &sMasterConfig) != HAL_OK)
		    {
		        return FALSE;
		    }

		    return TRUE;
}

然后是定時器開啟、關閉、中斷服務函式

inline void
vMBPortTimersEnable(  )
{
    /* Enable the timer with the timeout passed to xMBPortTimersInit( ) */
		__HAL_TIM_CLEAR_IT(&htim2,TIM_IT_UPDATE);//避免程式一上電就進入定時器中斷
		__HAL_TIM_ENABLE_IT(&htim2,TIM_IT_UPDATE);
		__HAL_TIM_SET_COUNTER(&htim2, 0);		// 清空計數器
	    __HAL_TIM_ENABLE(&htim2);				// 使能定時器

}

inline void
vMBPortTimersDisable(  )
{
    /* Disable any pending timers. */
    __HAL_TIM_DISABLE(&htim2);				// 禁能定時器
	__HAL_TIM_SET_COUNTER(&htim2,0);
	__HAL_TIM_DISABLE_IT(&htim2,TIM_IT_UPDATE);
	__HAL_TIM_CLEAR_IT(&htim2,TIM_IT_UPDATE);
}

/* Create an ISR which is called whenever the timer has expired. This function
 * must then call pxMBPortCBTimerExpired( ) to notify the protocol stack that
 * the timer has expired.
 */
void prvvTIMERExpiredISR( void )
{
    ( void )pxMBPortCBTimerExpired(  );
}

最后還有兩處修改,有的教程中并沒有提到這兩處修改,應該是與vMBPortSerialEnable中使用USART_IT_TC還是USART_IT_TXE中斷標志有關,如果使用USART_IT_TC中斷的話需要添加這兩處修改,就我目前使用USART_IT_TXE中斷標志的情況下,加上這兩處修改也并無問題,待后續研究明白了再更新

//啟動第一次發送,進入發送完成中斷
           xMBPortSerialPutByte( ( CHAR )*pucSndBufferCur );
           pucSndBufferCur++;  /* next byte in sendbuffer. */
           usSndBufferCount--;
   		//添加代碼end
 //插入代碼begin
                if(eStatus==MB_ENOERR)
                {
                	xMBRTUTransmitFSM();  //發送一幀資料中第一個位元組出發發送完成中斷
                }
                //插入代碼end

image
image

在系統的中斷處理.c中添加以下的函式宣告,有些教程是自己寫的中斷處理,這里我們還是用系統自己的,

定時器中斷處理函式:

void TIM2_IRQHandler(void)
{
  /* USER CODE BEGIN TIM2_IRQn 0 */

	  HAL_NVIC_ClearPendingIRQ(TIM2_IRQn);

  /* USER CODE END TIM2_IRQn 0 */
  HAL_TIM_IRQHandler(&htim2);
  /* USER CODE BEGIN TIM2_IRQn 1 */

  /* USER CODE END TIM2_IRQn 1 */
}

串口中斷函式:

void USART2_IRQHandler(void)
{
  /* USER CODE BEGIN USART2_IRQn 0 */
	  if(__HAL_UART_GET_IT_SOURCE(&huart2, UART_IT_RXNE)!= RESET)
	  		{
	  			prvvUARTRxISR();//接收中斷
	  		}

	  	if(__HAL_UART_GET_IT_SOURCE(&huart2, UART_IT_TXE)!= RESET)
	  		{
	  			prvvUARTTxReadyISR();//發送中斷
	  		}

	    HAL_NVIC_ClearPendingIRQ(USART2_IRQn);
  /* USER CODE END USART2_IRQn 0 */
  HAL_UART_IRQHandler(&huart2);
  /* USER CODE BEGIN USART2_IRQn 1 */

  /* USER CODE END USART2_IRQn 1 */
}

在檔案末尾user code 代碼段添加定時器中斷回呼函式:

/* USER CODE BEGIN 1 */
void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
{
  /* NOTE : This function Should not be modified, when the callback is needed,
            the __HAL_TIM_PeriodElapsedCallback could be implemented in the user file
   */
	if(htim->Instance == TIM2)
	{
		prvvTIMERExpiredISR( );
	}

}
/* USER CODE END 1 */



回到main函式

image
首先包含幾個頭檔案
在Private define段添加如下:

/* Private define ------------------------------------------------------------*/
/* USER CODE BEGIN PD */
//輸入暫存器起始地址
#define REG_INPUT_START       0x0001
//輸入暫存器數量
#define REG_INPUT_NREGS       8
//保持暫存器起始地址
#define REG_HOLDING_START     0x0001
//保持暫存器數量
#define REG_HOLDING_NREGS     8

//線圈起始地址
#define REG_COILS_START       0x0001
//線圈數量
#define REG_COILS_SIZE        16

//離散暫存器起始地址
#define REG_DISCRETE_START    0x0001
//離散暫存器數量
#define REG_DISCRETE_SIZE     16
/* USER CODE END PD */

Private variables段添加如下:

/* Private variables ---------------------------------------------------------*/

/* USER CODE BEGIN PV */
//輸入暫存器內容
uint16_t usRegInputBuf[REG_INPUT_NREGS] = {0x1000,0x1001,0x1002,0x1003,0x1004,0x1005,0x1006,0x1007};
//輸入暫存器起始地址
uint16_t usRegInputStart = REG_INPUT_START;

//保持暫存器內容
uint32_t usRegHoldingBuf[REG_HOLDING_NREGS] = {0x147b,0x3f8e,0x147b,0x400e,0x1eb8,0x4055,0x147b,0x408e};
//保持暫存器起始地址
uint16_t usRegHoldingStart = REG_HOLDING_START;

//線圈狀態
uint8_t ucRegCoilsBuf[REG_COILS_SIZE / 8] = {0x01,0x02};
//離散輸入狀態
uint8_t usRegDiscreteBuf[REG_DISCRETE_SIZE / 8] = {0x01,0x02};

uint8_t testfalg=0;

extern unsigned char NUM [];
/* USER CODE END PV */

Private function prototypes段添加如下:

/* USER CODE BEGIN PFP */
/**
 * @Brief : 讀輸入暫存器處理函式,功能碼04
 * @param  pucRegBuffer    保存輸入暫存器值的快取
 * @param  usAddress       暫存器地址
 * @param  usNRegs         讀取個數
 * @return?eMBErrorCode?
 */
eMBErrorCode
eMBRegInputCB( UCHAR * pucRegBuffer, USHORT usAddress, USHORT usNRegs )
{
    eMBErrorCode    eStatus = MB_ENOERR;
    int             iRegIndex;

    if( ( usAddress >= REG_INPUT_START )\
        && ( usAddress + usNRegs <= REG_INPUT_START + REG_INPUT_NREGS ) )
    {
        iRegIndex = ( int )( usAddress - usRegInputStart );
        while( usNRegs > 0 )
        {
            *pucRegBuffer++ = ( UCHAR )( usRegInputBuf[iRegIndex] >> 8 );
            *pucRegBuffer++ = ( UCHAR )( usRegInputBuf[iRegIndex] & 0xFF );
            iRegIndex++;
            usNRegs--;
        }
    }
    else
    {
        eStatus = MB_ENOREG;
    }

    return eStatus;
}

/**
 * @Brief : 讀保持暫存器處理函式,功能碼03
 * @param  pucRegBuffer
 * @param  usAddress
 * @param  usNRegs
 * @param  eMode
 * @return?eMBErrorCode?
 */
eMBErrorCode
eMBRegHoldingCB( UCHAR * pucRegBuffer, USHORT usAddress, USHORT usNRegs, eMBRegisterMode eMode )
{
	eMBErrorCode    eStatus = MB_ENOERR;
	int             iRegIndex;


	if((usAddress >= REG_HOLDING_START)&&\
		((usAddress+usNRegs) <= (REG_HOLDING_START + REG_HOLDING_NREGS)))
	{
		iRegIndex = (int)(usAddress - usRegHoldingStart);
		switch(eMode)
		{
			case MB_REG_READ://???? MB_REG_READ = 0
        	while(usNRegs > 0)
				{
					*pucRegBuffer++ = (uint8_t)(usRegHoldingBuf[iRegIndex] >> 8);
					*pucRegBuffer++ = (uint8_t)(usRegHoldingBuf[iRegIndex] & 0xFF);
          			iRegIndex++;
          			usNRegs--;
				}
        	break;
			case MB_REG_WRITE://???? MB_REG_WRITE = 1
			while(usNRegs > 0)
				{
					usRegHoldingBuf[iRegIndex] = *pucRegBuffer++ << 8;
          			usRegHoldingBuf[iRegIndex] |= *pucRegBuffer++;
          			iRegIndex++;
          			usNRegs--;
        		}
		}
	}
	else//錯誤
	{
		eStatus = MB_ENOREG;
	}

	return eStatus;
}

/**
  *****************************************************************************
  * @Name   : 操作線圈
  *
  * @Brief  : 對應功能????0x01 -> eMBFuncReadCoils
  *                    0x05 -> eMBFuncWriteCoil
  *                    0x15 -> 寫多個線???? eMBFuncWriteMultipleCoils
  *
  * @Input  : *pucRegBuffer:資料緩沖區,回應主機用
  *           usAddress:     暫存器地????
  *           usNRegs:       操作暫存器個????
  *           eMode:         功能????
  *
  * @Output : none
  *
  * @Return : Modbus狀???信????
  *****************************************************************************
**/
eMBErrorCode eMBRegCoilsCB( UCHAR * pucRegBuffer, USHORT usAddress, USHORT usNCoils, eMBRegisterMode eMode )
{
	eMBErrorCode	eStatus = MB_ENOERR;
	int 			iNCoils = ( int )usNCoils;
	unsigned short	usBitOffset;

	/* Check if we have registers mapped at this block. */
	if( ( usAddress >= REG_COILS_START ) &&	( usAddress + usNCoils <= REG_COILS_START + REG_COILS_SIZE ) )
	{
		usBitOffset = ( unsigned short )( usAddress - REG_COILS_START );
		switch ( eMode )
		{
			/* Read current values and pass to protocol stack. */
			case MB_REG_READ:
			while( iNCoils > 0 )
			{
				*pucRegBuffer++ = xMBUtilGetBits( ucRegCoilsBuf, usBitOffset,	( unsigned char )( iNCoils > 8 ? 8 : iNCoils ) );
				iNCoils -= 8;
				usBitOffset += 8;
			}
			break;

				/* Update current register values. */
			case MB_REG_WRITE:
			while( iNCoils > 0 )
			{
				xMBUtilSetBits( ucRegCoilsBuf, usBitOffset,	( unsigned char )( iNCoils > 8 ? 8 : iNCoils ),	*pucRegBuffer++ );
				iNCoils -= 8;
				usBitOffset += 8;
			}
			break;
		}

	}
	else
	{
		eStatus = MB_ENOREG;
	}
	return eStatus;
}


/**
  *****************************************************************************
  * @Name   : 操作離散寄存????
  *
  * @Brief  : 對應功能????0x02 -> eMBFuncReadDiscreteInputs
  *
  * @Input  : *pucRegBuffer:資料緩沖區,回應主機用
  *           usAddress:     暫存器地????
  *           usNRegs:       操作暫存器個????
  *
  * @Output : none
  *
  * @Return : Modbus狀???信????
  *****************************************************************************
**/
//eMBErrorCode eMBRegDiscreteCB( UCHAR * pucRegBuffer, USHORT usAddress, USHORT usNDiscrete )
//{
//	pucRegBuffer = pucRegBuffer;

//	return MB_ENOREG;
//}

eMBErrorCode
eMBRegDiscreteCB( UCHAR * pucRegBuffer, USHORT usAddress, USHORT usNDiscrete )
{
    eMBErrorCode    eStatus = MB_ENOERR;
    short           iNDiscrete = ( short )usNDiscrete;
    USHORT  usBitOffset;

    /* Check if we have registers mapped at this  block. */
    if( ( usAddress >= REG_DISCRETE_START ) && ( usAddress + usNDiscrete <= REG_DISCRETE_START + REG_DISCRETE_SIZE ) )
    {
        usBitOffset = ( USHORT )( usAddress - REG_DISCRETE_START );
        while( iNDiscrete > 0 )
        {
            *pucRegBuffer++ =
            xMBUtilGetBits( usRegDiscreteBuf, usBitOffset,( UCHAR )( iNDiscrete > 8 ? 8 : iNDiscrete ) );
            iNDiscrete -= 8;
            usBitOffset += 8;
        }
    }
    else
    {
        eStatus = MB_ENOREG;
    }
    return eStatus;
}
/* USER CODE END PFP */

新建變數estatus 前后說的這些添加內容必須夾在user code欄位中,否則cubemx修改工程重新生成代碼后,你修改的內容會消失
image
初始化、使能
image
因為前面在串口初始化中,串口選擇是被我寫死的,不能通過這里的第三個形參去選擇使用串口幾,但是通過簡單修改可以實作自由配置

最后在main函式中啟動輪詢就可以了
image

最后你把想傳出去的資料放在各個暫存器中,外部modbus主機就可以查詢到你放入暫存器的資料了

/* Private variables ---------------------------------------------------------*/

/* USER CODE BEGIN PV */
//輸入暫存器內容
uint16_t usRegInputBuf[REG_INPUT_NREGS] = {0x1000,0x1001,0x1002,0x1003,0x1004,0x1005,0x1006,0x1007};
//輸入暫存器起始地址
uint16_t usRegInputStart = REG_INPUT_START;

//保持暫存器內容
uint32_t usRegHoldingBuf[REG_HOLDING_NREGS] = {0x147b,0x3f8e,0x147b,0x400e,0x1eb8,0x4055,0x147b,0x408e};
//保持暫存器起始地址
uint16_t usRegHoldingStart = REG_HOLDING_START;

//線圈狀態
uint8_t ucRegCoilsBuf[REG_COILS_SIZE / 8] = {0x01,0x02};
//離散輸入狀態
uint8_t usRegDiscreteBuf[REG_DISCRETE_SIZE / 8] = {0x01,0x02};

uint8_t testfalg=0;

extern unsigned char NUM [];
/* USER CODE END PV */

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

標籤:其他

上一篇:MyBatis-Plus 可視化代碼生成器來啦,讓你的開發效率大大提速!!

下一篇:返回列表

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

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • freemodbus移植進STM32(包含HAL庫和標準庫兩種方法)

    #freemodbus移植 >基于freemodbus1.6 >使用HAL庫 >軟體:stm32cubemx stm32cubeide >>后續會更新標準庫的移植。以及rtos下的移植(盡量) ##下載freemodbus1.6 這個獲取方法網上到處都是,不細說了。 ##cubemx新建工程 新建工 ......

    uj5u.com 2023-05-23 08:08:17 more
  • MyBatis-Plus 可視化代碼生成器來啦,讓你的開發效率大大提速!!

    ## **前言** 在基于Mybatis的開發模式中,很多開發者還會選擇Mybatis-Plus來輔助功能開發,以此提高開發的效率。雖然Mybatis也有代碼生成的工具,但Mybatis-Plus由于在Mybatis基礎上做了一些調整,因此,常規的生成工具生成的代碼還有一些不太符合預期。而且對于多數 ......

    uj5u.com 2023-05-23 08:07:35 more
  • django 計算兩個TimeField的時差

    在 Django 中,你可以使用 datetime 模塊來計算兩個 TimeField 欄位的時間差。以下是一個示例: from datetime import datetime, timedelta # 假設有兩個 TimeField 欄位 time1 = obj.time_field1 time ......

    uj5u.com 2023-05-23 08:07:19 more
  • 【Java】參考傳遞?值傳遞?

    參考傳遞和值傳遞,從上學那會兒就開始強調的概念,不管你是計算機相關專業還是自學Java,一定聽過這么一句話: 方法呼叫引數如果是物件,那就是參考傳遞,如果是基本資料型別,就是值傳遞。 比如:function(Object o)就是參考傳遞,function(int i)就是值傳遞。這兩個概念似乎很好 ......

    uj5u.com 2023-05-23 08:07:13 more
  • Java入門9(HashSet,File檔案類)

    ## HashSet 1. jdk1.7之前,使用陣列加鏈表的方式實作 2. jdk1.8之后,在鏈表長度大于8并且陣列長度超過32的情況下,會轉成紅黑樹結構 3. HashSet的本質是一個HashMap,它所有的value都是一致的,傳入的引數作為key,因此HashSet中不允許重復資料 4. ......

    uj5u.com 2023-05-23 08:06:47 more
  • java 外殼加密,完美解決

    圣天諾LDK加密鎖(加密狗),對war包加密的測驗,測驗war包(或jar包)防止被反編譯的效果。 http://chinadlp.com/?list-DriveDownload.html 下載最新開發套件:Sentinel HASP/LDK9.0開發套件。完全默認安裝。 如果是有主鎖的正式用戶請導 ......

    uj5u.com 2023-05-23 08:06:25 more
  • Java使用HttpClient以multipart/form-data向介面上傳檔案

    ## 前言 對接某公司的介面,涉及到資質上傳等業務。需要對接他們的上傳附件介面。 JDK1.8 httpclient 4.x ## 封裝httpclient方法 ```java public static String postFileMultiPart(String url,Map reqPara ......

    uj5u.com 2023-05-23 08:06:11 more
  • Java集合中Set都有哪些特性?看這篇就夠了!

    **本文將為大家詳細講解Java中的,這是我們進行開發時經常用到的知識點,也是大家在學習Java中很重要的一個知識點,更是我們在面試時有可能會問到的問題。** **文章較長,干貨滿滿,建議大家收藏慢慢學習。文末有本文重點總結,主頁有全系列文章分享。技術類問題,歡迎大家和我們一起交流討論!** ### ......

    uj5u.com 2023-05-23 08:06:07 more
  • python呼叫父類方法的三種方式(super呼叫和父類名呼叫)

    ### 子類呼叫父類的方法的三種方式: - 父類名.方法名(self) - super(子類名,self).父類方法名() - super().父類方法名 注意:super()通過子類呼叫當前父類的方法,super默認會呼叫第一個父類的方法(適用于單繼承的多層繼承 如下代碼: ```python # ......

    uj5u.com 2023-05-23 08:05:26 more
  • 用Python將女朋友的照片做成壁紙軟體,實作桌面壁紙自動更換!

    話說兄弟們,女朋友生氣了都是怎么哄的? 不會吧不會吧,不會有人還是單身狗吧! 算了,還是回到正題吧,再說我要挨打了~ 今天咱們來交流一下程式員是怎么哄女朋友的,話不多說直接開始! 準備作業 1、環境 首先我們準備好環境和編輯器,我使用的是: Python 3.8 解釋器 Pycharm 編輯器 2、 ......

    uj5u.com 2023-05-23 08:05:18 more