一场由Parcelable引发的血案

ui8p0q1j 8年前
   <h2>问题背景</h2>    <p>前阵子接手了直播模块,有个需求需要在已有的AIDL接口中增加多一个int类型的参数B。由于该AIDL接口中已经有了一个自定义类型的参数A(已经实现Parcelable接口),我便将参数B追加到A的后面。嗯,炒鸡简单的,只是运行之后有问题而已(微笑脸)</p>    <p>有一个诡异的问题: <strong>无论B传入什么值,另一方接收到都始终为0(int的默认值),而A接收到的值却是正确的?!</strong></p>    <p>Take it easy! 作为共产主义的接班人,我当然是有办法的啦:</p>    <ol>     <li>怀疑是 <a href="/misc/goto?guid=4959009805721920577" rel="nofollow,noindex">Freeline</a> 不支持AIDL,改用AS重新build,问题依旧</li>     <li>怀疑是辣鸡AS的问题,重新启动再build,问题依旧</li>     <li>怀疑是工具链的版本问题,改为最新版本再次编译,问题依旧</li>     <li>怀疑是int的问题?将B改为float, boolean等其他的基本类型,问题依旧,取到的始终是默认值</li>     <li>抱着希望在StackOverflow和Google上逛了一圈,无果</li>     <li>求助群里的小伙伴,答曰没有遇到过此情况,并向我丢了一连窜的「233333」和一波表情</li>     <li>辞职</li>    </ol>    <p>就在我一筹莫展的时候,突然脑子一抽,试着把接口定义中A参数和B参数位置调换。震惊地发现,居然可以了!!</p>    <p>嗯,此篇文章完结,撒花~</p>    <h2>解决方案</h2>    <p>将基本类型的参数放在自定义类型的参数前面,虽然解决了问题,但是治标不治本,只能算个workaround。</p>    <p>不知道大家有没有发现:参数的定义顺序会影响结果这一行为,是不是跟实现Parcelable接口的时候有点类似?Parcelable中如果read的顺序和write的顺序不同的话,产生的结果也不同。</p>    <p>基于这点,我们怀疑问题出在自定义类型的参数A,先来看下相关代码:</p>    <pre>  <code class="language-java">//ILivePlayerService.aidl    interface ILivePlayerService {   //参数A:config, 参数B:type   void setPlayerConfig(in PlayerConfig config, int type)  }  </code></pre>    <pre>  <code class="language-java">//PlayerConfig.java    public class PlayerConfig implements Parcelable {      //省略其他代码....      String streamUrl;      int roomId;      int anchorId;        public PlayerConfig() {      }        protected PlayerConfig(Parcel in) {          //省略其他代码....          this.streamUrl = in.readString();          this.roomId = in.readInt();          this.anchorId = in.readInt();      }        @Override      public void writeToParcel(Parcel dest, int flags) {          //省略其他代码....          dest.writeString(this.streamUrl);          dest.writeInt(this.roomId);          //下面注释的这句,代码中是没有的。问题就出现在这里..          //dest.writeInt(this.anchorId);      }        @Override      public int describeContents() {          return 0;      }        public static final Creator<PlayerConfig> CREATOR = new Creator<PlayerConfig>() {          @Override          public PlayerConfig createFromParcel(Parcel source) {              return new PlayerConfig(source);          }            @Override          public PlayerConfig[] newArray(int size) {              return new PlayerConfig[size];          }      };  }  </code></pre>    <p>果然, PlayerConfig 没有正确地实现 Parcelable 接口,在写入的时候(详见上面代码中 writeToParcel 方法中的注释)漏掉了变量 anchorId ,而读取的时候却有。</p>    <p>论接手别人的代码是一种怎样的体验?</p>    <p>我们先把 writeToParcel 中漏掉的变量补上,再跑一下看问题是否解决了。</p>    <p>不出所料,那个诡异的问题没有了,可是为什么呢?我们来看下AIDL生成的Java代码:</p>    <pre>  <code class="language-java">//AIDL生成的 ILivePlayerService.java   //为了方便查看,格式了一下代码    @Override  public boolean onTransact(int code, android.os.Parcel data, android.os.Parcel reply, int flags) throws android.os.RemoteException {      switch (code) {          case INTERFACE_TRANSACTION: {              reply.writeString(DESCRIPTOR);              return true;          }          //setPlayerConfig方法          case TRANSACTION_setPlayerConfig: {              data.enforceInterface(DESCRIPTOR);              //参数A:config              com.kk.model.PlayerConfig _arg0;              if ((0 != data.readInt())) {               //在解析的时候,会调用PlayerConfig的createFromParcel方法                  _arg0 = com.kk.model.PlayerConfig.CREATOR.createFromParcel(data);              } else {                  _arg0 = null;              }              //参数B:type              int _arg1;              _arg1 = data.readInt();              this.setPlayerConfig(_arg0, _arg1);              reply.writeNoException();              return true;          }      }      return super.onTransact(code, data, reply, flags);  }  </code></pre>    <p>我们可以看到在解析数据的时候, <strong>参数A和参数B都是从data中解析的(共用一个Parcel源)</strong> 其中,解析参数A config 的时候会将 data 传入到自己实现的 createFromParcel 方法中进行处理,如下</p>    <pre>  <code class="language-java">//PlayerConfig.java 部分代码    public static final Creator<PlayerConfig> CREATOR = new Creator<PlayerConfig>() {      @Override      public PlayerConfig createFromParcel(Parcel source) {       //接受到aidl传入的data       return new PlayerConfig(source);      }  };            protected PlayerConfig(Parcel in) {      //aidl传入的data在这里解析   this.streamUrl = in.readString();   this.roomId = in.readInt();   this.anchorId = in.readInt();  }    @Override  public void writeToParcel(Parcel dest, int flags) {   //省略其他代码....   dest.writeString(this.streamUrl);   dest.writeInt(this.roomId);   //下面注释的这句,代码中是没有的。问题就出现在这里..   //dest.writeInt(this.anchorId);  }  </code></pre>    <p>由于 PlayerConfig 没有正确地实现 Parcelable ,只写入了1个int类型,但是却读取了2个,这就导致了参数A多读取了一个int… 等到参数B想从 data 中读取的时候,就会读取不到数值(返回默认值)…</p>    <p>这个涉及到了Parcel的内部机制,可以参考这篇文章</p>    <p><a href="/misc/goto?guid=4959675699969647215" rel="nofollow,noindex">http://blog.csdn.net/qinjuning/article/details/6785517</a></p>    <h2>小结</h2>    <p>这次的坑是因为没有正确实现Parcelable接口导致的。这很不应该,其实我们可以让工具来做这种体力活,比如AS中有个插件叫做「Android Parcelable code generator」就可以一键生成Parcelable代码,或者去Github搜搜Parcelable相关的注解库也行~</p>    <p>程序猿要对自己好点,能用工具完成的事情尽量不要自己写~</p>    <p>踩坑结束!</p>    <p> </p>    <p>来自:http://andydev.me/2017/05/31/trap-of-android-aidl/</p>    <p> </p>