- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
是否可以使用 Gob 编码将结构串联附加到同一文件中?它适用于写作,但是当我不止一次地使用解码器阅读时,我遇到了:
extra data in buffer
所以我首先想知道这是否可行,或者我是否应该使用 JSON 之类的东西来逐行附加 JSON 文档。因为另一种方法是序列化一个 slice ,但再次将其作为一个整体读取会破坏追加的目的。
最佳答案
gob
软件包并非设计为以这种方式使用。 gob 流必须由单个 gob.Encoder
写入, 它也必须由单个 gob.Decoder
读取.
这是因为 gob
package 不仅序列化你传递给它的值,它还传输数据来描述它们的类型:
A stream of gobs is self-describing. Each data item in the stream is preceded by a specification of its type, expressed in terms of a small set of predefined types.
这是编码器/解码器的状态——关于什么类型以及它们是如何传输的——后续的新编码器/解码器将不会(不能)分析“前面的”流来重建相同的状态并继续之前的编码器/解码器停止了。
当然,如果您创建单个 gob.Encoder
,您可以使用它来序列化任意数量的值。
您还可以创建 gob.Encoder
并写入一个文件,然后创建一个新的 gob.Encoder
, 并附加到同一个文件,但是您必须使用 2 gob.Decoder
s 读取这些值,与编码过程完全匹配。
作为演示,我们来看一个例子。此示例将写入内存缓冲区 ( bytes.Buffer
)。 2 个后续编码器将写入它,然后我们将使用 2 个后续解码器读取值。我们将写入此结构的值:
type Point struct {
X, Y int
}
为了简短、紧凑的代码,我使用了这个“错误处理程序”函数:
func he(err error) {
if err != nil {
panic(err)
}
}
现在是代码:
const n, m = 3, 2
buf := &bytes.Buffer{}
e := gob.NewEncoder(buf)
for i := 0; i < n; i++ {
he(e.Encode(&Point{X: i, Y: i * 2}))
}
e = gob.NewEncoder(buf)
for i := 0; i < m; i++ {
he(e.Encode(&Point{X: i, Y: 10 + i}))
}
d := gob.NewDecoder(buf)
for i := 0; i < n; i++ {
var p *Point
he(d.Decode(&p))
fmt.Println(p)
}
d = gob.NewDecoder(buf)
for i := 0; i < m; i++ {
var p *Point
he(d.Decode(&p))
fmt.Println(p)
}
输出(在 Go Playground 上尝试):
&{0 0}
&{1 2}
&{2 4}
&{0 10}
&{1 11}
请注意,如果我们只使用 1 个解码器来读取所有值(循环直到 i < n + m
,当迭代到达 n + 1
时,我们会收到您在问题中发布的相同错误消息,因为后续数据不是序列化的 Point
,而是新的 gob
流的开始。
所以如果你想坚持使用 gob
做你想做的事情的包,你必须稍微修改,增强你的编码/解码过程。当使用新的编码器时,您必须以某种方式标记边界(因此在解码时,您将知道必须创建一个新的解码器来读取后续值)。
您可以使用不同的技术来实现这一点:
这里需要注意的几点:
gob
如果只使用一个编码器,则包是最有效、最紧凑的,因为每次创建和使用新编码器时,都必须重新传输类型规范,从而导致更多开销,并使编码/解码过程较慢。如果你想寻找功能,你需要单独管理一个 index 文件,它会告诉新的编码器/解码器从哪个位置开始,这样你就可以寻找到那个位置,创建一个新的解码器,并从那里开始读取值。
关于serialization - 使用 Gob 以追加方式将日志写入文件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43199137/
在 Python 中,我可以附加到一个空数组,例如: >>> a = [] >>> a.append([1,2,3]) >>> a.append([1,2,3]) >>> a [[1, 2, 3],
我正在阅读(并慢慢尝试)在 php 中与 txt 文件交互的方法。我已经尝试过追加,它将数据添加到txt文件的末尾但是 a+ 与 a 有何不同 在 w3schools 中它说: 一个 append 。
我想执行一个非常简单的操作:合并两个形状文件。具体来说,我有美国每个州的人口普查区域形状文件,我想将它们合并到一个形状文件中。最终,我想获取组合的形状文件并在一组经纬度坐标上执行叠加,以确定我的坐标属
当我们使用 append 和 cut 运算符时会出现什么问题? append2([],L,L):-!. append2([H|T],L,[H|TL]):-append2(T,L,TL).
我有一个函数处理程序: function handler(data) { console.log(`1. ${data}`); } 我想在相同的范围内附加或重新定义,如下所示: let old
我目前正在使用很多这样的内容来重构应用程序: StringBuffer buff1 = new StringBuffer(""); buff1.append("some value A"); buff
我正在编写一些代码来对不同类型的啤酒进行一些计算。我有一个使用 GUI 的主类,并有一个 JTextArea 来打印输出。在主类中调用追加工作得很好,但是当我尝试从外部类调用追加来写入文本区域时...
我有一个像这样的 jquery block 。渲染 html 后,我看到 标签立即打开和关闭,同样的方式,立即打开和关闭。我在他的代码中做错了什么吗?有更好的方法来实现这个吗? 谢谢 $.each(f
我在尝试克隆父 div 然后将其直接附加到其自身下方时遇到一个问题。只要最后一个节点是,我的函数就可以正常工作如此选择: A B C 将导致 A A.1
我正在尝试在现有 td 末尾附加一个 td。下面是以下代码(我在 jqgrid 中执行)。 $("#list_toppager_center tr:first td:eq(7)").append("C
我正在尝试在 jQuery 中的以下追加方法上设置超时。我尝试过的所有操作都不断返回Uncaught SyntaxError:意外的标识符 这是我的代码: setTimeout("$('#us
我想用 c 打开一个文件,然后向其中添加一些内容并关闭它。我只是想知道 fopen 中的 a+ 自动导航到文件的最后一个字符。 最佳答案 是的。 为什么不尝试一下,或者阅读一下手册呢? 这里是:
在我的代码中,我有一个输入字段,它是一个循环的值。 用户在第一个字段中输入所需的值。 用户单击按钮/徽章(单击我添加项目符号)以附加到模式。 根据字段中的输入值显示带有项目符号数的模态框。 例如,如果
是否可以使用 QUrlQuery 在不对 url 进行 strip 化的情况下 append 数据? 使用下面的代码将删除“?”之后的所有内容和结果是: https://foobar.com/Info
好吧,我正在为 iPhone 制作一个简单的聊天应用程序,我很幸运,它运行良好并且看起来很棒但是我有一些问题,一个这样的问题是我向用户显示富文本的方式.. 目前我有一个荒谬的系统,它是这样工作的 {发
在 C# 中格式化我做的字符串: string a = String.Format("/blah/blah/{0}_{1}/blah.html", int1, int2) 在Python中,它会自动将
我有一个 300 万行的 .txt 文件。该文件包含如下所示的数据: # RSYNC: 0 1 1 0 512 0 #$SOA 5m localhost. hostmaster.localhost.
我有一个问题。可以删除使用 javascript 附加添加的元素? 当我尝试删除添加的跨度时,什么也没有发生。 像这样: $(document).ready(function(){ $('#
已关闭。此问题需要 debugging details 。目前不接受答案。 编辑问题以包含 desired behavior, a specific problem or error, and the
到目前为止这是我的代码,我想做的是说用户输入 1 2 3 然后按 -1,他或她将被要求输入另一组数字,比如 9 8 7,我的程序是什么假设要做的是将它们显示为 1 2 3 9 8 7,而是像这样显示它
我是一名优秀的程序员,十分优秀!