Android 如何使用XML和java代码定义自定义小部件

Android 如何使用XML和java代码定义自定义小部件,android,custom-controls,Android,Custom Controls,我想定义一个自定义小部件,它包含一些其他控件并具有自己的逻辑。我想用XML文件来定义UI,用java代码来定义逻辑。但我不知道怎么做 我看了这篇文章,但没有一篇有效的。下面是我的代码 XML: 在主布局中,我这样调用它: <com.example.MyView android:layout_width="wrap_content" android:layout_height="wrap_content"> </com.example.MyView

我想定义一个自定义小部件,它包含一些其他控件并具有自己的逻辑。我想用XML文件来定义UI,用java代码来定义逻辑。但我不知道怎么做

我看了这篇文章,但没有一篇有效的。下面是我的代码

XML:

在主布局中,我这样调用它:

<com.example.MyView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content">
</com.example.MyView>
<merge xmlns:android="http://schemas.android.com/apk/res/android>

    <Button android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="custom button 1"/>

    <Button android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="custom button 2"/>

</merge>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" 
   android:layout_width="match_parent"
   android:layout_height="match_parent">

    <Button android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="custom button 1"/>

    <Button android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="custom button 2"/>

</LinearLayout>

不幸的是,它在运行时引发异常:

05-26 14:11:25.199: ERROR/AndroidRuntime(15799): FATAL EXCEPTION: main
    java.lang.RuntimeException: Unable to start activity ComponentInfo{com.example/com.example.MyActivity}: android.view.InflateException: Binary XML file line #7: Error inflating class <unknown>
    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:1651)
    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:1667)
    at android.app.ActivityThread.access$1500(ActivityThread.java:117)
    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:935)
    at android.os.Handler.dispatchMessage(Handler.java:99)
    at android.os.Looper.loop(Looper.java:130)
    at android.app.ActivityThread.main(ActivityThread.java:3687)
    at java.lang.reflect.Method.invokeNative(Native Method)
    at java.lang.reflect.Method.invoke(Method.java:507)
    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:867)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:625)
    at dalvik.system.NativeStart.main(Native Method)
    Caused by: android.view.InflateException: Binary XML file line #7: Error inflating class <unknown>
    at android.view.LayoutInflater.createView(LayoutInflater.java:518)
    at com.android.internal.policy.impl.PhoneLayoutInflater.onCreateView(PhoneLayoutInflater.java:56)
    at android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:568)
    at android.view.LayoutInflater.rInflate(LayoutInflater.java:623)
    at android.view.LayoutInflater.inflate(LayoutInflater.java:408)
    at android.view.LayoutInflater.inflate(LayoutInflater.java:320)
    at android.view.LayoutInflater.inflate(LayoutInflater.java:276)
    at com.example.MyView.onFinishInflate(MyView.java:32)
    at android.view.LayoutInflater.rInflate(LayoutInflater.java:631)
    at android.view.LayoutInflater.inflate(LayoutInflater.java:408)
    at android.view.LayoutInflater.inflate(LayoutInflater.java:320)
    at android.view.LayoutInflater.inflate(LayoutInflater.java:276)
    at com.example.MyView.onFinishInflate(MyView.java:32)
    at android.view.LayoutInflater.rInflate(LayoutInflater.java:631)
    at android.view.LayoutInflater.inflate(LayoutInflater.java:408)
    at android.view.LayoutInflater.inflate(LayoutInflater.java:320)
    at android.view.LayoutInflater.inflate(LayoutInflater.java:276)
    at com.example.MyView.onFinishInflate(MyView.java:32)
    at android.view.LayoutInflater.rInflate(LayoutInflater.java:631)
    at android.view.LayoutInflater.inflate(LayoutInflater.java:408)
    at android.view.LayoutInflater.inflate(LayoutInflater.java:320)
    at android.view.LayoutInflater.inflate(LayoutInflater.java:276)
    at com.example.MyView.onFinishInflate(MyView.java:32)
    at android.view.LayoutInflater.rInflate(LayoutInflater.java:631)
    at android.view.LayoutInflater.inflate(LayoutInflater.java:408)
    at android.view.LayoutInflater.inflate(LayoutInflater.java:320)
    at android.view.LayoutInflater.inflate(LayoutInflater.java:276)
    ....
05-26 14:11:25.199:错误/AndroidRuntime(15799):致命异常:main
java.lang.RuntimeException:无法启动活动组件信息{com.example/com.example.MyActivity}:android.view.InflateException:二进制XML文件行#7:膨胀类时出错
在android.app.ActivityThread.performLaunchActivity(ActivityThread.java:1651)上
在android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:1667)上
在android.app.ActivityThread.access,售价1500美元(ActivityThread.java:117)
在android.app.ActivityThread$H.handleMessage(ActivityThread.java:935)上
位于android.os.Handler.dispatchMessage(Handler.java:99)
位于android.os.Looper.loop(Looper.java:130)
位于android.app.ActivityThread.main(ActivityThread.java:3687)
位于java.lang.reflect.Method.Invokenactive(本机方法)
位于java.lang.reflect.Method.invoke(Method.java:507)
在com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:867)上
位于com.android.internal.os.ZygoteInit.main(ZygoteInit.java:625)
在dalvik.system.NativeStart.main(本机方法)
原因:android.view.InflateException:二进制XML文件行#7:膨胀类时出错
在android.view.LayoutInflater.createView(LayoutInflater.java:518)
在com.android.internal.policy.impl.PhoneLayoutInflater.onCreateView(PhoneLayoutInflater.java:56)上
位于android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:568)
位于android.view.LayoutInflater.rInflate(LayoutInflater.java:623)
在android.view.LayoutInflater.充气(LayoutInflater.java:408)
在android.view.LayoutInflater.充气(LayoutInflater.java:320)
在android.view.LayoutInflater.inflate(LayoutInflater.java:276)
在com.example.MyView.onFinishInflate(MyView.java:32)
位于android.view.LayoutInflater.rInflate(LayoutInflater.java:631)
在android.view.LayoutInflater.充气(LayoutInflater.java:408)
在android.view.LayoutInflater.充气(LayoutInflater.java:320)
在android.view.LayoutInflater.inflate(LayoutInflater.java:276)
在com.example.MyView.onFinishInflate(MyView.java:32)
位于android.view.LayoutInflater.rInflate(LayoutInflater.java:631)
在android.view.LayoutInflater.充气(LayoutInflater.java:408)
在android.view.LayoutInflater.充气(LayoutInflater.java:320)
在android.view.LayoutInflater.inflate(LayoutInflater.java:276)
在com.example.MyView.onFinishInflate(MyView.java:32)
位于android.view.LayoutInflater.rInflate(LayoutInflater.java:631)
在android.view.LayoutInflater.充气(LayoutInflater.java:408)
在android.view.LayoutInflater.充气(LayoutInflater.java:320)
在android.view.LayoutInflater.inflate(LayoutInflater.java:276)
在com.example.MyView.onFinishInflate(MyView.java:32)
位于android.view.LayoutInflater.rInflate(LayoutInflater.java:631)
在android.view.LayoutInflater.充气(LayoutInflater.java:408)
在android.view.LayoutInflater.充气(LayoutInflater.java:320)
在android.view.LayoutInflater.inflate(LayoutInflater.java:276)
....

您不能像以前那样声明自定义视图。如果在主布局中使用自定义视图,将调用
onFinishInflate
方法来构造自定义视图,并在该方法中调用上面的布局。问题是该布局中已经有一个自定义视图引用(当
LayoutInflater
遇到自定义视图标记时,该视图引用将再次膨胀),因此您将处于循环膨胀状态

你可能想要这样的东西:

<com.example.MyView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content">
</com.example.MyView>
<merge xmlns:android="http://schemas.android.com/apk/res/android>

    <Button android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="custom button 1"/>

    <Button android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="custom button 2"/>

</merge>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" 
   android:layout_width="match_parent"
   android:layout_height="match_parent">

    <Button android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="custom button 1"/>

    <Button android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="custom button 2"/>

</LinearLayout>
编辑:在
onfinish充气
方法中充气内容布局似乎会再次触发该方法(至少在旧版本上)。您可以修改布局以使用
线性布局
,如下所示:

<com.example.MyView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content">
</com.example.MyView>
<merge xmlns:android="http://schemas.android.com/apk/res/android>

    <Button android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="custom button 1"/>

    <Button android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="custom button 2"/>

</merge>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" 
   android:layout_width="match_parent"
   android:layout_height="match_parent">

    <Button android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="custom button 1"/>

    <Button android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="custom button 2"/>

</LinearLayout>
这将在布局层次结构中引入另一个无用的元素,因此您可以简单地在自定义视图的构造函数中使用
merge
标记扩展布局:

public MyView(Context context) {
    super(context);
    LayoutInflater.from(getContext()).inflate(R.layout.test_view, this);
}

public MyView(Context context, AttributeSet attrs) {
    super(context, attrs);
    LayoutInflater.from(getContext()).inflate(R.layout.test_view, this);
}

public MyView(Context context, AttributeSet attrs, int defStyle) {
    super(context, attrs, defStyle);
    LayoutInflater.from(getContext()).inflate(R.layout.test_view, this);
}

merge
也不起作用。错误消息是
android.view.InflateException:Binary-XML文件行#5:error-inflating class
它在我的安卓4.1平板电脑上工作,但在我的安卓2.3手机上不工作。如何使其在安卓2.3(甚至2.2)上工作?@Freewind您是否将安卓名称空间添加到
merge
标记中<代码>我做了,和你的一模一样。