Как я могу решить эту проблему параллелизма в Android Java? - PullRequest
1 голос
/ 04 октября 2019

Поток getArt () блокируется, когда блокировка настроена на ожидание в течение 5 секунд, через 5 секунд поток возобновляется и возвращается ответ. Но поток прерывается после того, как установлена ​​блокировка.

Как я могу предотвратить прерывание потока и заставить его продолжать извлекать ответ, пока блокировка ожидает, и после того, как ответ будет возвращен, он должен mlock. notifyAll ();чтобы возобновить все, и мой код работает.

Я пробовал это без цикла while раньше, и он возвращает ответ после того, как значение используется как нуль другим методом.

Пожалуйста, помогите.

private void setArt(String imgArt){
    this.imgArt = imgArt;
}

private void setDone(){
    this.done = true;
}

private void getArt(){
    mLock = new Object();

    Thread thread = new Thread() {
        @Override
        public void run() {
            try {
                StringRequest stringRequest = new StringRequest(Request.Method.POST, String.format("%sFetchMP3Image.php", Station_Util.URL),
                        new Response.Listener<String>() {
                            @Override
                            public void onResponse(String response) {
                                try {
                                    System.out.println("Response received.");
                                    JSONObject Response = new JSONObject(response);
                                    String Status = Response.getString("Response");
                                    if (Status.equals("Success")) {
                                        JSONObject MP3File = Response.getJSONObject("MP3");
                                        String ImageURL = MP3File.getString("ImageURL");
                                        setArt(ImageURL);
                                        mLock.notifyAll();
                                        setDone();
                                    }
                                } catch (Exception e) {

                                }
                            }
                        },
                        new Response.ErrorListener() {
                            @Override
                            public void onErrorResponse(VolleyError error) {

                            }
                        }) {
                    @Override
                    protected Map<String, String> getParams() {
                        Map<String, String> params = new HashMap<String, String>();
                        params.put("MP3Name", "687956770_1134920615_1196563259.mp3");
                        return params;
                    }

                };

                Volley.newRequestQueue(My_Downloads.this).add(stringRequest);
            } catch (Exception e) {
            }
        }
    };
    thread.start();

    try {
        while(!this.done) {
            synchronized (mLock) {
                System.out.println("Thread Alive Before: " + thread.isAlive());
                System.out.println("Thread Alive Before: " + thread.getState());
                System.out.println("Thread Alive Before: " + thread.isInterrupted());
                mLock.wait(5000);
                System.out.println("Thread Alive After: " + thread.isAlive());
                System.out.println("Thread Alive After: " + thread.getState());
                System.out.println("Thread Alive After: " + thread.isInterrupted());
            }
        }
        SetList();
    } catch (InterruptedException e) {
        e.printStackTrace();
    }
}

РЕДАКТИРОВАТЬ 1: Я решил это самостоятельно, используя вторичный поток для чтения значения, а не поток пользовательского интерфейса, как я делал раньше. Спасибо за голосование и всех, кто пытался помочь.

Решение:

private void setArt(String imgArt){
    this.imgArt = imgArt;
}

private void setDone(){
    this.done = true;
}

private boolean getDone()
{
    return this.done;
}

private void setListed(){
    this.listed = true;
}

private boolean getListed(){
    return this.listed;
}

private void getArt(){
    mLock = new Object();

    final Thread thread = new Thread() {
        @Override
        public void run() {
            try {
                synchronized (mLock) {
                    StringRequest stringRequest = new StringRequest(Request.Method.POST, String.format("%sFetchMP3Image.php", Station_Util.URL),
                            new Response.Listener<String>() {
                                @Override
                                public void onResponse(String response) {
                                    try {
                                        System.out.println("Response received: " + response);
                                        JSONObject Response = new JSONObject(response);
                                        String Status = Response.getString("Response");
                                        if (Status.equals("Success")) {
                                            System.out.println("Status: " + Status);
                                            JSONObject MP3File = Response.getJSONObject("MP3");
                                            String ImageURL = MP3File.getString("ImageURL");
                                            System.out.println("ImageURL: " + ImageURL);
                                            setArt(ImageURL);
                                            //mLock.notifyAll();
                                            setDone();
                                            System.out.println("Done in API thread: " + getDone());
                                        }
                                    } catch (Exception e) {

                                    }
                                }
                            },
                            new Response.ErrorListener() {
                                @Override
                                public void onErrorResponse(VolleyError error) {

                                }
                            }) {
                        @Override
                        protected Map<String, String> getParams() {
                            Map<String, String> params = new HashMap<String, String>();
                            params.put("MP3Name", "687956770_1134920615_1196563259.mp3");
                            return params;
                        }

                    };

                    Volley.newRequestQueue(My_Downloads.this).add(stringRequest);
                }
            } catch(Exception e){
            }
        }
    };
    thread.start();

    Thread thread2 = new Thread() {
        @Override
        public void run() {
            try {
                while (!getDone()) {
                    synchronized (mLock) {
                        mLock.wait(250);
                        System.out.println("API Thread Alive: " + thread.isAlive());
                        System.out.println("Done in Reading Thread: " + getDone());

                        new Handler(Looper.getMainLooper()).post(new Runnable() {
                            @Override
                            public void run() {
                                System.out.println("UI Thread Done: " + getDone());
                                if (getDone()){
                                    setListed();
                                    SetList();
                                }
                            }
                        });
                    }
                }

                if (getDone() && !getListed()) {
                    setListed();
                    SetList();
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    };
    thread2.start();
}

1 Ответ

1 голос
/ 04 октября 2019

Не следует использовать метод notifyAll вне блока синхронизации. В противном случае будет выдано исключение java.lang.IllegalMonitorStateException .

Когда вы используете метод wait, переменная блокировки будет освобождена, и внутренний поток сможет использовать этот монитор.

...