- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我的应用程序是多线程的,具有密集的字符串处理。我们正在经历过多的内存消耗,分析表明这是由于 String 数据造成的。我认为内存消耗会从使用某种享元模式实现甚至缓存中受益匪浅(我确信字符串经常重复,尽管我没有这方面的任何硬数据)。
我看过 Java 常量池和 String.intern,但它似乎会引发一些 PermGen 问题。
在 Java 中实现应用程序范围内的多线程字符串池的最佳替代方案是什么?
编辑:另见我之前的相关问题:How does java implement flyweight pattern for string under the hood?
最佳答案
注意:此答案使用的示例可能与现代运行时 JVM 库无关。特别是,substring
示例在 OpenJDK/Oracle 7+ 中不再是问题。
我知道这与人们经常告诉您的相反,但有时显式创建新的 String
实例可能是减少内存的重要方法。
因为字符串是不可变的,所以有几种方法利用了这一事实并共享支持字符数组以节省内存。但是,有时这实际上可以通过防止对这些数组的未使用部分进行垃圾回收来增加内存。
例如,假设您正在解析日志文件的消息 ID 以提取警告 ID。您的代码看起来像这样:
//Format:
//ID: [WARNING|ERROR|DEBUG] Message...
String testLine = "5AB729: WARNING Some really really really long message";
Matcher matcher = Pattern.compile("([A-Z0-9]*): WARNING.*").matcher(testLine);
if ( matcher.matches() ) {
String id = matcher.group(1);
//...do something with id...
}
但是看看实际存储的数据:
//...
String id = matcher.group(1);
Field valueField = String.class.getDeclaredField("value");
valueField.setAccessible(true);
char[] data = ((char[])valueField.get(id));
System.out.println("Actual data stored for string \"" + id + "\": " + Arrays.toString(data) );
这是整个测试行,因为匹配器只是将一个新的 String 实例包装在相同的字符数据周围。比较将 String id = matcher.group(1);
替换为 String id = new String(matcher.group(1));
时的结果。
关于java - Java 中 String flyweight 实现的最佳替代方案,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2915511/
1、意图 运用共享技术有效的支持大量细粒度的对象 享元模式变化的是对象的存储开销 2、享元模式结构图 3、享元模式中主要角色 抽象享元(Flyweight)角色:此角色是所有的具体享元类的超
享元模式英文称为“Flyweight Pattern”,我非常感谢将Flyweight Pattern翻译成享元模式的那位强人,因为这个词将这个模式使用的方式明白得表示了出来;如果翻译成为羽量级模式
享元模式(Flyweight Pattern)主要用于减少创建对象的数量,以减少内存占用和提高性能 享元模式尝试重用现有的同类对象,如果未找到匹配的对象,则创建新对象 这种类型的设计模式属于结构型
这个问题已经有答案了: Run Jenkins job immediately (3 个回答) 已关闭 7 年前。 是否可以让我的工作作为“享元”工作执行,这样它就不会占用任何执行者? 最佳答案 任何
首先我将享元用于字符串,效果很好,但是当我将享元用于结构时。它不起作用。字符串的第一个测试用例是: static void testflyweightString() { char tmp[0]; v
我一直在阅读 boost::flyweight 的文档但我没有看到任何提及解除分配或引用计数政策。基本上,享元对象的行为应该像不同值的存储库,但不清楚当不再使用不同值时会发生什么。 是否已经支持?它可
使用 boost::flyweight 应该可以帮助我节省内存。我正在寻找对解决方案的有效性进行定量测量的方法。 有没有办法获取内部容器的 size()?如果它是基于散列的享元,有没有办法获取有关存储
这两种模式似乎达到了同样的目的。现实世界中有哪些不同的用例? 谢谢 最佳答案 Flyweight是当你有许多不同种类的单一事物时。 单例是当你有一个单一的东西。 例如,您可以使用享元模式来表示键盘字符
我想要一个 Flyweight 对象,因此我创建了一个对象并将其实例存储在 Map 中,如下所示: const FlyweightNumber = (function(){ "use stri
我正在开发一个新的应用程序,我将在其中同时打开一些窗口。我目前正在尝试设计 GUI,我正在为两种选择而苦苦挣扎: 我可以使用侧面导航面板并使用页面中心显示每个面板的内容。这些面板将根据享元模式存储,我
我的应用程序是多线程的,具有密集的字符串处理。我们正在经历过多的内存消耗,分析表明这是由于 String 数据造成的。我认为内存消耗会从使用某种享元模式实现甚至缓存中受益匪浅(我确信字符串经常重复,尽
我有一个关于享元选项的问题,给出下面的定义,基于 http://www.boost.org/doc/libs/1_40_0/libs/flyweight/test/test_basic.cpp typ
我无法理解如何将 boost::flyweight 用作 GOF 模式。有没有现成的例子? 例如,我希望它以下列方式使用。必须有一些享元容器,其中包含“胖”对象。这个容器可以为某些对象提供一些轻量级的
我正在尝试执行以下操作: boost::unordered_map, boost::flyweight > map; boost::flyweight foo(name);
我想知道为什么 std::map使用插入后不替换值。 示例: using std::string; using boost::flyweight; using std::map; int main()
所以我有一个字符串类型的享元: typedef boost::flyweight SymbolName_t; 我想将它的一个实例推送到它们的 vector 中,但天真的方法行不通: void Push
据我了解,享元设计模式与工厂或单例设计模式没有太大区别。 它只是一个生产不可变(和池化)对象的工厂。它只是一个单例,为(托管对象的)每种类型提供一个实例,而不是全局单个实例。 工厂和单例是创建模式,那
我的应用中有很多不同的屏幕一遍又一遍地引用相同的实体/业务对象。 目前,每个屏幕都引用每个对象的自己的拷贝。 此外,实体对象本身可能会公开对其他实体对象的访问,再次创建对象的新拷贝。 我正在尝试寻找缓
在打开 MFC 的情况下,在 VS2008 中编译以下代码时收到警告。 Boost 版本 1.39 include "boost/flyweight.hpp" include "boost/flywe
字符串已经在使用享元设计模式。汇集常见的 String 对象是否有益/性能好?因为字符串已经从字符串池中提取出来了吗? 最佳答案 字符串可以来自很多地方,默认情况下只有字符串文字在字符串池中。例如,当
我是一名优秀的程序员,十分优秀!