Skip to content

Commit 84e56f1

Browse files
committed
Upgrade Trie to NNBD
1 parent 266bd95 commit 84e56f1

File tree

2 files changed

+49
-52
lines changed

2 files changed

+49
-52
lines changed

lib/trie/trie.dart

+21-25
Original file line numberDiff line numberDiff line change
@@ -1,19 +1,16 @@
1-
// @dart=2.9
21
/// Trie is an ordered tree data structure used to store a dynamic set
32
/// or associative array where the keys are usually strings.
43
class Trie<V extends Comparable> {
54
/// Root of the trie.
6-
TrieNode root;
5+
TrieNode? root;
76

87
/// Separates the value into it's components.
98
final Function splitter;
109

1110
List _components;
1211

1312
/// Initialises trie with custom set of values.
14-
Trie(Set components, this.splitter) {
15-
_components = [...components];
16-
}
13+
Trie(Set components, this.splitter) : _components = [...components];
1714

1815
/// Generates trie of lower-case alphabets.
1916
Trie.ofAlphabets()
@@ -28,22 +25,22 @@ class Trie<V extends Comparable> {
2825

2926
/// Adds a [value] to the trie.
3027
void add(V value) {
31-
var list = _split(value);
28+
var list = _split(value) as List<V>;
3229
if (isEmpty) {
3330
root ??= TrieNode({..._components});
3431
}
35-
_add(root, list);
32+
_add(root!, list);
3633
}
3734

3835
/// Checks if [value] is contained in the trie.
3936
bool contains(V value) {
40-
var list = _split(value);
41-
return isEmpty ? false : _contains(root, list);
37+
var list = _split(value) as List<V>;
38+
return isEmpty ? false : _contains(root!, list);
4239
}
4340

4441
/// Deletes [value] from the trie.
4542
void delete(V value) {
46-
var list = _split(value);
43+
var list = _split(value) as List<V>;
4744
var returnValue = _delete(root, list);
4845
returnValue ?? nullify();
4946
}
@@ -61,11 +58,11 @@ class Trie<V extends Comparable> {
6158
var path = _indexOf(value.first);
6259
value = value.sublist(1);
6360

64-
if (node.children[path] == null) {
65-
node.children[path] = TrieNode(components);
61+
if (node.children![path] == null) {
62+
node.children![path] = TrieNode(components);
6663
}
6764

68-
_add(node.children[path], value);
65+
_add(node.children![path]!, value);
6966
}
7067

7168
bool _contains(TrieNode node, List<V> value) {
@@ -75,8 +72,8 @@ class Trie<V extends Comparable> {
7572
var path = _indexOf(value.first);
7673
value = value.sublist(1);
7774

78-
if (node.children[path] != null) {
79-
return _contains(node.children[path], value);
75+
if (node.children![path] != null) {
76+
return _contains(node.children![path]!, value);
8077
} else {
8178
return false;
8279
}
@@ -85,7 +82,7 @@ class Trie<V extends Comparable> {
8582
/// Traverses the path following [value] and marks
8683
/// `node.isValue` to `false` at end. Deletes values eagerly i.e.
8784
/// cleans up any parent nodes that are no longer necessary.
88-
TrieNode _delete(TrieNode node, List<V> value) {
85+
TrieNode? _delete(TrieNode? node, List<V> value) {
8986
if (value.isEmpty) {
9087
// In case trie is empty and an empty value is passed.
9188
if (node == null) return null;
@@ -94,7 +91,7 @@ class Trie<V extends Comparable> {
9491

9592
// Checks all the children. If null, then deletes the node.
9693
var allNull = true;
97-
for (var child in node.children) {
94+
for (var child in node.children!) {
9895
if (child != null) {
9996
allNull = false;
10097
break;
@@ -110,16 +107,16 @@ class Trie<V extends Comparable> {
110107
value = value.sublist(1);
111108

112109
// Path to value doesn't exist.
113-
if (node.children[path] == null) {
110+
if (node.children![path] == null) {
114111
return node;
115112
}
116113

117-
node.children[path] = _delete(node.children[path], value);
114+
node.children![path] = _delete(node.children![path]!, value);
118115

119116
// Delete node if all children are null.
120-
if (node.children[path] == null) {
117+
if (node.children![path] == null) {
121118
var allNull = true;
122-
for (var child in node.children) {
119+
for (var child in node.children!) {
123120
if (child != null) {
124121
allNull = false;
125122
break;
@@ -150,11 +147,10 @@ class TrieNode<V extends Comparable> {
150147
bool isValue = false;
151148

152149
/// Connection to [children].
153-
List<TrieNode> children;
150+
List<TrieNode?>? children;
154151

155152
/// Initializes the node to have as many [children]
156153
/// as there are components in the trie.
157-
TrieNode(Set components) {
158-
children = List<TrieNode>(components.length);
159-
}
154+
TrieNode(Set components)
155+
: children = List<TrieNode?>.filled(components.length, null);
160156
}

test/trie/trie_test.dart

+28-27
Original file line numberDiff line numberDiff line change
@@ -1,64 +1,65 @@
1-
// @dart=2.9
21
import 'package:test/test.dart';
32
import 'package:algorithms/trie/trie.dart';
43

54
void main() {
6-
Trie emptyTrie, trie, customTrie;
5+
Trie? emptyTrie, trie, customTrie;
76

87
setUp(() {
98
emptyTrie = Trie.ofAlphabets();
109

1110
trie = Trie.ofAlphabets();
12-
trie.add('algorithms');
11+
trie!.add('algorithms');
1312

1413
customTrie = Trie(
1514
{'a', 'b', 'f', 'o', 'r', 'z'}, (value) => value.toString().split(''));
16-
customTrie.add('foo');
15+
customTrie!.add('foo');
1716
});
1817

1918
test('Components', () {
20-
expect(trie.components,
19+
expect(trie!.components,
2120
equals({...List.generate(26, (i) => String.fromCharCode(122 - i))}));
2221

23-
expect(customTrie.components, equals({'f', 'o', 'b', 'a', 'r', 'z'}));
22+
expect(customTrie!.components, equals({'f', 'o', 'b', 'a', 'r', 'z'}));
2423
});
2524

2625
test('Empty trie', () {
27-
expect(emptyTrie.isEmpty, isTrue);
28-
expect(trie.isEmpty, isFalse);
26+
expect(emptyTrie!.isEmpty, isTrue);
27+
expect(trie!.isEmpty, isFalse);
2928
});
3029

3130
test('Add value', () {
32-
trie.add('hi');
33-
expect(trie.root.children[7].children[8].isValue, isTrue);
31+
trie!.add('hi');
32+
expect(trie!.root!.children![7]!.children![8]!.isValue, isTrue);
3433

35-
customTrie.add('bar');
36-
expect(customTrie.root.children[1].children[0].children[4].isValue, isTrue);
34+
customTrie!.add('bar');
35+
expect(customTrie!.root!.children![1]!.children![0]!.children![4]!.isValue,
36+
isTrue);
3737

38-
customTrie.add('baz');
39-
expect(customTrie.root.children[1].children[0].children[5].isValue, isTrue);
38+
customTrie!.add('baz');
39+
expect(customTrie!.root!.children![1]!.children![0]!.children![5]!.isValue,
40+
isTrue);
4041
});
4142

4243
test('Contains value', () {
43-
expect(emptyTrie.contains('test'), isFalse);
44+
expect(emptyTrie!.contains('test'), isFalse);
4445

45-
expect(trie.contains('algorithm'), isFalse);
46-
expect(trie.contains('algorithms'), isTrue);
46+
expect(trie!.contains('algorithm'), isFalse);
47+
expect(trie!.contains('algorithms'), isTrue);
4748

48-
expect(customTrie.contains('boar'), isFalse);
49-
expect(customTrie.contains('foo'), isTrue);
49+
expect(customTrie!.contains('boar'), isFalse);
50+
expect(customTrie!.contains('foo'), isTrue);
5051
});
5152

5253
test('Delete value', () {
53-
emptyTrie.delete('');
54-
emptyTrie.delete('test');
54+
emptyTrie!.delete('');
55+
emptyTrie!.delete('test');
5556

56-
expect(trie.contains('algorithms'), isTrue);
57-
trie.delete('algorithms');
58-
expect(trie.contains('algorithms'), isFalse);
57+
expect(trie!.contains('algorithms'), isTrue);
58+
trie!.delete('algorithms');
59+
expect(trie!.contains('algorithms'), isFalse);
5960

60-
expect(customTrie.contains('foo'), isTrue);
61-
customTrie.delete('foo');
62-
expect(customTrie.contains('foo'), isFalse);
61+
expect(customTrie!.contains('foo'), isTrue);
62+
customTrie!.delete('foo');
63+
expect(customTrie!.contains('foo'), isFalse);
6364
});
6465
}

0 commit comments

Comments
 (0)