为啥当我调用 UpdateData(FALSE) 时我的对话框没有立即更新?

Posted

技术标签:

【中文标题】为啥当我调用 UpdateData(FALSE) 时我的对话框没有立即更新?【英文标题】:Why isn't my dialog box updated immediately when I call UpdateData(FALSE)?为什么当我调用 UpdateData(FALSE) 时我的对话框没有立即更新? 【发布时间】:2011-02-17 06:15:25 【问题描述】:

谁能告诉我为什么在我的基于 VC++ 对话框的应用程序中调用消息框之前调用 UpdateData(FALSE) 不起作用?运行时窗口保持不变,直到我调用:

MessageBoxW(cDisp, L"!!!Data Count!!!", MB_OK | MB_ICONINFORMATION)  

我的一段代码sn-p是这样的……

for(int j=0;j<50;j++)
        


              if(ferr==0)
                
                    double X[15][4],splusn[15],m_dBiasC,WOld[1][4],alpha,err,WNew[1][4],y[15][1],WTransOld[4][1];

                    //do
                    // 
                            int iGraphX,iGraphY;
                            CString cDisp1(""),cDisp;
                            m_dBiasC=m_dC;
                            err=m_dError;
                            alpha=m_dAlpha;
                            char ch;
                            double dummy1,dummy2;
                            cDisp.Format(L"%d",j);
                                m_sCount.Format(L"%d",m_iInterval);
                                UpdateData(FALSE);
                                BeginWaitCursor();
                            for(int i=0;i<15;i++)
                                
                                    f[0]>>dummy1>>ch>>dummy2;
                                    Time[i]=dummy1;
                                    m_dAccX[i]=dummy2;
                                    if(!f[0].good())
                                        
                                            MessageBox(L"The end of first data file encountered",L"Caution");
                                            MessageBoxW(L"Quiting Program ",L"Caution");
                                            ferr=2;
                                            //break;
                                            exit(1);
                                        
                                    iGraphX=static_cast<int>(Time[i]*100.0);
                                    iGraphY=static_cast<int>(m_dAccX[i]);
                                    m_Graph[0].RemovePoint(0,0);
                                    m_Graph[0].AddPoint(iGraphX,iGraphY);
                                    cDisp.Format(L"%lf  %lf",Time[i],m_dAccX[i]);
                                    cDisp1+=cDisp;
                                    cDisp1+="\r\n";
                                    f[1]>>dummy1>>ch>>dummy2;
                                    Time[i]=dummy1;
                                    m_dAccY[i]=dummy2;
                                    if(!f[1].good())
                                        
                                            MessageBox(L"The end of second data file encountered",L"Caution");
                                            MessageBoxW(L"Quiting Program ",L"Caution");
                                            ferr=3;
                                            //break;
                                            exit(1);
                                        
                                    iGraphX=static_cast<int>(Time[i]*50.0);
                                    iGraphY=static_cast<int>(m_dAccY[i]*10);
                                    m_Graph[1].RemovePoint(0,0);
                                    m_Graph[1].AddPoint(iGraphX,iGraphY);

                                    f[2]>>dummy1>>ch>>dummy2;
                                    Time[i]=dummy1;
                                    m_dAccZ[i]=dummy2;
                                    if(!f[2].good())
                                        
                                            MessageBox(L"The end of third data file encountered",L"Caution");
                                            MessageBoxW(L"Quiting Program ",L"Caution");
                                            ferr=4;
                                            //break;
                                            exit(1);
                                        
                                    iGraphX=static_cast<int>(Time[i]*50.0);
                                    iGraphY=static_cast<int>(m_dAccZ[i]);
                                    m_Graph[2].RemovePoint(0,0);
                                    m_Graph[2].AddPoint(iGraphX,iGraphY);
                                    f[3]>>dummy1>>ch>>dummy2;
                                    Time[i]=dummy1;
                                    m_dECG[i]=dummy2;
                                    if(!f[3].good())
                                        
                                            MessageBox(L"The end of fourth data file encountered",L"Caution");
                                            MessageBoxW(L"Quiting Program ",L"Caution");
                                            ferr=5;
                                            //break;
                                            exit(1);
                                        
                                    iGraphX=static_cast<int>(Time[i]*100.0);
                                    iGraphY=static_cast<int>(m_dECG[i]*100);
                                    m_Graph[3].RemovePoint(0,0);
                                    m_Graph[3].AddPoint(iGraphX,iGraphY);
                                
                            Sleep(500);
                            GetDlgItem(IDC_DISPLAYFILE)->SetWindowTextW(cDisp1);
                            UpdateData(FALSE);
                            EndWaitCursor();
                            cDisp.Format(L"Data Read till now: %d",((j+1)*15));
                            MessageBox(cDisp,L"!!!Data Count!!!",MB_OK|MB_ICONINFORMATION);
                            UpdateData(FALSE);
                            //This part is for adaptive filter calculations

                            for(int r=0;r<15;r++)
                                
                                    splusn[r]=static_cast<double>(m_dECG[r]);
                                    X[r][0]=m_dBiasC;
                                    X[r][1]=static_cast<double>(m_dAccX[r]*m_dScaleX);
                                    X[r][2]=static_cast<double>(m_dAccY[r]*m_dScaleY);
                                    X[r][3]=static_cast<double>(m_dAccZ[r]*m_dScaleZ);      
                                
                            WOld[0][0]=m_dW0;
                            WOld[0][1]=m_dW1;
                            WOld[0][2]=m_dW2;
                            WOld[0][3]=m_dW3;
                            for(int q=0;q<4;q++)
                                WNew[0][q]=WOld[0][q];
                            for(int p=0;p<15;p++)
                                
                                    for(int iRectCnt=0;iRectCnt<60;iRectCnt++)
                                        
                                            f[4]<<setiosflags(ios::fixed)<<setw(8)<<setprecision(4)<<p<<"       ";
                                            for(int iWCnt=0;iWCnt<4;iWCnt++) 
                                                f[4]<<setw(8)<<setprecision(2)<<WOld[0][iWCnt];
                                            f[4]<<endl;
                                            y[p][0]=0.0;
                                            for(int q=0;q<4;q++)
                                                WTransOld[q][0]=WOld[0][q];
                                            for(int r=0;r<4;r++)
                                            
                                                y[p][0]=y[p][0]+(X[p][r]*WTransOld[r][0]);
                                            
                                            err=splusn[p]-y[p][0];
                                            for(int q=0;q<4;q++)
                                                WNew[0][q]=WOld[0][q]+(2.0*alpha*err*X[p][q]);
                                            for(int q=0;q<4;q++)
                                                WOld[0][q]=WNew[0][q];
                                         
                                    cDisp.Format(L"%lf",err);
                                    GetDlgItem(IDC_ERROR)->SetWindowTextW(cDisp);
                                    splusn[p]=splusn[p]-err;
                                 
                            for(int p=0;p<15;p++)
                            
                                iGraphX=static_cast<int>(Time[p]*100.0);
                                iGraphY=static_cast<int>(splusn[p]*100);
                                m_Graph[4].RemovePoint(0,0);
                                m_Graph[4].AddPoint(iGraphX,iGraphY);
                            
                            cDisp.Format(L"%lf",WNew[0][0]);
                            GetDlgItem(IDC_W0C)->SetWindowTextW(cDisp);
                            cDisp.Format(L"%lf",WNew[0][1]);
                            GetDlgItem(IDC_W1C)->SetWindowTextW(cDisp);
                            cDisp.Format(L"%lf",WNew[0][2]);
                            GetDlgItem(IDC_W2C)->SetWindowTextW(cDisp);
                            cDisp.Format(L"%lf",WNew[0][3]);
                            GetDlgItem(IDC_W3C)->SetWindowTextW(cDisp);
                            m_sCount.Format(L"%d",0);


                            // End of Adaptive filter 
                            //close files here if already end of file is encountered
                            if((f[0].eof()||f[1].eof()||f[2].eof()||f[3].eof()))
                            
                                MessageBox(L"End of File encountered");
                                ferr=1;
                                exit(1);
                            


                     //while(!(f[0].eof())&&!(f[1].eof())&&!(f[2].eof())&&!(f[3].eof()));
       

    

【问题讨论】:

我们无所不知,无所不知,所以是的,我们可以告诉您您的代码出了什么问题,而不会看到它的任何 sn-p。 ...我们可以告诉你,但我们不这样做,因为我们真的是坏人! 对不起 Mehrdad,您的意思是我应该在问题中添加我的代码的 sn-p 吗?如果是的话,我可以这样做...... 有趣的阅读:Avoiding UpdateData。建议的替代方法是访问 控制变量 中的值,而不是使用 UpdateData 函数。 @sspdd:哈哈,是的,我的意思是你应该在你的代码中添加一个 sn-p,所以我们看看事情是如何相互作用的。 (一般来说,即使看起来很明显,最好放一段代码,这样人们可以更真实地看到正在发生的事情。)感谢您发布它。 :) 【参考方案1】:

该代码有很多问题,我什至不知道从哪里开始。对于初学者:

    你为什么打电话给Sleep()??我无法想象您需要在设计良好的应用程序中执行此操作的原因。 特别是不要重复,正如您在代码中显示的那样。 Sleep 函数使您的线程暂停指定的毫秒数。尝试完全拨打Sleep 的电话,看看这是否不能解决您的问题。如果线程被挂起,UI 将如何更新?

    为什么要将UpdateDataGetDlgItem 混合使用并调用SetWindowText?如果您让 MFC 负责协调成员变量的值与对话框控件中显示的值,则不需要自己设置它们的属性。

您也没有向我们展示足够多的代码来了解CDisp1 是什么,或者IDC_DISPLAYFILE 是什么。您最终要完成什么?您想在处理数据时进行实时更新吗?如果我们知道目标是什么,我们当然可以就如何构建您的代码提供更好的帮助,而不仅仅是不工作的代码。

更新:啊,我看到你刚刚在评论中添加了上述代码在 for 循环中。所以你的问题实际上很简单:for 循环将 CPU 置于一个紧密循环中,除了你的代码之外什么都没有执行。当您的循环执行指定次数时,所有消息处理(尤其是更新 UI)都会暂停。

调用UpdateData 对对话框中控件的值没有任何结果,因为您的计算机的处理器当前 循环。显示消息框会通过创建自己的消息泵来暂时阻止 for 循环的执行,从而允许更新 UI。将代码从for 循环中取出,并删除对Sleep 的调用,您的问题就会消失。

从 cmets 看来,您最好的做法是创建一个工作线程并将循环放置在那里,将其与您的主 (GUI) 线程隔离开来。然后,工作线程可以向您的主线程发送消息以更新对话框中的控件。完整的解决方案和更多信息已作为此问题的答案提供:How can I show a modeless dialog and display information in it immediately?

【讨论】:

我相信你可以提供帮助,因为你已经提供了足够的信息......你是对的我需要在处理数据时实时更新......实际上我的代码 sn-p 非常大...... .如果你们没有任何异议,那么只有我可以将它与我的问题一起发布.. @sspdd:真正的问题可能是您不应该首先尝试在 GUI 线程中运行循环。分拆一个单独的线程并在那里进行处理。其他任何东西都是黑客,简单明了。如果您不反对,您可以自己抽出消息循环,但我不能说我有多强烈不同意该选择。示例代码见this thread。 好吧,如果我将代码从 for 循环中取出,那么如何连续处理我的数据?实际上,我正在阅读 4 个包含电压随时间变化的文件,并将它们绘制在屏幕上。对于每个循环索引,文件指针在文件中向下移动并再读取 15 个数据。然后将它们绘制在屏幕上。 你允许我在我的问题中添加完整的代码 sn-p 吗? @sspdd:当然,您可以添加代码。不是让你这样做吗?【参考方案2】:

可能是因为对 MessageBox 的调用包含消息泵。使用睡眠意味着 UI 不会响应。

【讨论】:

那么请告诉我如何在仍然使用 Sleep() 的同时避免出现 MessageBox,因为我需要让用户在屏幕上查看更改? @sspdd 您需要了解消息队列的工作原理,而此类程序永远不需要Sleep

以上是关于为啥当我调用 UpdateData(FALSE) 时我的对话框没有立即更新?的主要内容,如果未能解决你的问题,请参考以下文章

UpdateData(TRUE)与UpdateData(FALSE)的使用

UpdateData()函数

MFC编辑框关联的变量怎么用UpdateData(FALSE)无法直接写到编辑框中啊,而是需要手动点编辑框

当我在 firestore 中的 updateData 比 updateData 功能不起作用时?

为啥 UpdateData() 在 MFC CEdit 控件的 EN_CHANGE 处理程序中不起作用

VC中编辑框更新SetDlgItemText()与UpdateData()的区别