2015-09-03 4 views
-3

В моей MainActivity вершине я сделал:"К сожалению, приложение остановилось" из-за NetworkOnMainThreadException

private static final int MY_DATA_CHECK_CODE = 0; 
public static MainActivity currentActivity; 
TextToSpeech mTts; 
private String targetURL; 
private String urlParameters; 
private Button btnClick; 
private String clicking = "clicked"; 
private String[] ipaddresses = new String[]{ 
private String iptouse = ""; 
private TextView text; 
private boolean connectedtoipsuccess = false; 
private int counter = 0; 
private NotificationCompat.Builder mbuilder; 
private Timer timer = new Timer(); 
private TextView text1, text2, text3; 
private Button startButton; 
private Button pauseButton; 
private TextView timerValue; 
private long startTime = 0L; 
private Handler customHandler = new Handler(); 
long timeSwapBuff = 0L; 
private int servercheckCounter = 0; 
private byte[] checkServer = null; 
private boolean connectedSuccess = false; 

Тогда внутри OnCreate я сделал:

startTime = SystemClock.uptimeMillis(); 
customHandler.postDelayed(updateTimerThread,0); 

Тогда в updateTimerThread я сделал:

private Runnable updateTimerThread = new Runnable() { 
    public void run() { 
     long updatedTime = 0L; 
     long timeInMilliseconds = 0L; 
     timeInMilliseconds = SystemClock.uptimeMillis() - startTime; 
     updatedTime = timeSwapBuff + timeInMilliseconds; 
     servercheckCounter +=1 ; 
     if (servercheckCounter == 10) { 
      if(connectedSuccess == true) { 
       checkServer = Get(iptouse + "result"); 
       if (checkServer != null) { 
        String a = null; 
        try { 
         a = new String(checkServer, "UTF-8"); 
         textforthespeacch = a; 
         MainActivity.currentActivity.initTTS(); 
        } catch (UnsupportedEncodingException e) { 
         e.printStackTrace(); 
        } 
       } 
      } 
      servercheckCounter = 0; 
     } 
     int secs = (int) (updatedTime/1000); 
     int mins = secs/60; 
     secs = secs % 60; 
     int milliseconds = (int) (updatedTime % 1000); 
     timerValue.setText("" + mins + ":" + String.format("%02d", secs) + ":" + String.format("%03d", milliseconds) + " counter: " + String.format("%01d",servercheckCounter)); 
     customHandler.postDelayed(this, 100); 
    } 
}; 

Таймер работает без остановки, и я делал это каждую секунду, когда он отправит команду на веб-сервер, который у меня есть мой компьютер, это строка в потоке таймера, которые посылают команду:

checkServer = Get(iptouse + "result"); 

Метод Get является:

private byte[] Get(String urlIn) { 
    URL url = null; 
    String urlStr = urlIn; 
    if (urlIn != null) 
     urlStr=urlIn; 

    try { 
     url = new URL(urlStr); 
    } catch (MalformedURLException e) { 
     e.printStackTrace(); 
     return null; 
    } 
    HttpURLConnection urlConnection = null; 
    try { 
     urlConnection = (HttpURLConnection) url.openConnection(); 
     InputStream in = new BufferedInputStream(urlConnection.getInputStream()); 

     byte[] buf=new byte[10*1024]; 
     int szRead = in.read(buf); 
     byte[] bufOut; 
     if (szRead==10*1024) { 
      throw new AndroidRuntimeException("the returned data is bigger than 10*1024.. we don't handle it.."); 
     } else { 
      bufOut = Arrays.copyOf(buf, szRead); 
     } 

     return bufOut; 
    } catch (IOException e) { 
     e.printStackTrace(); 
     return null; 
    } finally { 
     if (urlConnection!=null) 
      urlConnection.disconnect(); 
    } 
} 

Что я пытаюсь сделать, это послать команду каждую секунду веб-сервер на моем компьютере, и если/когда что-то случится на веб-сервере, оно вернет клиенту значение java - это клиент.

Проблема заключается в том, что после того, как она делает эту линию:

checkServer = Get(iptouse + "result"); 

Один или два раза он бросает ошибку на экране андроид устройства. Я попытался отладить его, но я не смог найти точное место и какую причину и как исправить эту проблему.

я добавил точку останова в методе Получить на линии:

InputStream in = new BufferedInputStream(urlConnection.getInputStream()); 

Это добраться до этой линии остановки на этой линии, то, когда я делаю по-прежнему иногда останавливаются на этой линии еще раз иногда бросить ошибку иногда после одного раза выбросить ошибку.

И я использую мою программу в большем количестве мест для отправки команд на веб-сервер, и он работает нормально только в этом месте, где я пытаюсь отправить команду каждую секунду, что она создает проблему.

То, что я пытаюсь сделать, это то, что называется пул, я думаю.

Возможно, проблема в том, что я посылаю команду каждую секунду, и метод Get не может справиться с ней, поскольку он слишком быстр?

Дело в том, что оно не останавливается ни на одном из мест catch в методе Get.

Это сообщение об ошибке от LogCat:

09-03 17:16:46.476 17224-17224/com.test.webservertest E/AndroidRuntime﹕ FATAL EXCEPTION: main 
Process: com.adilipman.webservertest, PID: 17224, android.os.NetworkOnMainThreadException 
at android.os.StrictMode$AndroidBlockGuardPolicy.onNetwork(StrictMode.java:1147) 
at libcore.io.BlockGuardOs.recvfrom(BlockGuardOs.java:249) 
at libcore.io.IoBridge.recvfrom(IoBridge.java:553) 
at java.net.PlainSocketImpl.read(PlainSocketImpl.java:485) 
at java.net.PlainSocketImpl.access$000(PlainSocketImpl.java:37) 
at java.net.PlainSocketImpl$PlainSocketInputStream.read(PlainSocketImpl.java:237) 
at com.android.okio.Okio$2.read(Okio.java:113) 
at com.android.okio.RealBufferedSource.indexOf(RealBufferedSource.java:147) 
at com.android.okio.RealBufferedSource.readUtf8LineStrict(RealBufferedSource.java:94) 
at com.android.okhttp.internal.http.HttpConnection.readResponse(HttpConnection.java:175) 
at com.android.okhttp.internal.http.HttpTransport.readResponseHeaders(HttpTransport.java:101) 
at com.android.okhttp.internal.http.HttpEngine.readResponse(HttpEngine.java:616) 
at com.android.okhttp.internal.http.HttpURLConnectionImpl.execute(HttpURLConnectionImpl.java:379) 
at com.android.okhttp.internal.http.HttpURLConnectionImpl.getResponse(HttpURLConnectionImpl.java:323) 
at com.android.okhttp.internal.http.HttpURLConnectionImpl.getInputStream(HttpURLConnectionImpl.java:190) 
at com.adilipman.webservertest.MainActivity.Get(MainActivity.java:582) 
at com.adilipman.webservertest.MainActivity.access$700(MainActivity.java:72) 
at com.adilipman.webservertest.MainActivity$3.run(MainActivity.java:188) 
at android.os.Handler.handleCallback(Handler.java:739) 
at android.os.Handler.dispatchMessage(Handler.java:95) 
at android.os.Looper.loop(Looper.java:135) 
at android.app.ActivityThread.main(ActivityThread.java:5274) 
at java.lang.reflect.Method.invoke(Native Method) 
at java.lang.reflect.Method.invoke(Method.java:372) 
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:909) 
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:704) 
+5

Что говорит ваш логарифм? –

+3

, если только мы можем получить журнал ошибок также –

+4

Поскольку вы смешиваете код пользовательского интерфейса с сетевым кодом, я предполагаю, что это «NetworkOnMainThreadException». Пробег «Runnable» в «Handler» в основном потоке по-прежнему работает в основном потоке пользовательского интерфейса. http://stackoverflow.com/questions/6343166/android-os-networkonmainthreadexception – laalto

ответ

0

Исключение, которое вызывается, когда приложение пытается выполнить операцию сети на ее главном потоке. Это делается только для приложений, ориентированных на SDK Honeycomb или выше, поскольку ранее SDK поддерживали сетевое взаимодействие в основном потоке. Вы можете предотвратить его с помощью StrictMode, как показано ниже:

if (android.os.Build.VERSION.SDK_INT > 9) { 
    StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder().permitAll().build()); 
} 

Однако это крайне нежелательно, и вы всегда должны использовать AsyncTask для сетевых задач.

Смежные вопросы