- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我正在编写一个遍历目录系统的程序,为找到的文件创建哈希值,然后如果发现具有相同哈希键(重复文件)的任何其他文件,它们将被分组在一起。我打算使用嵌入在另一个链表中的链表。顶级链表包含哈希键,然后与该键关联的节点包含重复文件的值。目前,我在尝试初始化顶级列表时遇到了困难。我成功地将第一个键值传递给链表,创建了 headList
。但是,在初始传递并继续遍历之后, key 的值丢失了,我不确定为什么。我使用 gdb 对其进行了追踪,并发现当我返回到 searchDirects
函数时该值在某处被删除了。
这是我的代码:
#define _GNU_SOURCE // for asprintf(), if needed
#include <unistd.h>
#include <stdio.h>
#include <dirent.h>
#include <string.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <stdlib.h>
#include <openssl/md5.h>
//stat -c "%s %n" filename
//The above will display filename and size of file
//gcc -g -Wall -o file file.c -lssl -lcrypto
//the above is needed to link libraries so the encryption will work
#define table_size 20
#define BUFFER_SIZE 4096 //max path buffer size
#define X (37)
void directoryCheck(char *fileName, char *dirValue);
//This will be the struct that contains the binned hash keys (the rows)
struct LinkNode {
char pathValue[BUFFER_SIZE]; // this will be used to hold the directory value
struct LinkNode *nextNode;
};
struct List {
struct LinkNode *headNode; // Embedded linked list
unsigned char *key; // Hash key value
struct List *nextList;
//char *value;
}*headList;
void printAllThatShit(struct List **headList)
{
struct List *temp;
temp = *headList;
while(temp != NULL)
{
printf("Here is the key when sent to print function:\n");
for(int i = 0; i < 16; i++)
printf("%02x", temp->key[i]);
printf("\n");
temp = temp->nextList;
}
}
//Function to add to List
void addToList(unsigned char *key, char* dirValue)
{
//head is Null, so we will put first key in
if(headList == NULL)
{
printf("Adding to head of list!\n");
headList = malloc(sizeof(struct List));
headList->nextList = NULL;
headList->key = key;
/*printf("Here is the key: %s\n", headList->key);
for(int i = 0; i < 16; i++)
printf("%02x", headList->key[i]);
printf(" %s\n", dirValue);*/
}
else
{
currentList = headList;
if(currentList != 0)
{
while(currentList->nextList != 0)
{
if(currentList->key == key)
{
printf("Found the same key!\n");
return;
}
currentList = currentList->nextList;
}
currentList->nextList = malloc(sizeof(struct List));
currentList = currentList->nextList;
currentList->key = key;
currentList->nextList = NULL;
}
}
printAllThatShit(&headList); //After this initial print
//with headList, the key value is junked
}
void deleteList(struct List **headList)
{
struct List *current, *next;
current = *headList;
while(current != NULL)
{
printf("Here is the current value: %s", current->key);
next = current->nextList;
free(current);
current = next;
}
*headList = NULL;
}
void md5Hash(char *path)
{
unsigned char key[MD5_DIGEST_LENGTH]; //16 bytes for the output
struct stat statbuf;
FILE *inFile;
MD5_CTX mdx;
int bytes, i, size;
stat(path, &statbuf);
size = statbuf.st_size;
inFile = fopen(path, "rb");
/*if(size == -1)
{
fprintf(stderr, "Unable to open %s", File);
return;
}*/
if(inFile == NULL)
{
fprintf(stderr, "Unable to open %s", path);
return;
}
unsigned char data[size];
//Initialize the structure
MD5_Init(&mdx);
//fread read the file byte-by-byte for 1024 bytes, and reads it into
//the buffer(data). The function returns how many bytes that were
//successfully read. Then, MD5_Update hashes and updates the structure using
//the bytes in the data buffer, and goes in 1 byte increments.
while((bytes = fread(data, 1, size, inFile)) != 0)
MD5_Update(&mdx, data, bytes);
MD5_Final(key, &mdx); //Place the final 16 byte output in key
for(i = 0; i < MD5_DIGEST_LENGTH; i++)
printf("%02x", key[i]);
printf(" %s\n", path);
fclose(inFile);
printf("Here is that file path while in the hashing function: %s\n", path);
addToList(key, path);
}
void newFile(char *fileName, char *dirValue)
{
printf("Made it to add a new file!\n");
char *appendPath = NULL;
asprintf(&appendPath,"%s/%s", dirValue, fileName);
printf("Here is that file you are now sending to be Hashed: %s\n", fileName);
md5Hash(appendPath);
}
//CHeck if the argument is a file
int is_regular_file(const char *path)
{
struct stat path_stat;
stat(path, &path_stat);
return S_ISREG(path_stat.st_mode);
}
void searchDirects(char *path, int depth)
{
DIR *dp; // represents directory stream
struct dirent *entry; // This is used for traversing directories
struct stat statbuf; // this is so you can use the stat()
int file;
stat(path, &statbuf);
dp = opendir(path);
if(dp)
{
while((entry = readdir(dp)) != NULL)
{
if(entry->d_type == DT_REG)
{
printf("Found a file in the directory!\n");
newFile(entry->d_name, path); //send the file name and directory to be added
}
else if(strcmp(".",entry->d_name) == 0 || strcmp("..",entry->d_name) == 0)
{
printf("Found files with . or ..!\n");
continue;
}
else
{
printf("Attempting to check a directory\n");
directoryCheck(entry->d_name, path);
}
printf("[%s]\n", entry->d_name);
}
closedir(dp);
}
}
void directoryCheck(char *fileName, char *dirValue)
{
char *appendPath;
asprintf(&appendPath,"%s/%s", dirValue, fileName);
searchDirects(appendPath, 1);
free(appendPath);
}
int main(int argc, char * argv[])
{
headList = NULL;
//headNode = NULL;
//struct node* newnode = (struct node*)malloc(20 * sizeof(struct node));
if(argc <= 1)
{
return 0;
}
int i = 0;
for(i = 1; i < argc; i++)
{
if(is_regular_file(argv[i]))
{
printf("Put function to handle file\n");
//getHashKey(argv[i]);
md5Hash(argv[i]);
}
else
searchDirects(argv[i], 1);
}
printAllThatShit(&headList);
printf("Going to delete the list now!\n");
deleteList(&headList);
printf("Scan of current directory:\n");
printf("Scan of current directory: %s\n", argv[1]);
printf("done.\n");
//free(newnode);
exit(0);
}
我知道我有很多不同的错误,此时很可能是内存泄漏。但是,我只是想弄清楚为什么键值在初始传递后在 headList->key 中丢失。并且,对于添加的节点,任何后续传递的 key 都会丢失。我认为这可能是因为我创建哈希键并传递它的方式,但是当我运行 gdb 时,我意识到当我返回到 while() 搜索目录时(如上所述), key 丢失了。感谢您提供任何帮助或见解。
最佳答案
问题是由一个常见错误引起的。在 addToList
中:
headList->key = key;
这会创建一个指向 key
缓冲区的指针。但是,addToList
在 md5Hash
函数中是这样调用的:
unsigned char key[MD5_DIGEST_LENGTH];
addToList(key, path);
在该代码中,key
是一个局部变量。当 md5Hash
函数退出时,它超出范围。因此链表 key
字段是一个无效指针。由于从无效指针访问内存是未定义的行为,所有的赌注都在那时取消了。
一个修复方法是memcpy
键值到链表中。
将 key
定义为数组而不是指针。
struct List {
struct LinkNode *headNode; // Embedded linked list
unsigned char key[MD5_DIGEST_LENGTH]; // Hash key value
struct List *nextList;
//char *value;
}*headList;
在 addToList
中:
// REMOVE THIS LINE:
// headList->key = key;
// REPLACE WITH THIS:
memcpy(headList->key, key, sizeof(headList->key));
关于c - 链表中的键值被丢弃,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41709789/
#include using namespace std; class C{ private: int value; public: C(){ value = 0;
这个问题已经有答案了: What is the difference between char a[] = ?string?; and char *p = ?string?;? (8 个回答) 已关闭
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 7 年前。 此帖子已于 8 个月
除了调试之外,是否有任何针对 c、c++ 或 c# 的测试工具,其工作原理类似于将独立函数复制粘贴到某个文本框,然后在其他文本框中输入参数? 最佳答案 也许您会考虑单元测试。我推荐你谷歌测试和谷歌模拟
我想在第二台显示器中移动一个窗口 (HWND)。问题是我尝试了很多方法,例如将分辨率加倍或输入负值,但它永远无法将窗口放在我的第二台显示器上。 关于如何在 C/C++/c# 中执行此操作的任何线索 最
我正在寻找 C/C++/C## 中不同类型 DES 的现有实现。我的运行平台是Windows XP/Vista/7。 我正在尝试编写一个 C# 程序,它将使用 DES 算法进行加密和解密。我需要一些实
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
有没有办法强制将另一个 窗口置于顶部? 不是应用程序的窗口,而是另一个已经在系统上运行的窗口。 (Windows, C/C++/C#) 最佳答案 SetWindowPos(that_window_ha
假设您可以在 C/C++ 或 Csharp 之间做出选择,并且您打算在 Windows 和 Linux 服务器上运行同一服务器的多个实例,那么构建套接字服务器应用程序的最明智选择是什么? 最佳答案 如
你们能告诉我它们之间的区别吗? 顺便问一下,有什么叫C++库或C库的吗? 最佳答案 C++ 标准库 和 C 标准库 是 C++ 和 C 标准定义的库,提供给 C++ 和 C 程序使用。那是那些词的共同
下面的测试代码,我将输出信息放在注释中。我使用的是 gcc 4.8.5 和 Centos 7.2。 #include #include class C { public:
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我的客户将使用名为 annoucement 的结构/类与客户通信。我想我会用 C++ 编写服务器。会有很多不同的类继承annoucement。我的问题是通过网络将这些类发送给客户端 我想也许我应该使用
我在 C# 中有以下函数: public Matrix ConcatDescriptors(IList> descriptors) { int cols = descriptors[0].Co
我有一个项目要编写一个函数来对某些数据执行某些操作。我可以用 C/C++ 编写代码,但我不想与雇主共享该函数的代码。相反,我只想让他有权在他自己的代码中调用该函数。是否可以?我想到了这两种方法 - 在
我使用的是编写糟糕的第 3 方 (C/C++) Api。我从托管代码(C++/CLI)中使用它。有时会出现“访问冲突错误”。这使整个应用程序崩溃。我知道我无法处理这些错误[如果指针访问非法内存位置等,
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
我有一些 C 代码,将使用 P/Invoke 从 C# 调用。我正在尝试为这个 C 函数定义一个 C# 等效项。 SomeData* DoSomething(); struct SomeData {
这个问题已经有答案了: Why are these constructs using pre and post-increment undefined behavior? (14 个回答) 已关闭 6
我是一名优秀的程序员,十分优秀!