欢迎访问 生活随笔!

生活随笔

当前位置: 首页 >

自己在CODING过程中遇到的问题以及解决(C/VC)

发布时间:2025/3/21 22 豆豆
生活随笔 收集整理的这篇文章主要介绍了 自己在CODING过程中遇到的问题以及解决(C/VC) 小编觉得挺不错的,现在分享给大家,帮大家做个参考.

自己对于写程序还处于菜鸟阶段,很多用法尚未涉及或者还不清晰。总是想着等每个程序写完再来总结,却发现早已无从下手。于是想到了这个方法,每次遇到问题并解决之后第一时间到这个日志上记录下来。于是有了此文。

                                                                                                                                                                                                ivandark   处于边缘的菜的不能再菜的程序猴子一年半

                                                                                                                                                                                                                                       ——2012.11.10

1.声明空间并初始化内存

声明空间: 一维:int *a;   a = (int*)malloc(sizeof(int)*size);  //size为数组大小

                    二维:int **a;  a=(int**)malloc(sizeof(int*)*row);  //

                                 for(int i=0;i<col;i++)

                                 {

                                       a[i] = (int*)malloc(sizeof(int)*col);

                                  }

初始化内存:因为不初始化,动态声明的数组赋值不正常

                       一维和二维的初始化是不一样的。

                       一维:memset(a,0,sizeof(a));

                      二维:边分配内存边初始化:                          

                               int **a;  a=(int**)malloc(sizeof(int*)*row);  //

                                 for(int i=0;i<col;i++)

                                 {

                                       a[i] = (int*)malloc(sizeof(int)*col);

                                       memset(a[i],0,sizeof(int)*col); // 貌似只能这么写 不可以效仿一维的格式

                                  }

删除:  free(a); //一维和二维的一样

             如果你像上面对a分配刚好的够用的内存并对其赋值运算,再释放会出现错误after normal block(#207) at  。

          经过查找发现不少人曾经出现过此问题。

          原因描述(引用网络):这是典型的内存溢出错误,常在内存的delete处发生,而且一般在debug版本中可能出现,release版本中可能并不报错.出现这个错误的原因一般都是malloc/new申请的内存溢出.因为在c++中,如果用new分配一段内存,操作的时候改变了该部分的大小,在delete时就会出错.

因为申请了一个size为5的内存,但是strcpy过去了一个size为6的字符串,因此破坏了这个指针,运行debug版本的时候就 会出现先前的错误,但是在release版本中,溢出一个字节的内存很有可能是没有错误的,然后潜在的隐患是肯定存在的,因此,我们在debug遇到这样 的错误时候一定要仔细检查对new/malloc出的指针的操作. 

            解决方案:我是为a数组多分配了一个(int**)内存 程序运行正常

2.C语言读取txt文本

txt格式是我们利用编程语言处理数据时,数据常用的存储格式。

下面介绍常用的用于读写的函数:

打开:FILE *fp;     if((fp=fopen("E:\\test.txt","rt"))==NULL)   //" "里面的是文件路径  "rt"表示读取方式     {printf("cannot open file\n");return;     }

关闭:fclose(fp);

读取:这里分为很多函数:

          fscanf();这个是个人感觉比较好用的。从一个流中执行格式化输入,fscanf遇到空格和换行时结束,注意空格时也结束。这与fgets有区别,fgets遇到空格不结束。比如我们想把fp中的数据存到二维数组a中 ("%f"表示以浮点形式读取)

               for(i=0;i<M;i++){for(j=0;j<N;j++)fscanf(fp,"%f",&a[i][j]);}

             fread();     fread( void *buffer, size_t size, size_tcount, FILE *stream);根据msdn中的说明不难使用,这里不赘述。

写入:

           fwrite();  :fwrite(buffer,size,count,fp);  (1)buffer:是一个指针,对fwrite来说,是要输出数据的地址。  (2)size:要写入的字节数;  (3)count:要进行写入size字节的数据项的个数;  (4)fp:目标文件指针。

3.VC读取txt文本

对话框形式打开:

void COpenDataDlg::OnBtnOpendata() {// TODO: Add your control notification handler code herestatic char BASED_CODE file[] = "TXT Files (*.TXT)|*.txt|所有文件(*.*)|*.*||"; //文件类型说明字符串CFileDialog SelectFile (TRUE,NULL,NULL,OFN_HIDEREADONLY|OFN_OVERWRITEPROMPT,file,NULL); //文件对话框初始化(点击浏览时候的对话框) 第一个TRUE表示以打开方式;false是以保存的方式;file表示打开的是什么文件其他参数 显示对话框 隐藏只读,SelectFile.DoModal(); //弹出对话框CString FileName; FileName=SelectFile.GetPathName();//得到所选文件路径m_FileName = FileName;UpdateData(FALSE);//把后台数据刷新到前台显示} 读取:跟C语言相同。好像可以用vc中的方法 CFile什么的,但是暂时没搞懂,先放着。
输出:跟C相同。
char str[50]; //数组大小可以根据输出的内容大小而定,这里为了简便声明了静态数组CFile file(_T("E:\\result.txt"), CFile::modeCreate | CFile::modeWrite);for(i=0; i<8; i++) //8是要输出的字符的个数 可以根据需要变化 或者先求一个长度再用长度变量替换8{sprintf(str, "%d,%d\r\n",i,showID[i]); //showID里存储的是要写入数据file.Write(str, strlen(str) * sizeof(char));}file.Close();MessageBox("结果已输出到E:\\result.txt");

4.opencv取像素灰度宏

(CV_IMAGE_ELEM(pImg1,uchar,i,j) //注意i j i代表y轴方向 j代表x轴方向 别弄反了

5.程序在调试态下正常,执行时候mfc基础类停止运行

DAMAGE:After normal block(#****)

这种问题通常是由于动态分配内存导致的。可以检查一下以下三种情况:

(1)数组访问越界 int a[5]; a[5] = 3; //越界了
(2)动态分配数组越界 int *a; int lenth1 = 2; int lenth2 = 3; a = (int*)malloc(sizeof(int)*lenth1); memset(a,0,sizeof()*lenth2); free(a);这里初始化的数组长度比定义的要长,出错 (3)动态声明的空间没有被释放 所有动态声明的数组在一个函数结束之后一定一定要free掉! 但是在子函数中好像是无所谓的。只要在功能的主函数中释放即可。


6.最近用过的Opencv 函数

(1)CvMat 创建并初始化矩阵


CvMat Ma;cvInitMatHeader(&Ma,row,col,CV_64FC1,FC,CV_AUTOSTEP);//CV_64FC1 表示64位浮点数 可以根据需要换成其他

(2)cvEigenVV 

见另一篇转载文章   http://blog.csdn.net/ivandark/article/details/8228029

(3)cvGetSize 得到矩阵尺寸 返回CvSize结构体

       例如: CvSize size;

                   size = cvGetSize(IplImage* img);  //返回图像长宽


(4) cvSaveImage

  保存图像到文件

  int cvSaveImage( const char* filename, const CvArr* image );  //filename 直接给路径名,用字符串  指针用图像的指针 比如IplImage   filename   文件名。   image   要保存的图像。   函数cvSaveImage保存图像到指定文件。 图像格式的的选择依赖于filename的扩展名,请参考cvLoadImage。只有8位单通道或者3通道(通道顺序为'BGR' )可以使用这个函数保存。如果格式,深度或者通道不符合要求,请先用cvCvtScale 和cvCvtColor转换;或者使用通用的cvSave保存图像为XML或者YAML格式。

  保存图像到文件   int cvSaveImage( const char* filename, const CvArr* image );   filename   文件名。   image   要保存的图像。   函数cvSaveImage保存图像到指定文件。图像格式的的选择依赖于filename的扩展名,请参考cvLoadImage。只有8位单通道或者3通道(通道顺序为'BGR' )可以使用这个函数保存。如果格式,深度或者通道不符合要求,请先用cvCvtScale 和cvCvtColor转换;或者使用通用的cvSave保存图像为XML或者YAML格式。
(5)初始化 IplImage* 明确  IplImage* 中的灰度数据存在imageData里 IplImage* KL_Img1;CvSize size = cvGetSize(pImg1);KL_Img1 = cvCreateImage(size, 8, 1); // 创建8位单通道图像memset(KL_Img1->imageData, 0.0,sizeof(KL_Img1->imageData));// 将内存拷贝到图像中的imageData中

7.如何重绘 所谓重绘,就是讲程序中变量保存的数据源源不断的刷新到界面上。实现过程大概可以分为如下几步: (1)以绘制点为例,先在VIew里设置一个变量IsDraw,初始值为False。 (2)我们需要把在函数func1中绘制的点保存在一个数组里,我是在Doc类下新建了一个数组空间。 (3)当自己写的绘制函数执行完毕后,所有要重绘的点数据也全部存到数组中了,这里把IsDraw赋值为TRUE (4)然后在OnDraw函数中开始重绘,重绘内容就是func1中画的东西,只不过这次我们要从Doc里的数组调取数据而已。因为如果不执行func1,Ondraw是没的可画的,所以在重绘前还要加上if判断,如果IsDraw==TRUE,则执行重绘。
8.C语言取整问题 (1)向下取整 直接强制转换。如int i = 2.5  , i=2 (2)向上取整 函数ceil 。 如 int i = ceil(2.5) ,i=3   9.C语言txt文件读取操作 (1)打开     FILE *fp;    fp = fopen("E:\\.txt","r");   if(!fp)//容错   {         printf("ERROR");   return 0;   } (2)读取——非循环 只读一个数值   fscanf(fp,"%d",&N); (3)读取——循环 读一组数存入数组 for(i=0;i<N;i++) {       fscanf(fp."%d",&a[i]);   //读取浮点数一定要用 %lf 不要用%f 原因:http://blog.csdn.net/lutx/article/details/5072043 }      (4)写文件    mkdir("E:\\STU");                        //在硬盘新建文件夹和文件!!!
 fp2 = fopen("E:\\STU\\result.txt","w");
for(i=0;i<N;i++) {       fprintf(fp,"%d\t",result[i]);         if((i+1)%10 == 0)       {             fprintf(fp,"\n");       } } (5)文件指针问题      fgetc(fp); //可以使当前文件指针向后移动一位。在跳过空格 回车 很有用       while(!feof(fp))//读到文件末就停止 否则继续读   (6)fseek介绍 <整理并摘抄自百度百科>   A.函数原型:int fseek(FILE *stream, long offset, int fromwhere); //负责文件指针操纵:如果执行成功则返回0(文件stream的内部指针以参数3为基准,偏移参数2的量),失败返回-1   B.参数3可以有: (偏移起始位置:起始位置0(SEEK_SET),当前位置1(SEEK_CUR),文件尾2(SEEK_END))   C.主要作用:移动文件内部指针的位置,可以用来计算文件长度   例子: int main() {FILE *fp;fp = fopen("E:\\a.txt","r");fseek(fp,0L,SEEK_END);//移动文件指针到末尾int len = ftell(fp1);//顺便求个长度呗 经过试验 txt中 空格算一个长度 回车算两个//ftell用于得到文件位置指针当前位置相对于文件首的偏移字节数fseek(fp,0L,SEEK_SET);//把文件指针再移动到开始fseek(fp,100L,SEEK_SET); //再把文件指针移动到离文件开头处100字节fseek(fp,100L,SEEK_CUR); //再把文件指针移动到离当前位置处100字节fseek(fp,100L,SEEK_END); //再把文件指针移动到离文件结尾处100字节fseek(fp,-100L,SEEK_SET);//再把文件指针"退回"到离文件结尾处100字节return 0; }
 (7)读取字符串   A.fgetc函数:从文件中读取一个字符,读取一个字节后,光标位置后移一个字节。   例子: while(!feof(fp1)) //从文件中读取字符串 并计算字母字符的个数{ch = fgetc(fp1); //一个一个字符的读if(ch==EOF)//文件末端{printf("%d",count);printf("\n");}if(isalpha((int)ch))//判断ch是否为字母 字母的ASCII码:ASCII码:65~90号为26个大写英文字母,97~122号为26个小写英文字母!!!{str[count] = ch; //存入字符串中count++;}}
  B.fputc函数————将字符ch写到文件指针fp所指向的文件的当前写指针的位置。 例子: char msg[] = "Hello world"; int i = 0; //方法一while (msg[i]&&(i<strlen(msg))) { fputc(msg[i], fp1); i++; } //方法二fprintf(fp1,"%s",msg);
C.ctype.h —— 处理字符常用的头文件 用例: //isalpha:检查ch是否是字母 是字母则返回非0 否则返回0if(isalpha((int)ch))...//判断ch是否为字母 char ch//isdigit:检查ch是否为数字0-9 是则返回非0 否则返回0//islower:检查是否为小写字母//isupper:检查是否为大写字母//isalnum:检查是否是字母或数字//ispunct:是否是标点字符//isspace:是否是空格符/跳格符/控制字符/换行符ch = tolower((int)ch);//大写字母转成小写字母 ; toupper小写转大写
D.fgets —— 读取字符串 char *fgets(char *s, int n, FILE *stream); fgets函数的调用形式如下:fgets(str,n,fp); 此处,fp是文件指针;str是存放在字符串的起始地址;n是一个int类型变量。 函数的功能是从fp所指文件中读入n-1个字符放入str为起始地址的空间内; 如果在未读满n-1个字符之时,已读到一个换行符或一个EOF(文件结束标志),则结束本次读操作,读入的字符串中最后包含读到的换行符。 因此,确切地说,调用fgets函数时,最多只能读入n-1个字符。读入结束后,系统将自动在最后加'\0',并以str作为函数值返回。   通俗的讲: 比如有文件 Hello world! Hello Hello world! 代码为: int main() {FILE *fp1;fp1 = fopen("E:\\test.txt","r");char msg1[20];char msg2[20];fgets(msg1,4,fp1); //m=4fgets(msg2,7,fp1); //n=7printf("%s\n",msg1);printf("%s\n",msg2);return 0;}
情况一:(m=4 n=7) 输出: Hel lo wor 情况二:(m=4,n=20)如果使用fgets() 读取文件的时候bufsize大于该行的字符总数加2(多出来的两个,一个保存文件本身的'\n'换行,一个保存字符串本身的结束标识'\0'),文件并不会继续读下去,仅仅只是这一行读取完,随后指向文件的指针会自动偏移至下一行。 Hel lo world! 情况三:(m=15,n=4) Hello world! Hel 情况四:(m=15,n=20) Hello world!   Hello Hello world! (8)按行读取数据<待续>      10.C语言读取raw     //开始读
 unsigned char *tmp = (unsigned char*)malloc(sizeof(unsigned char)*256*256);
 memset(tmp,0,sizeof(tmp));
 fread(tmp,sizeof(unsigned char),256*256,fp1); //格式转换后再处理 int* data = (int*)malloc(sizeof(int)*256*256);
 memset(data,0,sizeof(data));
 for(i=0;i<=255*255;i++)
 {
  data[i] = (int)tmp[i];
 } //开始写 fwrite(tmp,sizeof(unsigned char),256*256,fp2);  //tmp 为 unsigned char
 

总结

以上是生活随笔为你收集整理的自己在CODING过程中遇到的问题以及解决(C/VC)的全部内容,希望文章能够帮你解决所遇到的问题。

如果觉得生活随笔网站内容还不错,欢迎将生活随笔推荐给好友。