我可以用Python将内存中的对象上传到FTP吗?
我现在正在做的事情是:
mysock = urllib.urlopen('http://localhost/image.jpg')
fileToSave = mysock.read()
oFile = open(r"C:\image.jpg",'wb')
oFile.write(fileToSave)
oFile.close
f=file('image.jpg','rb')
ftp.storbinary('STOR '+os.path.basename('image.jpg'),f)
os.remove('image.jpg')
把文件写到硬盘上,然后立刻把它们删除,这样的操作感觉是在给系统增加额外的负担,应该尽量避免。那么,我能不能直接把内存中的对象上传到FTP服务器,用Python来实现呢?
3 个回答
1
import urllib
import ftplib
ftp = ftplib.FTP(...)
f = urllib.urlopen('http://localhost/image.jpg')
ftp.storbinary('STOR image.jpg', f)
这段代码的意思是……
首先,它会做一些初始化的工作,比如设置一些变量或者准备一些数据。接着,它会进入一个循环,这个循环会重复执行一些操作,直到满足某个条件为止。这个条件可能是某个变量达到了特定的值,或者是用户输入了某个指令。
在循环内部,代码会执行一些具体的任务,比如计算、判断或者输出结果。最后,当循环结束后,代码可能会进行一些收尾工作,比如清理资源或者输出最终的结果。
总的来说,这段代码的结构很常见,主要是通过循环和条件判断来实现某种功能。
2
你可以使用任何在内存中的“文件样对象”,比如 BytesIO
:
from io import BytesIO
它可以在二进制模式下工作,配合 FTP.storbinary
使用:
f = BytesIO(b"the contents")
ftp.storbinary("STOR /path/file.txt", f)
也可以在 ASCII/文本模式下使用,配合 FTP.storlines
:
f = BytesIO(b"the contents")
ftp.storlines("STOR /path/file.txt", f)
如果你想看更高级的例子,可以参考:
6
因为有一种叫做鸭子类型的概念,你的代码中的文件对象(f
)只需要支持.read(blocksize)
这个调用,就可以和storbinary
一起使用。当遇到这样的问题时,我会去查阅源代码,这里指的是lib/python2.6/ftplib.py:
def storbinary(self, cmd, fp, blocksize=8192, callback=None):
"""Store a file in binary mode. A new port is created for you.
Args:
cmd: A STOR command.
fp: A file-like object with a read(num_bytes) method.
blocksize: The maximum data size to read from fp and send over
the connection at once. [default: 8192]
callback: An optional single parameter callable that is called on
on each block of data after it is sent. [default: None]
Returns:
The response code.
"""
self.voidcmd('TYPE I')
conn = self.transfercmd(cmd)
while 1:
buf = fp.read(blocksize)
if not buf: break
conn.sendall(buf)
if callback: callback(buf)
conn.close()
return self.voidresp()
正如注释所说,它只需要一个类似文件的对象,实际上它甚至不需要特别像文件,只要有read(n)
这个功能就可以了。StringIO就提供了这样的“内存文件”服务。