gpt4 book ai didi

java - 使用线程和异步任务无法防止 ANR 错误

转载 作者:太空宇宙 更新时间:2023-11-04 09:43:48 27 4
gpt4 key购买 nike

这几天我遇到了一个大问题。事实上,在我的 Activity 中,我调用了使用异步任务执行的 NTP 服务器,并且执行了多个数据库操作(我使用的是 Firebase 数据库),每当此 Activity 打开时,都会出现 ANR 错误。我读到,对于长任务,我们应该使用工作线程,因此我将数据库操作放在一个线程中,但即使如此,我仍然收到 ANR 错误。您还有什么建议?(NTP服务器返回正确的结果,并且每次都完成数据库中的所有操作,即使我收到ANR错误。)

public class Start extends AppCompatActivity {
//declaring variables
@Override
protected void onCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_start);


new Thread(new Runnable() {
public void run(){
finishQuiz();
}
}).start();

final DatabaseReference myRef = FirebaseDatabase.getInstance().getReference();

myRef.addValueEventListener(new ValueEventListener() {
//in this part, I call getNetworkTime, and under specific conditions I set an intent AlarmManager

}

这是我在线程内调用的函数 finishQuiz() :

 public void finishQuiz() {

// DEFINING THE VARIABLES NEEDED

myRef.addListenerForSingleValueEvent( new ValueEventListener() {
String dateD, hourD, userID, nom, prenom;
@Override
public void onDataChange(@NonNull DataSnapshot dataSnapshot) {
Date d = new Date();
timee = d.getTime();
try {
timee = new Start.getNetworkTime().execute().get();
} catch (Exception e) {
timee = d.getTime();
}


String dateD = dataSnapshot.child("quiz").child("dateD").getValue(String.class);
String hourD = dataSnapshot.child("quiz").child("heureD").getValue(String.class);
if (!(dateD.equals("")) && !(hourD.equals(""))) {

long nbQ = dataSnapshot.child("quiz").child("questions").getChildrenCount();
int dureeQ = Integer.valueOf(dataSnapshot.child("quiz").child("duree").getValue().toString());
String[] D = dateD.split("-", 3);
String[] H = hourD.split(":", 3);
Calendar calDebut = Calendar.getInstance();
calDebut.set(Integer.valueOf(D[2]), Integer.valueOf(D[1]) - 1, Integer.valueOf(D[0]), Integer.valueOf(H[0]), Integer.valueOf(H[1]), Integer.valueOf(H[2]));
Date dateDebutQ = calDebut.getTime();
long timeQFinInMillis = dateDebutQ.getTime() + (nbQ + 1) * dureeQ * 1000;

if (timee > timeQFinInMillis) {



try {
final SimpleDateFormat formatter = new SimpleDateFormat("dd-MM-yyyy HH:mm");
Calendar caldateQ = Calendar.getInstance();
String[] dateDT = dateD.split("-", 3);
String[] hourDT = hourD.split(":", 3);
caldateQ.set(Integer.valueOf(dateDT[2]), Integer.valueOf(dateDT[1]) - 1, Integer.valueOf(dateDT[0]), Integer.valueOf(hourDT[0]), Integer.valueOf(hourDT[1]), Integer.valueOf(hourDT[2]));
long c = caldateQ.getTimeInMillis();
List<Date> listPast = new ArrayList<>();
List<Date> listFuture = new ArrayList<>();
for (DataSnapshot ds : dataSnapshot.child("next quizs").getChildren()) {
String dateNext = ds.getKey();
try {

Date dateNx = formatter.parse(dateNext);
Date dateQQQ = caldateQ.getTime();
Date dateQQ = formatter.parse(formatter.format(dateQQQ));
long timeNx = dateNx.getTime();
if (timeNx < timee) {
listPast.add(dateNx);

}
else
{
listFuture.add(dateNx);
}
} catch (Exception e) {

}
}

for (int i = 0; i < listPast.size(); i++) {
myNextQuiz.child(formatter.format(listPast.get(i))).removeValue();
}

if (listFuture.size()==0) {
myQuiz.child("dateD").setValue("");
myQuiz.child("heureD").setValue("");
myQuiz.child("duree").setValue(10);
myQuiz.child("questions").setValue("This will be removed anyways");
} else
//I get ANR error in this case
{



Date minuD = listFuture.get(0);
for (int k = 1; k < listFuture.size(); k++) {
if ((listFuture.get(k).compareTo(minuD) < 0)) {
minuD = listFuture.get(k);
}
}
String date = formatter.format(minuD);
String[] DH = date.split(" ", 2);
myQuiz.child("dateD").setValue(DH[0]);
myQuiz.child("heureD").setValue(DH[1] + ":00");
String ddd = dataSnapshot.child("next quizs").child(date).child("duree").getValue().toString();
myQuiz.child("duree").setValue(ddd);
myQuiz.child("questions").removeValue();
int j = 1;
for (DataSnapshot ds : dataSnapshot.child("next quizs").child(date).child("questions").getChildren()) {
String quest = ds.child("question").getValue(String.class);
String sol = ds.child("sol").getValue(String.class);
String answA = ds.child("A").getValue(String.class);
String answB = ds.child("B").getValue(String.class);
String answC = ds.child("C").getValue(String.class);
String answD = ds.child("D").getValue(String.class);
myQuiz.child("questions").child("" + j).child("question").setValue(quest);
myQuiz.child("questions").child("" + j).child("sol").setValue(sol);
myQuiz.child("questions").child("" + j).child("A").setValue(answA);
myQuiz.child("questions").child("" + j).child("B").setValue(answB);
myQuiz.child("questions").child("" + j).child("C").setValue(answC);
myQuiz.child("questions").child("" + j).child("D").setValue(answD);
j++;
}
}

myPreQuiz.child(dateD + " " + hourD).child("dateD").setValue(dateD);
myPreQuiz.child(dateD + " " + hourD).child("heureD").setValue(hourD);
for (DataSnapshot ds : dataSnapshot.child("current quiz").child("number").getChildren()) {
userID = ds.getKey();
nom = dataSnapshot.child("utilisateurs").child(userID).child("Nom").getValue(String.class);
prenom = dataSnapshot.child("utilisateurs").child(userID).child("Prenom").getValue(String.class);
myPreQuiz.child(dateD + " " + hourD).child("gagnants").child(userID).setValue(nom + " " + prenom);
}
myCurrQuiz.child("currentQ").setValue("0");
myCurrQuiz.child("number").removeValue();

} catch (Exception e) {
//show Exception

} }}}

@Override
public void onCancelled(@NonNull DatabaseError databaseError) {

}
});}

这是我的异步任务:

private class getNetworkTime extends AsyncTask<Void, Void, Long> {

@Override
protected Long doInBackground(Void... params) {
String TIME_SERVER = "2.android.pool.ntp.org";
try{
NTPUDPClient timeClient = new NTPUDPClient();
InetAddress inetAddress = InetAddress.getByName(TIME_SERVER);
TimeInfo timeInfo = timeClient.getTime(inetAddress);
long returnTime = timeInfo.getMessage().getTransmitTimeStamp().getTime(); //server time

return returnTime;
} catch (Exception e) {
Date d = new Date();
long l=d.getTime();
return l;
}
}


protected void onPostExecute(Long result) {

}

@Override
protected void onPreExecute() {

}

@Override
protected void onProgressUpdate(Void... values) {
}
}

如果有帮助,在我使用 NTP 服务器之前,我的代码曾经完美地工作,并且我在没有 AlarmManager 部分的情况下进行了测试,它也工作得很好,而且,在函数 finishQuiz() 中,正如我在评论中提到的,只有特定情况才会出现 ANR 错误(这恰好是我的应用程序最常见的情况),我的意思是,如果不输入该 block ,就不会出现 ANR 错误。

最佳答案

即使您开始在后台线程上执行 finishQuiz 方法,Firebase 回调也会在主 (UI) 线程上回调。这可能会导致您的 ANR。

尝试在后台线程上执行 onDataChange() 回调中的代码。

new Thread(new Runnable() {
public void run(){

// PUT YOUR CODE FROM onDataChange() HERE, or use AsyncTask


}
}).start();

请参阅:Firebase Android: onDataChange() event always executed in Main UI Thread?

关于java - 使用线程和异步任务无法防止 ANR 错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55661035/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com