结构型设计模式—组合模式

结构型设计模式—组合模式

欢迎长按图片加好友,我会第一时间和你分享持续更多的开发知识,面试资源,学习方法等等。

组合模式(Composite Pattern)是一种结构型设计模式,它允许你将对象组合成树形结构来表示“部分-整体”的层次结构。通过这种模式,客户端可以统一地处理单个对象和对象组合。

想象一下,你家里有各种电器,比如电视、空调、冰箱、洗衣机等。这些电器可以分为不同的房间,如客厅、卧室、厨房等。每个房间可能有多个电器,而整个家就是由这些房间和电器组成的。

  • 叶子对象:每个电器(如电视、冰箱)就是一个叶子对象,它们没有子对象。
  • 组合对象:每个房间(如客厅、卧室)就是一个组合对象,它包含了多个电器。
  • 根对象:整个家庭就是一个顶层的组合对象,它包含了多个房间。

通过组合模式,你可以将家庭结构视为一个整体。无论是处理单个电器(叶子对象),还是处理整个家庭(根对象),你都可以使用统一的方式。例如,你可以写一个功能来关闭所有电器,无论是关闭单个电器,还是关闭某个房间里的所有电器,或者关闭整个家里的所有电器,这个功能的实现方式都是一样的。

组合模式概述

对于树形结构,当容器对象(例如文件夹)的某一个方法被调用时,将遍历整个树形结构,寻找也包含这个方法的成员对象(可以是容器对象,也可以是叶子对象)并调用执行,牵一而动百,其中使用了递归调用的机制来对整个结构进行处理。由于容器对象和叶子对象在功能上的区别,在使用这些对象的代码中必须有区别地对待容器对象和叶子对象,而实际上大多数情况下希望一致地处理它们,因为对于这些对象的区别对待将会使得程序非常复杂。组合模式为解决此类问题而诞生,它可以让叶子对象和容器对象的使用具有一致性。

组合模式定义如下:

组合模式(Composite Pattern):**组合多个对象形成树形结构以表示具有“部分—整体”关系的层次结构。**组合模式对单个对象(即叶子对象)和组合对象(即容器对象)的使用具有一致性,又可以称为“部分—整体”(Part-Whole)模式,它是一种对象结构型模式。

在组合模式中引入了抽象构件类Component,它是所有容器类和叶子类的公共父类,客户端针对Component进行编程。组合模式结构如图所示。

在组合模式结构图中包含以下3个角色:

  1. Component(抽象构件):它可以是接口或抽象类,为叶子构件和容器构件对象声明接口,在该角色中可以包含所有子类共有行为的声明和实现。在抽象构件中定义了访问及管理它的子构件的方法,例如增加子构件、删除子构件、获取子构件等。
  2. Leaf(叶子构件):它在组合模式结构中表示叶子节点对象。叶子节点没有子节点,它实现了在抽象构件中定义的行为。对于那些访问及管理子构件的方法,可以通过捕获异常等方式进行处理。
  3. Composite(容器构件):它在组合模式结构中表示容器节点对象。容器节点包含子节点,其子节点可以是叶子节点,也可以是容器节点。它提供一个集合用于存储子节点,实现了在抽象构件中定义的行为,包括那些访问及管理子构件的方法,在其业务方法中可以递归调用其子节点的业务方法。

组合模式的关键是定义了一个抽象构件类,它既可以代表叶子,又可以代表容器。客户端针对该抽象构件类进行编程,无须知道它到底表示的是叶子还是容器,可以对其进行统一处理。同时容器对象与抽象构件类之间还建立一个聚合关联关系,在容器对象中既可以包含叶子,也可以包含容器,以此实现递归组合,形成一个树形结构。

如果不使用组合模式,客户端代码将过多地依赖于容器对象复杂的内部实现结构。容器对象内部实现结构的变化将引起客户代码的频繁变化,从而带来了代码维护复杂、可扩展性差等弊端。组合模式的引入将在一定程度上解决这些问题。

下面通过简单的示例代码来分析组合模式中各个角色的用途和实现。

  1. Component(组件):定义了组合对象和叶子对象的共同接口或抽象类。所有对象(包括组合对象和叶子对象)都需要实现该接口或继承该抽象类。
  2. Leaf(叶子节点):叶子节点是组合中的基本元素,没有子节点。在你的案例中,具体的电器(如 TV 和 AirConditioner)就是叶子节点。
  3. Composite(组合节点):组合节点包含子节点,可以是叶子节点或其他组合节点。它实现了 Component 的接口,并且可以包含其他组件(包括叶子和组合)。在你的案例中,房间(Room)和房子(House)就是组合节点。
// 1. Component(组件):定义了组合和叶子节点的共同接口或抽象类
abstract class Appliance {public abstract void turnOff();
}// 2. Leaf(叶子节点):表示具体的电器,如 TV 和 AirConditioner,没有子节点
class TV extends Appliance {@Overridepublic void turnOff() {System.out.println("Turning off the TV");}
}class AirConditioner extends Appliance {@Overridepublic void turnOff() {System.out.println("Turning off the Air Conditioner");}
}// 3. Composite(组合节点):包含子节点的组合对象,既可以包含叶子节点,也可以包含其他组合节点
class Room extends Appliance {private List<Appliance> appliances = new ArrayList<>();public void addAppliance(Appliance appliance) {appliances.add(appliance);}@Overridepublic void turnOff() {for (Appliance appliance : appliances) {appliance.turnOff();}}
}// 另一个 Composite(组合节点):表示整个家,可以包含多个房间(Room)
class House extends Appliance {private List<Appliance> rooms = new ArrayList<>();public void addRoom(Room room) {rooms.add(room);}@Overridepublic void turnOff() {for (Appliance room : rooms) {room.turnOff();}}
}// 4. 客户端代码
public class CompositePatternDemo {public static void main(String[] args) {// 创建房间及其电器Room livingRoom = new Room();livingRoom.addAppliance(new TV());livingRoom.addAppliance(new AirConditioner());Room bedroom = new Room();bedroom.addAppliance(new TV());// 创建家并添加房间House house = new House();house.addRoom(livingRoom);house.addRoom(bedroom);// 关闭整个家中的所有电器house.turnOff();}
}
  1. Component(组件):Appliance 抽象类是 Component 角色,它定义了 turnOff() 方法,这是叶子节点和组合节点的共同接口。
  2. Leaf(叶子节点):TV 和 AirConditioner 类是 Leaf 角色,它们实现了 Appliance 的 turnOff() 方法,并且没有子节点。
  3. Composite(组合节点):Room 和 House 类是 Composite 角色,它们实现了 Appliance 的 turnOff() 方法,并包含了其他 Appliance 对象作为子节点。它们可以包含 Leaf 角色或其他 Composite 角色。

通过这个代码示例,你可以清楚地看到组合模式中的三个角色是如何协同工作的。客户端可以通过 Component 接口统一操作单个对象(叶子节点)或组合对象(组合节点)。

透明组合模式和安全组合模式

透明组合模式和安全组合模式的存在,是因为开发者在设计软件时需要在灵活性安全性之间做出权衡。这两种变体的出现,反映了不同的设计需求和设计哲学。透明组合模式安全组合模式是组合模式的两种变体,它们处理组件的方式略有不同,主要区别在于如何管理组合对象和叶子对象。

透明组合模式(Transparent Composite Pattern)

透明组合模式的设计初衷是为了简化客户端的使用,让客户端不必区分是操作叶子节点还是组合节点,统一使用同一套接口进行操作。

  • 透明性:在透明组合模式中,叶子节点和组合节点都实现了相同的接口,包括对组合节点有意义的操作(如 add()remove())。这样,客户端代码可以对组件进行统一的操作,而不需要区分当前操作的对象是叶子节点还是组合节点。
  • 简化客户端代码:由于客户端可以忽略对象的类型(叶子节点还是组合节点),可以更加方便地操作对象树结构。比如,客户端可以遍历整个树结构,执行 turnOff() 操作而不关心对象是叶子还是组合。
  • 灵活性:透明组合模式非常灵活,能够处理复杂的对象层次结构,特别是在需要频繁操作和修改对象结构的场景中。

但是同时透明组合模式的灵活性是以接口的安全性为代价的。叶子节点实现了组合操作(如 add()remove()),尽管这些操作对它们无意义。这可能导致在客户端误用这些操作,从而产生运行时错误。

下面通过简单的示例代码来分析透明组合模式中各个角色的用途和实现。

在透明组合模式中,Component 接口中包含了组合操作的方法(如 add、remove),叶子节点和组合节点都实现这些方法。

// Component(组件):定义了组合对象和叶子节点的共同接口
abstract class Appliance {// 透明组合模式中,add 和 remove 方法在 Component 中定义// 叶子节点将实现这些方法,但它们在叶子节点中没有实际意义public void add(Appliance appliance) {throw new UnsupportedOperationException();}public void remove(Appliance appliance) {throw new UnsupportedOperationException();}public abstract void turnOff();
}// Leaf(叶子节点):表示具体的电器,如 TV 和 AirConditioner,没有子节点
class TV extends Appliance {@Overridepublic void turnOff() {System.out.println("Turning off the TV");}
}class AirConditioner extends Appliance {@Overridepublic void turnOff() {System.out.println("Turning off the Air Conditioner");}
}// Composite(组合节点):包含子节点的组合对象,既可以包含叶子节点,也可以包含其他组合节点
class Room extends Appliance {private List<Appliance> appliances = new ArrayList<>();@Overridepublic void add(Appliance appliance) {appliances.add(appliance);}@Overridepublic void remove(Appliance appliance) {appliances.remove(appliance);}@Overridepublic void turnOff() {for (Appliance appliance : appliances) {appliance.turnOff();}}
}// 另一个 Composite(组合节点):表示整个家,可以包含多个房间(Room)
class House extends Appliance {private List<Appliance> rooms = new ArrayList<>();@Overridepublic void add(Appliance appliance) {rooms.add(appliance);}@Overridepublic void remove(Appliance appliance) {rooms.remove(appliance);}@Overridepublic void turnOff() {for (Appliance room : rooms) {room.turnOff();}}
}// 客户端代码
public class TransparentCompositePatternDemo {public static void main(String[] args) {// 创建房间及其电器Room livingRoom = new Room();livingRoom.add(new TV());livingRoom.add(new AirConditioner());Room bedroom = new Room();bedroom.add(new TV());// 创建家并添加房间House house = new House();house.add(livingRoom);house.add(bedroom);// 关闭整个家中的所有电器house.turnOff();}
}
  • Component(组件):Appliance 是抽象类,定义了组合操作方法(addremove),叶子节点虽然实现了这些方法,但在叶子节点中它们抛出了 UnsupportedOperationException
  • Leaf(叶子节点):TV 和 AirConditioner 是叶子节点,虽然它们实现了 addremove 方法,但这些方法在叶子节点中没有实际用途。
  • Composite(组合节点):Room 和 House 是组合节点,包含了 addremove 方法,允许子节点(叶子或组合)被添加或移除,并实现了统一的 turnOff 操作。

安全组合模式

安全组合模式的设计初衷是为了提供更高的安全性和清晰的接口设计。它避免了叶子节点实现不必要的方法,从而减少了误用的风险。

  • 安全性:安全组合模式通过将组合操作(如 add()、remove())限定在组合节点中,从而避免叶子节点中出现这些无意义的操作。这种方式确保了客户端只能在组合节点上调用这些方法,而不会在叶子节点上误调用。
  • 清晰的接口设计:安全组合模式将叶子节点和组合节点的职责分离得更加明确,接口设计更加简洁,减少了误解和误用的可能性。
  • 降低出错风险:因为叶子节点没有实现组合操作,所以客户端无法误用这些方法,降低了程序出错的风险。

但是安全组合模式的严格区分增加了客户端代码的复杂性。客户端需要知道它正在处理的是叶子节点还是组合节点,从而调用不同的方法。这种情况下,客户端代码可能变得更加复杂,需要进行类型检查或使用多态来处理不同的情况。

下面通过简单的示例代码来分析安全组合模式中各个角色的用途和实现。

在安全组合模式中,Component 接口只包含通用操作(如 turnOff),组合操作(如 addremove)仅在组合节点中定义,叶子节点不需要实现这些方法。

// Component(组件):定义了通用操作,不包含组合操作
abstract class Appliance {public abstract void turnOff();
}// Leaf(叶子节点):表示具体的电器,如 TV 和 AirConditioner,没有子节点,也不实现组合操作
class TV extends Appliance {@Overridepublic void turnOff() {System.out.println("Turning off the TV");}
}class AirConditioner extends Appliance {@Overridepublic void turnOff() {System.out.println("Turning off the Air Conditioner");}
}// Composite(组合节点):包含子节点的组合对象,定义组合操作如 add 和 remove
class Room extends Appliance {private List<Appliance> appliances = new ArrayList<>();// 组合操作在 Composite 中定义,安全组合模式下,叶子节点不需要实现这些方法public void add(Appliance appliance) {appliances.add(appliance);}public void remove(Appliance appliance) {appliances.remove(appliance);}@Overridepublic void turnOff() {for (Appliance appliance : appliances) {appliance.turnOff();}}
}// 另一个 Composite(组合节点):表示整个家,可以包含多个房间(Room)
class House extends Appliance {private List<Appliance> rooms = new ArrayList<>();public void add(Room room) {rooms.add(room);}public void remove(Room room) {rooms.remove(room);}@Overridepublic void turnOff() {for (Appliance room : rooms) {room.turnOff();}}
}// 客户端代码
public class SafeCompositePatternDemo {public static void main(String[] args) {// 创建房间及其电器Room livingRoom = new Room();livingRoom.add(new TV());livingRoom.add(new AirConditioner());Room bedroom = new Room();bedroom.add(new TV());// 创建家并添加房间House house = new House();house.add(livingRoom);house.add(bedroom);// 关闭整个家中的所有电器house.turnOff();}
}
  • Component(组件):Appliance 只定义了 turnOff 方法,没有组合操作(addremove),叶子节点和组合节点都继承了这个接口。
  • Leaf(叶子节点):TV 和 AirConditioner 是叶子节点,它们只实现了 turnOff 方法,不需要实现组合操作,因此接口更加简洁明了。
  • Composite(组合节点):Room 和 House 是组合节点,它们定义了组合操作方法(addremove),并通过 turnOff 方法遍历并关闭所有子节点。

透明组合模式与安全组合模式总结

  • 透明组合模式:优先考虑灵活性和统一性,适合处理复杂的对象结构,客户端代码简单,但接口不够安全,可能导致误用。
  • 安全组合模式:优先考虑安全性和接口清晰性,避免叶子节点实现无意义的操作,减少误用风险,但客户端代码可能会更复杂。

这两种模式之间的选择,取决于具体应用场景和开发者的设计需求。如果你的应用需要大量的对象层次结构操作,且不希望让客户端区分叶子和组合对象,透明组合模式可能更合适;如果你需要确保接口的安全性,并减少误用的风险,安全组合模式则可能是更好的选择。

组合模式总结

组合模式使用面向对象的思想来实现树形结构的构建与处理,描述了如何将容器对象和叶子对象进行递归组合,实现简单,灵活性好。由于在软件开发中存在大量的树形结构,因此组合模式是一种使用频率较高的结构型设计模式。

优点

  1. 组合模式可以清楚地定义分层次的复杂对象,表示对象的全部或部分层次。它让客户端忽略了层次的差异,方便对整个层次结构进行控制。
  2. 客户端可以一致地使用一个组合结构或其中单个对象,不必关心处理的是单个对象还是整个组合结构,简化了客户端代码。
  3. 在组合模式中增加新的容器构件和叶子构件都很方便,无须对现有类库进行任何修改,符合开闭原则。
  4. 组合模式为树形结构的面向对象实现提供了一种灵活的解决方案。通过叶子对象和容器对象的递归组合,可以形成复杂的树形结构,但对树形结构的控制却非常简单。

缺点

组合模式的主要缺点是:在增加新构件时很难对容器中的构件类型进行限制。有时希望一个容器中只能有某些特定类型的对象,例如在某个文件夹中只能包含文本文件。使用组合模式时,不能依赖类型系统来施加这些约束,因为它们都来自相同的抽象层。在这种情况下,必须通过在运行时进行类型检查来实现,这个实现过程较为复杂。

适用场景

  1. 在具有整体和部分的层次结构中,希望通过一种方式忽略整体与部分的差异,客户端可以一致性地对待它们。
  2. 在一个使用面向对象语言开发的系统中需要处理一个树形结构。
  3. 在一个系统中能够分离出叶子对象和容器对象,而且它们的类型不固定,将来需要增加一些新的类型。

案例

Sunny软件公司欲开发一个界面控件库。界面控件分为两大类:一类是单元控件,例如按钮、文本框等;另一类是容器控件,例如窗体、中间面板等。试用组合模式设计该界面控件库。

我们将控件分为两大类:

  1. 单元控件:如按钮、文本框等,这些控件不能包含其他控件。
  2. 容器控件:如窗体、中间面板等,这些控件可以包含其他控件(包括单元控件和容器控件)。

角色说明

  • Component(组件):定义所有控件的通用接口,既包括单元控件也包括容器控件的通用操作。
  • Leaf(叶子节点):表示单元控件,如按钮、文本框等。这些控件没有子控件。
  • Composite(组合节点):表示容器控件,如窗体、中间面板等,可以包含其他控件(包括叶子节点和组合节点)。

下面是使用 Java 实现的界面控件库的组合模式设计:

import java.util.ArrayList;
import java.util.List;// Component(组件):定义所有控件的通用接口
abstract class UIComponent {public void add(UIComponent component) {throw new UnsupportedOperationException();}public void remove(UIComponent component) {throw new UnsupportedOperationException();}public UIComponent getChild(int index) {throw new UnsupportedOperationException();}public abstract void render(); // 渲染控件
}// Leaf(叶子节点):表示单元控件,如按钮、文本框等
class Button extends UIComponent {@Overridepublic void render() {System.out.println("Rendering Button");}
}class TextBox extends UIComponent {@Overridepublic void render() {System.out.println("Rendering TextBox");}
}// Composite(组合节点):表示容器控件,如窗体、中间面板等
class Panel extends UIComponent {private List<UIComponent> children = new ArrayList<>();@Overridepublic void add(UIComponent component) {children.add(component);}@Overridepublic void remove(UIComponent component) {children.remove(component);}@Overridepublic UIComponent getChild(int index) {return children.get(index);}@Overridepublic void render() {System.out.println("Rendering Panel");for (UIComponent component : children) {component.render(); // 递归渲染子控件}}
}class Window extends UIComponent {private List<UIComponent> children = new ArrayList<>();@Overridepublic void add(UIComponent component) {children.add(component);}@Overridepublic void remove(UIComponent component) {children.remove(component);}@Overridepublic UIComponent getChild(int index) {return children.get(index);}@Overridepublic void render() {System.out.println("Rendering Window");for (UIComponent component : children) {component.render(); // 递归渲染子控件}}
}// 客户端代码
public class CompositePatternDemo {public static void main(String[] args) {// 创建单元控件Button button1 = new Button();TextBox textBox1 = new TextBox();// 创建容器控件并添加单元控件Panel panel = new Panel();panel.add(button1);panel.add(textBox1);// 创建窗体并添加面板Window window = new Window();window.add(panel);// 渲染整个界面window.render();}
}

代码解析

  • Component(组件):UIComponent 是抽象类,定义了所有控件的通用接口,包含了组合操作(addremovegetChild)和渲染操作(render)。
  • Leaf(叶子节点):Button 和 TextBox 是单元控件,只实现了渲染操作(render),不包含组合操作(addremove 等),这些操作在 UIComponent 中抛出了 UnsupportedOperationException
  • Composite(组合节点):Panel 和 Window 是容器控件,它们实现了组合操作(addremovegetChild),并且在 render 方法中递归地渲染它们包含的所有子控件。

优点

一致性:客户端可以统一处理单元控件和容器控件,而不需要区分它们的类型。

灵活性:通过组合模式,可以轻松地构建复杂的控件层次结构。

可扩展性:可以方便地增加新的控件类型(叶子或组合)而不需要修改现有代码。

这个设计使得开发人员能够轻松地扩展控件库,并通过统一的接口来处理各种控件,极大地提高了代码的可维护性和可重用性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/53017.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

网络安全服务基础Windows--第15节-CA与HTTPS理论

公钥基础设施&#xff08;Public Key Infrastructure&#xff0c;简称 PKI&#xff09;是指⼀套由硬件、软件、⼈员、策略和程序组成的系统&#xff0c;⽤于创建、管理、分发、使⽤、存储和撤销数字证书。PKI 的核⼼⽬的是通过使⽤公钥加密技术来确保电⼦通信的安全性。PKI 为数…

Linux之grep命令

在文本文件中过滤&#xff0c;包含指定字符串的行 – grep [选项] 字符串 文本文件...• 常用命令选项 – -v&#xff0c;取反匹配 – -i&#xff0c;忽略大小写 ]# grep root /etc/passwd #包含root的行 ]# grep -v root /etc/passwd #不包含root ]# grep ROOT…

数据库中的“加速器”:深度解析索引的重要性及最佳实践

文章目录 **为什么要使用索引&#xff1f;****索引的作用&#xff1a;** **索引的分类****1. 按索引字段的个数分类****2. 按照索引值的唯一性分类** **如何定义索引&#xff1f;****1. 创建索引****语句中的关键要素&#xff1a;****创建索引的示例&#xff1a;** **2. 删除索…

【保姆级教程】使用 PyTorch 自定义卷积神经网络(CNN) 实现图像分类、训练验证、预测全流程【附数据集与源码】

《博主简介》 小伙伴们好&#xff0c;我是阿旭。专注于人工智能、AIGC、python、计算机视觉相关分享研究。 &#x1f44d;感谢小伙伴们点赞、关注&#xff01; 《------往期经典推荐------》 一、AI应用软件开发实战专栏【链接】 项目名称项目名称1.【人脸识别与管理系统开发…

Android Camera系列(二):TextureView+Camera

两岸猿声啼不住&#xff0c;轻舟已过万重山—李白 Android Camera系列&#xff08;一&#xff09;&#xff1a;SurfaceViewCamera Android Camera系列&#xff08;二&#xff09;&#xff1a;TextureViewCamera Android Camera系列&#xff08;三&#xff09;&#xff1a;GLS…

设计模式学习-简单的命令模式例子

上一章节介绍过了命令模式&#xff0c;这一篇文章就简单的做一个小案例来巩固学习 搭建场景 简单的搭建一个场景就行 &#xff0c;随便准备一个物体放在场景中位置Reset一下即可。 代码编写 定义接口&#xff08;或者抽象类&#xff09;ICommand 用来规范Command的行为。注意…

SpringCloud开发实战(三):集成Eureka注册中心

目录 SpringCloud开发实战&#xff08;一&#xff09;&#xff1a;搭建SpringCloud框架 SpringCloud开发实战&#xff08;二&#xff09;&#xff1a;通过RestTemplate实现远程调用 Eureka简介 Eureka 是一个基于 Java 的开源技术&#xff0c;最广为人知的是作为 Netflix 开发…

Java 输入与输出之 NIO.2【AIO】【Path、Paths、Files】【walkFileTree接口】探索之【三】

在JDK 1.7 版本中对NIO进行了完善&#xff0c;推出了NIO.2&#xff0c;也称为AIO&#xff08;异步IO&#xff09;&#xff0c;在处理大量并发请求时具有优势&#xff0c;特别是在网络编程和高并发场景下&#xff0c;表现得更为出色。 对于输出流和输入流而言&#xff0c;操作的…

【GIS开发小课堂】vue3+Cesium.js三维WebGIS项目实战(一)

随着市场对数字孪生的需求日益增多&#xff0c;对于前端从业者的能力从对框架vue、react的要求&#xff0c;逐步扩展到2D、3D空间的交互&#xff0c;为用户提供更紧密的立体交互。近年来前端对GIS的需求日益增多。 本文档详细介绍了使用Vue3和Cesium.js构建三维WebGIS项目的步骤…

024集—— 正则表达式、replace、DateTime日期的用法——C#学习笔记

DateTime 是一个struct结构体。 代码如下&#xff1a; using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks;namespace ConsoleApp1 {internal class Program{static void Main(string[] args){args new s…

神策SDK不支持Windows客户端全埋点,怎么实现用户统计分析?

本文将介绍&#xff0c;ClkLog针对神策不支持全埋点的客户端实现用户访问基础统计分析 1。 客户遇到的问题 ClkLog的用户访问基础统计分析功能是基于神策SDK的全埋点来实现的。 我们遇到有些客户是使用C、C#等语言来开发的客户端&#xff0c;然而神策此类SDK&#xff08;如C, C…

psql常见报错解决

问题 解决 要在管理员模式下启动 pg_ctl start -D "D:\Program\PostgreSQL\data" 注册成服务 D:\Program\PostgreSQL\bin\pg_ctl.exe register -N "postgresql" -D "D:\Program\PostgreSQL\data" -U "postgres" -P "postgre…

守护夏日清凉:EasyCVR+AI视频智能管理方案为水上乐园安全保驾护航

随着夏季的来临&#xff0c;水上乐园成为了人们避暑消夏、亲子互动的理想去处。然而&#xff0c;随着游客量的激增&#xff0c;如何确保水上乐园的安全与秩序&#xff0c;提升游客体验&#xff0c;成为了管理者亟待解决的问题。为此&#xff0c;引入一套高效、智能的视频监控方…

workman和GateWay学习笔记

前言 workman支持Udp GateWay是基于workman的二次封装&#xff0c;更适合长链接场景 windows安装workman composer create-project workerman/webman windows运行workman cd webman php windows.php windows访问 http://ip地址:8787 将workman引入thinkphp框架理念

OZON户外运动产品有哪些好卖的

Top1 运动水壶 Спортивная бутылка 780 мл 商品id&#xff1a;1613789852 月销量&#xff1a;819 OZON热销文具产品&#xff1a;m6z.cn/5H6fQR (复制浏览器打开) 780毫升的容量设计&#xff0c;既不会过于笨重&#xff0c;也能满足用户在运动或户外活…

【自动驾驶】决策规划算法 | 数学基础(二)凸优化与非凸优化

写在前面&#xff1a; &#x1f31f; 欢迎光临 清流君 的博客小天地&#xff0c;这里是我分享技术与心得的温馨角落。&#x1f4dd; 个人主页&#xff1a;清流君_CSDN博客&#xff0c;期待与您一同探索 移动机器人 领域的无限可能。 &#x1f50d; 本文系 清流君 原创之作&…

初识Linux · 有关makefile

目录 前言&#xff1a; 1 makefile的简单使用 2 makefile介绍 前言&#xff1a; 我们上文介绍了gcc和g的基本使用&#xff0c;带了许多的子指令&#xff0c;但是有的时候啊&#xff0c;一个一个敲指令确实有点麻烦了&#xff0c;此时&#xff0c;一个工具就能派上用场&…

Linux学习-虚拟化平台安装和使用

注&#xff1a;系统使用Rock8.6 下载链接 通过百度网盘分享的文件&#xff1a;cirros.qcow2&#xff0c;node_base.xml等2个文件 链接&#xff1a;https://pan.baidu.com/s/1hupGQsMjrXMgngCy3lQLhw?pwdhlr6 提取码&#xff1a;hlr6[rootharbor ~]# cat /etc/redhat-releas…

Django+Vue宠物服务管理系统的设计与实现

目录 1 项目介绍2 项目截图3 核心代码3.1 需要的环境3.2 Django接口层3.3 实体类3.4 config.ini3.5 启动类3.5 Vue 4 数据库表设计5 文档参考6 计算机毕设选题推荐7 源码获取 1 项目介绍 博主个人介绍&#xff1a;CSDN认证博客专家&#xff0c;CSDN平台Java领域优质创作者&…

2024-09-03作业

作业结果 作业代码 #include <iostream> using namespace std; class RMB { friend const RMB operator-(const RMB &L,const RMB &R); friend const RMB operator--(RMB &O,int); private: int yuan; int jiao; int fen; static…