各位論壇好友,大家好,真心求幫忙!
我這邊專案需要,參考龔建偉老師SCOMMV2.3串口除錯助手,改寫了一個串口除錯,需要對開發板進行操作,波特率設定115200;改寫的串口界面,接受資料存在延時;即使給開發板斷電后,仍然過一段時間才停止接受資料,而不是立刻停止。這樣延時現象,我不容易通過設計的串口界面,按任意鍵進入系統uboot進行更進一步操作開發。
這個軟體是結合專案設計的,延時現象困擾了1周了,網上說有用執行緒處理的,但不熟悉,求幫忙指點。
串口接受資料核心代碼如下,使用CSerialPort類撰寫的串口,幫忙分析下哪里有問題,謝謝各位了:
static long rxdatacount=0; //該變數用于接收字符計數
LONG CSCOMMDlg::OnCommunication(WPARAM ch, LPARAM port)//通信,訊息回應
{ //定義變數:接受字符ch和埠port
if (port <= 0 || port > 11) //port代表串口號,取值1-10
return -1;
rxdatacount++; //接收的位元組計數
CString strTemp;
strTemp.Format("%ld",rxdatacount);//輸出長整型的接收資料計數rxdatacount
strTemp="RX:"+strTemp;
//m_ctrlRXCOUNT.SetWindowText(strTemp); //顯示接收計數
if(m_bStopDispRXData) //如果選擇了“停止顯示”接收資料,則回傳
return -1; //注意,這種情況下,計數仍在繼續,只是不顯示
//若設定了“自動清空”,則達到50行后,自動清空接收編輯框中顯示的資料
if((m_ctrlAutoClear.GetCheck())&&(m_ctrlReceiveData.GetLineCount()>=50))//程式打開,默認選擇自動清空
{
m_ReceiveData.Empty();
UpdateData(FALSE);
}
//如果沒有“自動清空”,資料行達到400后,也自動清空
//因為資料過多,影響接收速度,顯示是最費CPU時間的操作
if(m_ctrlReceiveData.GetLineCount()>400)
{
m_ReceiveData.Empty();//資料超過400行,清空界面
m_ReceiveData="https://bbs.csdn.net/topics/***The Length of the Text is too long, Emptied Automaticly!!!***/r/n";
UpdateData(FALSE);
}
CString str; //將字符送入臨時變數str存放
str.Format("%c",ch); //字符格式進行顯示
//以下是將接收的字符加在字串的最后,這里費時很多
//但考慮到資料需要保存成檔案,所以沒有用List Control
int nLen=m_ctrlReceiveData.GetWindowTextLength();//回傳指定視窗的標題文本的字符長度
m_ctrlReceiveData.SetSel(nLen, nLen); //在編輯控制元件選擇一定范圍的字符
m_ctrlReceiveData.ReplaceSel(str); //注釋后,視窗不接受資料
nLen+=str.GetLength();
m_ReceiveData+=str; //將臨時變數str存放的字符,加入到接收框
return 0;
}
uj5u.com熱心網友回復:
資料量大的話 建議不要單字訊息, 效率很低下, 盡量批量讀和處理; 如果用訊息機制,也推薦塊傳輸方式比如 OnComm(BYTE *pDataBuff, int iBuffBytesLen, int iPort)
uj5u.com熱心網友回復:
嗯,115200的波特率,接收4418主板內核資訊輸出,資料量有些大;分析發現,mfc串口界面的視窗顯示的資料存在延時,比接收的資料較慢,不能實作資料顯示與接收同步,這樣就不方便操作串口界面進入uboot做進一步開發了。像secureCRT這種串口,就不存在延時,操作方便;但我的串口界面給專案用的,需要額外添加其他功能,內核資訊顯示只是其中一部分,需要自行設計。將串口的資料接收和顯示分開處理,使用執行緒如何處理,可以給些參考代碼或提示嗎,謝謝!
串口初始化代碼如下:
BOOL CSCOMMDlg::OnInitDialog()//初始化設定
{
//CDialog::OnInitDialog();
//BOOL b = CDialog::OnInitDialog(0,fdAll,true);
BOOL b = CDialog::OnInitDialog();
// Add "About..." menu item to system menu.
// IDM_ABOUTBOX must be in the system command range.
ASSERT((IDM_ABOUTBOX & 0xFFF0) == IDM_ABOUTBOX);
ASSERT(IDM_ABOUTBOX < 0xF000);
CMenu* pSysMenu = GetSystemMenu(FALSE);
if (pSysMenu != NULL)
{
CString strAboutMenu;
strAboutMenu.LoadString(IDS_ABOUTBOX);
if (!strAboutMenu.IsEmpty())
{
pSysMenu->AppendMenu(MF_SEPARATOR);
pSysMenu->AppendMenu(MF_STRING, IDM_ABOUTBOX, strAboutMenu);
}
}
// Set the icon for this dialog. The framework does this automatically
// when the application's main window is not a dialog
SetIcon(m_hIcon, TRUE); // Set big icon
SetIcon(m_hIcon, FALSE); // Set small icon
// TODO: Add extra initialization here
//CurSel,即current selection,當前選項
m_Com.SetCurSel(0); //串口選擇第一個,即為com1
m_Speed.SetCurSel(11); //波特率case為11對應115200
m_Parity.SetCurSel(0); //校驗位選擇
m_DataBits.SetCurSel(0);//資料位選擇
m_StopBits.SetCurSel(0);//停止位選擇
m_hIconRed = AfxGetApp()->LoadIcon(IDI_ICON_RED);
m_hIconOff = AfxGetApp()->LoadIcon(IDI_ICON_OFF);
m_hIconGreen= AfxGetApp()->LoadIcon(IDI_ICON_GREEN);
// m_ctrlIconDatareceive.SetIcon(m_hIconOff);
//對這些串口變數賦初值
m_nBaud=115200; //設定波特率
m_nCom=1; //設定埠
m_cParity='N';
m_nDatabits=8;
m_nStopbits=1;
m_dwCommEvents = EV_RXFLAG | EV_RXCHAR; //串口事件
//if (m_Port.InitPort(this, 1, 9600,'N',8,1,dwCommEvents,512))
CString strStatus;
//if(),如果啟動串口成功
if (m_Port.InitPort(this, m_nCom, m_nBaud,m_cParity,m_nDatabits,m_nStopbits,m_dwCommEvents,1024))
{
m_Port.StartMonitoring();//啟動監測輔助執行緒
//列印串口狀態及引數
strStatus.Format("COM%d OPENED,%d,%c,%d,%d",m_nCom, m_nBaud,m_cParity,m_nDatabits,m_nStopbits);
m_ctrlIconOpenoff.SetIcon(m_hIconRed);
//m_ctrlIconOpenoff.SetIcon(m_hIconOff);
//"當前狀態:串口打開,無奇偶校驗,8資料位,1停止位");
}
else //如果啟動失敗
{
AfxMessageBox("沒有發現此串口");
m_ctrlIconOpenoff.SetIcon(m_hIconOff);
}
m_ctrlPortStatus.SetWindowText(strStatus);//顯示strStatus的字符:COM%d OPENED,
CEdit* pEdit=(CEdit*)GetDlgItem(IDC_EDIT_CYCLETIME);//IDC_EDIT_CYCLETIME為自動發送周期 控制元件
CString strText;
strText.Format("%d",m_nCycleTime); //m_nCycleTime為1000ms
pEdit->SetWindowText(strText); //Display cycle time
//m_strSendData="http://www.gjwtech.com";
char cr=13,lf=10;;
m_strSendData+=cr+lf;
m_ctrlAutoClear.SetCheck(1);
m_ctrlEditSendFile.SetWindowText("No File Selected!");
m_animIcon.SetImageList(IDB_ANIM_IMGLIST,4,RGB(0,0,0));
SetTimer(4,200,NULL);
UpdateData(FALSE);
ShowWindow(SW_SHOW);
// return TRUE; // return TRUE unless you set the focus to a control
return b;
}
uj5u.com熱心網友回復:
不要浪費時間研究NT5.1內核(包括正版XP和盜版Win7)的串口功能了,被回教做了手腳。由于NT6內核已經發布,NT5.1內核逐步停止維護了。
上海方正處可能拷貝得到正版Windows 6、SDK及配套編譯器。
轉載請註明出處,本文鏈接:https://www.uj5u.com/gongcheng/130519.html
上一篇:如何將當前CView傳入DLL
下一篇:VC新手,想用MFC做一個界面,在Llistcontrol中的某一列插入checkbox,不會實作,請牛人幫忙,最好有實體代碼!
