Skip to content

Commit

Permalink
feat: census
Browse files Browse the repository at this point in the history
  • Loading branch information
aelbozie committed Jan 30, 2024
1 parent 5ab9d7f commit aecfc6a
Show file tree
Hide file tree
Showing 6 changed files with 504 additions and 0 deletions.
41 changes: 41 additions & 0 deletions census/HELP.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
# Help

## Running the tests

To run the tests run the command `go test` from within the exercise directory.

If the test suite contains benchmarks, you can run these with the `--bench` and `--benchmem`
flags:

go test -v --bench . --benchmem

Keep in mind that each reviewer will run benchmarks on a different machine, with
different specs, so the results from these benchmark tests may vary.

## Submitting your solution

You can submit your solution using the `exercism submit census.go` command.
This command will upload your solution to the Exercism website and print the solution page's URL.

It's possible to submit an incomplete solution which allows you to:

- See how others have completed the exercise
- Request help from a mentor

## Need to get help?

If you'd like help solving the exercise, check the following pages:

- The [Go track's documentation](https://exercism.org/docs/tracks/go)
- The [Go track's programming category on the forum](https://forum.exercism.org/c/programming/go)
- [Exercism's programming category on the forum](https://forum.exercism.org/c/programming/5)
- The [Frequently Asked Questions](https://exercism.org/docs/using/faqs)

Should those resources not suffice, you could submit your (incomplete) solution to request mentoring.

To get help if you're having trouble, you can use one of the following resources:

- [How to Write Go Code](https://golang.org/doc/code.html)
- [Effective Go](https://golang.org/doc/effective_go.html)
- [Go Resources](http://golang.org/help)
- [StackOverflow](http://stackoverflow.com/questions/tagged/go)
20 changes: 20 additions & 0 deletions census/HINTS.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
# Hints

## 1. Create a new resident

- Revisit the [struct concept](/tracks/go/concepts/structs) for information on how to create structs.

## 2. Validate resident information

- The zero value for a string is `""`.
- The zero value for a map is `nil`.
- Revisit the [maps concept](/tracks/go/concepts/maps) for information on how to determine if a map key exists.

## 3. Delete resident information

- A resident is considered deleted when all of their fields are set to their zero value.
- Revisit the [zero values concept](/tracks/go/concepts/zero-values) for the zero value for each Go type.

## 4. Count the residents

- Revisit the [range iteration concept](/tracks/go/concepts/range-iteration) for information on how to range over a slice.
136 changes: 136 additions & 0 deletions census/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,136 @@
# Census

Welcome to Census on Exercism's Go Track.
If you need help running the tests or submitting your code, check out `HELP.md`.
If you get stuck on the exercise, check out `HINTS.md`, but try and solve it without using those first :)

## Introduction

Go does not have a concept of empty, null, or undefined for variable values. Variables declared without an explicit initial value default to the zero value for their respective type.

The zero value for primitive types such as booleans, numeric types, and strings are `false`, `0`, and `""`, respectively.

The identifier `nil`, meaning zero, is the zero value for more complex types such as pointers, functions, interfaces, slices, channels, and maps.

The following table details the zero value for Go's types.

| Type | Zero Value |
| --------- | ---------- |
| boolean | `false` |
| numeric | `0` |
| string | `""` |
| pointer | `nil` |
| function | `nil` |
| interface | `nil` |
| slice | `nil` |
| channel | `nil` |
| map | `nil` |

You may have noticed struct types are absent from the above table. That is because the zero value for a struct type depends on its fields. Structs are set to their zero value when all of its fields are set to their respective zero value.

## Instructions

It is your job to prepare the city's IT system for an upcoming census. Specifically, you are responsible for the program that will process the data from the census workers.

The program must be able to create a new resident in the system when given a resident's information. Additionally, you will create functions that ensure the required information is present in the resident's data and delete a resident's data. Lastly, you will count the residents to provide an accurate census count.

## 1. Create a new resident

When a census worker collects a resident's information, they need to register that resident by entering their name, age, and address into the system.

Implement the function `NewResident` that accepts three arguments:

- The name of the resident.
- The age of the resident.
- The address of the resident.

The function should return a pointer to a `Resident` struct that holds this information.

```go
name := "Matthew Sanabria"
age := 29
address := map[string]string{"street": "Main St."}

NewResident(name, age, address)
// => &{Matthew Sanabria 29 map[street:Main St.]}
```

## 2. Validate required information

Residents may be reluctant to provide personal data to census workers. In those cases it's necessary to determine if the resident provided the required information to be counted in the census.

In order to be counted, a resident must provide a non-zero value for their name and an address that contains a non-zero value for the `street` key. All other information, such as the resident's age, is optional. Implement the `HasRequiredInfo` method that returns a boolean indicating if the resident has provided the required information.

```go
name := "Matthew Sanabria"
age := 0
address := make(map[string]string)

resident := NewResident(name, age, address)

resident.HasRequiredInfo()
// => false
```

## 3. Delete resident information

Life moves fast and mistakes happen. A resident can move out of the city. A census worker can make mistakes when collecting data. In those cases, it's necessary to have the ability to delete a resident's data so they will not be counted.

Implement the `Delete` method that sets all of the fields the resident to their zero value.

```go
name := "Matthew Sanabria"
age := 29
address := map[string]string{"street": "Main St."}

resident := NewResident(name, age, address)

fmt.Println(resident)
// Output: &{Matthew Sanabria 29 map[street:Main St.]}

resident.Delete()

fmt.Println(resident)
// Output: &{ 0 map[]}
```

## 4. Count the residents

Now that the system supports census data, it's time to perform the census and count the residents!

Implement the function `Count` that accepts one argument:

- A slice of pointers to `Resident` structs.

The function should return an integer indicating the number of residents that were counted in the census. A resident can only be counted if they provided the required information to the census worker.

```go
name1 := "Matthew Sanabria"
age1 := 29
address1 := map[string]string{"street": "Main St."}

resident1 := NewResident(name1, age1, address1)

name2 := "Rob Pike"
age2 := 0
address2 := make(map[string]string)

resident2 := NewResident(name2, age2, address2)

residents := []*Resident{resident1, resident2}

Count(residents)
// => 1
```

## Source

### Created by

- @jamessouth
- @sudomateo

### Contributed to by

- @tehsphinx
- @eklatzer
46 changes: 46 additions & 0 deletions census/census.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,46 @@
// Package census simulates a system used to collect census data.
package census

// Resident represents a resident in this city.
type Resident struct {
Name string
Age int
Address map[string]string
}

// NewResident registers a new resident in this city.
func NewResident(name string, age int, address map[string]string) *Resident {
r := Resident{
Name: name,
Age: age,
Address: address,
}

return &r
}

// HasRequiredInfo determines if a given resident has all of the required information.
func (r *Resident) HasRequiredInfo() bool {
if r.Address["street"] == "" || r.Name == "" {
return false
}
return true
}

// Delete deletes a resident's information.
func (r *Resident) Delete() {
r.Name = ""
r.Address = nil
r.Age = 0
}

// Count counts all residents that have provided the required information.
func Count(residents []*Resident) int {
var count int
for _, i := range residents {
if i.HasRequiredInfo() {
count += 1
}
}
return count
}
Loading

0 comments on commit aecfc6a

Please sign in to comment.