转载自http://blog.csdn.net/u012759136/article/details/52232266 原文做者github地址python
关于Tensorflow读取数据,官网给出了三种方法:git
对于数据量较小而言,可能通常选择直接将数据加载进内存,而后再分batch
输入网络进行训练(tip:使用这种方法时,结合yield
使用更为简洁,你们本身尝试一下吧,我就不赘述了)。可是,若是数据量较大,这样的方法就不适用了,由于太耗内存,因此这时最好使用tensorflow提供的队列queue
,也就是第二种方法 从文件读取数据。对于一些特定的读取,好比csv文件格式,官网有相关的描述,在这儿我介绍一种比较通用,高效的读取方法(官网介绍的少),即便用tensorflow内定标准格式——TFRecords
github
TFRecords实际上是一种二进制文件,虽然它不如其余格式好理解,可是它能更好的利用内存,更方便复制和移动,而且不须要单独的标签文件(等会儿就知道为何了)… …总而言之,这样的文件格式好处多多,因此让咱们用起来吧。网络
TFRecords文件包含了tf.train.Example
协议内存块(protocol buffer)(协议内存块包含了字段 Features
)。咱们能够写一段代码获取你的数据, 将数据填入到Example
协议内存块(protocol buffer),将协议内存块序列化为一个字符串, 而且经过tf.python_io.TFRecordWriter
写入到TFRecords文件。函数
从TFRecords文件中读取数据, 可使用tf.TFRecordReader
的tf.parse_single_example
解析器。这个操做能够将Example
协议内存块(protocol buffer)解析为张量。ui
接下来,让咱们开始读取数据之旅吧~spa
咱们使用tf.train.Example
来定义咱们要填入的数据格式,而后使用tf.python_io.TFRecordWriter
来写入。.net
import os import tensorflow as tf from PIL import Image cwd = os.getcwd() ''' 此处我加载的数据目录以下: 0 -- img1.jpg img2.jpg img3.jpg ... 1 -- img1.jpg img2.jpg ... 2 -- ... ... ''' writer = tf.python_io.TFRecordWriter("train.tfrecords") for index, name in enumerate(classes): class_path = cwd + name + "/" for img_name in os.listdir(class_path): img_path = class_path + img_name img = Image.open(img_path) img = img.resize((224, 224)) img_raw = img.tobytes() #将图片转化为原生bytes example = tf.train.Example(features=tf.train.Features(feature={ "label": tf.train.Feature(int64_list=tf.train.Int64List(value=[index])), 'img_raw': tf.train.Feature(bytes_list=tf.train.BytesList(value=[img_raw])) })) writer.write(example.SerializeToString()) #序列化为字符串 writer.close()
关于Example
Feature
的相关定义和详细内容,我推荐去官网查看相关API。code
基本的,一个Example
中包含Features
,Features
里包含Feature
(这里没s)的字典。最后,Feature
里包含有一个 FloatList
, 或者ByteList
,或者Int64List
blog
就这样,咱们把相关的信息都存到了一个文件中,因此前面才说不用单独的label文件。并且读取也很方便。
for serialized_example in tf.python_io.tf_record_iterator("train.tfrecords"): example = tf.train.Example() example.ParseFromString(serialized_example) image = example.features.feature['image'].bytes_list.value label = example.features.feature['label'].int64_list.value # 能够作一些预处理之类的 print image, label
一旦生成了TFRecords文件,接下来就可使用队列(queue
)读取数据了。
def read_and_decode(filename): #根据文件名生成一个队列 filename_queue = tf.train.string_input_producer([filename]) reader = tf.TFRecordReader() _, serialized_example = reader.read(filename_queue) #返回文件名和文件 features = tf.parse_single_example(serialized_example, features={ 'label': tf.FixedLenFeature([], tf.int64), 'img_raw' : tf.FixedLenFeature([], tf.string), }) img = tf.decode_raw(features['img_raw'], tf.uint8) img = tf.reshape(img, [224, 224, 3]) img = tf.cast(img, tf.float32) * (1. / 255) - 0.5 label = tf.cast(features['label'], tf.int32) return img, label
以后咱们能够在训练的时候这样使用
img, label = read_and_decode("train.tfrecords") #使用shuffle_batch能够随机打乱输入 img_batch, label_batch = tf.train.shuffle_batch([img, label], batch_size=30, capacity=2000, min_after_dequeue=1000) init = tf.initialize_all_variables() with tf.Session() as sess: sess.run(init) threads = tf.train.start_queue_runners(sess=sess) for i in range(3): val, l= sess.run([img_batch, label_batch]) #咱们也能够根据须要对val, l进行处理 #l = to_categorical(l, 12) print(val.shape, l)
至此,tensorflow高效从文件读取数据差很少完结了。
恩?等等…什么叫差很少?对了,还有几个注意事项:
第一,tensorflow里的graph可以记住状态(state
),这使得TFRecordReader
可以记住tfrecord
的位置,而且始终能返回下一个。而这就要求咱们在使用以前,必须初始化整个graph,这里咱们使用了函数tf.initialize_all_variables()
来进行初始化。
第二,tensorflow中的队列和普通的队列差很少,不过它里面的operation
和tensor
都是符号型的(symbolic
),在调用sess.run()
时才执行。
第三, TFRecordReader
会一直弹出队列中文件的名字,直到队列为空。
record reader
解析tfrecord文件batcher
)QueueRunner