C traps and Puzzles
下面的例子都在Ubuntu8.04 GCC下编译的结果,有些没有给结果
编程一定要自己动手试一试!
1.定义与声明,定义要分配内存,声明只是声明在别处定义了
int a; //定义
extern int a; //声明
char str[100]
extern char str[] //ok
char * str[]
extern char str[] //error
char str[100]
extern char * str //error
数组和指针是两码事!只是在某些情况下为了简化编译都当做指针处理
2.register 不能用取址运算符,因为可能不存放在内存中.
#include <stdio.h>
int main(void)
{
register int r=10;
int * ptr=&r;
printf("%d\n",*ptr);
return 0;
}
3.sizeof是关键字不是函数也不是宏
(define 不是关键字),sizeof在编译的时候就确定了,由编译器算出
int i=0;
sizeof(i) //4 ok
sizeof(int) //4 ok
sizeof i //4 ok
sizeof int //error!
int *p=NULL;
sizeof(p) //ok 4
sizeof(*p) //ok 4!
int a[100];
sizeof(a) //100
sizeof(&a) //4
sizeof(&a[0]) //4
enum{A,B,C}e;
sizeof(e) //4
4.void 指针
ANSI规定以不能对void 指针进行算法操作
void * vPtr;
vPtr++; //error
vPtr+=1; //error
但在GNU中
用GCC编译都是合法的!
用G++编译cpp文件确是不合法的!
VC没试过不知道。
5.volatile关键字
告诉编译器每次使用变量值的时候都是从内存中读出,而不进行任何优化,常用在多线程编程中。
6.空结构体空类
struct st{}st;
class cl{}cl;
g++编译结果sizeof(st)=1 sizeof(cl)=1;
gcc sizeof(st)=0!
7.fleible array
C99中,结构体的最后一个元素允许是大小未知的数组,但这个数组前面必须有一个其他类型的成员
struct F{int i; char a[0];} sizeof(F)=4
struct F1{int i;char a[];} sizeof(F1)=4
F1 * f1;
f1=(F1 *)malloc(sizeof(struct F1)+100*sizeof(char));/*给f1分配4+100byte的空间,这样数组a就有100个元素了*/
sizeof(f1)=4 //!还是4,说明sizeof是在编译的时候就确定了!
8.编写程序测试big/small endian
bool smallEndian(void)
{
union endian{int i;char ch;}
相关文档:
int main()
{
printf(&unix["\021%six\012\0"], (unix)["have"] + "fun" - 0x60);
}
gcc -S编译成汇编代码如下:
.file "test.c"
.section .rodata
.LC0:
&nbs ......
突然发现自己连一元二次方程怎么算的都不知道了。想了半天,拿起笔来才顺手些了给x2+2x+1=0.悔恨啊。
#include "iostream"
#include "cmath"
using namespace std;
int main(){
double a,b,c;
double delta,x1,x2;
int sign,stop;
cout<<"输入3个系数a(a!=0),b,c"<<endl;
cin>>a>>b& ......
1.介绍一下STL,详细说明STL如何实现vector。
Answer:
STL (标准模版库,Standard Template Library)它由容器算法迭代器组成。
STL有以下的一些优点:
可以方便容易地实现搜索数据或对数据排序等一系列的算法;
&nb ......
转帖:一辉的文章
在Linux下做开发难免要接触makefile,整个项目的构建都依赖于它。100个developer有100种makefile的写法,在一个较大的项目中,各种各样的makefile无论在开发、后期维护还是整个系统的持续集成都是一个负担。
有幸参与重构一个遗留系统的makefile,以下是一些心得和一个makefile模板。
重构目的:
1.清 ......
CreateThread函数
前面讲过,进程的主线程会在CreateProcess调用时自动创建。假如你要手工创建线程,你可以调用CreateThread函数:
HANDLE CreateThread(
PSECURITY_ATTRIBUTES psa,
DWORD cbStackSize,
PTHREAD_START_ROUTINE pfnStartAddr,
PVOID pvParam,
DWORD dwCreateFlags,
PDWORD pdwThreadID) ......