学习日志之struct.c的运行及解释

1.struct.c的代码展示
2.运行实例
3.详细解释及知识点

1.strruct.c的代码如下

#include <stdio.h>
#include <stdlib.h>

typedef struct {
    int a[2];
    double d;
} struct_t;

double fun(int i) {
    volatile struct_t s;
    s.d = 3.14;
    s.a[i] = 1073741824; /* Possibly out of bounds */
    return s.d; /* Should be 3.14 */
}

int main(int argc, char *argv[]) {
    int i = 0;
    if (argc >= 2)
	i = atoi(argv[1]);
    double d = fun(i);
    printf("fun(%d) --> %.10f\n", i, d);
    return 0;
}

2.运行实例

gec@ubuntu:/mnt/hgfs/share/csapp_code$ ./a.out 0
fun(0) --> 3.1400000000
gec@ubuntu:/mnt/hgfs/share/csapp_code$ ./a.out 1
fun(1) --> 3.1400000000
gec@ubuntu:/mnt/hgfs/share/csapp_code$ ./a.out 2
fun(2) --> 3.1399998665
gec@ubuntu:/mnt/hgfs/share/csapp_code$ ./a.out 3
fun(3) --> 2.0000006104
gec@ubuntu:/mnt/hgfs/share/csapp_code$ ./a.out 4
fun(4) --> 3.1400000000
段错误 (核心已转储)

3.详细解释
为什么当i>1时就会有问题?
在结构体的存储区由低地址到高地址依次存放int型a[0],a[1],double型的d,当i>1时,1073741824会覆盖掉d中的数据。由于d的存储区由低到高是按尾数,阶码,符号的顺序存放,所以i=2时没有i=3时覆盖的数据造成的影响严重。

知识点

越界访问和缓冲区溢出
C语言程序中对数组的访问可能会有意或无意地超越数组存储区范围
而无法发现。
数组存储区可看成是一个缓冲区,超越数组存储区范围的写入操作
称为缓冲区溢出
–例如,对于一个有10个元素的char型数组,其定义的缓冲区有10个字
节。若写一个字符串到这个缓冲区,那么只要写入的字符串多于9个字
符(结束符’\0’占一个字节) ,就会发生“写溢出“ 。
缓冲区溢出是一种非常普遍、非常危险的漏洞,在各种操作系统、
应用软件中广泛存在。
缓冲区溢出攻击是利用缓冲区溢出漏洞所进行的攻击。利用缓冲区
溢出攻击,可导致程序运行失败、系统关机、重新启动等后果。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
可以使用时间戳来判断哪个日志文件是最先创建的,从而进行删除。以下是一个简单的示例代码: ```c #include <stdio.h> #include <stdlib.h> #include <time.h> #include <sys/stat.h> #include <string.h> #define LOG_FILE_PREFIX "./log_" #define LOG_FILE_SUFFIX ".txt" #define MAX_LOG_FILES 10 int main() { // 获取当前时间戳 time_t now = time(NULL); // 构造日志文件名 char log_file_name[100]; sprintf(log_file_name, "%s%ld%s", LOG_FILE_PREFIX, now, LOG_FILE_SUFFIX); // 创建新的日志文件 FILE *log_file = fopen(log_file_name, "w"); if (log_file == NULL) { printf("Failed to create log file!\n"); exit(1); } // 关闭日志文件 fclose(log_file); // 获取日志文件列表 struct stat st; char cmd[100]; sprintf(cmd, "ls %s*%s | sort -n", LOG_FILE_PREFIX, LOG_FILE_SUFFIX); FILE *fp = popen(cmd, "r"); if (fp == NULL) { printf("Failed to execute command!\n"); exit(1); } // 读取日志文件列表 char log_files[MAX_LOG_FILES][100]; int i = 0; while (fgets(log_files[i], 100, fp) != NULL) { // 去掉换行符 log_files[i][strlen(log_files[i]) - 1] = '\0'; // 记录文件信息 stat(log_files[i], &st); time_t file_time = st.st_mtime; // 如果文件是最先创建的,就删除它 if (i >= MAX_LOG_FILES - 1) { remove(log_files[0]); for (int j = 0; j < i - 1; j++) { strcpy(log_files[j], log_files[j + 1]); } i--; } // 插入新的文件 if (now < file_time) { for (int j = i - 1; j >= 0; j--) { strcpy(log_files[j + 1], log_files[j]); } strcpy(log_files[0], log_file_name); i++; break; } i++; } // 输出日志文件列表 printf("Log files:\n"); for (int j = 0; j < i; j++) { printf("%s\n", log_files[j]); } // 关闭文件指针 pclose(fp); return 0; } ``` 这个程序会在当前目录下创建一个新的日志文件,然后列出所有以 `./log_` 开头、以 `.txt` 结尾的文件,并根据它们的创建时间戳来判断哪些文件是最先创建的。如果文件列表中的文件数超过了 `MAX_LOG_FILES`,就删除最先创建的文件。最后输出日志文件列表。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值