- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
我在使用 json.Unmarshal
时遇到了以下怪癖。运行以下代码时,出现错误 json: Unmarshal(non-pointer map[string]string)
func main() {
m := make(map[string]string)
data := `{"foo": "bar"}`
err := json.Unmarshal([]byte(data), m)
if err != nil {
log.Fatal(err)
}
fmt.Println(m)
}
查看 documentation对于 json.Unmarshal
,似乎没有迹象表明需要指针。我能找到的最接近的是以下行
Unmarshal parses the JSON-encoded data and stores the result in the value pointed to by v.
关于 Unmarshal 遵循的映射协议(protocol)的行同样不清楚,因为它没有引用指针。
To unmarshal a JSON object into a map, Unmarshal first establishes a map to use. If the map is nil, Unmarshal allocates a new map. Otherwise Unmarshal reuses the existing map, keeping existing entries. Unmarshal then stores key-value pairs from the JSON object into the map. The map's key type must either be a string, an integer, or implement encoding.TextUnmarshaler.
为什么我必须将指针传递给 json.Unmarshal,尤其是当映射已经是引用类型时?我知道如果我将一个映射传递给一个函数,并将数据添加到映射中,映射的基础数据将被更改(请参阅 the following playground example ),这意味着如果我将指针传递给 map 。有人可以解决这个问题吗?
最佳答案
如文档中所述:
Unmarshal uses the inverse of the encodings that Marshal uses, allocating maps, slices, and pointers as necessary, with ...
Unmarshal
可以分配变量(映射、 slice 等)。如果我们将 map
而不是 pointer 传递给 map
,那么新分配的 map
将不可见给来电者。以下示例 ( Go Playground) 对此进行了演示:
package main
import (
"fmt"
)
func mapFunc(m map[string]interface{}) {
m = make(map[string]interface{})
m["abc"] = "123"
}
func mapPtrFunc(mp *map[string]interface{}) {
m := make(map[string]interface{})
m["abc"] = "123"
*mp = m
}
func main() {
var m1, m2 map[string]interface{}
mapFunc(m1)
mapPtrFunc(&m2)
fmt.Printf("%+v, %+v\n", m1, m2)
}
其中的输出是:
map[], map[abc:123]
如果需求说一个函数/方法可以在必要时分配一个变量并且新分配的变量需要对调用者可见,解决方案将是:(a)变量必须在函数的返回 语句 或 (b) 可以将变量分配给函数/方法参数。因为在 go
中 everything 都是按值传递的,所以在 (b) 的情况下,参数必须是一个指针。下图说明了上述示例中发生的情况:
m1
和m2
都指向nil
。mapFunc
会将m1
指向的值复制到m
,结果m
也将指向无
map 。m1
指向的地址(不是地址m1
) 将被复制到 m
。在这种情况下,m1
和 m
都指向相同的 map 数据结构,因此通过 m1
修改 map 项将对 m
也可见。 mapFunc
函数中,分配新 map 并将其赋值给m
。无法将其分配给 m1
。 在指针的情况下:
mapPtrFunc
时,m2
的地址会被复制到mp
。 mapPtrFunc
中,新 map 被分配并分配给*mp
(不是mp
)。由于 mp
是指向 m2
的指针,将新映射分配给 *mp
将更改 m2
指向的值。注意mp
的值是不变的,即m2
的地址。 关于json - 如果 map 是引用类型,为什么 json.Unmarshal 需要指向 map 的指针?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45122496/
有人能给我解释一下我下面程序中的一点吗我编写的以下程序是为了理解 int * const * var 的含义; #include "iostream" using namespace std ; in
我正在摆弄 C 中的指针,但仍然不确定一些非常基础的知识。我想出了以下示例代码: #include int main(void) { int num = 42; /
以下代码产生警告: const char * mystr = "\r\nHello"; void send_str(char * str); void main(void){ send_str
我正在尝试在我的 Linux 计算机上升级 cmake。这是我使用的命令。 删除旧的 cmake: sudo apt purge --auto-remove cmake 从 https://cmake
我试图借助 char 指针来追踪 union 第一个字节的值。我假设 byte1 的输出应该是 ff,但实际上是 ffffffff。如果我错了请纠正我。 #include #include #in
我正在使用微 Controller 进行一些 ADC 测量。当我尝试使用 -O2 优化编译以下代码时遇到问题,当代码中存在 PrintVal() 函数时,MCU 卡住。我做了一些调试,结果发现,当我添
#include #include using namespace std; int main () { vector qwerty; qwerty.push_back(5);
我有我的 woking setup.py 文件;其中包含以下行: home = os.environ["HOME"] # home = /home/dr in my machine; distu
这段代码 #include #include static_assert(std::is_same_v::value_type, volatile int>); 在最新的 GCC 和 clang
我对 C 中的前向声明有疑问。 代码 typedef struct yhash_s t_yhash;// forward declaration struct yhash_s { size_t
我想提取成员指针指向的成员的类型。 template void demo(myClass& instance, void* ptr) { instance.*member = *reinter
我正在尝试使用指针将一段 C 代码转换为 Swift 3。这是 C 代码中的相关部分。 Float32 sampleArray[256] = { // Array is 256 Float value
您可能知道,VLA's haves pros and cons 和它们在 C11 中是可选的。 我想使 VLA 成为可选项的主要原因是:“堆栈可能会爆炸”: int arr[n]; /* where
这段代码有什么错误?为什么我不能按照我尝试的方式取消引用该元素。 #include typedef struct { int value; struct node * left;
有什么方法可以在 JavaScript 中创建\返回指向变量的指针吗? 比如,在 PHP 中: function func() { ..... return &$result; } 我
如果您想使用方法的指针作为参数,则需要将该方法键入作为对象的函数,就像这样好 : type TAcceptor = function(filename:string):boolean of objec
很简单的问题: 我对 C++ 中的智能指针有点陌生。我想我得到了所有权的东西,但我不知道如何访问他们实际指向的内容。当我尝试使用对象的成员函数/变量时,我只是得到了 unique_ptr 类的函数,这
我得到了一个点的方位 Angular 、指南针方向和一个可以将箭头设置到某个方向的 api(0 是顶部,90 是右侧,180 是底部,360 是顶部) 如果我希望箭头指向我采用方位 Angular 形
我正在尝试找到一种方法,从单元格中获取位于当前工作表左侧(托盘下方)的工作表单元格中的数据。 我知道如何通过调用其他工作表 =Sheet1!A1 但现在我需要一些最好的解释 =Sheet[-1]!A1
所以我在 MATLAB 中有一个 for 循环,其中向量 x 将通过一个函数,比如 cos(x).^2,或者不同的选择,比如 sin(x).^2 + 9.*x。用户将在 for 循环之前选择他想使用的
我是一名优秀的程序员,十分优秀!