


What are Go's visibility rules for identifiers (exported vs. unexported)?
Jul 31, 2025 am 08:23 AMIn Go, identifier visibility is determined by capitalization: 1) Exported identifiers start with a capital letter and are visible outside the package, forming the public API; 2) Unexported identifiers start with a lowercase letter and are accessible only within the same package; 3) Package-level scope allows exported identifiers to be accessed externally while unexported ones remain internal; 4) No keywords like public or private are used—visibility is purely lexical; 5) This design supports encapsulation, intentional API design, and controlled access to internal state, as seen in structs with mixed exported and unexported fields and associated methods that safely expose private data.
In Go, the visibility of identifiers—such as variables, functions, types, and methods—is determined by a simple naming convention based on the first letter of the identifier. This governs whether an identifier is exported (visible outside the package) or unexported (visible only within the same package).

Exported vs. Unexported Identifiers
Go uses capitalization to control visibility:
-
Exported identifiers start with a capital letter (e.g.,
MyVar
,DoSomething
,Person
).- These are visible to code in other packages.
- They are part of the package's public API.
-
Unexported identifiers start with a lowercase letter (e.g.,
myVar
,doSomething
,person
).- These are only visible within the same package.
- They are used for internal implementation details.
For example:

package counter var counter int = 0 // unexported: only visible inside 'counter' package var Counter int = 0 // exported: visible to other packages func increment() { // unexported function counter } func Increment() { // exported function increment() }
In another package:
package main import "counter" func main() { counter.Increment() // OK: Increment is exported // counter.counter // Compile error: counter is unexported // counter.increment() // Compile error: increment is unexported }
Scope Rules Summary
- Package-level scope: Only identifiers starting with a capital letter can be accessed from outside the package.
- Internal scope: Unexported identifiers are accessible throughout the same package, across multiple
.go
files. - No access modifiers: Unlike languages such as Java or C#, Go doesn’t use keywords like
public
,private
, orprotected
. Visibility is purely lexical.
Key Implications
- Encapsulation: Use unexported identifiers to hide implementation details and prevent external dependencies on internal logic.
- API design: Carefully choose which types, functions, and variables to export—once exported, changing them can break compatibility.
- Testing: You can test unexported functions by placing test files in the same package (e.g.,
package counter
instead ofpackage counter_test
if using a separate package for tests).
Examples in Practice
type Person struct { Name string // exported field age int // unexported field } func NewPerson(name string, age int) *Person { return &Person{ Name: name, age: age, } }
Here, Name
can be accessed directly from outside the package, but age
cannot. Instead, you might provide a method:
func (p *Person) Age() int { return p.age // safe access to unexported field }
This allows controlled access while keeping internal state encapsulated.
Basically, Go’s visibility rules are simple but powerful: uppercase = public, lowercase = private, all at the package level. No extra keywords, just consistent naming.
The above is the detailed content of What are Go's visibility rules for identifiers (exported vs. unexported)?. For more information, please follow other related articles on the PHP Chinese website!

Hot AI Tools

Undress AI Tool
Undress images for free

Undresser.AI Undress
AI-powered app for creating realistic nude photos

AI Clothes Remover
Online AI tool for removing clothes from photos.

Clothoff.io
AI clothes remover

Video Face Swap
Swap faces in any video effortlessly with our completely free AI face swap tool!

Hot Article

Hot Tools

Notepad++7.3.1
Easy-to-use and free code editor

SublimeText3 Chinese version
Chinese version, very easy to use

Zend Studio 13.0.1
Powerful PHP integrated development environment

Dreamweaver CS6
Visual web development tools

SublimeText3 Mac version
God-level code editing software (SublimeText3)

Hot Topics

Go's switch statement will not be executed throughout the process by default and will automatically exit after matching the first condition. 1. Switch starts with a keyword and can carry one or no value; 2. Case matches from top to bottom in order, only the first match is run; 3. Multiple conditions can be listed by commas to match the same case; 4. There is no need to manually add break, but can be forced through; 5.default is used for unmatched cases, usually placed at the end.

Usereflect.ValueOfandreflect.TypeOftogetruntimevaluesandtypes;2.Inspecttypedetailswithreflect.TypemethodslikeName()andKind();3.Modifyvaluesviareflect.Value.Elem()andCanSet()afterpassingapointer;4.CallmethodsdynamicallyusingMethodByName()andCall();5.R

In Go, to break out of nested loops, you should use labeled break statements or return through functions; 1. Use labeled break: Place the tag before the outer loop, such as OuterLoop:for{...}, use breakOuterLoop in the inner loop to directly exit the outer loop; 2. Put the nested loop into the function, and return in advance when the conditions are met, thereby terminating all loops; 3. Avoid using flag variables or goto, the former is lengthy and easy to make mistakes, and the latter is not recommended; the correct way is that the tag must be before the loop rather than after it, which is the idiomatic way to break out of multi-layer loops in Go.

Usecontext.WithTimeouttocreateacancellablecontextwithadeadlineandalwayscallcancel()toreleaseresources.2.ForHTTPrequests,settimeoutsusinghttp.Client.Timeoutorusecontextviahttp.NewRequestWithContextforper-requestcontrol.3.Ingoroutineswithchannels,usese

Usecontexttopropagatecancellationanddeadlinesacrossgoroutines,enablingcooperativecancellationinHTTPservers,backgroundtasks,andchainedcalls.2.Withcontext.WithCancel(),createacancellablecontextandcallcancel()tosignaltermination,alwaysdeferringcancel()t

UsestructswithPERJSontagsFeRpredictabledatoensurefast, safeparsingwithcompile-timetypesafety.2.avoidmap [string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] interface {string] }duetoreflectionoverheadandruntimetypeassertionsunlessdealingwithtrulydynamicJSON.3.Usejson.RawMessagefordeferredorselectivep

InitializeaGomodulewithgomodinit,2.InstallgqlgenCLI,3.Defineaschemainschema.graphqls,4.Rungqlgeninittogeneratemodelsandresolvers,5.Implementresolverfunctionsforqueriesandmutations,6.SetupanHTTPserverusingthegeneratedschema,and7.RuntheservertoaccessGr

Gooffersfasterexecutionspeedduetocompilationtonativemachinecode,outperforminginterpretedlanguageslikePythonintaskssuchasservingHTTPrequests.2.Itsefficientconcurrencymodelusinglightweightgoroutinesenablesthousandsofconcurrentoperationswithlowmemoryand
