我正在尝试用jsf和CDI编写一个简单的登录表单。问题是,当我注入SessionScoped bean时,它不能像我期望的那样工作。这是我的豆子
@Named
@SessionScoped
public class LoginInfo implements Serializable {
private String uname;
private String pass;
private String pagename;
private int count;
public LoginInfo() {
}
public void increment() {
count++;
}
}
这是我的控制器:
@Named
@RequestScoped
public class LoginPageMg {
@Inject
LoginInfo lo;
public LoginPageMg() {
}
public void login() {
lo.increment();
lo.setPagename("aaa");
int x = 8;
}
}
还有一个简单的Jsf表单,它调用login函数并显示LoginInfo类的计数器字段。
<h:form prependId="false" id="mainform" styleClass="login-box">
<p:inputText value="#{loginPageMg.uname}"/>
<p:password value="#{loginPageMg.pass}"/>
<h:outputLabel id="counter" value="#{loginInfo.count}"></h:outputLabel>
<p:commandButton update="counter"
action="#{loginPageMg.login}"
value="login"></p:commandButton>
</h:form>
通过单击登录按钮和调试变量,我可以看到“lo”是这样的:
lo = {LoginInfo $ Proxy $ _ $$ _ WeldClientProxy @ 16688}“com.mg.LoginInfo@703ec5d5”
在线int x = 8我可以看到“lo”变量根本没有改变,但是在我的jsf页面中我可以看到每次按下登录按钮时计数器都会增加,并且bean在刷新页面后保持该值。
我正在使用Wildfly 15 Jsf 2.3.4 CDI 1.1
我正在使用Wildfly 15 Jsf 2.3.4 CDI 1.1
WildFly 15意味着EE 8兼容性,因此它带有CDI 2.0。
现在尽可能多地回答你的问题:
什么是WeldClientProxy?
Weld是大多数EE服务器使用的CDI参考实现,包括WildFly。 WeldClientProxy
是一种内部构造,用于注射普通的范围豆(几乎所有除了@Dependent
)。它不是bean的实际实例,您可以将其视为“包装器”,它知道如何检索实际的上下文实例并委托对它的调用。这样,您可以重用代理实例,同时仍然指向引擎盖下的正确实例(因此引用不需要更改)。
为什么SessionScoped bean有两个不同的实例?这是正常的还是我做错了什么?
只有一个,客户端代理不是实际的实例。你基本上通过看到每次增加count
并在请求之间保持状态来验证。
我怎样才能注入jsf所做的相同实例?
JSF实际上并没有注入任何内容,而是使用表达式语言根据其名称查找bean。 CDI允许您通过@Named
定义此名称。然后,JSF再次通过代理使用您注入的同一个bean。