- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我花了一些时间找出一种可持续的方法来过滤 ERROR
我的应用程序实时记录 JBoss
服务器。与Log4j
我可以轻松使用SocketAppender
我按照它通知 ERROR
的方式配置了它实时登录。不幸的是我不能用 JBoss AS 7
来做到这一点。我找不到 SocketAppender
除了 SMTPAppender
之外的其他实现方式。所以我想出了一个办法来解决这个问题。请看看我到目前为止做了什么。
我为 Logger
编写了一种包装器.
我的应用程序的示例类
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class MyClass {
private Logger logger= LoggerFactory.getLogger(MyClass.class);
public void updateData(String data){
logger.info("updating data with {} ",data);
try {
// do somthing
}catch (Exception e){
//something went wrong
logger.error("update fail ",e);// I want to get this message real time
}
}
}
现在我想到了一个方法来获取 ERROR
日志。
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class MyLogger {
private Logger logger = null;
private MyLogger() {
}
public static MyLogger getLogger(Class clazz) {
MyLogger logger = new MyLogger();
logger.logger = LoggerFactory.getLogger(clazz);
return logger;
}
public void info(String format, Object... arguments) {
logger.info(format,arguments);
}
public void error(String format, Object... arguments) {
// now i can send this message real time,
// i am calling web service with error log by a new Thread
logger.error(format,arguments); // say line number 10
}
}
MyLogger
使用方法如下
public class MyClass {
private static MyLogger logger= MyLogger.getLogger(MyClass.class);
public void updateData(String data){
logger.info("updating data with {} ",data);
try {
// do somthing
}catch (Exception e){
//something went wrong
logger.error("update fail ",e); // say line number 14
}
}
}
此时我的要求符合,但这会产生一个新问题。
以下是JBoss
的记录器格式配置
<pattern-formatter pattern="%K{level}%d{HH:mm:ss,SSS} %-5p (%F:%L) (%t) %s%E%n"/>
现在%F
-文件名将变为MyLogger
类(class)。和%L
- 行号将成为MyLogger
的行号这会导致日志不准确。
部分示例日志
(MyLogger.java : 10) // but this should be (MyClass.java : 14)
现在我已将此模式更改为
<pattern-formatter pattern="%K{level}%d{HH:mm:ss,SSS} %-5p [%c:%L] (%t) %s%E%n"/>
这几乎可以工作,但是 %c
-class name 给出了预期的正确类。但无法获取行号(正确的行号)。
更改记录器 paten 后我得到了什么
(com.my.test.MyClass : 10) // but this should be (com.my.test.MyClass : 14)
这是我的问题。
SocketAppender
或 JMSAppender
)与 JBoss AS7
一起使用?(在 Jboss AS 7 上没有找到有效的解决方案)我已经在 JBossDeveloper
中提出了问题。但至少目前还没有任何评论。
https://developer.jboss.org/thread/249729
任何帮助都非常感谢。
最佳答案
如果您使用的是 JBoss 7.2 (EAP 6.1) 或更高版本,则可以使用 Log4j 附加程序 作为自定义处理程序 ( check this issue )。
例如。
<custom-handler name="server" class="org.apache.log4j.net.SocketAppender" module="org.apache.log4j">
<level name="INFO"/>
<formatter>
<pattern-formatter pattern="%d{HH:mm:ss,SSS} %-5p [%c] (%t) %s%E%n"/>
</formatter>
<properties>
<property name="RemoteHost" value="10.10.10.10"/>
<property name="Port" value="1234"/>
<property name="LocationInfo" value="true"/>
<property name="ReconnectionDelay" value="10000"/>
</properties>
</custom-handler>
在其他情况下,使其在 JBoss AS 7 中工作的唯一方法是为 SocketAppender
开发自定义 java.util.logging.Handler
。
查看Creating a custom logging handler in JBOSS as 7.1.0 Final和 Custom log handlers on 7.0.1创建自定义日志处理程序,然后将它们委托(delegate)给 Log4jAppenderHandler
。
例如(示例代码):
import java.util.Collections;
import java.util.logging.Filter;
import java.util.logging.Formatter;
import java.util.logging.Handler;
import java.util.logging.LogRecord;
import org.apache.log4j.Level;
import org.apache.log4j.Logger;
import org.apache.log4j.net.SocketAppender;
import org.apache.log4j.net.SyslogAppender;
import org.apache.log4j.spi.LocationInfo;
import org.apache.log4j.spi.LoggingEvent;
import org.apache.log4j.spi.ThrowableInformation;
import org.jboss.logmanager.ExtLogRecord;
import org.jboss.logmanager.log4j.handlers.Log4jAppenderHandler;
public class SocketHandler extends Handler {
private final Log4jAppenderHandler log4j = new Log4jAppenderHandler(new SocketAppender());
@Override
public void publish(LogRecord record) {
if (record == null) {
return;
}
ExtLogRecord extRecord;
if (record instanceof ExtLogRecord) {
extRecord = (ExtLogRecord) record;
} else {
extRecord = ExtLogRecord.wrap(record);
}
if (isLoggable(record)) {
LoggingEvent event = new LoggingEvent(extRecord.getLoggerClassName(),
Logger.getLogger(extRecord.getLoggerClassName()), extRecord.getMillis(),
Level.toLevel(extRecord.getLevel().toString()), extRecord.getFormattedMessage(),
extRecord.getThreadName(), extRecord.getThrown() == null ? null
: new ThrowableInformation(extRecord.getThrown()), extRecord.getNdc(),
new LocationInfo(new Throwable(), extRecord.getLoggerClassName()),
Collections.singletonMap("org.jboss.logmanager.record", extRecord));
((SyslogAppender) log4j.getAppender()).doAppend(event);
flush();
}
}
@Override
public void flush() {
log4j.flush();
}
@Override
public void close() throws SecurityException {
log4j.close();
}
@Override
public void setFilter(Filter filter) {
log4j.setFilter(filter);
}
@Override
public void setFormatter(Formatter formatter) {
log4j.setFormatter(formatter);
}
public void setThreshold(String threshold) {
((SocketAppender) log4j.getAppender()).setThreshold(Level.toLevel(threshold));
}
public void setRemoteHost(String remoteHost) {
((SocketAppender) log4j.getAppender()).setRemoteHost(remoteHost);
}
public void setPort(String port) {
try {
((SocketAppender) log4j.getAppender()).setPort(Integer.parseInt(port));
} catch (Exception e) {}
}
public void setReconnectionDelay(String reconnectionDelay) {
try {
((SocketAppender) log4j.getAppender()).setReconnectionDelay(Integer.parseInt(reconnectionDelay));
} catch (Exception e) {}
}
public void setLocationInfo(String locationInfo) {
try {
((SocketAppender) log4j.getAppender()).setLocationInfo(Boolean.parseBoolean(locationInfo));
} catch (Exception e) {}
}
}
我认为最好的方法是这样,这样您就不会在应用程序中实现日志记录。
我希望这会有所帮助。
关于java - 如何在以下上下文中获得正确的记录器行号,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27039249/
出现以下错误 Network access for Distributed Transaction Manager (MSDTC) has been disabled. Please enable D
在调试应用程序时出现以下错误。 The CLR has been unable to transition from COM context 0x3b2d70 to COM context 0x3b2
在 GAE Go 中,为了记录,我们需要使用 appengine.NewContext(r) 创建一个新的上下文,它返回 context.Context。 如何使用此上下文在请求范围内设置/获取变量?
我想使用 Puppeteer 从放置在页面上 iframe 内的选择器中获取数据,该页面在与其父框架域不同的域上运行。因此,我不是任何域的所有者 - 无法使用 frame.postMessage。 试
我正在尝试获取可用的应用程序上下文并想切换到 webview 上下文,但 appium 仅获取 Navive App。 应用程序还启用了 WebView。 Appium 版本:1.10.1 Chrom
这个问题在这里已经有了答案: How to fix this nullOk error when using the flutter_svg package? (7 个回答) 7 个月前关闭。 当我尝
我观看了关于 Core Data 的 2016 WWDC 视频并查看了各种教程。我见过使用 Core Data Framework 创建对象以持久保存到 managedObjectContext 中的
这是代码 obj = { a: 'some value'; m: function(){ alert(this.a); } } obj.m(); 结果是'som
我正在尝试做类似的事情 $(".className").click(function() { $(this).(".anotherClass").css("z-index","1");
var User = { Name: "Some Name", Age: 26, Show: function() { alert("Age= "+this.Age)}; }; fun
我目前正在使用我见过的常见 Context 模式,它允许子组件通过传递修饰函数来更新父组件的状态(即 Provider)通过共享的 Context。 我遇到的问题是,修改函数只引用原始状态,不引用最新
有没有办法让 React Context类型安全与流类型? 例如: Button.contextTypes = { color: React.PropTypes.string }; 最佳答案 不幸
我想知道是否有一种方法可以为不同的功能使用不同的上下文类。 我希望有一个功能使用 MinkExtensions 进行浏览器测试,另一个功能使用和 HTTP 客户端(如 Guzzle)进行 API 测试
我有这个配置文件 apiVersion: v1 clusters: - cluster: server: [REDACTED] // IP of my cluster name: stag
我在实现非抢先式调度时遇到了用于初始化TCB的代码。 typedef struct TCB_t { struct TCB_t *next; struct TCB_t
我想将一个函数设置为数组中每个元素的属性,但使用不同的参数调用它。我想我会使用匿名函数来解决它: for ( var i = 0; i < object_count; i++ ) { obje
这个问题已经有答案了: How to access the correct `this` inside a callback (15 个回答) 已关闭 7 年前。 我正在做一些练习,但我在管道方法中丢
我正在尝试通过 Java 和 Android Studio 学习和制作 Android 应用程序。我对Java的了解程度是两年前几个小时的youtube学习和大学基础类(class)。不过我确实知道如
我在(这个)上遇到了问题。错误ImageView无法应用。我在 fragment 类中执行此代码。 ViewFlipper v_flipper; @Nullable @Override public
我想使用 openGL 的某些功能,但与渲染视觉内容无关。有没有办法在没有任何依赖性的情况下创建它(不是对 Windows,也不是某些包[SDL,SFML,GLUT])?只允许使用没有外部库的库,就像
我是一名优秀的程序员,十分优秀!