Linux文件原生操作

Linux 中一切皆文件,那么 Linux 文件是什么?

在 Linux 中的文件

可以是:传统意义上的有序数据集合,即:文件系统中的物理文件

也可以是:设备,管道,内存。。。(Linux 管理的一切对象)

Linux 中的文件描述符

文件描述符是一个整数值,他在内核中被用于标识打开的文件或其它 I/O 资源

当打开一个文件时,系统会分配一个文件描述符来唯一标识该文件

文件描述符的范围通常是 0 到 1023

  • 0、1、2 被系统占用,分别代表标准输入、标准输出和标准错误
  • 进程中可用的文件描述符范围是 3 到 1023

Linux 原生文件编程接口

示例 -- 文件复制

文件复制实现

test1.c

#define _LARGEFILE_SOURCE
#define _LARGEFILE64_SOURCE
#define _FILE_OFFSET_BITS 64

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <memory.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

static int copy(int src, int des)
{
    int ret = 0;
    char buf[256] = {0};
    
    do
    {
        ret = read(src, buf, sizeof(buf));
        ret = (ret > 0) ? write(des, buf, ret) : ret;
    } while( ret == sizeof(buf) );
    
    return (ret >= 0);
}

int main(int argc, char* argv[]) 
{
    int src = 0;
    int des = 0;
    
    if( argc == 3 )
    {
        if( (src = open(argv[1], O_RDONLY)) == -1 )
        {
            printf("source: open error...\n");
            exit(-1);
        }
        
        if( (des = open(argv[2], O_CREAT | O_WRONLY, 0777)) == -1 )
        {
            printf("dest: open error...\n");
            exit(-1);
        }
        
        if( copy(src, des) )
        {
            printf("succeed: %s ==> %s\n", argv[1], argv[2]);
        }
        else
        {
            printf("failed...\n");
        }
        
        close(src);
        close(des);
    }
    else
    {
        printf("invalid parameters...\n");
    }
    
    return 0;
}

第 1 - 3 行的宏定义是为了可以拷贝更大的文件

第 19 - 22 行,一直不断的从源文件读取 256 字节,写入到目的文件,直到最后写入的字节不足256字节,就认为是拷贝结束了

程序运行结果如下图所示:

示例 -- 外存数组

需求:

  • 创建一个可以存储 "无限" 个元素的数组
  • 数组大小可动态扩大,即:动态向数组中追加元素
  • 提供统一访问数据元素的方式,即:以 0 下标作为起始位置

解决方案 => 时间换空间

C 语言中的数组将数据存储于内存中,使用数组前必须定义大小

  • 优点:访问速度快    缺点:大小必须固定

若要实现可 "无限追加" 的数组,则需要将数据存储于外存中

  • 将数组元素存储于文件中 (不必预先定义大小,可实时拓展)
  • 根据数组元素大小实时定位文件读写位置,并读写固定大小的数据
  • 优点:数组大小不受限制    缺点:访问速度较慢

外存数组接口设计

关键代码设计

外存数组应用示例

外存数组实现

ext_array.h

#ifndef _EXT_ARRAY_H_
#define _EXT_ARRAY_H_

typedef void EArray;

#define EArray_Init(n, type)    EArray_Init_(n, sizeof(type)) 

#define EArray_Get(type, arr, index)  ({    \
    type ret = {0};                         \
    EArray_Get_(arr, index, &ret);          \
    ret;                                    \
})

#define EArray_Set(type, arr, index, v)  ({ \
    type val = v;                           \
    EArray_Set_(arr, index, &val);          \
})

#define EArray_Append(type, arr, v)  ({ \
    type val = v;                       \
    EArray_Append_(arr, &val);          \
})

EArray* EArray_Init_(unsigned int n, unsigned int esize);

int EArray_Get_(EArray* arr, unsigned int index, void* e);
int EArray_Set_(EArray* arr, unsigned int index, const void* e);

int EArray_Append_(EArray* arr, const void* e);

unsigned int EArray_Length(EArray* arr);

void EArray_Release(EArray* arr);

#endif

ext_array.c

#define _LARGEFILE_SOURCE
#define _LARGEFILE64_SOURCE
#define _FILE_OFFSET_BITS 64

#include <time.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include "ext_array.h"

typedef struct _ext_array_
{
    int fd;
    char name[128];
    unsigned int esize;
    unsigned int length;
}ExtArray;

EArray* EArray_Init_(unsigned int n, unsigned int esize)
{
    ExtArray* ret = malloc(sizeof(ExtArray));
    
    if( ret )
    {
        time_t t = {0};
    
        time(&t);
        
        sprintf(ret->name, "./%ld", t);
    
        if( (ret->fd = open(ret->name, O_CREAT | O_RDWR, 0777)) != -1 )
        {
            int i = 0;
            
            for(i=0; i<n*esize; i++)
            {
                char c = 0;
                write(ret->fd, &c, 1);
            }
            
            ret->esize = esize;
            ret->length = n;
        }
        else
        {
            free(ret);
            ret = NULL;
        }
    }
    
    return ret;
}

int EArray_Get_(EArray* arr, unsigned int index, void* e)
{
    int ret = -1;
    ExtArray* ea = arr;
    
    if( ea && e && (index < ea->length) )
    {
        lseek(ea->fd, index * ea->esize, SEEK_SET);
        ret = read(ea->fd, e, ea->esize);
    }
    
    return ret;
}

int EArray_Set_(EArray* arr, unsigned int index, const void* e)
{
    int ret = -1;
    ExtArray* ea = arr;
    
    if( ea && e && (index < ea->length) )
    {
        lseek(ea->fd, index * ea->esize, SEEK_SET);
        ret = write(ea->fd, e, ea->esize);
    }
    
    return ret;
}

int EArray_Append_(EArray* arr, const void* e)
{
    int ret = -1;
    ExtArray* ea = arr;
    
    if( ea && e )
    {
        lseek(ea->fd, 0, SEEK_END);
        ret = write(ea->fd, e, ea->esize);
        
        if( ret != -1 )
        {
            ea->length += 1;
        }
    }
    
    return ret;
}

unsigned int EArray_Length(EArray* arr)
{
    int ret = -1;
    ExtArray* ea = arr;
    
    if( ea )
    {
        ret = ea->length;
    }
    
    return ret;
}

void EArray_Release(EArray* arr)
{
    ExtArray* ea = arr;
    
    if( ea )
    {
        close(ea->fd);
        unlink(ea->name);
        free(ea);
    }
}

test2.c


#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <memory.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

#include "ext_array.h"

int main(int argc, char* argv[]) 
{
    EArray* arr = EArray_Init(5, int);
    int i = 0;
    
    printf("length = %d\n", EArray_Length(arr));
    
    for(i=0; i<EArray_Length(arr); i++)
    {
        EArray_Set(int, arr, i, i + 1);
    }
    
    EArray_Append(int, arr, 1000);
    
    printf("length = %d\n", EArray_Length(arr));
    
    for(i=0; i<EArray_Length(arr); i++)
    {
        int val = EArray_Get(int, arr, i);
        printf("val = %d\n", val);
    }
    
    EArray_Release(arr);
    
    return 0;
}


EArray_Init_(...) 函数用于创建一个随机文件名的文件,用于存储数组的内容,将这片预定义的空间先写为0

EArray_Get_(...) 和 EArray_Set_(...) 函数通过 lseek(...) 定位到要读写元素的位置,然后进行读写

EArray_Append_(...) 函数通过 lseek(...) 定位到文件的末尾,然后再进行写入操作

我们定义 EArray_Get、EArray_Set 和 EArray_Append 这三个宏,是因为这三个函数末尾带下划线的版本需要传入元素的地址而不是数据,而传入地址,我们每次调用的时候都必须先定义出一个变量出来,再传入这个变量的地址,这样使用很不方便,所以才定义这三个宏,可以直接传入数据

Linux 一切皆文件?

ASCII C 文件操作:

  • stdin => 标准输入流,stdout => 标准输出流,stderr => 标准错误输出

Linux 原生文件操作:

  • 1 => 显示设备,0 => 输入设备,2 => 错误设备

以文件的方式操作设备

通过 ASCII C 文件操作 => 输入 / 输出

通过 Linux 文件操作 => 输入 / 输出

设备文件操作

test3.c


#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <memory.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

static void ascii_file_io()
{
    char buf[16] = {0};
    int i = 0;
    char c = 0;
    
    do
    {
        fread(&c, sizeof(c), 1, stdin);
        
        if( c == '\n' )
            break;
        else
            buf[i++] = c;
        
    } while( i < 16);
    
    fwrite("output: ", 1, 8, stdout);
    fwrite(buf, 1, i, stdout);
    fwrite("\n", 1, 1, stdout);
}

static void linux_file_io()
{
    char buf[16] = {0};
    int i = 0;
    /*
    do
    {
        char c = 0;
        
        read(0, &c, 1);
        
        if( c == '\n' )
            break;
        else
            buf[i++] = c;
        
    } while( i < 16);*/
    
    i = read(0, buf, sizeof(buf));
    
    write(1, "output: ", 8);
    write(1, buf, i);
    write(1, "\n", 1);
}
int main(int argc, char* argv[]) 
{
    // ascii_file_io();
    linux_file_io();
    
    return 0;
}

第 50 行,我们读取文件描述符 0(标准输入),标准输入默认为行缓冲,所以我们在键盘上输入换行符,read 才会返回

程序运行结果如下图所示:

运行环境要求 硬件: x86或x64 CPU 下载文件夹所在卷容量大于4G 软件: Linux 32位或64位x86系列CPU Python3 >= 3.4 Py3.PyQt5 >= 5.2 详细的软件依赖参见发行版的打包文件。 其它: 拥有迅雷账户(会员/非会员均可) 许可证 GPLv3 -- Xware Desktop部分 迅雷协议 -- Xware部分 新增64位版本(迅雷下载核心1.0.31): 附件: xware-desktop_0.13.20141115_amd64.deb [1.71 MiB] 被下载 2099 次 Xware Desktop已打包,64位打包系统是kubuntu15.04,凡是64位的ubuntu15.04系,应该都能用. 迅雷版本是1.0.31,包含数个linux原生二进制程序,建议双击安装包进行安装,它还要额外的pyQt5图形库,会自动从ubuntu仓库下载. 第一次先不要登陆,先要设置.启动迅雷软件->菜单栏->文件->设置 ->挂载->添加下载目录,比如我的是/home/name/download. ->启动与登陆->xwared托管->我们选systemd托管,ETM选随xwared启动 至此设置告一段落.重启PC后,如下图会有两个迅雷的原生进程: 附件: thunder.png thunder.png [ 23.98 KiB | 被浏览 52484 次 ] 在我的kubuntu开始菜单的因特网子菜单里会有个xwared Desktop的迅雷图标,点击它就能启动迅雷了. 第一次登陆后会有激活码,点一下即可成功. 就能跟windows里的迅雷7一样用啦,速度嘛,跟windows完全一样,免费用户同样没有高速通道,要vip才能用. 32位迅雷下载核心更新至1.0.25 附件: xware-desktop_0.10.20140702_i386.deb [1.61 MiB] 被下载 7829 次 当然,有能力的同学可以自行编译,在Ubuntu上编译安装说明如下 : 编译环境: 安装必备的软件。 sudo apt-get install git build-essential devscripts 下载源代码。 git clone git://github.com/Xinkai/XwareDesktop.git 这会在当前目录下生成一个名为XwareDesktop的子目录。 切换到源代码目录XwareDesktop。 cd XwareDesktop 打包 你需要切换到XwareDesktop的源代码目录。 列出缺失的编译依赖。 dpkg-checkbuilddeps。 如果没有列出任何东西,跳过步骤2。 安装缺失的编译依赖。 sudo apt-get install 制作安装包。 dpkg-buildpackage 执行这条命令后会在你当前目录下生成数个包,其中包括xware-desktop_??????.deb安装包。 安装 回到图形化界面,在XwareDesktop的源代码目录的上级目录,你应该能找到名为xware-desktop_??????.deb的安装包文件, 双击它,按提示安装。 浏览器扩展整合 Xware Desktop接受来自命令行的参数作为新任务的网址,格式为 xware-desktop http://www.website.com/file1 ftp://www.website.com/file2 ... 以Firefox上的Flashgot为例,打开其选项。添加一个新的下载器,程序设置为xware-desktop,参数设置为[URL]。 来自命令行的参数支持http,https,ftp,ed2k,magnet,flashget,qqdl,thunder等多种协议, 也同样支持本地的torrent文件。 注:本软件同linux版115网盘存在冲突,出现如下提示: This application failed to start because it could not find or load the Qt platform plugin "xcb". Reinstalling the application may fix this problem.
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值