使画布视图无效时单击按钮后应用程序崩溃(Android Studio,Java)

Posted

技术标签:

【中文标题】使画布视图无效时单击按钮后应用程序崩溃(Android Studio,Java)【英文标题】:App crashes after button click when invalidating canvas view (Android Studio, Java) 【发布时间】:2018-05-31 01:59:53 【问题描述】:

当我调用invalidate() 刷新视图时,我的应用程序在单击按钮后崩溃。我不知道为什么。如果我删除canvasView.invalidate(),应用程序不会崩溃。是否允许从另一个班级拨打invalidate()?我是 android Studio 的新手。当然,有人可以帮助我吗? 编辑:问题解决了,答案如下,我加了LogCat。

代码如下:

public class MainActivity extends AppCompatActivity implements 
View.OnClickListener 

    CanvasView canvasView;
    Button btnChangeColor;
    static boolean colorRed;

    @Override
    protected void onCreate(Bundle savedInstanceState) 
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        CanvasView canvasView = new CanvasView(this, null);
        btnChangeColor = (Button) findViewById(R.id.button);

        btnChangeColor.setOnClickListener(this);

        colorRed = false;

    

    @Override
    public void onClick(View v) 
        if(colorRed) 
            colorRed = false;
            canvasView.invalidate();
            return;
        

        else 
            colorRed = true;
            canvasView.invalidate();
            return;
        
    





class CanvasView extends View 

    Paint paint;

    public CanvasView(Context context, @Nullable AttributeSet attrs) 
        super(context, attrs);

        paint = new Paint();
    


    @Override
    protected void onDraw(Canvas canvas) 
        super.onDraw(canvas);

        if(MainActivity.colorRed == true)
            paint.setColor(Color.RED);
        else
            paint.setColor(Color.BLACK);

        paint.setStyle(Paint.Style.FILL);

        canvas.drawCircle(400, 400, 50, paint);
    

LogCat:

05-31 21:10:41.341 27902-27902/? I/art: 后期启用 -Xcheck:jni 重新初始化属性:dalvik.vm.checkjni=false

05-31 21:10:41.351 27902-27909/? I/art:调试器不再处于活动状态 启动阻塞式 GC 检测

05-31 21:10:41.425 27902-27902/? W/System:ClassLoader 引用了未知路径:/data/app/com.example.andreas.drawing_exp-2/lib/arm64

05-31 21:10:41.433 27902-27902/? I/InstantRun:启动即时运行服务器:是主进程

05-31 21:10:41.450 27902-27902/? I/HwCust:为类 android.app.HwCustActivityImpl 找到构造函数

05-31 21:10:41.465 27902-27902/? I/HwCust:为类 android.app.HwCustHwWallpaperManagerImpl 找到构造函数

05-31 21:10:41.477 27902-27902/? W/art:在 Android 4.1 之前,方法 android.graphics.PorterDuffColorFilter android.support.graphics.drawable.VectorDrawableCompat.updateTintFilter(android.graphics.PorterDuffColorFilter, android.content.res.ColorStateList, android.graphics.PorterDuff$Mode)错误地覆盖了 android.graphics.drawable.Drawable 中的 package-private 方法

05-31 21:10:41.533 27902-27913/? I/art:后台粘性并发标记扫描 GC 释放 2801(763KB) AllocSpace objects,0(0B) LOS objects,22% free,3MB/4MB,paused 5.329ms total 13.242ms

05-31 21:10:41.567 27902-27902/? W/VRSystemServiceManager:虚拟现实服务不活跃

05-31 21:10:41.568 27902-27902/? I/HwSecImmHelper: mSecurityInputMethodService 为空

05-31 21:10:41.571 27902-27902/? I/HwPointEventFilter: 支持 AFT

05-31 21:10:41.615 27902-27922/? I/OpenGLRenderer:初始化的 EGL,版本 1.4

05-31 21:10:41.621 27902-27922/? W/链接器:/vendor/lib64/libhwuibp.so:未使用的 DT 条目:类型 0xf arg 0xe3a

05-31 21:10:41.629 27902-27902/? W/art:在 Android 4.1 之前,方法 int android.support.v7.widget.DropDownListView.lookForSelectablePosition(int, boolean) 会错误地覆盖 android.widget.ListView 中的包私有方法

05-31 21:10:48.728 27902-27902/com.example.andreas.drawing_exp I/hwaps: JNI_OnLoad

05-31 21:10:48.802 27902-27902/com.example.andreas.drawing_exp E/AndroidRuntime: 致命异常: main 进程:com.example.andreas.drawing_exp,PID:27902 java.lang.NullPointerException:尝试从空对象引用上的字段“boolean com.example.andreas.drawing_exp.CanvasView.colorRed”读取 在 com.example.andreas.drawing_exp.MainActivity.onClick(MainActivity.java:37) 在 android.view.View.performClick(View.java:5646) 在 android.view.View$PerformClick.run(View.java:22473) 在 android.os.Handler.handleCallback(Handler.java:761) 在 android.os.Handler.dispatchMessage(Handler.java:98) 在 android.os.Looper.loop(Looper.java:156) 在 android.app.ActivityThread.main(ActivityThread.java:6523) 在 java.lang.reflect.Method.invoke(本机方法) 在 com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:942) 在 com.android.internal.os.ZygoteInit.main(ZygoteInit.java:832)

05-31 21:10:48.837 27902-27902/com.example.andreas.drawing_exp I/Process:发送信号。 PID:27902 SIG:9

【问题讨论】:

请同时发布您的日志 阅读***.com/questions/23353173/…,了解如何查找错误源的提示。 由于颜色与画布的关系比活动更多,所以colorRed 变量应该在CanvasView 类中。使用static在类之间共享变量是对static的不当使用。 Unfortunately MyApp has stopped. How can I solve this?的可能重复 感谢您的提示。我发现了我的应用程序崩溃的原因。如果您有兴趣,请在下面查看我的答案。 【参考方案1】:

您错误地使用了invalidate()

这必须从 UI 线程调用。要从非 UI 线程调用,请调用 postInvalidate()

您使用postInvalidate() 来执行此操作。

【讨论】:

即使我调用 canvasView.postInvalidate(),应用程序也会崩溃。什么是 UI 线程? 请注意,所有事件处理程序都在 UI 线程上运行,包括 OP 中的 onClick() 方法,其中类 invalidate()【参考方案2】:

在观看了一些有关如何在 YouTube 中自定义视图中绘图的教程后,我发现了我的应用程序崩溃的原因。 这是因为我以错误的方式初始化了画布视图类。我没有为视图添加 ID,也没有使用“findViewById”进行初始化,而是使用了类的构造函数。但无论如何,谢谢你的回答! 这是更改后的代码的样子(原始代码的简短摘录):

public class MainActivity extends AppCompatActivity implements 
View.OnClickListener

    CanvasView canvasView;
    Button btnChangeColor;

    @Override
    protected void onCreate(Bundle savedInstanceState) 
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        canvasView = (CanvasView) findViewById(R.id.canvasview);
        btnChangeColor = (Button) findViewById(R.id.button);

        btnChangeColor.setOnClickListener(this);

        canvasView.colorRed = false;

    

【讨论】:

注意,使用构造函数创建自定义视图并没有错。但是,您还需要将新视图添加到活动的视图层次结构中。假设 CanvasView 在 XML 布局中用作标记,则在此处使用您的解决方案要简单得多。

以上是关于使画布视图无效时单击按钮后应用程序崩溃(Android Studio,Java)的主要内容,如果未能解决你的问题,请参考以下文章

从表视图中删除数据会使应用程序崩溃

UINavigationController 应用程序在单击“返回”按钮时崩溃

单击按钮时如何保存画布绘图并显示它

在执行功能时单击导航控制器上的后退按钮时应用程序崩溃

如何在单选按钮单击时使按钮无效?

搜索后重新加载视图会使应用程序崩溃