使画布视图无效时单击按钮后应用程序崩溃(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)的主要内容,如果未能解决你的问题,请参考以下文章