- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我想创建一个类来管理 C++ 中的标记语言(例如 HTML)。我希望我的类(class)保留属性和子标签。问题是,给定封装的容器,如何正确抽象访问和返回什么,以便提供一种简单的方法来检查返回的值是否有效。
我将包含两个映射的类定义为私有(private)成员(名义上是 std::map<std::string, Tag> _children;
和 std::map<std::string, std::string> _attr;
)。我定义了两个函数来填充这些字段,我想定义两个函数来读取访问存储的元素。
问题是,我不想破坏我的抽象,因为我这样做是为了提高我的 c++ 技能,我想找到正确的方法(或更简洁的方法,或标准方法)去做。
一个基本的解决方案是简单地调用 return map.find(s);
,但是我必须将函数的返回类型定义为 std::map<std::string, Tag>::const_iterator
,这会破坏抽象。所以我可以取消引用 map.find()
返回的迭代器,但如果值不在 map 中,我将取消引用不可取消引用的迭代器 ( _children.cend()
)。
到目前为止我定义了什么:
using namespace std;
class Tag {
static const regex re_get_name, re_get_attributes;
string _name;
map<string,string> _attr;
map<string,Tag> _children;
public:
Tag(const string &toParse) {
/* Parse line using the regex */
}
const string& name() const {
return _name;
}
Tag& add_child(const Tag& child) {
_children.insert(child._name, child);
return *this;
}
SOMETHING get_child(const string& name) const {
map<string,Tag>::const_iterator val = _children.find(name);
/* Do something here, but what ? */
return something;
}
SOMETHING attr(const string& name) const {
map<string, string>::const_iterator val = _attr.find(name);
/* Do something here, but what ? */
return something;
}
};
const regex Tag::re_get_name("^<([^\\s]+)");
const regex Tag::re_get_attributes(" ([^\\s]+) = \"([^\\s]+)\"");
在 C++ 中处理这种情况的正确方法是什么?我应该创建自己的 Tag::const_iterator
吗?类型?如果是这样,怎么办?我是否应该采用更“C”的方法,我只将返回类型定义为 Tag&
并返回 NULL
如果 map 不包含我的 key ?我是否应该使用静态成员更面向对象 static const Tag NOT_FOUND
,如果该元素不在我的 map 中,则返回对此对象的引用?我也想过抛出异常,但是异常管理在C++中似乎相当繁重且效率低下。
最佳答案
std::optional
可以帮助你,但需要一个 C++17 就绪的标准库,所以与此同时你也可以使用 boost::optional
这是或多或少是一样的,因为 AFAIK std::optional
的设计是基于 boost 的。 (因为 boost 通常是新 C++ 标准提案的来源)
尽管由于您的方法存在普遍问题,我不愿意向您提出建议,但我还是为您写了一个,但请考虑代码后的要点:
#include <string>
#include <regex>
#include <map>
#include <boost/optional.hpp>
class Tag {
static const std::regex re_get_name, re_get_attributes;
using string = std::string;
string _name;
std::map<string,string> _attr;
std::map<string,Tag> _children;
public:
Tag(const string &toParse) {
/* Parse line using the regex */
}
const string& name() const {
return _name;
}
Tag& add_child(const Tag& child) {
_children.emplace(child._name, child);
return *this;
}
boost::optional<Tag> get_child(const string& name) const {
auto val = _children.find(name);
return val == _children.cend() ? boost::optional<Tag>{} : boost::optional<Tag>{val->second};
}
boost::optional<string> attr(const string& name) const {
auto val = _attr.find(name);
return val == _attr.cend() ? boost::optional<string>{} : boost::optional<string>{val->second};
}
};
如您所见,您基本上只是在重新实现 std::map
的容器语义,而且还使用某种内置的解析器逻辑。我强烈反对这种方法,因为解析很快就会变得非常丑陋,并且将值生成代码混合到一个容器中,即应该用作值类会使事情变得更糟。
我的第一个建议是只声明/使用你的 Tag
类/结构作为值类,所以只包含 std::maps 作为公共(public)成员。将您的解析函数与 Tag 容器一起放在命名空间中,并在需要时让它们只是函数或不同的类。
我的第二个建议是小建议:不要以_
为前缀,它是保留的并且被认为是不好的风格,但您可以将它用作后缀。也不要在类/函数/命名空间 block 之外使用 using 命名空间指令,即全局,它在 .cpp 中是糟糕的样式,在 header /.h/.hpp 中是非常糟糕的样式
我的第三个建议:使用 boost spirit qi 解析器框架,您只需按照我的建议首先声明您的值类,而 qi 会通过 boost fusion 自动填充它们。如果你已经知道 EBNF 表示法,你可以在 C++ 中编写类似 EBNF 的语法,编译器将通过模板魔法生成一个解析器。然而,气,尤其是融合存在一些问题,但从长远来看,它使事情变得容易得多。正则表达式最多只完成一半的解析逻辑。
关于c++ - 在不破坏 C++ 抽象的情况下处理对存储在私有(private)映射中的值的封装访问的标准方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54294044/
如果需要在类外访问静态(例如单例),可以选择公共(public)静态而不是私有(private)静态,而当不需要公开函数时首选私有(private)静态(否则未命名的命名空间就可以了)——在这种情况下
在互联网上进行了一些搜索,但找不到简单的答案。我的问题集是在 Android 框架中使用 Java,但我相信这也是标准的 Java 行为。我理解 final 和 private 的定义,它们都用于变量
我有这个代码: public final class Board { private final int[][] blocks; private final int N; pr
对我来说,过去作为 Objective-C 开发人员很简单。一个类需要公开的每个字段都是一个属性,每个私有(private)字段都是一个没有 getter 或 setter 的实例变量。但我经常看到人
按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the
我有一个在 Docker 容器中运行的应用程序。它需要来自公司私有(private) NPM 注册表(Sinopia)的一些私有(private)模块,并且访问这些需要用户身份验证。 Dockerfi
我试图理解 C# 使用 getters 和 setters 自动声明变量与 java 声明之间的区别。 在java中我通常这样做: private int test; public int getTe
我在 Azure 中创建了 VNET。我放入了一个子集 Azure Private Link,它在 VNET 之外和另一台虚拟机中调用 Azure Function。 当我尝试通过专用 IP 调用专用
我在 Azure 中创建了 VNET。我放入了一个子集 Azure Private Link,它在 VNET 之外和另一台虚拟机中调用 Azure Function。 当我尝试通过专用 IP 调用专用
我目前正在使用 Objective-C(适用于 iPhone)构建游戏。 为此,出于性能/复杂性原因,我略微打破了 MVC,并为 View (渲染器)提供了对模型的直接引用。这是因为它应该以 60fp
我已经在 ubuntu 上成功配置了 2 个虚拟主机站点(基于名称的虚拟主机)。我的 apache 版本是 2.2.22。 这两个站点都在本地主机上工作。 /etc/hosts 条目 127.0.0.
考虑下面的类 public class A { private final Map cache; public HeavyObject getThing(); } 假设不能泄漏对缓存
我有一个类,它有一个方法,我希望它只能被它的子对象访问,而不能被这个包中的其他类访问。 Modifier | Class | Package | Subclass | World ———————
本文实例讲述了JavaScript中的公有、私有、特权和静态成员用法。分享给大家供大家参考。具体分析如下: 下面的内容是在《JavaScript.DOM高级程序设计》里面摘抄出来的,比较容易理解,
我有一个用例,我已将其简化为以下程序: public class A { private int x = 100; class B { private int y = ne
问题: 类声明如下: class Select { public: template static Iterator function(Iterator , Iterator , bo
我是一名初级 PHP 程序员。我还有很多东西要学。这就是我问这个问题的原因。在一个类中,您有一个公共(public)函数,您可以从该类外部调用它。有时你有一个私有(private)函数,你可以在私有(
问题是: 何时使用私有(private)函数,何时使用嵌套函数? (我在问 F# 但也许答案可能与其他功能语言相关) 一个小例子 namespace SomeName module BinaryRea
我发现工作表中仍然可以使用私有(private)函数。它们是隐藏的,但如果用户输入他们的名字,他们就会被调用。为什么?它应该以这种方式工作吗?有没有办法完全阻止用户定义的函数在 VBA 项目之外使用?
所以我最近开始尝试使用 Kotlin,我偶然发现了这个: If a top-level declaration is marked private, it is private to the pack
我是一名优秀的程序员,十分优秀!