java从ManagedBeans中丢失值

java从ManagedBeans中丢失值,java,jsf,managed-bean,Java,Jsf,Managed Bean,我尝试编写一个应用程序来创建PDF文件,它使用JavaServerFaces。当我给从bean到工厂类的文本框赋值时,我遇到了一个问题,即​​我们迷路了。我想知道为什么会发生这种情况,已经尝试了许多解决方案,但它们会导致代码的扩展,并且没有帮助 Bean代码: @ManagedBean @ViewScoped public class PanelInneBean extends AbstractPanel implements Serializable { private static

我尝试编写一个应用程序来创建PDF文件,它使用JavaServerFaces。当我给从bean到工厂类的文本框赋值时,我遇到了一个问题,即​​我们迷路了。我想知道为什么会发生这种情况,已经尝试了许多解决方案,但它们会导致代码的扩展,并且没有帮助

Bean代码:

@ManagedBean
@ViewScoped
public class PanelInneBean extends AbstractPanel implements Serializable {
    private static final long serialVersionUID = 1L;
    private final int CODE = 4;
    private boolean use;
    private String tytul, opis;
    private PdfFactory pdf = PdfFactory.getPdfObject();

    public PanelInneBean() {
    use = false;
    }

    public boolean getUse() {
    return use;
    }

    public String getTytul() {
    return tytul;
    }

    public void setTytul(String tytul) {
    this.tytul = tytul;
    }

    public String getOpis() {
    return opis;
    }

    public void setOpis(String opis) {
    this.opis = opis;
    }

    public int getCode() {
    return CODE;
    }

    private void add() {
    use = true;
    }

    public void addBean() {
    add();
    pdf.addElement(this);
    System.out.println("InnePanel after pdf.addElement() this.opis:" + this.opis);
    // This sysout prints the correct value after give object to factory
    }
}
出厂代码:

  public int addElement(PdfElement element) {
    pdfType = true;
    if (element.getUse()) {
        elementList.add(element);
        return 1;
    }
    return 0;
    }

    public void prepare() {
    for (PdfElement element : elementList) {
        System.out.println("element.code:" + element.getCode());
        switch (element.getCode()) {
        case 0:
        if (nF != null)
            break;
        nF = new NaglowekFactory(element, Counter.getNumber());
        break;
        case 1:
        if (pF != null)
            break;
        pF = new ProduktyFactory(element, Counter.getNumber());
        prodSum = pF.getProdukty().getSuma();
        euroData = pF.getProdukty().getEuroData();
        break;
        case 2:
        if (mF != null)
            break;
        mF = new MontazFactory(element, Counter.getNumber());
        servSum = mF.getMontaz().getSuma();
        break;
        case 3:
        if (uF != null)
            break;
        uF = new UslugiFactory(element, Counter.getNumber());
        asmSum = uF.getUslugi().getSuma();
        break;
        case 4:
        if (iF != null)
            break;
        iF = new InneFactory(element, Counter.getNumber());
            //here Opis value is empty
        break;
        }
    }
    System.out.println("factory.prepare() ->");
    }
Infactory的构造函数:

    PanelInneBean inne;

public InneFactory(PdfElement element, int order) {
    inne = (PanelInneBean) element;
    System.out.println("innerFactory constructor, inne.getTytul(): "
        + inne.getTytul());

        //here values are empty

    this.order = order;
    list = new ArrayList<Element>();
}

public int getOrder() {
    return order;
}
panelinnebeninne;
公共innfactory(PdfElement元素,整数顺序){
inne=(PanelInneBean)元素;
System.out.println(“内部工厂构造函数,inne.getTytul():”
+inne.getTytul();
//这里的值为空
这个。顺序=顺序;
列表=新的ArrayList();
}
public int getOrder(){
退货单;
}

我做错了什么?

我认为您对
@ViewScoped
bean的实际工作方式有一个常见的误解。bean对象本身在每个Faces请求时创建,并在每个Faces响应后销毁

然而,在重新创建bean之后,JSF将进入第一个生命周期阶段RestoreView。在这个阶段,它将从视图状态获取这个bean的最后一个好值,并将它们应用于bean管理的属性

基本上,您正在调用一个操作,该操作生成一个与加载页面的请求不同的请求,此时您添加到
PdfFactory
的引用现在丢失,因为最后一个对象中的PdfFactory已被销毁

解决这个问题的最佳方法是将
PdfFactory
变成托管Bean。您可以在
@PostConstruct
方法中初始化工厂,并且可以使用
@ManagedProperty
注释将其他托管bean注入PdfFactory bean

@ManagedBean
@RequestScoped
public PdfFactory extends ... implements Serializable {

  @ManagedProperty("#{myViewBean}")
  private MyViewBean viewBean;

  @PostConstruct
  public initializeFactory() {
    //Do stuff
  }

  public void prepare() {
    // prepare stuff
  }
}

我认为您对
@ViewScoped
bean的实际工作方式有一个常见的误解。bean对象本身在每个Faces请求时创建,并在每个Faces响应后销毁

然而,在重新创建bean之后,JSF将进入第一个生命周期阶段RestoreView。在这个阶段,它将从视图状态获取这个bean的最后一个好值,并将它们应用于bean管理的属性

基本上,您正在调用一个操作,该操作生成一个与加载页面的请求不同的请求,此时您添加到
PdfFactory
的引用现在丢失,因为最后一个对象中的PdfFactory已被销毁

解决这个问题的最佳方法是将
PdfFactory
变成托管Bean。您可以在
@PostConstruct
方法中初始化工厂,并且可以使用
@ManagedProperty
注释将其他托管bean注入PdfFactory bean

@ManagedBean
@RequestScoped
public PdfFactory extends ... implements Serializable {

  @ManagedProperty("#{myViewBean}")
  private MyViewBean viewBean;

  @PostConstruct
  public initializeFactory() {
    //Do stuff
  }

  public void prepare() {
    // prepare stuff
  }
}

当我有几个ManagedBeans将数据发送到工厂时会发生什么?只需将它们添加到@ManagedProperty?@insect您就可以了是的。。。有很多方法可以做到这一点,只是要意识到您的工厂应该是一个具有某种范围的ManagedBean。在对象创建时创建Factory对象是您丢失数据的原因。感谢您的建议,我还有一个问题要问您,(或其他人)Factory类内部有私有类,例如
OtherFactory
,它具有适当的ManagedBean对象(
Other
)。我应该如何使用注释@ManagedProperty
OtherFactory
创建对象或在
OtherFactory
内部创建特定对象
Other
?@insect Hmmm我不确定是否诚实。我从未尝试过将工厂模式与托管Bean(视图控制器模式)合并。也许你必须重新考虑你的设计。。。或者干脆让PdfFactory
@ApplicationScoped
,把它当作一个单独的对象?然后,您所要做的就是在所有视图中将PdfFactory作为ManagedProperty引用。当我有两个ManagedBeans将数据发送到factory时会发生什么?只需将它们添加到@ManagedProperty?@insect您就可以了是的。。。有很多方法可以做到这一点,只是要意识到您的工厂应该是一个具有某种范围的ManagedBean。在对象创建时创建Factory对象是您丢失数据的原因。感谢您的建议,我还有一个问题要问您,(或其他人)Factory类内部有私有类,例如
OtherFactory
,它具有适当的ManagedBean对象(
Other
)。我应该如何使用注释@ManagedProperty
OtherFactory
创建对象或在
OtherFactory
内部创建特定对象
Other
?@insect Hmmm我不确定是否诚实。我从未尝试过将工厂模式与托管Bean(视图控制器模式)合并。也许你必须重新考虑你的设计。。。或者干脆让PdfFactory
@ApplicationScoped
,把它当作一个单独的对象?然后,您所要做的就是在所有视图中将PdfFactory作为ManagedProperty引用。