<address id="tl19n"></address>
      <address id="tl19n"><nobr id="tl19n"></nobr></address>

        <address id="tl19n"></address>

          <address id="tl19n"></address>
          <form id="tl19n"></form>

              首頁 > QQ技巧 > Java陷阱 慎用入參做返回值

              Java陷阱 慎用入參做返回值

              時間:2021-04-21 20:44 作者:QQ地帶 我要評論

              正常情況下,在Java中入參是不建議用做返回值的。除了造成代碼不易理解、語義不清等問題外,可能還埋下了陷阱等你入坑。
               
              問題背景
              比如有這么一段代碼:
               
              @Named
              public class AService {   
              private SupplyAssignment localSupply = new SupplyAssignment();
                  @Inject
                  private BService bervice;
               
                  public List<Supply> calcSupplyAssignment()
                     List<Supply> supplyList = bService.getLocalSupplyList(this.localSupply);
                      …
                     return supplyList;
                  }
              }
              上面代碼,服務A希望調用服務B,以獲取supplyList,但同時,服務A又希望修改localSupply的狀態值,未能避免修改calcSupplyAssignment接口的(不想改返回的類型),將localSupply作為了入參但同時也用作了返回值。
               
              服務B代碼如下:
               
              @Named
              public class BService {   
               
              public List<Supply> getLocalSupplyList (SupplyAssignment localSupply)
                  SupplyAssignment supplyAssignment = this.getSupplyAssignment();
                      // 希望localSupply被重新賦值后返回
                      localSupply = supplyAssignment;
                      …
                      return supplyList;
               
                  }
              }
              在服務B代碼內部,服務A的入參localSupply被傳入,希望重新被supplyAssignment賦值而后返回新值。然而,這樣做是無效的。
               
              問題原因
              先來看下編程語言中關于參數傳遞的類型:
               
              值傳遞(pass by value)是指在調用函數時將實際參數復制一份傳遞到函數中,這樣在函數中如果對參數進行修改,將不會影響到實際參數。
              引用傳遞(pass by reference)是指在調用函數時將實際參數的地址直接傳遞到函數中,那么在函數中對參數所進行的修改,將影響到實際參數。
              因為Java程序設計語言是采用的值傳遞,因為Java沒有指針的概念。也就是說方法得到的是所有參數值的一個拷貝,方法并不能修改傳遞給它的任何參數變量的內容。
               
              因此,上述代碼中,服務A調用服務B時,服務B的參數localSupply實際上是服務A的localSupply的一個拷貝,當然,這兩個都是指向了同一個地址對象supplyAssignment1。
               
               
               
              當在服務B內部對參數localSupply進行重新賦值是localSupply = supplyAssignment,實際上,只是對B的參數localSupply做了從新賦值,B的參數localSupply會指向一個新的地址對象supplyAssignment2。
               
               
               
              從上圖可以清晰看到,因此,服務A的localSupply和B的參數localSupply已經指向了不同的對象了,對B的參數localSupply做任何的修改,都不會影響服務A的localSupply的原值。這就是問題的原因,你希望服務B來修改服務A入參的狀態,并將改后的值返回給服務A,但并不奏效。
               
              解決方案
              方案1:入參不要用作返回值
              當然,這個是最清晰的且易于理解的,但這會導致有的接口的返回類型產生變化。
               
              有時確實想要入參做返回值,那看方案2。
               
              方案2:入參不要賦值新對象
              這個方案就是直接在入參的對象上做狀態的修改,而不要去賦值新對象。還是這個圖:
               
               
               
              在這個圖中,只要我們是一直在B的參數localSupply修改的是supplyAssignment1的狀態值,那結果就能反饋到服務A的localSupply上。如何實現?看下下面代碼:
               
              @Named
              public class BService {   
               
                  public List<Supply> getLocalSupplyList (SupplyAssignment localSupply)
                      
                      SupplyAssignment supplyAssignment = this.getSupplyAssignment();
               
                      // 針對localSupply不能新建引用,只能重新賦值屬性
                      BeanUtils.copyProperties(supplyAssignment, localSupply);
                      …
                      return supplyList;
               
                  }
               
              }
              在上面的方法中,我們用到了Spring的工具類BeanUtils,該類的copyProperties方法的實質是將supplyAssignment的屬性值,賦值到了localSupply的屬性上。這意味著我們是修改的B的參數localSupply上的屬性,而并未新建對象。

              標簽: java
              頂一下
              (0)
              0%
              踩一下
              (0)
              0%

              Google提供的廣告

              国产古代一级a毛片,国产成人午夜在线直播,write.as 当众老师
                <address id="tl19n"></address>
                  <address id="tl19n"><nobr id="tl19n"></nobr></address>

                    <address id="tl19n"></address>

                      <address id="tl19n"></address>
                      <form id="tl19n"></form>