tag 标签: labwindows

相关博文
  • 热度 24
    2013-6-14 15:08
    1031 次阅读|
    0 个评论
    摘要:        雷达用于实现对目标的准确定位,帮助武器系统锁定目标,继而命中并摧毁目标。因此,雷达性能的优劣直接影响了对目标的定位精度,继而影响了整个武器系统的功效。本课题研究某型号雷达的自动测试系统,用于在研制、生产、维护过程中对雷达的性能指标进行详细、全面的测试,有力保证了雷达工作的可靠性。         本文详细分析了某型号雷达的测试需求,提出了基于PXI总线平台的雷达自动测试系统总体方案。整个系统由PXI机箱及嵌入式控制器、数据采集卡、专用串行通讯卡、信号转接组合、供电组合、自检组合、运动目标支架以及回波信号模拟器组成。该系统实现了雷达的供电控制及状态监测,雷达基本电气参数的测量,以及与雷达的同步485定时通讯、专用异步422通讯和基于TCP、UDP协议的以太网通讯等功能,且满足测试系统小型化的需求。       详细介绍了自研的专用串行通讯卡和自检组合的设计方法。专用串行通讯卡集标准异步422通讯、专用异步422通讯、同步485定时通讯功能于一身,分别实现了运动目标支架的控制、遥测设备与雷达通讯的模拟、飞控机与雷达定时通讯的模拟,且各通讯能够并行工作,降低了系统成本。自检组合基于DSP和FPGA的系统架构,实现了对雷达工作流程、测试信号以及多种通讯功能的模拟,为系统全面自检奠定了基础,其中对以太网通讯接口的模拟采用以太网协议芯片W5300实现,缩短了开发周期、降低了成本。       使用 LabWindows/CVI 软件平台完成了测试系统软件的开发,系统软件采用 VISA 技术、ODBC 技术、多线程技术进行设计,主要由测试管理软件、仪器驱动软件、手动测试软件、自动测试软件四部分组成。整个软件采用层次化、模块化设计思想,满足通用性和可移植性的要求。       系统联调结果表明,该雷达自动测试系统能够完成现阶段雷达的全部测试项目,符合雷达的测试需求。   作者:梁宇,哈尔滨工业大学   论文来自:万方知识服务平台 感谢关注!   更多信息与我们交流: WIZnet邮箱:wiznetbj@wiznettechnology.com WIZnet中文主页:http://www.iwiznet.cn WIZnet中文博客:http://blog.iwiznet.cn WIZnet企业博客:http://e.weibo.com/wiznet2012  
  • 热度 19
    2012-7-18 00:15
    7422 次阅读|
    6 个评论
           这是 一个 FPGA 的串口通信上位机软件 , 要求不高,顶多算是个测试用 面板。 我觉得这就是个小 case 。可是实际上这 不到 1 千行的 代码,却把我折腾了好长时间。      这个软件的大概内容是接受 fpga 定时发送的数据,把数据保存在一个文本文档里面就 基本 OK 了。让人蛋疼的是通信的速度,每隔 10ms fpga 就要发送 37 字节的数据到上位机,波特率采用 115200 。 数据格式为 4 位帧头, 32 位数据, 1 位帧尾( 0xa5 )。 按照串口的通信协议来说,至少一个起始位(低电平) 8 个数据位,一个终止位。照这样算, 10ms 的时间,满打满算才能接收 110 字节的。而 PC 机作为串口上位机,为非实时系统,稍微其他什么程序影响一下,延迟了几个毫秒响应中断,很可能就处理不过来数据的。 为了让串口中断不影响程序的运行,就用 CmtScheduleThreadPoolFunction 函数插入了一个子线程,然后再子线程里面添加串口中断回调函数( InstallComCallback (COM, LWRS_RECEIVE, 37, 0, ComCallback, callbackdata);    )。公司里以前做数据采集都是这样插入子线程来做的。令人头痛的是这次居然要在子线程里面插入中断回调函数。退出时居然不能关闭串口,停在 CloseCom 函数了。以前都是中断完了,直接退出时关闭串口,没有问题。看了 CloseCom 的函数说明,确实可以自动取消中断并退出的。后来狂看 labwindows 的帮助文档,发现了用     InstallComCallback (COM, LWRS_RECEIVE, 0, 0, ComCallback, callbackdata);        可以取消回调函数。这样就可以只在退出时调用 CloseCom ,终于可以运行正常了。 这样程序差不多可以了,试着跑了一下。发现有很小的几率发生串口不能接收到数据的情况。而且,一旦发生,程序就再也不能接收到数据,除非重启。我想可能是因为,串口回调函数里面的程序还是太多了,有时候运行不过来,毕竟是非实时系统嘛。然后我把串口中断里面的程序尽量精简,只是接收到数后就保存在一个大数组里面,还有一些必须进行的操作,其他的都不要了。并且把接收到 37 字节产生中断,改成了接收到 185 字节产生中断( InstallComCallback (COM, LWRS_RECEIVE, 185, 0, ComCallback, callbackdata);    )等所有的数据都接收完毕,然后再把数据按照通信协议的格式挑出来,并处理保存在文本文档中。 这样就差不多了(虽然就写了这么点,当时我调了好几天呢)。下面附上部分源代码: #define COM 1 static int panelHandle,pathSelect; static void*callbackdata; void ComCallback(int COMport, int eventMask, void *callbackdata); int CVICALLBACK Lel1Test (void *functionData); void printspace(int inputdat) ; void DataTransfer(void);   static int intWorkStatus=0; int lv1TestID; static int intTestTimes=0; int intTotalTimes; //unsigned char chSaveData ; FILE *fp; static long count = 0; int intPrintData ; //int SavePattern; //0默认路径,1是选择的路径 //char dirSlected ; static int status ;   unsigned char buffer ;    //在main函数里面打开的串口 int main (int argc, char *argv ; char FileName ; char DataPrint ; int j; long i; InstallComCallback (COM, LWRS_RECEIVE, 185, 0, ComCallback, callbackdata); FlushInQ (COM); return 0;   } //串口中断回调函数 void ComCallback(int COMport,int eventMask,void *callbackdata) { int process;  long j; int num; //获取输入队列长度,有可能会在大于185,不信你仔细看帮助文档 num= GetInQLen(COM); //读取输入队列存入缓存中 if(intWorkStatus==1) {   ComRd(COM,buffer ,num); count=count+num;       intTestTimes=intTestTimes+5; //控制进度条显示 process = intTestTimes*100/(intTotalTimes); if(process100) { process=100; } SetCtrlAttribute (panelHandle, PANEL_NUMERICSLIDE, ATTR_CTRL_VAL,process); //获取静/动态     GetCtrlAttribute (panelHandle, PANEL_BINARYSWITCH, ATTR_CTRL_VAL, status ); if(intTestTimes==intTotalTimes+5) { intWorkStatus=3;//工作状态3处于终止状态 intTestTimes=0; //把采集次数置 SetCtrlAttribute (panelHandle, PANEL_TIMER, ATTR_ENABLED, 1); InstallComCallback (COM, LWRS_RECEIVE, 0, 0, ComCallback, callbackdata);  FlushInQ (COM); } }     }  //这个函数太二了,printf函数明明可以控制输出格式的。当时忘了。。 //没办法,在这之前我做单片机编程相对比较多,单片机程序从来不用printf //这个函数的。。。。。。。 void printspace(int inputdata) { if(inputdata10) fprintf(fp,"%s","      "); else if(inputdata100) fprintf(fp,"%s","     "); else if(inputdata1000) fprintf(fp,"%s","    "); else if(inputdata10000) fprintf(fp,"%s","   "); else fprintf(fp,"%s","  ");  } //最后数据处理的部分让我放到了一个定时器的回调函数中 int CVICALLBACK timer (int panel, int control, int event, void *callbackData, int eventData1, int eventData2) { char *time; char *date; char Dir ; char FileName ; char DataPrint ; int SaveDefault; int SelectStatus; int j; long i; int savePattern; unsigned char last, current; int errorNum = 0; switch (event) { case EVENT_TIMER_TICK: SetCtrlAttribute (panelHandle, PANEL_TIMER, ATTR_ENABLED, 0); SetCtrlAttribute (panelHandle, PANEL_CHECKBOX, ATTR_DIMMED, 0); // CloseCom(COM);           SetCtrlAttribute (panelHandle, PANEL_COMMANDBUTTON_PAUSE, ATTR_DIMMED, 1); SetCtrlAttribute (panelHandle, PANEL_COMMANDBUTTON_STOP, ATTR_DIMMED, 1); SetCtrlAttribute (panelHandle, PANEL_NUMERICSLIDE, ATTR_DIMMED, 1); SetCtrlAttribute (panelHandle, PANEL_COMMANDBUTTON_START, ATTR_DIMMED, 0); SetCtrlAttribute (panelHandle, PANEL_COMMANDBUTTON_START, ATTR_LABEL_TEXT,"开始采集"); SetCtrlAttribute (panelHandle, PANEL_NUMERICSLIDE, ATTR_CTRL_VAL,0); date="\0"; time="\0"; date=DateStr(); time=TimeStr(); DataTransfer();   GetCtrlVal (panelHandle, PANEL_CHECKBOX, SaveDefault); if(SaveDefault==0) { SelectStatus=DirSelectPopup("","请选择保存路径",1,1,Dir); if(SelectStatus) { sprintf(FileName,"%s\\DataSaved.txt",Dir); savePattern=1; } else { GetProjectDir(Dir);   sprintf(FileName,"%s\\DataSaved.txt",Dir); savePattern=0; } } else { GetProjectDir(Dir);   sprintf(FileName,"%s\\DataSaved.txt",Dir); savePattern=0; } fp = fopen (FileName, "w"); //写入文件我用的fprintf函数,也可以用sprinf先把这些转换成字符串再用fwrite //函数写入     fprintf(fp,"%s ",time);     fprintf(fp,"%s\n\n",date); fprintf(fp,"%s","编号   "); fprintf(fp,"%s","通道1  "); fprintf(fp,"%s","通道2  "); fprintf(fp,"%s","通道3  "); fprintf(fp,"%s","通道4  "); fprintf(fp,"%s","通道5  "); fprintf(fp,"%s","通道6  "); fprintf(fp,"%s","通道7  "); fprintf(fp,"%s","通道8  "); fprintf(fp,"%s","通道9  "); fprintf(fp,"%s","通道10 "); fprintf(fp,"%s","通道11 "); fprintf(fp,"%s","通道12 "); fprintf(fp,"%s","通道13 "); fprintf(fp,"%s","通道14 "); fprintf(fp,"%s","通道15 "); fprintf(fp,"%s","通道16 "); fprintf(fp,"%s","静/动态 \n"); for(i=0;i       {   fprintf(fp,"%d",i+1);     printspace(i+1); for(j=0;j17;j++) { fprintf(fp,"%d",intPrintData );     printspace(intPrintData ); } fprintf(fp,"%s","\n"); }         fclose (fp); if(savePattern) MessagePopup("数据采集完毕","测试结果已经成功保存至所选目录"); else MessagePopup("数据采集完毕","测试结果已经成功保存至默认目录"); break; } return 0; } //把接收到的数据转换成要保存的数据,2字节数代表一个数据,一共16个数据 void DataTransfer(void) {   long i;   int j;   int k;   int saveNum=0;   i=0;   for(i=0;i   {                       //0xaa,0x55,0x80,0x20是串口通讯的帧头,0xa5是帧尾     if(buffer ==0xaabuffer ==0x55buffer ==0x80   buffer ==0x20buffer ==0xa5)   {   for(j=0;j16;j++)   {   intPrintData =((int)(buffer ))*256+buffer ;   }   intPrintData =status ;   saveNum++;   if(saveNum==intTotalTimes)   break;   }   } } ;i++) ;i++)            
  • 热度 21
    2012-4-6 16:14
    6421 次阅读|
    8 个评论
    第4节  利用Labwindows/CVI软件对摄像头的控制
    利用Labwindows/CVI软件对摄像头的控制 一、概述 首先,先介绍一下windows系统在驱动摄像头的时候都使用到一个链接库,传统的是avicap32.dll和avicap32.lib。这两个文件是比较通用的,而且也是比较早期的文件。现在也有比较新的链接库,例如DirectShowLib.dll。在使用这个链接库来实现视频的捕捉时,主要使用下面这个函数来打开摄像头: int capCreateCaptureWindowA(String lpszWindowName,Long dwStyle,Long x,Long y,Long nWidth,Long nHeight,Long hwndParent,Long nID); 返回值为打开的句柄值。 String lpszWindowName:视频窗口的名字 Long dwStyle:样式 Long x:左边距(相对于主窗口,没有边距值为0) Long y:上边距(相对于主窗口,没有边距值为0) Long nWidth:视频窗口的宽度(注:此项不会改变摄像头的分辨率,超出部分不会被刷新) Long nHeight:视频窗口的高度 Long hwndParent:主窗口句柄,主要是用来显示摄像头采集到得视频。 Long nID:标识符(一般为0) 摄像头打开后,要对摄像头相关信息进行参数设置,比如摄像头的分辨率等。摄像头参数的设置使用到得函数是windows.h声明到的SendMessage。函数的原型为: LRESULT SendMessage(HWND hWnd,UNIT Msg,WPARAM wParam,LPARAM IParam); hWnd:其窗口程序将接收消息的窗口的句柄。如果此参数为HWND_BROADCAST,则消息将被发送到系统中所有顶层窗口,包括无效或不可见的非自身拥有的窗口,被覆盖的窗口和弹出式窗口,但消息不被发送到子窗口。 Msg:指定被发送的消息。 wParam:指定附加的消息指定信息。 IParam:指定附加的消息指定信息。 返回值:返回值指定消息处理的结果,依赖于所发送的消息。 Msg是指定发送的消息,主要有一下这些消息指令:   WM_CAP_START= WM_USER=1024;   // start of unicode messages   WM_CAP_UNICODE_START= WM_USER+100; //开始   WM_CAP_GET_CAPSTREAMPTR= (WM_CAP_START+ 1); //获得 CAPSTR EAMPTR   WM_CAP_SET_CALLBACK_ERROR= (WM_CAP_START+ 2); //设置回调错误   WM_CAP_SET_CALLBACK_STATUS= (WM_CAP_START+ 3); //设置回调状态   WM_CAP_SET_CALLBACK_YIELD= (WM_CAP_START+ 4); //设置回调出产   WM_CAP_SET_CALLBACK_FRAME= (WM_CAP_START+ 5); //设置回调结构   WM_CAP_SET_CALLBACK_VIDEOSTREAM= (WM_CAP_START+ 6); //设置回调视频流   WM_CAP_SET_CALLBACK_WAVESTREAM= (WM_CAP_START+ 7); //设置回调视频波流   WM_CAP_GET_USER_DATA =(WM_CAP_START+ 8); //获得使用者数据   WM_CAP_SET_USER_DATA =(WM_CAP_START+ 9) ; //设置使用者数据   WM_CAP_DRIVER_CONNECT =(WM_CAP_START+ 10); //驱动程序连接   WM_CAP_DRIVER_DISCONNECT =(WM_CAP_START+ 11); //断开启动程序连接   WM_CAP_DRIVER_GET_NAME =(WM_CAP_START+ 12); //获得驱动程序名字   WM_CAP_DRIVER_GET_VERSION =(WM_CAP_START+ 13); //获得驱动程序版本   WM_CAP_DRIVER_GET_CAPS =(WM_CAP_START+ 14); //获得驱动程序性能   WM_CAP_FILE_SET_CAPTURE_FILE =(WM_CAP_START+ 20); //设置捕获文件的文件名   WM_CAP_FILE_GET_CAPTURE_FILE =(WM_CAP_START+ 21); //获得捕获文件的文件名   WM_CAP_FILE_SAVEAS =(WM_CAP_START+ 23); //另存文件为   WM_CAP_FILE_SAVEDIB =(WM_CAP_START+ 25); //保存文件   // out of order to save on ifdefs   WM_CAP_FILE_ALLOCATE =(WM_CAP_START+ 22); //分派文件, 为捕获文件建一个指定大小的文件   WM_CAP_FILE_SET_INFOCHUNK =(WM_CAP_START+ 24); //设置开始文件   WM_CAP_EDIT_COPY =(WM_CAP_START+ 30); //编辑复制,把图象考入剪贴板   WM_CAP_SET_AUDIOFORMAT =(WM_CAP_START+ 35); //设置音频格式   WM_CAP_GET_AUDIOFORMAT =(WM_CAP_START+ 36); //捕获音频格式   WM_CAP_DLG_VIDEOFORMAT =(WM_CAP_START+ 41); //1065 打开视频格式设置对话框, 选择数字视频的框架大小和视频图像的色深,以及捕获视频图像的压缩格式。   WM_CAP_DLG_VIDEOSOURCE =(WM_CAP_START+ 42); //1066 打开属性设置对话框,设置对比度、亮度等。(视频源对话框) 选择视频输入通道和视频图像的动态参数。   WM_CAP_DLG_VIDEODISPLAY =(WM_CAP_START+ 43); //1067 打开视频显示对话框   WM_CAP_GET_VIDEOFORMAT =(WM_CAP_START+ 44); //1068 获得视频格式   WM_CAP_SET_VIDEOFORMAT =(WM_CAP_START+ 45); //1069 设置视频格式   WM_CAP_DLG_VIDEOCOMPRESSION =(WM_CAP_START+ 46); //1070 打开压缩设置对话框   WM_CAP_SET_PREVIEW =(WM_CAP_START+ 50); //设置预览模式   WM_CAP_SET_OVERLAY =(WM_CAP_START+ 51); //设置覆盖   WM_CAP_SET_PREVIEWRATE =(WM_CAP_START+ 52); //设置预览模式下图像的帧速度   WM_CAP_SET_SCALE =(WM_CAP_START+ 53); //设置预览视频的缩放比例   WM_CAP_GET_STATUS =(WM_CAP_START+ 54); //获得状态   WM_CAP_SET_SCROLL =(WM_CAP_START+ 55); //设置卷   WM_CAP_GRAB_FRame =(WM_CAP_START+ 60); //逮捕结构   WM_CAP_GRAB_FRame_NOSTOP =(WM_CAP_START+ 61); //停止逮捕结构,截取当前图象   WM_CAP_SEQUENCE =(WM_CAP_START+ 62); //次序,捕捉到文件   WM_CAP_SEQUENCE_NOFILE =(WM_CAP_START+ 63); //没有文件   WM_CAP_SET_SEQUENCE_SETUP =(WM_CAP_START+ 64); //设置当前捕获的帧频率   WM_CAP_GET_SEQUENCE_SETUP =(WM_CAP_START+ 65); //获得当前捕获的帧频率   WM_CAP_SET_MCI_DEVICE =(WM_CAP_START+ 66); //设置媒体控制接口   WM_CAP_GET_MCI_DEVICE =(WM_CAP_START+ 67); //获得媒体控制接口   WM_CAP_STOP =(WM_CAP_START+ 68); //停止捕捉   WM_CAP_ABORT =(WM_CAP_START+ 69); //异常中断   WM_CAP_SINGLE_FRame_OPEN =(WM_CAP_START+ 70); //打开单一的结构   WM_CAP_SINGLE_FRame_CLOSE =(WM_CAP_START+ 71); //关闭单一的结构   WM_CAP_SINGLE_FRame =(WM_CAP_START+ 72); //单一的结构   WM_CAP_PAL_OPEN =(WM_CAP_START+ 80); //打开视频   WM_CAP_PAL_SAVE =(WM_CAP_START+ 81); //保存视频   WM_CAP_PAL_PASTE =(WM_CAP_START+ 82); //粘贴视频   WM_CAP_PAL_AUTOCREATE =(WM_CAP_START+ 83); //自动创造   WM_CAP_PAL_MANUALCREATE =(WM_CAP_START+ 84); //手动创造   // Following added post VFW 1.1   WM_CAP_SET_CALLBACK_CAPCONTROL =(WM_CAP_START+ 85); // 设置收回的错误   WM_CAP_END =WM_CAP_SET_CALLBACK_CAPCONTROL; 二、编程实例 说明:此实例采用的是labwindows/CVI8.5虚拟软件实现的摄像头采集图像程序。 Project”新建一个工程文件camera.prj,然后添加一个*.uir面板,命名为camera.uir,并添加到工程文件中。然后对camera.uir面板进行编辑。如图: 接着,为这面板上的控件进行编辑。显示面板采用的是CANVAS控件,定义变量命为:CANVAS_DISPLAY,用于后面显示时调用,“开始视频”控件定义函数名:StartCB,“退出视频”控件函数名:stopCB,“保存图像”控件函数名:recordbmp,“录像”控件函数名:recordstream,“停止录像”控件函数名:stopstream,显示模式控件采用的Ring控件,定义函数名:method。最后退出面板控件定义函数QuitCallback。在面板的最左边空白处,添加了三个canvas控件,用于在保存图像时,对新保存的三个图像进行显示。 All code”为此面板自动生成c文件代码。保存c源文件为camera.c。接着为这些函数添加代码: (1)为“开始视频”控件函数startCB添加代码如下: int Display_width; int Display_height; int Display_top; int Display_left; switch (event) { case EVENT_COMMIT: //获得面板句柄,用于在调用摄像头函数时,需要使用的句柄handle GetPanelAttribute (panelHandle, ATTR_SYSTEM_WINDOW_HANDLE, &handle); //获得屏幕中图像要显示的位置 GetCtrlAttribute (panelHandle, PANEL_CANVAS_DISPLAY, ATTR_LEFT, &Display_left); GetCtrlAttribute (panelHandle, PANEL_CANVAS_DISPLAY, ATTR_TOP, &Display_top); GetCtrlAttribute (panelHandle, PANEL_CANVAS_DISPLAY, ATTR_HEIGHT, &Display_height); GetCtrlAttribute (panelHandle, PANEL_CANVAS_DISPLAY, ATTR_WIDTH, &Display_width); //打开摄像头 result = capCreateCaptureWindowA("",WS_CHILD | WS_VISIBLE ,Display_left,Display_top,Display_width,Display_height,handle,0); //设置摄像头函数 SendMessage((HWND)result, WM_CAP_SET_CALLBACK_VIDEOSTREAM, 0, 0); SendMessage((HWND)result, WM_CAP_SET_CALLBACK_ERROR, 0, 0); SendMessage((HWND)result, WM_CAP_SET_CALLBACK_STATUSA, 0, 0); SendMessage((HWND)result, WM_CAP_DRIVER_CONNECT, 0, 0); SendMessage((HWND)result, WM_CAP_SET_SCALE, 1, 0); SendMessage((HWND)result, WM_CAP_SET_PREVIEWRATE, 66, 0); SendMessage((HWND)result, WM_CAP_SET_OVERLAY, 1, 0); SendMessage((HWND)result, WM_CAP_SET_PREVIEW, 1, 0); break; } return 0; (2)为“停止视频”控件函数stopCB添加源代码如下: if (result != 0) //如果有摄像头开启 { //发送一个断开连接命令 SendMessage((HWND)result, WM_CAP_DRIVER_DISCONNECT, 0, 0); } (3)为“保存图像”控件函数recordbmp添加源代码如下: //定义BMP位图ID号 int bitmapID; //定义文件存储路径 char path ; //定义静态变量,设置循环次数 static int i; i = 0; switch (event) { case EVENT_COMMIT: //保存位图 0) { SendMessage((HWND)result,WM_CAP_SAVEDIB,0,(LPARAM)path); //获得BMP位图ID GetBitmapFromFile (path, &bitmapID); switch (i) { //设置三个位图浏览器 case 0: CanvasDrawBitmap (panelHandle, PANEL_CANVAS_SNAP0, bitmapID, VAL_ENTIRE_OBJECT, VAL_ENTIRE_OBJECT); break; case 1: CanvasDrawBitmap (panelHandle, PANEL_CANVAS_SNAP1, bitmapID, VAL_ENTIRE_OBJECT, VAL_ENTIRE_OBJECT); break; case 2: CanvasDrawBitmap (panelHandle, PANEL_CANVAS_SNAP2, bitmapID, VAL_ENTIRE_OBJECT, VAL_ENTIRE_OBJECT); break; } //删除BMP位图ID DiscardBitmap (bitmapID); //循环累加,共三次,三次之后回零 i++; if (i == 3) { i = 0; } } break; } return 0; (4)为“开始录像”控件函数recordstream添加源代码如下: char path ; switch (event) { case EVENT_COMMIT: //保存录像 0) { SendMessage((HWND)result,WM_CAP_FILE_SET_CAPTURE_FILEA,0,(LPARAM)path); SendMessage((HWND)result, WM_CAP_SEQUENCE, 0, 0); } break; } return 0; (5)为“停止录像”控件函数stopstream添加源代码如下: switch (event) { case EVENT_COMMIT: //发送WM_CAP_STOP停止命令 SendMessage((HWND)result, WM_CAP_STOP, 0, 0); break; } return 0; } (6)为“显示模式”控件添加函数method代码如下: //获得制式种类 GetCtrlVal (panelHandle, PANEL_RING, &kind); switch (kind) { case 0: //160*120制式 SetCtrlAttribute (panelHandle, PANEL_CANVAS_DISPLAY, ATTR_WIDTH, 160); SetCtrlAttribute (panelHandle, PANEL_CANVAS_DISPLAY, ATTR_HEIGHT, 120); break; case 1: //1176*144制式 SetCtrlAttribute (panelHandle, PANEL_CANVAS_DISPLAY, ATTR_WIDTH, 176); SetCtrlAttribute (panelHandle, PANEL_CANVAS_DISPLAY, ATTR_HEIGHT, 144); break; case 2: //320*240制式 SetCtrlAttribute (panelHandle, PANEL_CANVAS_DISPLAY, ATTR_WIDTH, 320); SetCtrlAttribute (panelHandle, PANEL_CANVAS_DISPLAY, ATTR_HEIGHT, 240); break; case 3: //352*288制式 SetCtrlAttribute (panelHandle, PANEL_CANVAS_DISPLAY, ATTR_WIDTH, 352); SetCtrlAttribute (panelHandle, PANEL_CANVAS_DISPLAY, ATTR_HEIGHT, 288); break; case 4: //1默认制式 SetCtrlAttribute (panelHandle, PANEL_CANVAS_DISPLAY, ATTR_WIDTH, 592); SetCtrlAttribute (panelHandle, PANEL_CANVAS_DISPLAY, ATTR_HEIGHT, 333); break; } //获得屏幕中图像要显示的位置 GetCtrlAttribute (panelHandle, PANEL_CANVAS_DISPLAY, ATTR_LEFT, &left); GetCtrlAttribute (panelHandle, PANEL_CANVAS_DISPLAY, ATTR_TOP, &top); GetCtrlAttribute (panelHandle, PANEL_CANVAS_DISPLAY, ATTR_HEIGHT, &height); GetCtrlAttribute (panelHandle, PANEL_CANVAS_DISPLAY, ATTR_WIDTH, &width); //重新定位 SetCtrlAttribute (panelHandle, PANEL_CANVAS_DISPLAY, ATTR_TOP, (333-height)/2+55); SetCtrlAttribute (panelHandle, PANEL_CANVAS_DISPLAY, ATTR_LEFT, (592-width)/2+53); //断开连接 SendMessage((HWND)result, WM_CAP_DRIVER_DISCONNECT, 0, 0); //调用开始按钮事件 startCB(panel, PANEL_CANVAS_DISPLAY, EVENT_COMMIT, NULL, 0, 0); 三、总结 采用Labwindows/CVI软件实现了摄像头的视频和录像功能,而且可以选择不同的显示分辨率,设计面板较为简单且美观,控件使用方面快捷!主要使用到得函数来自于动态链接库avicap32.dll中的摄像头开启函数capCreateCaptureWindowA,然后通过windows.h声明的SendMessage函数来发送摄像头的显示相关的命令来设置摄像头。