背景
自己寫了一個(gè)自定義的ViewGroup的流式布局,但是調(diào)用addView方法添加xml中的View的時(shí)候報(bào)錯(cuò):
The specified child already has a parent.You must call removeView() on the child's parent first."
探索
二話不說,點(diǎn)源碼看一下,順著addView的流程,最后走到了addViewInner()這個(gè)方法,看下代碼
private void addViewInner(View child, int index, LayoutParams params,
boolean preventRequestLayout) {
if (mTransition != null) {
// Don't prevent other add transitions from completing, but cancel remove
// transitions to let them complete the process before we add to the container
mTransition.cancel(LayoutTransition.DISAPPEARING);
}
if (child.getParent() != null) { //這里拋出了我遇到的錯(cuò)誤的異常
throw new IllegalStateException("The specified child already has a parent. " +
"You must call removeView() on the child's parent first.");
}
if (mTransition != null) {
mTransition.addChild(this, child);
}
if (!checkLayoutParams(params)) {
params = generateLayoutParams(params);
}
if (preventRequestLayout) {
child.mLayoutParams = params;
} else {
child.setLayoutParams(params);
}
if (index < 0) {
index = mChildrenCount;
}
addInArray(child, index);
// tell our children
if (preventRequestLayout) {
child.assignParent(this);
} else {
child.mParent = this;
}
if (child.hasFocus()) {
requestChildFocus(child, child.findFocus());
}
AttachInfo ai = mAttachInfo;
if (ai != null && (mGroupFlags & FLAG_PREVENT_DISPATCH_ATTACHED_TO_WINDOW) == 0) {
boolean lastKeepOn = ai.mKeepScreenOn;
ai.mKeepScreenOn = false;
child.dispatchAttachedToWindow(mAttachInfo, (mViewFlags&VISIBILITY_MASK));
if (ai.mKeepScreenOn) {
needGlobalAttributesUpdate(true);
}
ai.mKeepScreenOn = lastKeepOn;
}
if (child.isLayoutDirectionInherited()) {
child.resetRtlProperties();
}
dispatchViewAdded(child);
if ((child.mViewFlags & DUPLICATE_PARENT_STATE) == DUPLICATE_PARENT_STATE) {
mGroupFlags |= FLAG_NOTIFY_CHILDREN_ON_DRAWABLE_STATE_CHANGE;
}
if (child.hasTransientState()) {
childHasTransientStateChanged(child, true);
}
if (child.getVisibility() != View.GONE) {
notifySubtreeAccessibilityStateChangedIfNeeded();
}
if (mTransientIndices != null) {
final int transientCount = mTransientIndices.size();
for (int i = 0; i < transientCount; ++i) {
final int oldIndex = mTransientIndices.get(i);
if (index <= oldIndex) {
mTransientIndices.set(i, oldIndex + 1);
}
}
}
if (mCurrentDragStartEvent != null && child.getVisibility() == VISIBLE) {
notifyChildOfDragStart(child);
}
}
從注釋中可以看到代碼哪里錯(cuò)了,瞬間明白,其實(shí),在Android中,一個(gè)View只能屬于一個(gè)特定的ViewGroup,我自定義了一個(gè)ViewGroup,然后又addView()加載xml中的一個(gè)view的時(shí)候,這個(gè)View的外層布局是一個(gè)LinearLayout。到了addInnerView的時(shí)候。進(jìn)入if條件語句,拋出異常,崩潰。
if (child.getParent() != null) { //這里拋出了我遇到的錯(cuò)誤的異常
throw new IllegalStateException("The specified child already has a parent. " +
"You must call removeView() on the child's parent first.");
}
解決辦法
把View的父ViewGroup解除關(guān)系,再在自己的ViewGroup添加就好。
if (childView != null) {
ViewGroup parentViewGroup = (ViewGroup) childView.getParent();
if (parentViewGroup != null ) {
parentViewGroup.removeView(childView);
}
}
到這里,這個(gè)問題就搞定了。OK!收集小問題,一點(diǎn)一滴。