كيف يمكنني الحصول على حجم الملف في C ؟ [مكررة]
سؤال
ممكن مكررة:
كيف يمكنك تحديد حجم ملف في C ؟
كيف يمكنني معرفة حجم الملف فتحت مع تطبيق مكتوب في C ؟ أود أن أعرف حجم لأنني أريد أن أضع المحتوى من تحميل الملف إلى سلسلة ، والتي تخصص باستخدام malloc()
.مجرد كتابة malloc(10000*sizeof(char));
هو IMHO فكرة سيئة.
المحلول
عليك أن تسعى إلى نهاية الملف ثم نسأل عن الموقف:
fseek(fp, 0L, SEEK_END);
sz = ftell(fp);
يمكنك ثم طلب مرة أخرى ، على سبيل المثال:
fseek(fp, 0L, SEEK_SET);
أو (إذا كان يسعى إلى الذهاب إلى البداية)
rewind(fp);
نصائح أخرى
باستخدام معيار المكتبة:
على افتراض أن تنفيذ مجد يدعم SEEK_END:
fseek(f, 0, SEEK_END); // seek to end of file
size = ftell(f); // get current file pointer
fseek(f, 0, SEEK_SET); // seek back to beginning of file
// proceed with allocating memory and reading the file
لينكس/POSIX:
يمكنك استخدام stat
(إذا كنت تعرف اسم الملف) ، أو fstat
(إذا كان لديك ملف واصف).
هنا هو مثال على القانون الأساسي:
#include <sys/stat.h>
struct stat st;
stat(filename, &st);
size = st.st_size;
Win32:
يمكنك استخدام GetFileSize أو GetFileSizeEx.
إذا كان لديك ملف واصف fstat()
يعود القانون الأساسي هيكل التي تحتوي على حجم الملف.
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
// fd = fileno(f); //if you have a stream (e.g. from fopen), not a file descriptor.
struct stat buf;
fstat(fd, &buf);
off_t size = buf.st_size;
انتهى بي الأمر جعل مجرد قصيرة وحلوة fsize
وظيفة(ملاحظة أي خطأ التحقق)
int fsize(FILE *fp){
int prev=ftell(fp);
fseek(fp, 0L, SEEK_END);
int sz=ftell(fp);
fseek(fp,prev,SEEK_SET); //go back to where we were
return sz;
}
انها نوع من السخيف أن المكتبة القياسية C لا يملك مثل هذه الوظيفة ، ولكن أستطيع أن أرى لماذا سيكون من الصعب وليس كل "ملف" لديها حجم(على سبيل المثال ، /dev/null
)
هل تعتبر لا الحوسبة حجم الملف فقط المتزايد الصفيف إذا لزم الأمر ؟ هنا مثال (مع التحقق من الخطأ حذفها):
#define CHUNK 1024
/* Read the contents of a file into a buffer. Return the size of the file
* and set buf to point to a buffer allocated with malloc that contains
* the file contents.
*/
int read_file(FILE *fp, char **buf)
{
int n, np;
char *b, *b2;
n = CHUNK;
np = n;
b = malloc(sizeof(char)*n);
while ((r = fread(b, sizeof(char), CHUNK, fp)) > 0) {
n += r;
if (np - n < CHUNK) {
np *= 2; // buffer is too small, the next read could overflow!
b2 = malloc(np*sizeof(char));
memcpy(b2, b, n * sizeof(char));
free(b);
b = b2;
}
}
*buf = b;
return n;
}
هذه الميزة من العمل حتى للتيارات التي من المستحيل الحصول على حجم الملف (مثل stdin).
إذا كنت على لينكس ، تنظر بجدية فقط باستخدام g_file_get_contents وظيفة من سطحي.أنه يتعامل مع كل كود تحميل ملف تخصيص الذاكرة و التعامل مع الأخطاء.
كيفية استخدام lseek/fseek/ستات/fstat للحصول على حجم الملف ?
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <sys/stat.h>
void
fseek_filesize(const char *filename)
{
FILE *fp = NULL;
long off;
fp = fopen(filename, "r");
if (fp == NULL)
{
printf("failed to fopen %s\n", filename);
exit(EXIT_FAILURE);
}
if (fseek(fp, 0, SEEK_END) == -1)
{
printf("failed to fseek %s\n", filename);
exit(EXIT_FAILURE);
}
off = ftell(fp);
if (off == (long)-1)
{
printf("failed to ftell %s\n", filename);
exit(EXIT_FAILURE);
}
printf("[*] fseek_filesize - file: %s, size: %ld\n", filename, off);
if (fclose(fp) != 0)
{
printf("failed to fclose %s\n", filename);
exit(EXIT_FAILURE);
}
}
void
fstat_filesize(const char *filename)
{
int fd;
struct stat statbuf;
fd = open(filename, O_RDONLY, S_IRUSR | S_IRGRP);
if (fd == -1)
{
printf("failed to open %s\n", filename);
exit(EXIT_FAILURE);
}
if (fstat(fd, &statbuf) == -1)
{
printf("failed to fstat %s\n", filename);
exit(EXIT_FAILURE);
}
printf("[*] fstat_filesize - file: %s, size: %lld\n", filename, statbuf.st_size);
if (close(fd) == -1)
{
printf("failed to fclose %s\n", filename);
exit(EXIT_FAILURE);
}
}
void
stat_filesize(const char *filename)
{
struct stat statbuf;
if (stat(filename, &statbuf) == -1)
{
printf("failed to stat %s\n", filename);
exit(EXIT_FAILURE);
}
printf("[*] stat_filesize - file: %s, size: %lld\n", filename, statbuf.st_size);
}
void
seek_filesize(const char *filename)
{
int fd;
off_t off;
if (filename == NULL)
{
printf("invalid filename\n");
exit(EXIT_FAILURE);
}
fd = open(filename, O_RDONLY, S_IRUSR | S_IRGRP);
if (fd == -1)
{
printf("failed to open %s\n", filename);
exit(EXIT_FAILURE);
}
off = lseek(fd, 0, SEEK_END);
if (off == (off_t)-1)
{
printf("failed to lseek %s\n", filename);
exit(EXIT_FAILURE);
}
printf("[*] seek_filesize - file: %s, size: %lld\n", filename, off);
if (close(fd) == -1)
{
printf("failed to close %s\n", filename);
exit(EXIT_FAILURE);
}
}
int
main(int argc, const char *argv[])
{
int i;
if (argc < 2)
{
printf("%s <file1> <file2>...\n", argv[0]);
exit(0);
}
for(i = 1; i < argc; i++)
{
seek_filesize(argv[i]);
stat_filesize(argv[i]);
fstat_filesize(argv[i]);
fseek_filesize(argv[i]);
}
return 0;
}
#include <stdio.h>
#define MAXNUMBER 1024
int main()
{
int i;
char a[MAXNUMBER];
FILE *fp = popen("du -b /bin/bash", "r");
while((a[i++] = getc(fp))!= 9)
;
a[i] ='\0';
printf(" a is %s\n", a);
pclose(fp);
return 0;
}
HTH