#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <sys/time.h>
#include <unistd.h>
#include <glog/logging.h>
#include <gflags/gflags.h> // 读文件类
class CFileReader
{
public:
CFileReader()
: _buffer(NULL)
{
} ~CFileReader()
{
free(_buffer);
} bool open(const char* filepath)
{
// 以O_DIRECT方式打开文件
int fd = ::open(filepath, O_RDONLY | O_DIRECT);
if (-1 == fd)
{
LOG(ERROR) << "open " << filepath << " error: " << strerror(errno);
return false;
} // 取得文件大小,以便一次性将文件读取出来
struct stat st;
if (-1 == fstat(fd, &st))
{
LOG(ERROR) << "stat " << filepath << " error: " << strerror(errno);
close(fd);
return false;
} // 分配足以容纳整个文件的Buffer
// 由于以O_DIRECT方式读取,所以需要按页对齐
size_t size = st.st_size + (getpagesize() - st.st_size%getpagesize());
posix_memalign((void**)&_buffer, getpagesize(), size);
if (NULL == _buffer)
{
LOG(ERROR) << "malloc failed";
close(fd);
return false;
} // 将整个文件读取_buffer中
int bytes_read = read(fd, _buffer, size);
if (-1 == bytes_read)
{
LOG(ERROR) << "read " << filepath << " error: " << strerror(errno);
close(fd);
free(_buffer);
_buffer = NULL;
return false;
}
else if (bytes_read != size)
{
// 两组测试输出数据:
// FileSize(212000000) => AlignSize(212000768) -> RealSize(212000000)
// FileSize(2120000000) => AlignSize(2120003584) -> RealSize(2120000000)
printf("FileSize(%d) => AlignSize(%d) -> RealSize(%d)\n", st.st_size, size, bytes_read);
} return true;
} // 从文件中读取一个节点数据
// offset:偏移量
// return:返回指向记录的指针
template <class P>
const P* get_record(uint64_t offset) const
{
return reinterpret_cast<P*>(_buffer + offset);
} // 取得文件所有的记录
template <class P>
const P** get_all_record() const
{
return reinterpret_cast<P**>(_buffer);
} private:
char* _buffer;
}; // 用于计时
class TimeWatcher
{
public:
TimeWatcher(const std::string& tip)
: _tip(tip)
{
struct timeval now;
gettimeofday(&now, NULL); _now_msec = (now.tv_sec * 1000) + (now.tv_usec / 1000);
} ~TimeWatcher()
{
struct timeval now;
gettimeofday(&now, NULL); time_t cur_msec = (now.tv_sec * 1000) + (now.tv_usec / 1000);
LOG(INFO) << _tip << " spend " << cur_msec - _now_msec << "ms";
} private:
std::string _tip;
time_t _now_msec;
}; struct User
{
int32_t age;
int32_t hight;
int32_t weight;
char bitmap[50*4];
}; // 用于生成测试文件
bool make_test_file(const char* filepath, int num_records)
{
int fd = open(filepath, O_WRONLY | O_CREAT, S_IRUSR | S_IWUSR);
if (-1 == fd)
{
LOG(ERROR) << "open " << filepath << " error: " << strerror(errno);
return false;
} User user;
TimeWatcher time_watcher("write"); for (int i=0; i<num_records; ++i)
{
user.age = i;
user.hight = i + i;
user.weight = i * i; if (-1 == write(fd, &user, sizeof(user)))
{
LOG(ERROR) << "write " << filepath << " error: " << strerror(errno);
close(fd);
return false;
}
} close(fd);
return true;
} // 模拟随机读取
void random_read(const CFileReader& file_reader, int num_random)
{
int *index = new int[num_random]; for (int i=0; i<num_random; ++i)
{
srandom(time(NULL));
index[i] = random() % num_random;
} TimeWatcher time_watcher("randmon read");
for (int i=0; i<num_random; ++i)
{
file_reader.get_record<struct User>(index[i]);
}
} // 执行测试
void test()
{
int num_records1 = 1000000;
int num_records2 = 10000000;
std::string file1 = "./file_1000000";
std::string file2 = "./file_10000000"; if (make_test_file(file1.c_str(), 1000000)
&& make_test_file(file2.c_str(), 10000000))
{
printf("to read, press ENTER to continue ...\n");
getchar(); CFileReader file_reader1; {
TimeWatcher time_watcher("open");
if (!file_reader1.open(file1.c_str()))
{
return;
}
} random_read(file_reader1, 1000000);
random_read(file_reader1, 3000000); CFileReader file_reader2; {
TimeWatcher time_watcher("open");
if (!file_reader2.open(file2.c_str()))
{
return;
}
} random_read(file_reader2, 1000000);
random_read(file_reader2, 3000000);
}
} int main(int argc, char* argv[])
{
test();
return 0;
}

O_DIRECT方式读取文件示例的更多相关文章

  1. c++ 二进制方式读取文件 读取特殊类型数据

    #include <iostream> #include <fstream> using namespace std; /* 二进制方式进行读写文件,可以读写 各种各样数据类型 ...

  2. [JAVA 多种方式读取文件]

    package com.file; import java.io.BufferedReader; import java.io.File; import java.io.FileInputStream ...

  3. 文件结束符和C\C++读取文件方式

    http://www.cnblogs.com/cvbnm/articles/2003056.html 约定编译器为 gcc2/x86: 所以 char, unsigned char 为 8 位, in ...

  4. 精通awk系列(3):铺垫知识:读取文件的几种方式

    回到: Linux系列文章 Shell系列文章 Awk系列文章 读取文件的几种方式 读取文件有如下几种常见的方式: 下面使用Shell的read命令来演示前4种读取文件的方式(第五种按字节数读取的方式 ...

  5. 在Spring Boot快捷地读取文件内容的若干种方式

    引言: 在Spring Boot构建的项目中,在某些情况下,需要自行去读取项目中的某些文件内容,那该如何以一种轻快简单的方式读取文件内容呢?  基于ApplicationContext读取 在Spri ...

  6. C#中StreamReader类读取文件使用示例

    C#中StreamReader类读取文件使用示例 1.需要导入的命名空间是:System.IO; 2.操作的是字符,所以打开的是文本文件. 常用属性:   CurrentEncoding:对象正在使用 ...

  7. python中读取文件的read、readline、readlines方法区别

    #读取文件所有内容,返回字符串对象,python默认以文本方式读取文件,遇到结束符读取结束. fr = open('lenses.txt')read = fr.read()print(type(rea ...

  8. Win10系列:JavaScript写入和读取文件

    正如上面的内容中所提到的,文件保存选取器用于保存文件,通过Windows.Storage.Pickers命名空间中的FileSavePicker类的pickSaveFileAsync函数可以向指定的文 ...

  9. FileReader读取文件详解

    FileReader是一种异步文件读取机制,结合input:file可以很方便的读取本地文件. input:file 在介绍FileReader之前,先简单介绍input的file类型. <in ...

随机推荐

  1. ALSA声卡10_从零编写之数据传输_学习笔记

    1.引言 (1)应用程序使用声卡的时候,数据流程是:应用程序把数据发送给驱动,驱动把数据发送给硬件声卡,声卡把数据转换成声音数据播放出去. (2)可以使用两种方式发送数据 第一种:app发数据,等驱动 ...

  2. ThinkPHP 目录结构

    2.0 ThinkPHP 目录结构 在前面的博客中,通过一个简单的案例向大家演示了在ThinkPHP 框架下开发的大致法程,本篇博客将对ThinkPHP框架目录结构进行详细讲解. 要想在项目中熟练地使 ...

  3. 关于alter database open resetlogs及incarnation的一点理解

    关于alter database open resetlogs及incarnation的一点理解 不完全恢复只能做一次吗? 采用rman的默认设置,对数据库进行了backup database备份,进 ...

  4. oracle建立用户与授权(转载)

    创建表空间及用户: create tablespace 表空间名 datafile 'd:/seal.dbf' size 10M autoextend on;create user username ...

  5. poj-2828 Buy Tickets(经典线段树)

    /* Buy Tickets Time Limit: 4000MS Memory Limit: 65536K Total Submissions: 10207 Accepted: 4919 Descr ...

  6. 敌兵布阵hdu1166

    /* 敌兵布阵 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total Submi ...

  7. Linux 服务器--Iptables 端口转发

    日常Iptables 端口转发 需求:公司是局域网络,通过一个外网ip,进行互联网的访问.公司的云平台服务器在公网中,虚拟化平台中有一台内部服务器,用于公司某部门的使用,上面运行www 服务,ssh端 ...

  8. 读取resource下文件

    ArrayList<PatrolOper> patrolOpers = new ArrayList<>(); String jsonData = null; File json ...

  9. bootstrap 的页码显示问题-------------德州

    之前一个小bug,无论上边怎么搜索,下边的页码,不会改变 调整: 1,在mapper中添加计数, 2,找到service,queryPage中添加, 3,关键一部,如果没有会报错:,找不到该列 so, ...

  10. java.lang.IllegalStateException: getOutputStream() has already been called for this response解决方案

    异常产生原因:web容器生成的servlet代码中有out.write(""),这个和JSP中调用的response.getOutputStream()产生冲突.即Servlet规 ...