- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我正在尝试将文件读入 Common Lisp 中的 string(而不是 list),但是我最终在字符串。只有当文件包含换行符或制表符等字符时才会发生这种情况;空格似乎工作得很好。这是我的代码:
(defun load-file (filename)
(with-open-file (stream filename
:direction :input
:if-does-not-exist :error)
(let ((contents (make-string (file-length stream))))
(read-sequence contents stream)
contents)))
请注意:很遗憾,我不允许在此程序中使用循环或外部库。
最佳答案
这是一个老问题,答案是“不要那样做”。这样做的原因是 file-length
在许多有趣的情况下不能做你想做的事。特别是,file-length
的一个版本以您期望的方式工作,返回文件中的字符数,仅当以下一个或两个为真时才易于实现:
遗憾的是,对于我所知道的任何现代平台,这些都不是真的:
文件中的字符数不是其中字节数的固定倍数,原因至少有两个:
#\Return
#\Newline
),它们将被视为一个字符;对于像这样的平台,file-length
告诉你你想知道你正在读取的文件的内容的唯一方法是读取和解码整个文件文件,这显然是不可取的。实际上,file-length
仅告诉您文件的字节长度。
所以这种“计算出文件的长度并将其压缩成一大块”的技巧一般来说是行不通的,因为无法知道文件的长度以字符为单位没有阅读它。
有点烦人(我认为是 CL 的轻微缺陷)它不包含一个函数,其约定是“读取此文件并返回包含它的字符串”。
我相信,至少对于常见的编码,文件的字符长度永远不会长于字节长度。所以如果你愿意冒险一点,你可以做的一件事是分配一个数组,它是文件的字节长度,读取文件,然后记下你填充了多少数组(为了更聪明,使用可调阵列,读取后调整为合适的长度)。
请注意 Alexandria包含一个函数,read-file-into-string
,它可以满足您的需求,并且便携且速度可能很快。
这是一个相当简单的版本,我认为它适用于大多数情况(它根本不考虑字符串的元素类型):
(defun file->string (f &key (buffer-size 1024))
(with-open-file (in f :direction :input)
(with-output-to-string (out)
(loop with buffer = (make-string buffer-size)
for nchars = (read-sequence buffer in)
do (write-sequence buffer out :start 0 :end nchars)
while (= nchars buffer-size)))))
这是一个经过部分测试的、毛茸茸的函数,它试图变得更聪明,并处理文件的字节数比字符数短的情况(即使在正常的平台上,如果文件在读取时被附加到)。处理此问题的代码分支尚未经过测试:caveat emptor。
在大多数情况下,这也会减少数据的复制,但它返回的字符串通常会有一些浪费的空间。它假设填充指针很便宜(它们应该是)并且调整数组大小仅作为最后的手段才可以接受:因此当它需要使字符串更短时,它通过设置填充指针而不是调整它的大小来实现,只调整大小当它需要让它更长的时候。
它还温和地假设尾调用是优化的。
(defun file->string (f &key (element-type ':default)
(external-format ':default)
(growth-factor 0.1))
"Read a file into a string, dealing with character encoding issues"
;; This attempts to be efficient: it allocates a string which, if
;; there are slightly fewer characters than bytes in the file (which
;; is the case for common encodings, will be a little too large,
;; then reads the file into it in one fell swoop, setting the
;; fill-pointer correctly after doing so if needed. It also
;; attempts to deal with the case where the file is *shorter* in
;; bytes than it is in characters (this might be true if the file
;; was being appended to as the read is happening, or on some
;; platform which compresses files and reports the compressed
;; length), although this part of the code is untested.
;;
;; I am not sure if the use of LISTEN here is really right.
;;
(with-open-file (in f :direction :input
:element-type element-type
:external-format external-format)
(let* ((l (file-length in))
(buf (make-array (list l)
:element-type (stream-element-type in)
:adjustable t :fill-pointer t))
(n (read-sequence buf in)))
(cond ((< n l)
;; Just make the array seem a bit shorter: this is the
;; common case for things like UTF-8 and DOS line endings
(adjust-array buf (list l) :fill-pointer n))
((and (= n l) (not (listen in)))
;; We got the exact length of the string and the stream
;; is at EOF. So the string is fine as is: this will be
;; true for traditional Unix encodings where a character
;; is a byte and line endings are a single character.
buf)
(t
;; This is unexpected: the file is longer in characters
;; than it is in bytes. This code is UNTESTED since the
;; only case I can engineer for it involves a race
;; between something which is appending to the file and
;; this code, and that test is too hard to set up.
(labels ((get-more (start chunk-size)
(let ((size (+ start chunk-size)))
(adjust-array buf (list size) :fill-pointer size)
(let ((n (read-sequence buf in :start start)))
(cond ((< n chunk-size)
;; we're done: set the fill pointer
;; right and return
(adjust-array buf (list size)
:fill-pointer (+ start n)))
((and (= n chunk-size) (not (listen in)))
;; We're also done: we got the
;; exact number of characters we
;; had allocated fortuitously
buf)
(t
;; there is more to get
(get-more (+ start chunk-size) chunk-size)))))))
(get-more l (ceiling (* l growth-factor)))))))))
关于lisp - With-open-file 读取额外的字符,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48230636/
OpenAL.org && 创意开发网站已关闭。我选择替代版本 OpenAL Soft .我很担心,因为在 OpenAL Soft 的二进制安装中我找不到 alut.h header 。 alut.h
我使用 Android Studio 已经有一段时间了,但有一天应用程序突然出错了。当我尝试单击我的目录以查找要导入或打开的文件时,应用程序变得异常缓慢并且根本没有响应。当我最终成功切换到存储我的文件
自 Firefox 4 以来,这似乎是一个奇怪的功能变化。在使用 window.open() 打开一个窗口后,当用鼠标中键单击打开的窗口中的链接时(或右键单击并选择“在新窗口中打开”选项卡') 导致链
我无法从 Open::URI 的 rdoc 中得知当我这样做时返回的是什么: result = open(url) URL 返回 XML,但我如何查看/解析 XML? 最佳答案 open 返回一个 I
经常开发asp但对于细致的说法,真实不太清楚,这里简单的介绍下。 一般情况下 读取数据都是用rs.open sql,conn,1,1 修改数据:rs.open sql,conn,1,3 删除
关于 pathlib 标准库中的模块,是 path.open() 方法只是内置 open() 的“包装器”功能? 最佳答案 如果您阅读了 source code的 pathlib.Path.open你
我想将 Open Liberty 运行时的语言更改为 en_US从 Eclipse IDE 中,但我不知道如何。 也尝试使用 JVM 参数的首选项来设置它,但它没有用。 -Duser.language
这是我所拥有的: 参数“opener”未在可能的函数调用参数中列出。这是 PyCharm 错误还是其他原因? PyCharm 2018.3.5 社区版,Windows 7 上的 Python 3.6.
我正在使用 Tinkerpop 的 GraphFactory.open(Configuration 配置) Java 命令来访问 Neo4j 数据库。 一个最低限度的工作示例是: Configurat
这个问题在这里已经有了答案: What is the python "with" statement designed for? (11 个答案) 关闭 7 年前。 我没有使用过 with 语句,但
我正在玩 python 3.5 中的 open 函数。我不明白 opener 参数(最后一个参数)在 open 函数中的用法。根据 python 文档:可以通过将可调用对象作为打开器传递来使用自定义打
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 5 年前。 Improve th
我试图用 Python 来做一些模拟 3D 声音的工作。我试图运行此代码(答案中提供):Python openAL 3D sound类似,两次都收到: ModuleNotFoundError: No
我一直认为 open 和 io.open 可以互换。 显然不是,如果我相信这个片段: import ctypes, io class POINT(ctypes.Structure): _fie
这个问题在这里已经有了答案: What's the difference between io.open() and os.open() on Python? (7 个答案) 关闭 9 年前。 我是
我正在尝试更好地了解 WCF 的一些内部工作原理。我已经做了相当多的环顾四周,但我无法找到关于 ChannelFactory.Open() 与 IClientChannel.Open() 相比的明确解
这个问题在这里已经有了答案: What is the python "with" statement designed for? (11 个答案) 关闭 7 年前。 我知道有很多关于在 python
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界. 这篇CFSDN的博客文章adodb.recordset.open(rs.open)方法参数详解由
不久前我遇到了一个interesting security hole Link 看起来足够无害,但有一个漏洞,因为默认情况下,正在打开的页面允许打开的页面通过 window.opener 回调到它。有
这在我的应用程序上运行良好,但由于某种原因我无法让它在这里正常工作。无论如何,我的问题是,当我单击列表标题时,我想关闭之前打开的列表标题并仅保留事件的列表标题打开。目前它会打开我点击的所有内容,但也会
我是一名优秀的程序员,十分优秀!