mirror of
https://github.com/golang/go.git
synced 2025-12-08 06:10:04 +00:00
allow copy of struct containing unexported fields
An experiment: allow structs to be copied even if they contain unexported fields. This gives packages the ability to return opaque values in their APIs, like reflect does for reflect.Value but without the kludgy hacks reflect resorts to. In general, we trust programmers not to do silly things like *x = *y on a package's struct pointers, just as we trust programmers not to do unicode.Letter = unicode.Digit, but packages that want a harder guarantee can introduce an extra level of indirection, like in the changes to os.File in this CL or by using an interface type. All in one CL so that it can be rolled back more easily if we decide this is a bad idea. Originally discussed in March 2011. https://groups.google.com/group/golang-dev/t/3f5d30938c7c45ef R=golang-dev, adg, dvyukov, r, bradfitz, jan.mercl, gri CC=golang-dev https://golang.org/cl/5372095
This commit is contained in:
parent
0ed5e6a2be
commit
d03611f628
15 changed files with 56 additions and 221 deletions
|
|
@ -13,6 +13,14 @@ import (
|
|||
|
||||
// File represents an open file descriptor.
|
||||
type File struct {
|
||||
*file
|
||||
}
|
||||
|
||||
// file is the real representation of *File.
|
||||
// The extra level of indirection ensures that no clients of os
|
||||
// can overwrite this data, which could cause the finalizer
|
||||
// to close the wrong file descriptor.
|
||||
type file struct {
|
||||
fd int
|
||||
name string
|
||||
dirinfo *dirInfo // nil unless directory being read
|
||||
|
|
@ -32,8 +40,8 @@ func NewFile(fd int, name string) *File {
|
|||
if fd < 0 {
|
||||
return nil
|
||||
}
|
||||
f := &File{fd: fd, name: name}
|
||||
runtime.SetFinalizer(f, (*File).Close)
|
||||
f := &File{&file{fd: fd, name: name}}
|
||||
runtime.SetFinalizer(f.file, (*file).close)
|
||||
return f
|
||||
}
|
||||
|
||||
|
|
@ -71,6 +79,10 @@ func OpenFile(name string, flag int, perm uint32) (file *File, err error) {
|
|||
// Close closes the File, rendering it unusable for I/O.
|
||||
// It returns an error, if any.
|
||||
func (file *File) Close() error {
|
||||
return file.file.close()
|
||||
}
|
||||
|
||||
func (file *file) close() error {
|
||||
if file == nil || file.fd < 0 {
|
||||
return EINVAL
|
||||
}
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue