c++ 具有unique_ptr的单链表

bzzcjhmw  于 2022-11-27  发布在  其他
关注(0)|答案(1)|浏览(279)

我正在尝试使用智能指针(std::unique_ptr)来创建一个单向链表。下面是一个带有原始指针的单向链表的例子。

  1. struct Node {
  2. int data;
  3. Node *next = nullptr;
  4. Node(int data) : data{data}, next{nullptr} {}
  5. ~Node() { std::cout << "Destroy node with data: " << data << '\n'; }
  6. };
  7. void print_list(Node *head) {
  8. while (head != nullptr) {
  9. cout << head->data << " --> ";
  10. head = head->next;
  11. }
  12. cout << "nullptr" << std::endl;
  13. }
  14. void insert(Node *&head, int data) {
  15. Node *new_node = new Node{data};
  16. new_node->next = head;
  17. head = new_node;
  18. }
  19. int main(int argc, char *argv[]) {
  20. Node *head = nullptr;
  21. for (int i = 0; i < 5; ++i) {
  22. insert(head, i);
  23. }
  24. print_list(head);
  25. return 0;
  26. }

输出为:

  1. 4 --> 3 --> 2 --> 1 --> 0 --> nullptr

显然上面的代码中存在内存泄漏(没有调用析构函数)。现在我想用智能指针来实现同样的事情:

  1. struct Node {
  2. int data = 0;
  3. std::unique_ptr<Node> next;
  4. Node(int data) : data{data}, next{nullptr} {}
  5. ~Node() { std::cout << "Destroy node with data: " << data << '\n'; }
  6. };
  7. void print_list(std::unique_ptr<Node> head) {
  8. while (head != nullptr) {
  9. std::cout << head->data << " --> ";
  10. head = std::move(head->next);
  11. }
  12. std::cout << "nullptr" << std::endl;
  13. }
  14. void insert(std::unique_ptr<Node> &&head, int data) {
  15. std::unique_ptr<Node> new_node{std::make_unique<Node>(data)};
  16. new_node->next = std::move(head);
  17. head = std::move(new_node);
  18. }
  19. // g++ -std=c++17 -Wall 2_1.cpp && ./a.out
  20. int main(int argc, char *argv[]) {
  21. std::unique_ptr<Node> head{nullptr};
  22. for (int i = 0; i < 5; ++i) {
  23. insert(std::move(head), i);
  24. }
  25. print_list(std::move(head));
  26. return 0;
  27. }

输出为:

  1. 4 --> Destroy node with data: 4
  2. 3 --> Destroy node with data: 3
  3. 2 --> Destroy node with data: 2
  4. 1 --> Destroy node with data: 1
  5. 0 --> Destroy node with data: 0
  6. nullptr

我们可以观察到new_node的生命周期在insert()返回时结束,我想知道是否可以使用智能指针实现单向链表,并保留上面的函数接口。

xmakbtuz

xmakbtuz1#

首先,您的print_list实现存在问题(仅适用于unique_ptr的两个版本)。对于print_list,每次将head分配给不同的uniq_ptr时,实际上是在释放head中唯一的Node,这是不希望的。相反,在print_list中,您应该首先创建一个指向head的临时指针,然后只对临时指针进行迭代。
现在到了你的unique_ptr版本,你不需要把unique_ptr作为右值引用传递,你也可以把它作为左值引用传递。

  1. void print_list(const std::unique_ptr<Node>& head);
  2. void insert(std::unique_ptr<Node> &head, int data);

这允许您在不使用main中的std::move的情况下调用它们。
现在来看看定义。对于插入,你需要先用给定的值创建一个新的Node,然后把旧的head赋给新节点的next,并把新节点设为新的head

  1. void insert(std::unique_ptr<Node> &head, int data)
  2. {
  3. // Use `auto` to avoid typing `....<Node>` twice
  4. auto new_node = std::make_unique<Node>(data);
  5. new_node->next = std::move(head);
  6. head = std::move(new_node);
  7. }

或者,您也可以在Node的构造函数中再添加一个参数:

  1. Node(int data, std::unique_ptr<Node>&& next = nullptr)
  2. : data{data}, next{std::move(next)}
  3. {}

现在,您可以简单地创建new_node,如下所示:

  1. void insert(std::unique_ptr<Node> &head, int data)
  2. {
  3. // No need to assign `Node::next` separately
  4. auto new_node = std::make_unique<Node>(data, std::move(head));
  5. head = std::move(new_node);
  6. }

或者直接将新节点分配给head

  1. void insert(std::unique_ptr<Node> &head, int data)
  2. {
  3. head = std::make_unique<Node>(data, std::move(head));
  4. }

对于print_list,我们应该首先创建一个指向head底层对象的临时指针,然后通过将临时指针赋给下一个对象的底层对象来迭代列表:

  1. void print_list(const std::unique_ptr<Node>& head)
  2. {
  3. // Create a pointing to the underlying object
  4. // You can use `.get()` to get the underlying pointer
  5. auto current = head.get();
  6. // No need to explicit compare pointer types to `nullptr`
  7. while (current) {
  8. std::cout << current->data << " --> ";
  9. // Make `current` point to the next underlying object
  10. current = current->next.get();
  11. }
  12. std::cout << "nullptr" << std::endl;
  13. }

现在,您的main将如下所示:

  1. int main(int, char *[]) {
  2. std::unique_ptr<Node> head;
  3. for (int i = 0; i < 5; ++i) {
  4. insert(head, i);
  5. }
  6. print_list(head);
  7. return 0;
  8. }

Demo

展开查看全部

相关问题