C/C++中修饰符const、extern、stat…

本文深入解析C++中const和static的用法,包括它们在数组、指针、函数参数、返回值及成员函数中的应用,以及与全局作用域、名字空间作用域的关系。同时介绍了extern的使用场景,以及volatile类型修饰符的作用。
来自:网易博客 http://afericazebra.blog.163.com/blog/static/30050408200801494053866/
1.const的用法:
为什么使用const?
采用符号常量写出的代码更容易维护;
指针常常是边读边移动,而不是边写边移动;
许多函数参数是只读不写的。
const最常见用途是作为数组的界和switch分情况标号(也可以用枚举符代替)
用法1:常量
取代了C中的宏定义,声明时必须进行初始化。const限制了常量的使用方式,并没有描述常量应该如何分配。如果编译器知道了某const的所有使用,它甚至可以不为该const分配空间。最简单的常见情况就是常量的值在编译时已知,而且不需要分配存储。―《C++ Program Language》
用const声明的变量虽然增加了分配空间,但是可以保证类型安全。
C标准中,const定义的常量是全局的,C++中视声明位置而定。
用法2:指针和常量
使用指针时涉及到两个对象:该指针本身和被它所指的对象。将一个指针的声明用const“预先固定”将使那个对象而不是使这个指针成为常量。要将指针本身而不是被指对象声明为常量,必须使用声明运算符*const。
所以出现在 * 之前的const是作为基础类型的一部分:
char *const cp;   // 到char的const指针
char const *pc1;   // 到const char的指针
const char *pc2; // 到const char的指针(后两个声明是等同的)
从右向左读的记忆方式:
cp is a const pointer to char.
pc2 is a pointer to const char.
用法3:const修饰函数传入参数
将函数传入参数声明为const,以指明使用这种参数仅仅是为了效率的原因,而不是想让调用函数能够修改对象的值。同理,将指针参数声明为const,函数将不修改由这个参数所指的对象。
通常修饰指针参数和引用参数:
void Fun( const A *in); // 修饰指针型传入参数
void Fun(const A &in);   // 修饰引用型传入参数
用法4:修饰函数返回值
可以阻止用户修改返回值。返回值也要相应的付给一个常量或常指针。
用法5:const修饰成员函数
const对象只能访问const成员函数,而非const对象可以访问任意的成员函数,包括const成员函数;
const对象的成员是不能修改的,而通过指针维护的对象却是可以修改的;
const成员函数不可以修改对象的数据,不管对象是否具有const性质。编译时以是否修改成员数据为依据进行检查。

2.static的用法:
静态变量作用范围在一个文件内,程序开始时分配空间,结束时释放空间,默认初始化为0,使用时可以改变其值。
静态变量或静态函数只有本文件内的代码才能访问它,它的名字在其它文件中不可见。
用法1:函数内部声明的static变量,可作为对象间的一种通信机制
如果一局部变量被声明为static,那么将只有唯一的一个静态分配的对象,它被用于在该函数的所有调用中表示这个变量。这个对象将只在执行线程第一次到达它的定义时初始化。
用法2:局部静态对象
对于局部静态对象,构造函数是在控制线程第一次通过该对象的定义时调用。在程序结束时,局部静态对象的析构函数将按照他们被构造的相反顺序逐一调用,没有规定确切时间。
用法3:静态成员和静态成员函数
如果一个变量是类的一部分,但却不是该类的各个对象的一部分,它就被成为是一个static静态成员。一个static成员只有唯一的一份副本,而不像常规的非static成员那样在每个对象里各有一份副本。同理,一个需要访问类成员,而不需要针对特定对象去调用的函数,也被称为一个static成员函数。
类的静态成员函数只能访问类的静态成员(变量或函数)。
3.extern的用法
extern可以声明其他文件内定义的变量。在一个程序里,一个对象只能定义一次,它可以有多个声明,但类型必须完全一样。如果定义在全局作用域或者名字空间作用域里某一个变量没有初始化,它会被按照默认方式初始化。
将变量或函数声明成外部链接,即该变量或函数名在其它函数中可见。被其修饰的变量(外部变量)是静态分配空间的,即程序开始时分配,结束时释放。
在C++中,还可以指定使用另一语言链接,需要与特定的转换符一起使用。
extern “C” 声明语句
extern “C” { 声明语句块 }
extern 表明该变量在别的地方已经定义过了,在这里要使用那个变量。
static 表示静态的变量,分配内存的时候,存储在静态区,不存储在栈上面。static 作用范围是内部连接的关系,和extern有点相反,它和对象本身是分开存储的,extern也是分开存储的,但是extern可以被其他的对象用extern 引用,而static 不可以,只允许对象本身用它。
4.volatile的用法
类型修正符(type-modifier),限定一个对象可被外部进程(操作系统、硬件或并发进程等)改变。volatile与变量连用,可以让变量被不同的线程访问和修改。
声明时语法:
int volatile vInt;
常用于像中断处理程序之类的异步进程进行内存单元访问。
除了基本类型外,对用户定义类型也可以用volatile类型进行修饰。
注意:可以把一个非volatile int赋给volatile int,但是不能把非volatile对象赋给一个volatile对象。
一个有volatile标识符的类只能访问它接口的子集,一个由类的实现者控制的子集。用户只能用const_cast来获得对类型接口的完全访问。此外,volatile向const一样会从类传递到它的成员。
void handle_request(int client_fd) { char buffer[BUFFER_SIZE]; ssize_t bytes_read = read(client_fd, buffer, BUFFER_SIZE - 1); if (bytes_read <= 0) { perror("read failed or connection closed"); return; } buffer[bytes_read] = '\0'; // 解析HTTP请求方法 (GET/POST等) char *method = strtok(buffer, " "); if (!method) { send_error(client_fd, 400); return; } // 只处理GET请求 if (strcmp(method, "GET") != 0) { send_error(client_fd, 501); // 501 Not Implemented return; } // 获取请求路径 char *path = strtok(NULL, " "); if (!path) { send_error(client_fd, 400); return; } // 安全过滤:防止路径遍历攻击 if (strstr(path, "..") || strstr(path, "//")) { send_error(client_fd, 403); // 403 Forbidden return; } // 处理默认路径 if (strcmp(path, "/") == 0) { path = "/Index.html"; } // 构建完整文件路径 char full_path[256]; snprintf(full_path, sizeof(full_path), "%s%s", STATIC_DIR, path); // 检查文件是否存在 struct stat st; if (stat(full_path, &st) < 0) { send_error(client_fd, 404); return; } // 检查是否是目录 if (S_ISDIR(st.st_mode)) { // 如果是目录,尝试查找index.html char index_path[256]; snprintf(index_path, sizeof(index_path), "%s/index.html", full_path); if (stat(index_path, &st) < 0 || S_ISDIR(st.st_mode)) { send_error(client_fd, 403); // 目录浏览禁止 return; } strcpy(full_path, index_path); } // 打开文件 int fd = open(full_path, O_RDONLY); if (fd < 0) { perror("open file failed"); send_error(client_fd, 500); return; } // 获取内容类型 const char *content_type = get_content_type(full_path); // 构建HTTP响应头 char header[512]; int header_len = snprintf(header, sizeof(header), "HTTP/1.1 200 OK\r\n" "Server: SimpleWebServer\r\n" "Content-Type: %s\r\n" "Content-Length: %ld\r\n" // 兼容IE8需要明确长度 "Connection: close\r\n\r\n", // 兼容IE8需要显式关闭 content_type, st.st_size); // 发送响应头 if (send(client_fd, header, header_len, 0) < 0) { perror("send header failed"); close(fd); return; } // 发送文件内容 ssize_t bytes, total_sent = 0; char file_buffer[BUFFER_SIZE]; while ((bytes = read(fd, file_buffer, BUFFER_SIZE)) > 0) { ssize_t sent = send(client_fd, file_buffer, bytes, 0); if (sent < 0) { perror("send file content failed"); break; } total_sent += sent; } close(fd); printf("Served: %s (%ld bytes)\n", path, total_sent); } // 根据文件扩展名获取内容类型 const char *get_content_type(const char *path) { const char *ext = strrchr(path, '.'); if (!ext) return "text/plain"; // 常见MIME类型映射 if (strcmp(ext, ".html") == 0) return "text/html; charset=utf-8"; if (strcmp(ext, ".htm") == 0) return "text/html; charset=utf-8"; if (strcmp(ext, ".css") == 0) return "text/css"; if (strcmp(ext, ".js") == 0) return "application/javascript"; if (strcmp(ext, ".json") == 0) return "application/json"; if (strcmp(ext, ".png") == 0) return "image/png"; if (strcmp(ext, ".jpg") == 0) return "image/jpeg"; if (strcmp(ext, ".jpeg") == 0) return "image/jpeg"; if (strcmp(ext, ".gif") == 0) return "image/gif"; if (strcmp(ext, ".ico") == 0) return "image/x-icon"; if (strcmp(ext, ".svg") == 0) return "image/svg+xml"; if (strcmp(ext, ".txt") == 0) return "text/plain; charset=utf-8"; return "application/octet-stream"; // 默认二进制流 } // 发送错误响应 void send_error(int client_fd, int code) { const char *messages[] = { [400] = "Bad Request", [403] = "Forbidden", [404] = "Not Found", [500] = "Internal Server Error", [501] = "Not Implemented" }; const char *message = messages[code] ? messages[code] : "Unknown Error"; char response[512]; int len = snprintf(response, sizeof(response), "HTTP/1.1 %d %s\r\n" "Content-Type: text/html; charset=utf-8\r\n" "Connection: close\r\n\r\n" "<html><body><h1>%d %s</h1></body></html>", code, message, code, message); send(client_fd, response, len, 0); printf("Error %d: %s\n", code, message); }这段代码怎么区分这三种函数
最新发布
08-15
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值