>目标:我想在应用程序在后台时增加计数器,当应用程序现在处于恢复状态时,增量应该停止,然后计数器的最后一个值将显示在屏幕中。
实际:但是当我运行代码时,如果应用程序在后台,计数器会成功递增。但是当我恢复时,计数器仍在递增。如果此线程现在处于恢复状态,我该如何停止?
这是我在MainActivity中的代码.java
package com.example.background;
import androidx.appcompat.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;
public class MainActivity extends AppCompatActivity {
TextView timeTextView;
Button buttonCtr;
int counter=0;
boolean isResume;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
timeTextView = (TextView) findViewById(R.id.timeLbl);
buttonCtr = (Button) findViewById(R.id.ctrUp);
buttonCtr.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
counter++;
timeTextView.setText(String.valueOf(counter));
}
});
}
class backgroundProcess extends Thread {
@Override
public void run() {
while(isResume == false ) {
counter++;
System.out.println(counter);
}
System.out.println("LAST COUNT:" + counter);
}
}
@Override
protected void onPause() {
super.onPause();
isResume=false;
System.out.println("IM IN PAUSE");
System.out.println("IS RESUME: "+ isResume);
new Thread(new backgroundProcess()).start();
}
@Override
protected void onRestart() {
super.onRestart();
isResume=true; //to stop backgroundProcess Thread
System.out.println("IM IN RESUME");
System.out.println("IS RESUME: "+ isResume);
System.out.println("LAST COUNT:"+ counter);
timeTextView.setText(String.valueOf(counter));
}
}
这是我的XML文件的代码
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MainActivity">
<TextView
android:id="@+id/timeLbl"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="0"
android:textSize="18sp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintHorizontal_bias="0.482"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintVertical_bias="0.397" />
<Button
android:id="@+id/ctrUp"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginBottom="196dp"
android:text="Button"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintHorizontal_bias="0.475"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/timeLbl"
app:layout_constraintVertical_bias="0.54" />
</androidx.constraintlayout.widget.ConstraintLayout>
这里有两件事:
您没有使用任何同步基元。这意味着后台线程对
isResume
的读取是不可靠的;后台线程可能会显示isResume
false
,即使 UI 线程已将其设置为true
。在此特定实例中,您应该能够将isResume
声明为volatile
,这将消除缓存一致性问题。您正在以极高的速率向logcat转储信息,可能是每秒数十万行(
System.out.println(counter)
)。Android Studio 无法以这样的速度读取和显示日志猫信息。在这种情况下,它通常会滞后,这可能会导致您的线程看起来正在运行,即使它已经停止。
我还应该提到,您对counter
的访问不是线程安全的。与isResume
不同,您正在对来自两个线程的counter
执行读/写操作(++
),因此volatile
关键字不足以修复。您必须确保对counter
的所有访问都受到实际锁的保护(例如,将它们包装在synchronized(MainActivity.this)
块中)。