- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试找到解决由于使用以下形式的代码而导致的冗余字符串连接问题的最佳方法:
logger.debug("Entering loop, arg is: " + arg) // @1
在大多数情况下,logger.level
高于 debug
,并且 arg.toString()
和字符串连接是一种浪费,耗尽 CPU 周期并短暂耗尽内存。
在引入可变参数之前,推荐的方法是首先测试记录器级别:
if (logger.isDebugEnabled())
logger.debug("Entering loop, arg is: " + arg); // @2
但现在首选的形式是
logger.debug("Entering loop, arg is: {}", arg); // @3
在脚本中为每个 logger.debug
加上 if (logger.isDebugEnabled())
(以及其他方法的等价物)前缀并不是很困难,但是我正在尝试找到将第一种形式转换为第三种形式的最佳方法。
有什么建议吗?挑战在于在格式字符串中插入正确的数字大括号对 {}
。我希望 logback 能够在末尾附加占位符未涵盖的剩余参数,但我找不到它执行此操作的引用。
作为替代方案,我正在考虑编写一个粘贴在末尾的类Concatenator
并将第一种形式转换为
logger.debug(new Concatenator("Entering loop, arg is: ", arg)); // @4
Concatenator
类延迟对 arg.toString()
的调用和字符串连接,直到 logger
调用 toString()
,从而在记录器处于较高过滤级别时避免这两种情况。它确实增加了创建 Object[]
和 Concatenator
的开销,但这应该比替代方案便宜。
问题:
@1->@4
- 将+
替换为,
并包含在new Contatenator( ...)
) 更容易。我有什么遗漏的吗?@4
比 @1
好得多,我这样说对吗?public class Concatenator {
final Object[] input;
String output;
public Concatenator(Object... input) {
this.input = input;
}
public String toString() {
if (output == null) {
StringBuffer b = new StringBuffer();
for (Object s : input) b.append(s.toString());
output = b.toString();
}
return output;
}
public static void main(String args[]) {
new Concatenator("a", "b", new X());
System.out.println(new Concatenator("c", "d", new X()));
}
}
class X {
public String toString() {
System.out.println("X.toString");
return super.toString();
}
}
最佳答案
不幸的是,你的方法不会改变任何事情。事实上,它引入了一个额外的对象实例化/分配(您的连接器)。您还使用了 StringBuffer
,它引入了您不需要的同步开销。
问题出在 SLF4J 的 Logger.debug()
调用的方法签名上。第一个参数始终是String
。这意味着您必须调用:
logger.debug(new Concatenator("Entering loop, arg is: ", arg).toString());
这意味着...您所做的事情与 Java 要做的事情完全相同,但开销更大。
Java 编译器通过创建 StringBuilder
并执行您在 Concatenator
中所做的操作来处理字符串连接运算符 (+
) toString()
上的类。
logger.debug("Entering loop, arg is: " + arg);
变成:
logger.debug(new StringBuilder()
.append("Entering loop, arg is: ")
.append(arg).toString());
(如果您使用 javap
查看生成的字节码,您会发现情况就是如此。)
因此,您当前的方法将比现在的方法更昂贵。
编辑:因此,完成这项工作的方法是......
logger.debug("{}", new Concatenator("Entering loop, arg is: ", arg));
这样,您的Concatenator
就会作为Object
传递,除非记录器需要,否则不会调用它的toString()
。另外,将类中的 StringBuffer
替换为 StringBuilder
。
如果我没有直接回答你的问题......这比原来的更好吗? 可能;除非需要,否则不会发生字符串连接。然而,您正在引入对象实例化/分配。查看差异的唯一真正方法是对其进行分析/编写基准。
关于java - 将 logger.debug ("message: "+ 文本)转换为 logger.debug(消息 : {}", 文本),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21718213/
我尝试将 python 2.x 代码移植到 python 3。我正在努力解决的问题是 from mimetools import Message ... headers = Message(Strin
我有一个输入组件,它有三种类型的验证(required、validatorMessage、converterMessage),这个输入有自己的消息图标,整个表单有一个消息组件来显示所有组件的所有消息,
我有一个使用消息代理 (activemq) 的带有 spring 和 websockets 的 webapp。 这是我的配置类: @Configuration @EnableWebSocketMess
据我了解mbox Python 3.6 标准库中的类生成 email.message.Message 类型的旧式消息对象. 较新的类(class) email.message.EmailMessage
我使用的是mysql。 我有一个包含 userid、message_id、opened(true 或 false)、时间戳的消息表。 我希望所有未打开最近收到的 5 条消息的用户 这就是我现在拥有的:
我是 Android 的新手,发现要不断更新主视图,我必须创建一个线程来处理各种进程,然后将更新传回主视图。我决定使用 Handler 类来执行此操作。此示例中的 View 有一个用于激活代码的按钮和
我遇到了重定向符号的不同位置( > , &2 message message echo message >&2 message >&2 echo message message 对于所有表单,我得到了
我想使用 firebase 云函数发送通知,所以我尝试使用 firebase.messaging().getToken() 获取 token ,但我不断收到错误消息: TypeError: fireb
我实现了一个短信应用。现在我在使用 Oppo 设备时遇到了问题,因为无论何时收到消息,系统都会将默认应用程序更改为内置应用程序并显示此消息: For your messages security, S
我正在实现本指南:https://spring.io/guides/gs/centralized-configuration/关于Spring Cloud配置。 服务器: @EnableConfigS
我想在“匹配”之后,向所有比赛发送介绍信息。你知道一种轻松发送的方法吗?(我使用 Bluestacks) 提前致谢。问候。 最佳答案 只需传递 session.send()在session.dialo
在我们的应用程序中,我们使用 kafka 并有一个像这样的 Spring Cloud 输入流 @Component public interface SomeChannel { @Input(
这周我在通过 Node.js 库(代码相同,库版本相同等)向我的 iOS 设备发送消息时遇到了很多内部错误 很难调试,因为有时它会起作用。当我使用 for 循环发送 10 条消息时,我的设备将收到 3
我目前正在记录错误并希望获得尽可能多的描述性细节。我知道我可以捕获许多不同类型的异常,但 Exception.Message 之间有什么区别?和 Exception.InnerException.Me
我创建了一个映射到 MyBean.beansField 的表单。我使用 javax.validation.NotNull 注释来确保必须输入它。到目前为止一切正常,但错误消息如下所示: beansFi
我正在研究 Azure 服务总线。我的服务总线队列正在处理一条消息 3 次。我的消息锁定时间是5分钟。每条消息最多处理 2 分钟,但我不知道为什么队列会选择相同的消息并发送到处理,而重复的消息仅在 5
我正在使用最新的快速修复版本,即 1.6.0。我已经有针对 1.5.3 编写的代码,我想做的是将其升级到 1.6.0 我遇到的问题是,当我使用破解(msg,sessionID)方法时,它会抛出quic
当我调用 grails message()函数来查找和评估消息 key 对,它无法评估参数。 在我的 Controller 中,我调用消息函数: rejectWithError(message(cod
我使用一个小型 Spring 应用程序,其中数据库中的值很少,我想使用可变调用来检索它们。 API 在这里, @RestController @RequestMapping("/api/v1/prod
我在想在用以更好的方式,像这样: Please inform us about: 这个想法是以一种不同的方式向用户展示消息,具有更多的风格。这可能吗? 最佳答案 它们可通过 Fac
我是一名优秀的程序员,十分优秀!