Android“ เฉพาะเธรดดั้งเดิมที่สร้างลำดับชั้นมุมมองเท่านั้นที่สามารถแตะมุมมองได้”


940

ฉันสร้างเครื่องเล่นเพลงธรรมดาใน Android มุมมองสำหรับเพลงแต่ละเพลงมี SeekBar ซึ่งดำเนินการดังนี้:

public class Song extends Activity implements OnClickListener,Runnable {
    private SeekBar progress;
    private MediaPlayer mp;

    // ...

    private ServiceConnection onService = new ServiceConnection() {
          public void onServiceConnected(ComponentName className,
            IBinder rawBinder) {
              appService = ((MPService.LocalBinder)rawBinder).getService(); // service that handles the MediaPlayer
              progress.setVisibility(SeekBar.VISIBLE);
              progress.setProgress(0);
              mp = appService.getMP();
              appService.playSong(title);
              progress.setMax(mp.getDuration());
              new Thread(Song.this).start();
          }
          public void onServiceDisconnected(ComponentName classname) {
              appService = null;
          }
    };

    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.song);

        // ...

        progress = (SeekBar) findViewById(R.id.progress);

        // ...
    }

    public void run() {
    int pos = 0;
    int total = mp.getDuration();
    while (mp != null && pos<total) {
        try {
            Thread.sleep(1000);
            pos = appService.getSongPosition();
        } catch (InterruptedException e) {
            return;
        } catch (Exception e) {
            return;
        }
        progress.setProgress(pos);
    }
}

ใช้งานได้ดี ตอนนี้ฉันต้องการตัวจับเวลานับวินาที / นาทีของความคืบหน้าของเพลง ดังนั้นผมจึงใส่TextViewในรูปแบบที่ได้รับมันด้วยfindViewById()ในonCreate()และใส่ในrun()หลังprogress.setProgress(pos):

String time = String.format("%d:%d",
            TimeUnit.MILLISECONDS.toMinutes(pos),
            TimeUnit.MILLISECONDS.toSeconds(pos),
            TimeUnit.MINUTES.toSeconds(TimeUnit.MILLISECONDS.toMinutes(
                    pos))
            );
currentTime.setText(time);  // currentTime = (TextView) findViewById(R.id.current_time);

แต่บรรทัดสุดท้ายนั่นทำให้ฉันมีข้อยกเว้น:

android.view.ViewRoot $ CalledFromWrongThreadException: เฉพาะเธรดดั้งเดิมที่สร้างลำดับชั้นมุมมองเท่านั้นที่สามารถแตะมุมมองได้

แต่โดยพื้นฐานแล้วฉันกำลังทำสิ่งเดียวกับที่ฉันทำกับSeekBar- สร้างมุมมองonCreateจากนั้นแตะมันrun()- และมันก็ไม่ได้ทำให้ฉันมีข้อร้องเรียนนี้

คำตอบ:


1894

คุณต้องย้ายส่วนของงานพื้นหลังที่อัปเดต UI ไปยังเธรดหลัก มีรหัสง่าย ๆ สำหรับสิ่งนี้:

runOnUiThread(new Runnable() {

    @Override
    public void run() {

        // Stuff that updates the UI

    }
});

Activity.runOnUiThreadเอกสาร

เพียงซ่อนสิ่งนี้ไว้ภายในเมธอดที่กำลังทำงานอยู่ในพื้นหลังแล้วคัดลอกวางโค้ดที่ใช้การอัพเดตใด ๆ ที่อยู่ตรงกลางของบล็อก รวมเฉพาะโค้ดจำนวนน้อยที่สุดเท่าที่จะเป็นไปได้มิฉะนั้นคุณจะเริ่มเอาชนะวัตถุประสงค์ของเธรดพื้นหลัง


5
ทำงานเหมือนจับใจ สำหรับฉันปัญหาเดียวที่นี่คือฉันต้องการที่จะทำวิธีการerror.setText(res.toString());ภายใน () แต่ฉันไม่สามารถใช้ความละเอียดเพราะมันไม่ได้สุดท้าย .. เลวร้ายเกินไป
โนแลน

64
หนึ่งความคิดเห็นสั้น ๆ เกี่ยวกับเรื่องนี้ ฉันมีเธรดแยกต่างหากที่พยายามปรับเปลี่ยน UI และโค้ดด้านบนทำงานได้ แต่ฉันเรียก runOnUiThread จากวัตถุกิจกรรม ฉันต้องทำอะไรบางอย่างเช่น myActivityObject.runOnUiThread(etc)
เคอร์บี้

1
@Kirby ขอบคุณสำหรับการอ้างอิงนี้ คุณสามารถทำ 'MainActivity.this' ได้และมันก็ใช้ได้เช่นกันเพื่อที่คุณจะได้ไม่ต้องอ้างอิงกับคลาสกิจกรรมของคุณ
JRomero

24
ฉันใช้เวลาสักครู่เพื่อคิดออกว่าrunOnUiThread()เป็นวิธีการของกิจกรรม ฉันกำลังเรียกใช้รหัสของฉันในส่วน ฉันลงเอยด้วยการทำงานgetActivity().runOnUiThread(etc)และใช้งานได้ Fantastic !;
lejonl

เราสามารถหยุดงานที่กำลังทำอยู่ซึ่งเขียนในเนื้อความของ 'runOnUiThread' ได้หรือไม่?
Karan Sharma

143

ฉันแก้ไขสิ่งนี้ได้โดยใส่runOnUiThread( new Runnable(){ ..เข้าไปข้างในrun():

thread = new Thread(){
        @Override
        public void run() {
            try {
                synchronized (this) {
                    wait(5000);

                    runOnUiThread(new Runnable() {
                        @Override
                        public void run() {
                            dbloadingInfo.setVisibility(View.VISIBLE);
                            bar.setVisibility(View.INVISIBLE);
                            loadingText.setVisibility(View.INVISIBLE);
                        }
                    });

                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            Intent mainActivity = new Intent(getApplicationContext(),MainActivity.class);
            startActivity(mainActivity);
        };
    };  
    thread.start();

2
อันนี้สั่นสะเทือน ขอบคุณสำหรับข้อมูลที่สามารถใช้ในกระทู้อื่น ๆ ได้
Nabin

ขอขอบคุณเป็นเรื่องน่าเศร้าจริง ๆ ที่ได้สร้างเธรดเพื่อกลับไปยัง UI เธรด แต่โซลูชันนี้ได้บันทึกเคสของฉันแล้ว
Pierre Maoui

2
สิ่งสำคัญอย่างหนึ่งwait(5000);คือไม่ได้อยู่ใน Runnable ไม่เช่นนั้น UI ของคุณจะค้างระหว่างรอ คุณควรพิจารณาใช้AsyncTaskแทน Thread สำหรับการดำเนินการเช่นนี้
มาร์ติน

นี่แย่มากสำหรับการรั่วไหลของหน่วยความจำ
ราฟาเอลลิมา

ทำไมต้องกังวลกับบล็อกซิงโครไนซ์ รหัสข้างในนั้นดูปลอดภัยพอสมควร (แม้ว่าฉันพร้อมที่จะกินคำพูด)
David

69

โซลูชันของฉันสำหรับสิ่งนี้:

private void setText(final TextView text,final String value){
    runOnUiThread(new Runnable() {
        @Override
        public void run() {
            text.setText(value);
        }
    });
}

เรียกวิธีนี้ในเธรดพื้นหลัง


ข้อผิดพลาด: (73, 67) ข้อผิดพลาด: ไม่สามารถอ้างอิงชุดวิธีแบบไม่คงที่ (สตริง) จากบริบทแบบคงที่

1
ฉันมีปัญหาเดียวกันกับชั้นเรียนทดสอบ สิ่งนี้ได้ผลเหมือนมนต์เสน่ห์สำหรับฉัน แต่แทนที่ด้วยrunOnUiThread runTestOnUiThreadขอบคุณ
DaddyMoe

28

โดยปกติแล้วการกระทำใด ๆ ที่เกี่ยวข้องกับส่วนต่อประสานผู้ใช้จะต้องทำใน main หรือ UI thread นั่นคือสิ่งที่onCreate()ดำเนินการจัดการเหตุการณ์ วิธีหนึ่งที่จะตรวจสอบให้แน่ใจว่ามีการใช้runOnUiThread ()อีกวิธีหนึ่งคือการใช้ตัวจัดการ

ProgressBar.setProgress() มีกลไกที่มันจะทำงานบนเธรดหลักเสมอนั่นคือสาเหตุที่มันทำงาน

ดูเจ็บปวด Threading


บทความ Painless Threading ที่ลิงค์นั้นตอนนี้เป็น 404 นี่คือลิงค์ไปยังบล็อก (รุ่นเก่า) บน Painless Threading - android-developers.blogspot.com/2009/05/painless-threading.html
Tony Adams

20

ฉันอยู่ในสถานการณ์นี้ แต่ฉันพบวิธีแก้ปัญหาด้วยตัวจัดการวัตถุ

ในกรณีของฉันฉันต้องการที่จะปรับปรุง ProgressDialog กับรูปแบบการสังเกตการณ์ มุมมองของฉันใช้ผู้สังเกตการณ์และแทนที่วิธีการอัปเดต

ดังนั้นเธรดหลักของฉันสร้างมุมมองและเธรดอื่นเรียกเมธอดการอัพเดตที่อัพเดต ProgressDialop และ .... :

เฉพาะเธรดดั้งเดิมที่สร้างลำดับชั้นมุมมองเท่านั้นที่สามารถแตะมุมมองได้

เป็นไปได้ที่จะแก้ปัญหาด้วยตัวจัดการวัตถุ

ด้านล่างส่วนต่างๆของรหัสของฉัน:

public class ViewExecution extends Activity implements Observer{

    static final int PROGRESS_DIALOG = 0;
    ProgressDialog progressDialog;
    int currentNumber;

    public void onCreate(Bundle savedInstanceState) {

        currentNumber = 0;
        final Button launchPolicyButton =  ((Button) this.findViewById(R.id.launchButton));
        launchPolicyButton.setOnClickListener(new OnClickListener() {

            @Override
            public void onClick(View v) {
                showDialog(PROGRESS_DIALOG);
            }
        });
    }

    @Override
    protected Dialog onCreateDialog(int id) {
        switch(id) {
        case PROGRESS_DIALOG:
            progressDialog = new ProgressDialog(this);
            progressDialog.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);
            progressDialog.setMessage("Loading");
            progressDialog.setCancelable(true);
            return progressDialog;
        default:
            return null;
        }
    }

    @Override
    protected void onPrepareDialog(int id, Dialog dialog) {
        switch(id) {
        case PROGRESS_DIALOG:
            progressDialog.setProgress(0);
        }

    }

    // Define the Handler that receives messages from the thread and update the progress
    final Handler handler = new Handler() {
        public void handleMessage(Message msg) {
            int current = msg.arg1;
            progressDialog.setProgress(current);
            if (current >= 100){
                removeDialog (PROGRESS_DIALOG);
            }
        }
    };

    // The method called by the observer (the second thread)
    @Override
    public void update(Observable obs, Object arg1) {

        Message msg = handler.obtainMessage();
        msg.arg1 = ++currentPluginNumber;
        handler.sendMessage(msg);
    }
}

คำอธิบายนี้สามารถพบได้ในหน้านี้และคุณต้องอ่าน "ตัวอย่าง ProgressDialog ด้วยด้ายที่สอง"


10

คุณสามารถใช้ Handler เพื่อลบมุมมองโดยไม่รบกวน UI Thread หลัก นี่คือตัวอย่างรหัส

new Handler(Looper.getMainLooper()).post(new Runnable() {
                                                        @Override
                                                        public void run() {
                                                           //do stuff like remove view etc
                                                            adapter.remove(selecteditem);
                                                        }
                                                    });

7

ฉันเห็นว่าคุณยอมรับคำตอบของ @ providence แล้ว ในกรณีที่คุณสามารถใช้ตัวจัดการได้เช่นกัน! ก่อนอื่นให้ทำฟิลด์ int

    private static final int SHOW_LOG = 1;
    private static final int HIDE_LOG = 0;

จากนั้นสร้างอินสแตนซ์ของตัวจัดการเป็นฟิลด์

    //TODO __________[ Handler ]__________
    @SuppressLint("HandlerLeak")
    protected Handler handler = new Handler()
    {
        @Override
        public void handleMessage(Message msg)
        {
            // Put code here...

            // Set a switch statement to toggle it on or off.
            switch(msg.what)
            {
            case SHOW_LOG:
            {
                ads.setVisibility(View.VISIBLE);
                break;
            }
            case HIDE_LOG:
            {
                ads.setVisibility(View.GONE);
                break;
            }
            }
        }
    };

ทำวิธีการ

//TODO __________[ Callbacks ]__________
@Override
public void showHandler(boolean show)
{
    handler.sendEmptyMessage(show ? SHOW_LOG : HIDE_LOG);
}

สุดท้ายให้ใส่onCreate()วิธีนี้

showHandler(true);

7

ฉันมีปัญหาที่คล้ายกันและวิธีแก้ปัญหาของฉันน่าเกลียด แต่ก็ใช้งานได้:

void showCode() {
    hideRegisterMessage(); // Hides view 
    final Handler handler = new Handler();
    handler.postDelayed(new Runnable() {
        @Override
        public void run() {
            showRegisterMessage(); // Shows view
        }
    }, 3000); // After 3 seconds
}

2
@ R.jzadeh ดีใจที่ได้ยินเช่นนั้น ตั้งแต่ช่วงเวลาที่ผมได้เขียนคำตอบที่อาจจะเป็นตอนนี้คุณสามารถทำมันได้ดี :)
Błażej

6

ผมใช้กับHandler Looper.getMainLooper()มันทำงานได้ดีสำหรับฉัน

    Handler handler = new Handler(Looper.getMainLooper()) {
        @Override
        public void handleMessage(Message msg) {
              // Any UI task, example
              textView.setText("your text");
        }
    };
    handler.sendEmptyMessage(1);

5

ใช้รหัสนี้และไม่จำเป็นต้องใช้runOnUiThreadฟังก์ชัน:

private Handler handler;
private Runnable handlerTask;

void StartTimer(){
    handler = new Handler();   
    handlerTask = new Runnable()
    {
        @Override 
        public void run() { 
            // do something  
            textView.setText("some text");
            handler.postDelayed(handlerTask, 1000);    
        }
    };
    handlerTask.run();
}

5

นี่คือการโยนข้อผิดพลาดอย่างชัดเจน มันบอกว่าด้ายใดสร้างมุมมองเท่านั้นที่สามารถสัมผัสมุมมองของมัน เป็นเพราะมุมมองที่สร้างขึ้นอยู่ภายในพื้นที่ของเธรดนั้น การสร้างมุมมอง (GUI) เกิดขึ้นในเธรด UI (หลัก) ดังนั้นคุณจะต้องใช้เธรด UI เพื่อเข้าถึงวิธีการเหล่านั้นเสมอ

ป้อนคำอธิบายภาพที่นี่

ในภาพด้านบนตัวแปรความคืบหน้าอยู่ภายในพื้นที่ของเธรด UI ดังนั้นเฉพาะเธรด UI เท่านั้นที่สามารถเข้าถึงตัวแปรนี้ ที่นี่คุณกำลังเข้าถึงความคืบหน้าผ่านทางเธรดใหม่ () และนั่นเป็นสาเหตุที่คุณมีข้อผิดพลาด


4

นี้เกิดขึ้นกับฉันเมื่อฉันเรียกร้องให้มีการเปลี่ยนแปลง UI จากdoInBackgroundจากแทนการใช้AsynctaskonPostExecute

การจัดการกับ UI ในการonPostExecuteแก้ไขปัญหาของฉัน


1
ขอบคุณโจนาธาน นี่คือปัญหาของฉันเช่นกัน แต่ฉันต้องอ่านเพิ่มอีกนิดหน่อยเพื่อเข้าใจสิ่งที่คุณหมายถึงที่นี่ สำหรับคนอื่น ๆonPostExecuteก็เป็นวิธีการAsyncTaskแต่มันทำงานบนเธรด UI ดูที่นี่: blog.teamtreehouse.com/all-about-android-asynctasks
ciaranodc

4

Corotines Kotlin สามารถทำให้โค้ดของคุณกระชับและอ่านง่ายขึ้นเช่นนี้:

MainScope().launch {
    withContext(Dispatchers.Default) {
        //TODO("Background processing...")
    }
    TODO("Update UI here!")
}

หรือในทางกลับกัน:

GlobalScope.launch {
    //TODO("Background processing...")
    withContext(Dispatchers.Main) {
        // TODO("Update UI here!")
    }
    TODO("Continue background processing...")
}

3

ฉันทำงานกับคลาสที่ไม่มีการอ้างอิงถึงบริบท ดังนั้นจึงเป็นไปไม่ได้ที่ฉันจะใช้runOnUIThread();ฉันใช้view.post();และแก้ไขได้

timer.scheduleAtFixedRate(new TimerTask() {

    @Override
    public void run() {
        final int currentPosition = mediaPlayer.getCurrentPosition();
        audioMessage.seekBar.setProgress(currentPosition / 1000);
        audioMessage.tvPlayDuration.post(new Runnable() {
            @Override
            public void run() {
                audioMessage.tvPlayDuration.setText(ChatDateTimeFormatter.getDuration(currentPosition));
            }
        });
    }
}, 0, 1000);

การเปรียบเทียบaudioMessageและtvPlayDurationรหัสคำถามคืออะไร
gotwo

audioMessageเป็นวัตถุผู้ถือของมุมมองข้อความ tvPlayDurationเป็นมุมมองข้อความที่เราต้องการอัปเดตจากเธรด UI ที่ไม่ใช่ ในคำถามข้างต้นcurrentTimeเป็นมุมมองข้อความ แต่ไม่มีวัตถุตัวยึด
Ifta

3

เมื่อใช้AsyncTaskอัพเดต UI ในเมธอดonPostExecute

    @Override
    protected void onPostExecute(String s) {
   // Update UI here

     }

เรื่องนี้เกิดขึ้นกับฉัน ฉันกำลังอัปเดต ui ใน doinbackground ของ asynk task
mehmoodnisar125

3

ฉันกำลังประสบปัญหาที่คล้ายกันและไม่มีวิธีการใด ๆ ที่ได้กล่าวมาใช้งานได้สำหรับฉัน ในท้ายที่สุดสิ่งนี้ทำให้ฉัน:

Device.BeginInvokeOnMainThread(() =>
    {
        myMethod();
    });

ผมพบว่าอัญมณีนี้ที่นี่


2

นี่คือการติดตามสแต็กของข้อยกเว้นที่กล่าวถึง

        at android.view.ViewRootImpl.checkThread(ViewRootImpl.java:6149)
        at android.view.ViewRootImpl.requestLayout(ViewRootImpl.java:843)
        at android.view.View.requestLayout(View.java:16474)
        at android.view.View.requestLayout(View.java:16474)
        at android.view.View.requestLayout(View.java:16474)
        at android.view.View.requestLayout(View.java:16474)
        at android.widget.RelativeLayout.requestLayout(RelativeLayout.java:352)
        at android.view.View.requestLayout(View.java:16474)
        at android.widget.RelativeLayout.requestLayout(RelativeLayout.java:352)
        at android.view.View.setFlags(View.java:8938)
        at android.view.View.setVisibility(View.java:6066)

ดังนั้นถ้าคุณไปขุดแล้วคุณจะรู้

void checkThread() {
    if (mThread != Thread.currentThread()) {
        throw new CalledFromWrongThreadException(
                "Only the original thread that created a view hierarchy can touch its views.");
    }
}

โดยที่mThreadถูกเตรียมใช้งานใน constructor ดังนี้

mThread = Thread.currentThread();

ทั้งหมดที่ฉันพูดคือเมื่อเราสร้างมุมมองเฉพาะเราสร้างมันใน UI Thread และต่อมาพยายามปรับเปลี่ยนในเธรดผู้ทำงาน

เราสามารถตรวจสอบได้โดยใช้ข้อมูลโค้ดด้านล่าง

Thread.currentThread().getName()

เมื่อเราขยายเลย์เอาต์และหลังจากนั้นคุณจะได้รับข้อยกเว้น


2

หากคุณไม่ต้องการใช้runOnUiThreadAPI คุณสามารถนำไปใช้AsynTaskกับการดำเนินการที่ใช้เวลาไม่กี่วินาที แต่ในกรณีนี้หลังจากประมวลผลงานของคุณdoinBackground()แล้วคุณต้องส่งคืนมุมมองที่เสร็จสิ้นonPostExecute()ด้วย การติดตั้ง Android อนุญาตให้เธรด UI หลักเท่านั้นที่สามารถโต้ตอบกับมุมมองได้


2

หากคุณต้องการทำให้โมฆะ (ฟังก์ชันทาสี / โทรออกใหม่) จาก UI เธรดที่ไม่ใช่ของคุณให้ใช้ postInvalidate ()

myView.postInvalidate();

สิ่งนี้จะโพสต์คำขอที่ไม่ถูกต้องบน UI-thread

สำหรับข้อมูลเพิ่มเติม: what-do-postinvalidate-do


1

สำหรับฉันปัญหาคือว่าฉันโทรonProgressUpdate()จากรหัสของฉันอย่างชัดเจน สิ่งนี้ไม่ควรทำ ฉันโทรpublishProgress()แทนและนั่นก็แก้ไขข้อผิดพลาด


1

ในกรณีของฉันฉันมีEditTextในอะแดปเตอร์และมีอยู่แล้วในเธรด UI อย่างไรก็ตามเมื่อกิจกรรมนี้โหลดแสดงว่าเกิดข้อผิดพลาดนี้

โซลูชันของฉันคือฉันต้องลบ<requestFocus />ออกจาก EditText ใน XML


1

สำหรับผู้ที่ดิ้นรนใน Kotlin มันทำงานเช่นนี้:

lateinit var runnable: Runnable //global variable

 runOnUiThread { //Lambda
            runnable = Runnable {

                //do something here

                runDelayedHandler(5000)
            }
        }

        runnable.run()

 //you need to keep the handler outside the runnable body to work in kotlin
 fun runDelayedHandler(timeToWait: Long) {

        //Keep it running
        val handler = Handler()
        handler.postDelayed(runnable, timeToWait)
    }

0

แก้ไขได้: เพียงใส่วิธีนี้ใน doInBackround Class ... และส่งข้อความ

public void setProgressText(final String progressText){
        Handler handler = new Handler(Looper.getMainLooper()) {
            @Override
            public void handleMessage(Message msg) {
                // Any UI task, example
                progressDialog.setMessage(progressText);
            }
        };
        handler.sendEmptyMessage(1);

    }

0

ในกรณีของฉันผู้โทรมาหลายครั้งในระยะเวลาอันสั้นจะได้รับข้อผิดพลาดนี้ฉันเพียงแค่ทำการตรวจสอบเวลาที่ผ่านไปโดยไม่ทำอะไรเลยถ้าสั้นเกินไปเช่นเพิกเฉยถ้าฟังก์ชั่นเรียกน้อยกว่า 0.5 วินาที:

    private long mLastClickTime = 0;

    public boolean foo() {
        if ( (SystemClock.elapsedRealtime() - mLastClickTime) < 500) {
            return false;
        }
        mLastClickTime = SystemClock.elapsedRealtime();

        //... do ui update
    }

ทางออกที่ดีกว่าคือปิดใช้งานปุ่มเมื่อคลิกและเปิดใช้งานอีกครั้งเมื่อการดำเนินการเสร็จสมบูรณ์
lsrom

@lsrom ในกรณีของฉันไม่ง่ายนักเพราะผู้โทรนั้นเป็นห้องสมุดบุคคลที่สามภายในและไม่สามารถควบคุมได้
ผลไม้

0

หากคุณไม่พบ UIT คุณสามารถใช้วิธีนี้ได้

yourcurrentcontextหมายถึงคุณต้องแยกวิเคราะห์บริบทปัจจุบัน

 new Thread(new Runnable() {
        public void run() {
            while (true) {
                (Activity) yourcurrentcontext).runOnUiThread(new Runnable() {
                    public void run() { 
                        Log.d("Thread Log","I am from UI Thread");
                    }
                });
                try {
                    Thread.sleep(1000);
                } catch (Exception ex) {

                }
            }
        }
    }).start();

0

คำตอบของ Kotlin

เราต้องใช้ UI Thread สำหรับงานด้วยวิธีการจริง เราสามารถใช้ UI Thread ใน Kotlin:

runOnUiThread(Runnable {
   //TODO: Your job is here..!
})

@canerkaseler


0

ในKotlinเพียงแค่ใส่รหัสของคุณในวิธีกิจกรรม runOnUiThread

runOnUiThread{
    // write your code here, for example
    val task = Runnable {
            Handler().postDelayed({
                var smzHtcList = mDb?.smzHtcReferralDao()?.getAll()
                tv_showSmzHtcList.text = smzHtcList.toString()
            }, 10)

        }
    mDbWorkerThread.postTask(task)
}
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.