3.2 模板双向排序链表

C++ 精简教程 目录(必读)

本文素材来自于 内华达大学拉斯维加斯分校

双向排序链表

双向排序链表是一个双链表

在插入元素的时候需要保持元素的大小顺序

如果是字符,字符串,就按照字母表顺序

如果是数值类型, 就按照数值大小排序

双向链表的功能

(1)插入一个元素insert,保持元素升序排列

(2)删除指定值的所有元素 searchDelete

(3)顺序输出所有元素 print

(4)逆序输出所有元素 backwardsPrint

双链表模板类

我们的双链表类可以存储任意类型的元素。

比如,元素类型可以是 int类型;也可以是 string 类型;

定义双链表的时候只需要指定一个具体的类型,就会只存放这种类型的元素。

就像std::vector, std::list, std::map 这些标准库容器一样。

这种类在C++中可以定义成模板类。

创建双链表模板类

template<typename T>
class List
{
};
定义一个双链表类

上面的 template<typename T> 表示List这个类是一个模板类。

其中, typename T 表示 List类内部可以声明 T 类型的变量。

当我们创建 List对象的时候,需要给T指定具体类型。

例如:

List<int> listInt;//定义一个存放int类型数据的双链表对象
List<string> listString;//定义一个存放 string 类型数据的双链表对象
List<Student> listStudent;//定义一个存放 Student 类型数据的双链表对象

在定义模板类的时候需要template<typename T>,在类外部定义模板类的成员函数时,也需要。见后文完整的代码和测试用例部分。

管理元素的成员变量

元素的存放需要通过成员变量来管理

添加头节点 head(其next指向第一个节点的指针,一开始其next设置为空指针)。

添加尾节点 tail(其next指向最后一个节点的指针,一开始其next设置为空指针)。

注意:为何是头结点,尾结点,而不是头指针呢?

这是因为为了代码好写。

例如,当我们希望在当前节点 p 的前面插入一个新节点 new_node 的时候,我们希望可以做下面的操作:

p->prev->next = new_node;

可以看出来,我们的 p->prev 这个节点必须要存在, 只有存在才能访问。

但是我们不想去判断存在与否,这就必须要求 p->prev 这个节点一开始就存在。

为了做到这一点,我们只需要给 List 添加头节点 head 即可。

所以,一个空的链表就像下面这样:

class List
{
    Node m_head;//注意不是指针类型,所以可以直接访问该节点
    Node m_tail;//注意不是指针类型,所以可以直接访问该节点
};
一个空的双向链表,带有头节点和尾节点

其中,节点Node类型如下:

节点Node类型

struct Node
{
    Node() {}
	Node(T t):data(t){ }
	T data;
	Node* pre = nullptr;
	Node* next = nullptr;
};

添加一个元素

添加一个元素需要先找到插入位置,因为链表是有序的,所以需要先通过遍历查找到合适的插入位置。

假如这个位置是一个由Node* right指向的结点,我们需要把元素插入到这个节点之前。

查找 right 的过程如下:

Node* right = head.next;
while (right != &tail && right->data < p->data)
{
	right = right->next;
}
查找插入位置run(right)

下面是具体的插入过程(用right表示上面的run):

添加一个内部节点(不是push_back的情形),在j前面插入i

输出双链表的所有元素

输出双链表的所有元素,只需要创建一个指针变量,从指向第一个元素开始,循环指向下一个节点

每遇到一个节点就先输出这个节点的内容,并输出一个分隔符。

当移动到tail节点的时候,停止循环。

顺序打印print

Node* p = head.next;
while (p != &tail)
{
	cout << p->data<<" ";
	p = p->next;
}

逆序输出双链表的所有元素

逆序输出的逻辑和正序输出的print很像。

只不过是从最后一个节点开始往前打印。

结束的条件是打印到了head,就不用打印了。

逆序打印backwardsPrint

void List::backwardsPrint(void) const
{
    Node* p = tail.next;
    while (p != nullptr && p != &head)
    {
        cout  << p->data<< " ";
        //(1) your code

    }
    cout <<"." << endl;
}

删除给定值的元素

删除指定的元素,一般的情形如下图所示:

找到待删除的元素,用p指向;q 指向 待删除元素的后继。

删除元素 j

具体的删除过程:

删除中间元素 j

删除元素的实现

(1)使用循环遍历每个节点,如果节点的值是要删除的值,就删除这个节点

(2)遍历节点的指针往后移动一个节点到下一个节点

template<typename T>
void DoublyLinkedList<T>::searchDeleteLoop(T t)
{
	Node* p = head.next;
	while (p)
	{
		if (p->data == t)
		{
			//p=p->next is finished inner
                        //(2) your code

		}
		else
		{
			p = p->next;
		}
	}
}

这其中删除一个节点的动作由deleteNode来完成,具体实现如下:

//------下面的代码是用来测试你的代码有没有问题的辅助代码,你无需关注------
#include <algorithm>
#include <cstdlib>
#include <iostream> 
#include <vector>
#include <utility>
using namespace std;
struct Record { Record(void* ptr1, size_t count1, const char* location1, int line1, bool is) :ptr(ptr1), count(count1), line(line1), is_array(is) { int i = 0; while ((location[i] = location1[i]) && i < 100) { ++i; } }void* ptr; size_t count; char location[100] = { 0 }; int line; bool is_array = false; bool not_use_right_delete = false; }; bool operator==(const Record& lhs, const Record& rhs) { return lhs.ptr == rhs.ptr; }std::vector<Record> myAllocStatistic; void* newFunctionImpl(std::size_t sz, char const* file, int line, bool is) { void* ptr = std::malloc(sz); myAllocStatistic.push_back({ ptr,sz, file, line , is }); return ptr; }void* operator new(std::size_t sz, char const* file, int line) { return newFunctionImpl(sz, file, line, false); }void* operator new [](std::size_t sz, char const* file, int line)
{
	return newFunctionImpl(sz, file, line, true);
}void operator delete(void* ptr) noexcept { Record item{ ptr, 0, "", 0, false }; auto itr = std::find(myAllocStatistic.begin(), myAllocStatistic.end(), item); if (itr != myAllocStatistic.end()) { auto ind = std::distance(myAllocStatistic.begin(), itr); myAllocStatistic[ind].ptr = nullptr; if (itr->is_array) { myAllocStatistic[ind].not_use_right_delete = true; } else { myAllocStatistic[ind].count = 0; }std::free(ptr); } }void operator delete[](void* ptr) noexcept { Record item{ ptr, 0, "", 0, true }; auto itr = std::find(myAllocStatistic.begin(), myAllocStatistic.end(), item); if (itr != myAllocStatistic.end()) { auto ind = std::distance(myAllocStatistic.begin(), itr); myAllocStatistic[ind].ptr = nullptr; if (!itr->is_array) { myAllocStatistic[ind].not_use_right_delete = true; } else { myAllocStatistic[ind].count = 0; }std::free(ptr); } }
#define new new(__FILE__, __LINE__)
struct MyStruct { void ReportMemoryLeak() { std::cout << "Memory leak report: " << std::endl; bool leak = false; for (auto& i : myAllocStatistic) { if (i.count != 0) { leak = true; std::cout << "leak count " << i.count << " Byte" << ", file " << i.location << ", line " << i.line; if (i.not_use_right_delete) { cout << ", not use right delete. "; }	cout << std::endl; } }if (!leak) { cout << "No memory leak." << endl; } }~MyStruct() { ReportMemoryLeak(); } }; static MyStruct my; void check_do(bool b, int line = __LINE__) { if (b) { cout << "line:" << line << " Pass" << endl; } else { cout << "line:" << line << " Ohh! not passed!!!!!!!!!!!!!!!!!!!!!!!!!!!" << " " << endl; exit(0); } }
#define check(msg)  check_do(msg, __LINE__);
//------上面的代码是用来测试你的代码有没有问题的辅助代码,你无需关注------

#include <iostream>
#include <string>
#include <fstream>
#include <sstream>
#include <cassert>
using namespace std;

template<typename T>
class DoublyLinkedList
{
	struct Node
	{
		Node() {}
		Node(T t) :data(t) {}
		T data;
		Node* prev = nullptr;
		Node* next = nullptr;
	};
public:
	DoublyLinkedList();
	~DoublyLinkedList();
	void insert(T t);
	string print(void) const;
	void backwardsPrint(void) const;

	void searchDelete(T t);
private:
	//当前类不支持复制,仅限于单个对象独立使用,主要是用于降低难度
	DoublyLinkedList<T>(const DoublyLinkedList<T>& from) = delete;
	//当前类不支持复制,仅限于单个对象独立使用,主要是用于降低难度
	DoublyLinkedList<T>& operator=(const DoublyLinkedList<T>& from) = delete;

	void searchDeleteLoop(T t);
	void deleteNode(Node*& p);

	Node m_head;
	Node m_tail;
};

template<typename T>
void DoublyLinkedList<T>::deleteNode(Node*& p)
{
	//p is not nullptr
	assert(p);
	Node* q = p->next;
	//1 inner node
	/*(1)*/ p->prev->next = p->next;
	/*(2)*/ p->next->prev = p->prev;
	delete p;
	p = q;//p指向被删除节点的下一个节点
}
template<typename T>
void DoublyLinkedList<T>::backwardsPrint(void) const
{
	Node* p = m_tail.prev;
	while (p != nullptr && p != &m_head)
	{
		cout << p->data << " ";
		//(1) your code


	}
	cout << endl;
}
template<typename T>
void DoublyLinkedList<T>::searchDeleteLoop(T t)
{
	Node* p = m_head.next;
	while (p)
	{
		if (p->data == t)
		{
			//p=p->next is finished inner
			//(2) your code

		}
		else
		{
			p = p->next;
		}
	}
}

template<typename T>
void DoublyLinkedList<T>::searchDelete(T t)
{
	searchDeleteLoop(t);
}
template<typename T>
inline DoublyLinkedList<T>::DoublyLinkedList()
{
	//(3) your code


}
template<typename T>
DoublyLinkedList<T>::~DoublyLinkedList()
{
	Node* p = m_head.next;
	while (p != &m_tail)
	{
		Node* q = p;
		p = p->next;
		delete q;
	}
	m_head.next = &m_tail;
	m_tail.prev = &m_head;
}
template<typename T>
void DoublyLinkedList<T>::insert(T t)
{
	Node* p = new Node(t);

	Node* right = m_head.next;
	while (right != &m_tail && right->data < p->data)
	{
		right = right->next;
	}
	//inner insert
	/*(1)*/ right->prev->next = p;
	/*(2)*/ p->prev = right->prev;
	/*(3)*/ p->next = right;
	//(4) your code

}

template<typename T>
string DoublyLinkedList<T>::print(void) const
{
	ostringstream oss;
	Node* p = m_head.next;
	while (p != &m_tail)
	{
		oss << p->data << " ";
		p = p->next;
	}
	string result = oss.str();
	return result;
}


int main()
{
	DoublyLinkedList<int> dlist;
	dlist.insert(5);
	check(dlist.print() == "5 ");
	dlist.insert(2);
	check(dlist.print() == "2 5 ");
	dlist.insert(8);
	check(dlist.print() == "2 5 8 ");
	dlist.insert(7);
	check(dlist.print() == "2 5 7 8 ");
	dlist.insert(7);
	check(dlist.print() == "2 5 7 7 8 ");
	dlist.insert(8);
	check(dlist.print() == "2 5 7 7 8 8 ");
	dlist.insert(9);
	check(dlist.print() == "2 5 7 7 8 8 9 ");
	dlist.backwardsPrint();
	dlist.searchDelete(2);
	check(dlist.print() == "5 7 7 8 8 9 ");
	dlist.backwardsPrint();
	dlist.searchDelete(9);
	check(dlist.print() == "5 7 7 8 8 ");
	dlist.backwardsPrint();
	dlist.searchDelete(7);
	check(dlist.print() == "5 8 8 ");
	dlist.insert(7);
	check(dlist.print() == "5 7 8 8 ");
	dlist.searchDelete(7);
	check(dlist.print() == "5 8 8 ");
	dlist.searchDelete(5);
	check(dlist.print() == "8 8 ");
	dlist.searchDelete(8);
	check(dlist.print() == "");
	dlist.insert(7);
	check(dlist.print() == "7 ");
	dlist.backwardsPrint();
	return 0;
}

预期输出:

加油吧!祝你好运!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
Opencv3.2中的模板匹配函数是`cv2.matchTemplate()`。该函数用于在图像中寻找与给定模板最匹配的区域。 函数的语法如下: ``` cv2.matchTemplate(image, template, method[, result[, mask]]) ``` 参数解释: - `image`:待搜索的输入图像。 - `template`:要匹配的模板图像。 - `method`:匹配方法,指定了用于比较模板和输入图像的算法。常用的方法有: - `cv2.TM_SQDIFF`:平方差匹配法。 - `cv2.TM_SQDIFF_NORMED`:归一化平方差匹配法。 - `cv2.TM_CCORR`:相关匹配法。 - `cv2.TM_CCORR_NORMED`:归一化相关匹配法。 - `cv2.TM_CCOEFF`:相关系数匹配法。 - `cv2.TM_CCOEFF_NORMED`:归一化相关系数匹配法。 - `result`:可选参数,输出的匹配结果图像。 - `mask`:可选参数,掩码图像。如果提供了掩码图像,则只有掩码对应位置的像素才参与匹配。 函数返回一个包含匹配结果的灰度图像。你可以使用`cv2.minMaxLoc()`函数找到最佳匹配位置的坐标。 示例代码: ```python import cv2 import numpy as np # 读取输入图像和模板图像 image = cv2.imread('image.jpg') template = cv2.imread('template.jpg') # 进行模板匹配 result = cv2.matchTemplate(image, template, cv2.TM_CCOEFF_NORMED) # 找到最佳匹配位置的坐标 min_val, max_val, min_loc, max_loc = cv2.minMaxLoc(result) top_left = max_loc bottom_right = (top_left[0] + template.shape[1], top_left[1] + template.shape[0]) # 在原图像中标记匹配区域 cv2.rectangle(image, top_left, bottom_right, (0, 255, 0), 2) # 显示结果图像 cv2.imshow('Matched Image', image) cv2.waitKey(0) cv2.destroyAllWindows() ``` 这是一个简单的模板匹配示例,你可以根据自己的需求进行更复杂的操作和参数调整。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

C++程序员Carea

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值