Aclarirem certs conceptes de threading aplicats a Android.
Referències:
El mes important treballant amb threads en Android és saber que:
No seguir alguna d'aquests dues premisses ens portarà a alguna excepció d'execució.
Per executar tasques curtes disposàvem oficialment de l'objecte ad-hoc AsyncTask, però a partir de la API 30 d'Android ha quedat obsolet i es recomana utilitzar el generic java.util.concurrent package.
La forma oficial d'utilitzar threads actualment a Java és emprar la llibreria java.util.concurrent
.
Referències:
Resumint, la forma recomanda actualment per executar un thread és aquesta:
ExecutorService executor = Executors.newSingleThreadExecutor(); executor.execute(new Runnable() { @Override public void run() { // Tasques en background (xarxa) Handler handler = new Handler(Looper.getMainLooper()); handler.post(new Runnable() { @Override public void run() { // Tasques a la interfície gràfica (GUI) } }); } });
Si volem executar operacions en el UI Thread després d'un cert temps, aquest post clarifica com utilitzar un Handler i un Runnable.
Les tasques enviades al Handler
s'executaran en el mateix thread on s'ha creat. Per tant, si estem en un thread secundari, podem crear un Handler
que envii missatges al UI thread indicat que el seu Looper
és el MainLooper:
final Handler handler1 = new Handler(Looper.getMainLooper()); handler1.postDelayed(r, 1000);
on r
és un Runnable.
Seguint el què hem explicat, des d'un thread secundari no podem fer operacions sobre el UI. Si necessitem fer-ne, caldrà utilitzar:
Runnable
en el UI thread.Message
pot ser un objecte amb text i dades, però també poden ser porcions de codi que es poden executar de forma asíncrona quan el thread pugui.Handler
disposa de funcions com postDelayed
que permet l'execució diferida d'un codi.
Teniu aquest article que explica força coses de com funcionen els objectes Handler
, Looper
, Message
i altres.
Una aplicació típica és la utilització de threads per a comunicació. Tal i com hem dit, no podem executar funcions que bloqueginen el mainUIthread d'Android, i les comunicacions ho fan en tant que esperen la resposta remota.
Posem aquest codi d'una crida HTTP en una funció getDataFromUrl
que podem afegir a la nostra MainActivity
. Aquí tens més info sobre llibreries per a crides HTTP en Java.
Si l'executem dins el MainThread
ens saltarà una excepció NetworkOnMainThreadException
. Per això necessitem el codi del thread de mes amunt per poder-la executar.
String error = ""; // string field private String getDataFromUrl(String demoIdUrl) { String result = null; int resCode; InputStream in; try { URL url = new URL(demoIdUrl); URLConnection urlConn = url.openConnection(); HttpsURLConnection httpsConn = (HttpsURLConnection) urlConn; httpsConn.setAllowUserInteraction(false); httpsConn.setInstanceFollowRedirects(true); httpsConn.setRequestMethod("GET"); httpsConn.connect(); resCode = httpsConn.getResponseCode(); if (resCode == HttpURLConnection.HTTP_OK) { in = httpsConn.getInputStream(); BufferedReader reader = new BufferedReader(new InputStreamReader( in, "iso-8859-1"), 8); StringBuilder sb = new StringBuilder(); String line; while ((line = reader.readLine()) != null) { sb.append(line).append("\n"); } in.close(); result = sb.toString(); } else { error += resCode; } } catch (IOException e) { e.printStackTrace(); } return result; }
Per poder-se connectar a internet, cal activar el permís Android d'accés a Internet:
<uses-permission android:name="android.permission.INTERNET" /> <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
Les crides a APIs externes ens retornaran objectes JSON que cal descodificar. Per exemple, pots provar https://api.myip.com
per veure la teva IP. Des de la shell fariem:
$ curl https://api.myip.com {"ip":"139.47.113.84","country":"Spain","cc":"ES"}
Es recomana aquest exemple per descodificar els missatges JSON.
Ull, perquè si el server de myip.com rep moltes peticions seguides acaba per bloquejar-se ja que es pensa que som atacants.
Si veieu que amb el CURL no funciona, cerqueu una altra API. Cerqueu alguna que us agradi d'aquest llistat de public APIs. Per exemple, aquesta ens retorna la URL d'una imatge aleatòria de guineus:
$ curl https://randomfox.ca/floof/
Llençar comunicacions des d'un thread alternatiu:
getDataFromUrl
i crida-la amb una URL (per exemple https://api.myip.com
) al prémer el botó, directament des del onClick
.NetworkOnMainThreadException
.Executor
i crida la funció de xarxa per obtenir les dades d'una URL. Mostra les dades per la consola de debug amb Log.i
.Actualitzar la GUI:
TextView
en el projecte.TextView
. Comprova que si ho fem en el mateix cos del thread, la funció run
, funciona.
Anem a posar alguna acció gràfica que ens obligui a utilitzar el Handler
, que serà el què ens permetrà executar en el thread principal (el de gràfics).
Farem servir una ImageView
:
ImageView
al layout.Bitmap
. Ho podem fer així String urldisplay = "https://randomfox.ca/images/122.jpg"; Bitmap bitmap; try { InputStream in = new java.net.URL(urldisplay).openStream(); bitmap = BitmapFactory.decodeStream(in); } catch (Exception e) { Log.e("Error", e.getMessage()); e.printStackTrace(); }
ImageView
. Comprova que al fer el imageView.setImageBitmap(bitmap)
ens falla amb una excepció si ho fem al thread de comunicacions.Handler
com a l'exemple i comprova que ara canvia la imatge i no peta.
Si et queda temps, crida la API randomfox (explicada mes amunt) i obtingues una imatge diferent cada cop, i mostra-la al ImageView
.
Per si et resulta avorrit, mes feina (exercici optatiu):
Anem a provar amb una GUI més ambiciosa: una ListView
.
ArrayList<String>
com a model per no haver de fer layouts personalitzats.adapter.notifyDataSetChanged();
ens surt una ViewRootImpl$CalledFromWrongThreadException
.
handler.post
i comprova que funciona.
Per poder emprar la biblioteca de codi Java WebSockets en Android caldrà afegir algunes línies als arxius:
dependencies { ... implementation(libs.websocket) ... }
[versions] ... websocket = "1.5.7" [libraries] ... websocket = { group = 'org.java-websocket', name = "Java-WebSocket", version.ref = "websocket" }
Recordem que per poder-se connectar a internet, cal activar el permís Android d'accés a Internet:
<uses-permission android:name="android.permission.INTERNET" /> <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
Un cop fet tot això podrem emprar les biblioteques Java WebSockets al nostre codi:
import org.java_websocket.client.WebSocketClient; import org.java_websocket.drafts.Draft_6455; import org.java_websocket.handshake.ServerHandshake; ... WebSocketClient client = null; URI location = "ws://mywsserver.com"; try { client = new WebSocketClient(new URI(location), (Draft) new Draft_6455()); client.connect(); } catch (URISyntaxException e) { e.printStackTrace(); System.out.println("Error: " + location + " no és una direcció URI de WebSocket vàlida"); } client.send("Hola! M'acabo de connectar"); ...
Teniu més explicacions i exemples a la pàgina WebSockets Java d'aquesta wiki.