原创作品,转载请标明
http://blog.csdn.net/yming0221/article/details/7247075
更多查看
1、scanf()输入注意
当你使用scanf(“%d”,&a),然后用gets()读取下一行的一个字符串时,调用后好像gets()函数没有执行。
原因:这是由于scanf()函数不处理回车换行符。这样以来gets()函数读取的只是回车换行符。所以好像时gets()函数没有执行。
解决方法:可以在scanf()函数后插入一个getchar()函数来吃掉那个回车换行符。
2、scanf()的缓冲区问题
为什么说scanf()函数有问题呢?就因为它适用于结构化,格式相对整齐的数据输入。
例如下面的例子
#include <stdio.h>
int main(int argc,char **argv)
{
char a;
while(1)
{
scanf("%c",&a);
printf("a=%c\n",a);
}
return 0;
}
当输入连续asdfg时运行结果如下:
这里可以看出scanf()读取数据到缓冲区,当缓冲区不空时从缓冲区截取,当缓冲区为空时,再读一组数据到缓冲区。所以为保证安全
*可以在每次scanf()后加一个清空缓冲区的语句fflush(stdin),但是这不是所有的编译器都支持的,VC6.0支持,但是GCC不支持。
*使用scanf()的替代功能函数。可以先用fgets()读入一整行,然后用字符串处理函数进行字符串处理(strtok(),strtol(),atoi())
3、printf()和sprintf()函数
就像scanf()函数一样,这两个函数也存在缓冲区的问题,例如输出一个字符串,当你不可预测字符串的长度时,可能存在字符串的过长导致缓冲区溢出,从而影响其他内存区。
如下例子:
int main(int argc,char **argv)
{
char a[11]={0};
const char *p="12345678901234";
sprintf(a,"%s",p);
printf("a= %s\n",a);
return 0;
}
当p的长度大于系统给a分配的内存区,这样就会导致系统错误。
*当我们知道字符串的结构,我们可以使用sprintf(a,”%10s”,p);防止超越内存区。
*使用snprintf()函数原型
int snprintf(char *restrict buf, size_t n, const char * restrict
format, …);
例如
int main(int argc,char **argv)
{
char a[11]={0};
const char *p="12345678901234";
snprintf(a,sizeof(a),"%s",p);
printf("a= %s\n",a);
return 0;
}
4、GCC中qsort()函数使用方法
使用方法
*对int,char,double,float等数组排序方法
#include <stdlib.h>
#define MY_TYPE int
static int cmp(const void *a,const void *b)
{
return *(MY_TYPE *)a > *(MY_TYPE *)b ? 1 : -1;
}
int va[10]={6,3,2,4,7,1,9,8,5,0};
int main(int argc,char **argv)
{
int i;
for(i=0;i<10;i++)
{
printf("%d ",va[i]);
}
printf("\n");
qsort(va,10,sizeof(va[0]),cmp);
for(i=0;i<10;i++)
{
printf("%d ",va[i]);
}
printf("\n");
return 0;
}
*对结构体按一个关键字排序
#include <stdlib.h>
#define MY_TYPE _st
typedef struct _st
{
int va;
char c;
};
struct _st st[10];
static int cmp(const void *a,const void *b)
{
return (*(struct MY_TYPE *)a).va > (*(struct MY_TYPE *)b).va ? 1 : -1;
}
int main(int argc,char **argv)
{
int i;
for(i=0;i<10;i++)
{
st[i].va=10-i;
}
printf("\n");
qsort(st,10,sizeof(st[0]),cmp);
for(i=0;i<10;i++)
{
printf("%d ",st[i].va);
}
printf("\n");
return 0;
}
比较函数也可以写成如下形式
static int cmp(const void *a,const void *b)
{
return ((struct MY_TYPE *)a)->va > ((struct MY_TYPE *)b)->va ? 1 : -1;
}
*对结构体多级排序
int cmp( const void *a , const void *b )
{
struct In *c = (In *)a;
struct In *d = (In *)b;
if(c->x != d->x) return c->x - d->x;
else return d->y - c->y;
}
*对字符串数组排序
#include <stdlib.h>
#define LEN 30+1
char str[4][LEN];
static int cmp(const void *a,const void *b)
{
return strcmp((char *)a,(char *)b);
}
int main(int argc,char **argv)
{
int i;
strcpy(str[0],"abaa");
strcpy(str[1],"baaa");
strcpy(str[2],"baaaa");
strcpy(str[3],"aasasasaa");
printf("\n");
qsort(str,4,sizeof(str[0]),cmp);
for(i=0;i<4;i++)
{
printf("%s\n",str[i]);
}
printf("\n");
return 0;
}
5、程序中条件语句中的double值的相等判断
double a,b;
…..
if( a == b )
这样不对,应该使用
#include <math.h>
......
if (fabs(a - b)<= epsilon * fabs(a))
6、C语言中取整的方法
我们知道float或double化为整型可以使用
(int)(a+0.5);但是这样对负数不合适。
应该使用这样的方法
(int)(x<0 ? x – 0.5 : x + 0.5);
7、C中数学方法获得π 的值
可以自己定义也可以使用数学函数计算
4*atan(1.0) 或 acos(-1.0)
8、如何定义参数可变的函数
#include <stdio.h>
#include <stdlib.h>
#include <stdarg.h>
void __print(const char *fmt,...)
{
va_list argp;
printf("INFO: ");
va_start(argp,fmt);
vprintf(fmt,argp);
va_end(argp);
printf("\n");
}
int main(int argc,char **argv)
{
__print("%s","asddasda");
return 0;
}
注意参数个数至少是一个。
9、为什么有的人判断语句的写法如下
if(0 == x)
而不是if( x==0 )
当然这两个都没有错误。这是由于防止”==”写成”=”。
当把变量放在前面,“==”写成“=”编译器就会报错,不至于引发不应该出现的BUG
10、如何实现控制台的“转动的小棒”来显示程序的执行进度。
这里需要使用fflush(stdout);来清空输出缓冲区。
void print1(int ca)
{
switch(ca)
{
case 0:printf("\\");break;
case 1:printf("/");break;
case 2:printf("-");break;
}
}
int main(int argc,char **argv)
{
__print("%s","asddasda");
int i;
for(i=0;i<10;i++)
{
printf("\r");
print1(i%3);
fflush(stdout);
sleep(1);
}
return 0;
}