https://manuais.iessanclemente.net/index.php?title=PDM_Avanzado_Comunicacion_Descarga_de_arquivos&feed=atom&action=historyPDM Avanzado Comunicacion Descarga de arquivos - Historial de revisiones2024-03-29T14:37:54ZHistorial de revisiones de esta página en el wikiMediaWiki 1.36.2https://manuais.iessanclemente.net/index.php?title=PDM_Avanzado_Comunicacion_Descarga_de_arquivos&diff=53559&oldid=prevCarrion: /* Identificar o tipo de rede */2015-02-23T06:08:48Z<p><span dir="auto"><span class="autocomment">Identificar o tipo de rede</span></span></p>
<p><b>Página nueva</b></p><div>==Introdución==<br />
<br />
<u>Nota:</u> Se utilizades un dispositivo real para facer esta práctica teredes que ter conexión a Internet.<br />
<br />
Se utilizades o emulador o computador onde estea correndo debe ter conexión a Internet.<br />
<br />
<br />
O proceso que temos que seguir para descargar un arquivo é:<br />
<br />
* Identificar o tipo de conexión para saber se estamos conectados a Internet. Neste punto podemos indicar o usuario que o uso da aplicación pode levar un custo se está conectado pola rede móbil.<br />
* Descargar o arquivo.<br />
<br />
<br />
Para facer o anterior necesitaremos engadir ó arquivo '''AndroidManifiest.xml''' os seguintes permisos:<br />
<br />
<syntaxhighlight lang="java" line enclose="div" highlight="" ><br />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" /><br />
<uses-permission android:name="android.permission.INTERNET" /><br />
</syntaxhighlight> <br />
<br />
Xa que imos acceder a Internet e imos descargar o arquivo na tarxeta SD Externa.<br />
<br />
==Identificar o tipo de rede==<br />
<br />
Para obter información sobre a rede necesitamos engadir o seguinte permiso o arquivo de '''AndroidManifiest.xml''':<br />
<br />
<syntaxhighlight lang="java" line enclose="div" highlight="" ><br />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" /><br />
</syntaxhighlight> <br />
<br />
<br />
<br />
Cando estamos conectados cun dispositivo móbil a unha rede temos tres posibilidades:<br />
* Móbil<br />
* Ethernet (cable de rede)<br />
* WIFI<br />
<br />
<br />
Para obter o tipo de rede ó que estamos conectados deberemos usar un obxecto da [http://developer.android.com/reference/android/net/ConnectivityManager.html clase ConnectivityManager].<br />
<br />
Para obtelo, deberemos chamar ó [http://developer.android.com/reference/android/content/Context.html#getSystemService%28java.lang.String%29 método getSystemService] da seguinte forma:<br />
<br />
<syntaxhighlight lang="java" line enclose="div" highlight="" ><br />
ConnectivityManager connMgr = (ConnectivityManager)contexto.getSystemService(Context.CONNECTIVITY_SERVICE);<br />
</syntaxhighlight> <br />
<br />
<br />
Unha vez feito isto podemos obter información acerca da rede na que estamos conectados mediante un obxecto da [http://developer.android.com/reference/android/net/NetworkInfo.html clase NetworkInfo]:<br />
<br />
<syntaxhighlight lang="java" line enclose="div" highlight="" ><br />
NetworkInfo networkInfo=null;<br />
networkInfo = connMgr.getActiveNetworkInfo();<br />
</syntaxhighlight> <br />
<br />
<br />
NetworkInfo informa se estamos conectados e o tipo de rede ó que estamos conectados:<br />
<br />
<syntaxhighlight lang="java" line enclose="div" highlight="" ><br />
if (networkInfo != null && networkInfo.isConnected()) {<br />
switch(networkInfo.getType()){<br />
case ConnectivityManager.TYPE_MOBILE:<br />
break;<br />
case ConnectivityManager.TYPE_ETHERNET:<br />
// ATENCION API LEVEL 13 PARA ESTA CONSTANTE<br />
break;<br />
case ConnectivityManager.TYPE_WIFI:<br />
// NON ESTEAS MOITO TEMPO CO WIFI POSTO<br />
// MAIS INFORMACION EN http://www.avaate.org/<br />
break;<br />
}<br />
}<br />
else {<br />
// NON TEMOS REDE<br />
}<br />
</syntaxhighlight> <br />
<br />
<br />
<br />
<u>Nota:</u> Se necesitamos manexar a conexión WIFI podemos facer uso da clase WifiManager: http://developer.android.com/reference/android/net/wifi/WifiManager.html e engadir o permiso android.permission.ACCESS_WIFI_STATE no AndroidManifiest.xml.<br />
<br />
==Descargar o arquivo==<br />
<br />
Agora imos resolver o problema de descargar un arquivo dende Internet.<br />
<br />
Teremos que utilizar un InputStream para lela e un OutputStream para escribila a disco.<br />
<br />
<u>Nota:</u> O manexo de arquivos xa os vimos neste punto: http://manuais.iessanclemente.net/index.php/PDM_Avanzado_Datos_Persistentes_Arquivos<br />
<br />
O cartafol onde imos gardala será o predeterminado polo S.O. para gardar imaxes.<br />
<br />
<br />
O proceso será o seguinte:<br />
<br />
* Determinamos a dirección URL para descargar o arquivo:<br />
<br />
:<syntaxhighlight lang="java" line enclose="div" highlight="" ><br />
private String IMAXE_DESCARGAR="http://www.aspedrinas.com/imagenes/santiago/santiago1.jpg";<br />
URL url=null;<br />
try {<br />
url = new URL(IMAXE_DESCARGAR);<br />
} catch (MalformedURLException e1) {<br />
// TODO Auto-generated catch block<br />
e1.printStackTrace();<br />
return;<br />
}<br />
</syntaxhighlight> <br />
<br />
<br />
Se queremos obter só o nome do arquivo da URL podemos poñer este código:<br />
<br />
:<syntaxhighlight lang="java" line enclose="div" highlight="" ><br />
String nomeArquivo = Uri.parse(IMAXE_DESCARGAR).getLastPathSegment();<br />
</syntaxhighlight> <br />
<br />
<br />
* Agora necesitamos ler dende Internet o arquivo a descargar.<br />
<br />
A idea é moi sinxela. Temos que facer unha conexión có Servidor.<br />
Ó establecer dita conexión podemos especificar unha serie de parámetros coma son:<br />
<br />
* Tempo máximo de lectura (en milisegundos).<br />
* Tempo máximo para facer a conexión.<br />
* Método de transmisión (GET ou POST, entre outros).<br />
<br />
Todo isto se fai cun obxecto da [http://developer.android.com/reference/java/net/HttpURLConnection.html clase HttpURLConnection]:<br />
<br />
<syntaxhighlight lang="java" line enclose="div" highlight="" ><br />
HttpURLConnection conn=null;<br />
<br />
conn = (HttpURLConnection) url.openConnection();<br />
conn.setReadTimeout(10000); /* milliseconds */<br />
conn.setConnectTimeout(15000); /* milliseconds */<br />
conn.setRequestMethod("POST");<br />
conn.setDoInput(true); /* Indicamos que a conexión vai recibir datos */<br />
<br />
conn.connect();<br />
<br />
</syntaxhighlight> <br />
<br />
Ó facer o intento de conexión o servidor web pode darnos unha resposta de que todo é correcto ou un erro (por exemplo, recursos non atopado, que non deixe descargar,...)<br />
Comprobaremos por tanto que non tivemos ningún erro:<br />
<br />
<syntaxhighlight lang="java" line enclose="div" highlight="" ><br />
int response = conn.getResponseCode();<br />
if (response != HttpURLConnection.HTTP_OK){ <br />
// TRATAREMOS O ERRO<br />
return;<br />
}<br />
<br />
</syntaxhighlight> <br />
<br />
<br />
<u>Nota:</u> Aínda que dependemos do servidor, podemos obter o tamaño do que queremos descargar da seguinte forma: <br />
<br />
<syntaxhighlight lang="java" line enclose="div" highlight="" ><br />
int fileLength = conn.getContentLength(); // Non funciona sempre<br />
</syntaxhighlight> <br />
<br />
<br />
Isto nos pode servir para modificar unha barra de progreso a indicar canto lle queda por descargar....<br />
<br />
En caso de que o servidor non resposnda filelength terá de valor -1.<br />
<br />
<br />
* Definimos o OutputStream (para gardar o arquivo lido) e un InputStream para ler o contido de Internet:<br />
<br />
<syntaxhighlight lang="java" line enclose="div" highlight="" ><br />
OutputStream os;<br />
InputStream in;<br />
......<br />
os = new FileOutputStream(Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES+File.separator+nomeArquivo));<br />
in = conn.getInputStream();<br />
</syntaxhighlight> <br />
<br />
<u>Nota:</u> Necesitaremos usar try...catch<br />
<br />
<br />
* Agora só queda ler o contido e escribilo ó mesmo tempo como fixemos na unidade de arquivos:<br />
<br />
<syntaxhighlight lang="java" line enclose="div" highlight="" ><br />
byte data[] = new byte[1024]; // Buffer a utilizar<br />
int count;<br />
while ((count = in.read(data)) != -1) {<br />
os.write(data, 0, count);<br />
}<br />
os.flush();<br />
os.close();<br />
in.close();<br />
conn.disconnect();<br />
</syntaxhighlight> <br />
<br />
<br />
* '''O PUNTO MÁIS IMPORTATE''': Todo o anterior ten que facerse nun fío separado do principal.<br />
<br />
Como se vimos na [http://manuais.iessanclemente.net/index.php/Curso_Platega:_Desenvolvemento_de_aplicaci%C3%B3ns_para_dispositivos_m%C3%B3biles_con_Android._Avanzado#UNIDADE_3:_Threads_e_AsyncTask Unidade de Threads e AsynTask] o podemos facer utilizando un Thread ou un AsynTask.<br />
<br />
<br />
<br />
<br />
<u>Aclaración:</u> Estamos a amosar como descargar un arquivo calquera de Internet e unha vez baixado facer algunha operación sobre el.<br />
<br />
Se queremos descargar unha imaxe podemos cargar directamente un Bitmap dende o InputStream desta forma:<br />
<br />
<syntaxhighlight lang="java" line enclose="div" highlight="" ><br />
Bitmap bitmap = BitmapFactory.decodeStream(in);<br />
</syntaxhighlight><br />
<br />
==Caso Práctico==<br />
<br />
O obxectivo desta práctica é comprobar como podemos descargar un arquivo dende Internet (neste exemplo é unha imaxe).<br />
<br />
A dirección URL está posta na propia aplicación do exemplo. O alumno pode cambiala por unha súa pero tendo en conta que hai sitios web que van responder cun erro de conexión (403, forbidden) e outros teñen unha visualización diferente para móbiles que para PC, polo que unha dirección probada dende un PC pode funcionar pero que cando se pon na aplicación móbil pode dar outro erro (307, redirect).<br />
<br />
Neste exemplo utilizamos un Thread sen paso de mensaxes. <br />
<br />
Isto non será a forma adecuada de implementalo, xa que o lóxico sería premer o botón de Descargar Imaxe e 'informar' a activity cando rematou de descargala. <br />
<br />
Unha forma elegante de facelo é utilizar un diálogo de progreso como vimos nas unidades anteriores.<br />
<br />
[[Imagen:PDM_Avanzada_DatosPersistentes_20.jpg|400px|center]]<br />
<br />
<br />
===Preparación===<br />
<br />
Engadimos no AndroidManifiest.xml os seguintes permisos:<br />
<br />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" /><br />
<uses-permission android:name="android.permission.INTERNET" /><br />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" /><br />
<br />
<br />
* Lembrar que no caso de utilizar un emulador será necesario que teña a tarxeta externa SD.<br />
<br />
<br />
===Creamos a Activity ===<br />
<br />
*Nome do proxecto: '''UD5_01_Comunicacion'''<br />
*Nome da activity: '''UD5_01_Comunicacion.java'''<br />
<br />
<br />
<br />
'''Código do layout xml'''<br/><br />
<syntaxhighlight lang="xml" line enclose="div" highlight="" ><br />
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"<br />
xmlns:tools="http://schemas.android.com/tools"<br />
android:layout_width="match_parent"<br />
android:layout_height="match_parent"<br />
tools:context="${relativePackage}.${activityClass}" ><br />
<br />
<ImageView<br />
android:id="@+id/ud5_imgvwImaxeDescargada"<br />
android:layout_width="300dp"<br />
android:layout_height="300dp"<br />
android:layout_alignParentBottom="true"<br />
android:layout_centerHorizontal="true"<br />
android:contentDescription="IMAXE DESCARGADA"<br />
android:src="@drawable/ic_launcher" /><br />
<br />
<Button<br />
android:id="@+id/ud5_btnDescargarImaxe"<br />
android:layout_width="wrap_content"<br />
android:layout_height="wrap_content"<br />
android:layout_centerHorizontal="true"<br />
android:layout_marginTop="34dp"<br />
android:text="DESCARGAR IMAXE" /><br />
<br />
<Button<br />
android:id="@+id/ud5_btnVisualizarImaxe"<br />
android:layout_width="wrap_content"<br />
android:layout_height="wrap_content"<br />
android:layout_below="@+id/ud5_btnDescargarImaxe"<br />
android:layout_centerHorizontal="true"<br />
android:text="VISUALIZAR IMAXE" /><br />
<br />
</RelativeLayout><br />
</syntaxhighlight> <br />
<br />
<br />
'''Código da clase UD5_01_Comunicacion'''<br/><br />
'''Obxectivo:''' Descargar unha imaxe de Internet.<br />
<syntaxhighlight lang="java" line enclose="div" highlight="29,30,36-56,59-108,122-124,137-141,153-157" ><br />
import java.io.File;<br />
import java.io.FileNotFoundException;<br />
import java.io.FileOutputStream;<br />
import java.io.IOException;<br />
import java.io.InputStream;<br />
import java.io.OutputStream;<br />
import java.net.HttpURLConnection;<br />
import java.net.MalformedURLException;<br />
import java.net.URL;<br />
<br />
import android.app.Activity;<br />
import android.content.Context;<br />
import android.graphics.Bitmap;<br />
import android.graphics.BitmapFactory;<br />
import android.net.ConnectivityManager;<br />
import android.net.NetworkInfo;<br />
import android.net.Uri;<br />
import android.os.Bundle;<br />
import android.os.Environment;<br />
import android.util.Log;<br />
import android.view.View;<br />
import android.view.View.OnClickListener;<br />
import android.widget.Button;<br />
import android.widget.ImageView;<br />
import android.widget.Toast;<br />
<br />
public class UD5_01_Comunicacion extends Activity {<br />
<br />
public static enum TIPOREDE{MOBIL,ETHERNET,WIFI,SENREDE};<br />
private TIPOREDE conexion;<br />
<br />
private final String IMAXE_DESCARGAR="http://www.aspedrinas.com/imagenes/santiago/santiago1.jpg";<br />
private File rutaArquivo;<br />
private Thread thread;<br />
<br />
private TIPOREDE comprobarRede(){<br />
NetworkInfo networkInfo=null;<br />
<br />
ConnectivityManager connMgr = (ConnectivityManager)getSystemService(Context.CONNECTIVITY_SERVICE);<br />
networkInfo = connMgr.getActiveNetworkInfo();<br />
<br />
if (networkInfo != null && networkInfo.isConnected()) {<br />
switch(networkInfo.getType()){<br />
case ConnectivityManager.TYPE_MOBILE:<br />
return TIPOREDE.MOBIL;<br />
case ConnectivityManager.TYPE_ETHERNET:<br />
// ATENCION API LEVEL 13 PARA ESTA CONSTANTE<br />
return TIPOREDE.ETHERNET;<br />
case ConnectivityManager.TYPE_WIFI:<br />
// NON ESTEAS MOITO TEMPO CO WIFI POSTO<br />
// MAIS INFORMACION EN http://www.avaate.org/<br />
return TIPOREDE.WIFI;<br />
}<br />
}<br />
return TIPOREDE.SENREDE; <br />
}<br />
<br />
<br />
private void descargarArquivo() {<br />
URL url=null;<br />
try {<br />
url = new URL(IMAXE_DESCARGAR);<br />
} catch (MalformedURLException e1) {<br />
// TODO Auto-generated catch block<br />
e1.printStackTrace();<br />
return;<br />
}<br />
<br />
HttpURLConnection conn=null;<br />
String nomeArquivo = Uri.parse(IMAXE_DESCARGAR).getLastPathSegment();<br />
rutaArquivo = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES+File.separator+nomeArquivo);<br />
try {<br />
<br />
conn = (HttpURLConnection) url.openConnection();<br />
conn.setReadTimeout(10000); /* milliseconds */<br />
conn.setConnectTimeout(15000); /* milliseconds */<br />
conn.setRequestMethod("POST");<br />
conn.setDoInput(true); /* Indicamos que a conexión vai recibir datos */<br />
<br />
conn.connect();<br />
<br />
int response = conn.getResponseCode();<br />
if (response != HttpURLConnection.HTTP_OK){ <br />
return;<br />
}<br />
OutputStream os = new FileOutputStream(rutaArquivo);<br />
InputStream in = conn.getInputStream();<br />
byte data[] = new byte[1024]; // Buffer a utilizar<br />
int count;<br />
while ((count = in.read(data)) != -1) {<br />
os.write(data, 0, count);<br />
}<br />
os.flush();<br />
os.close();<br />
in.close();<br />
conn.disconnect();<br />
Log.i("COMUNICACION","ACABO");<br />
}<br />
catch (FileNotFoundException e) {<br />
// TODO Auto-generated catch block<br />
Log.e("COMUNICACION",e.getMessage());<br />
} catch (IOException e) {<br />
// TODO Auto-generated catch block<br />
e.printStackTrace();<br />
Log.e("COMUNICACION",e.getMessage());<br />
}<br />
<br />
}<br />
<br />
private void xestionarEventos(){<br />
<br />
<br />
Button btnDescargarImaxe=(Button) findViewById(R.id.ud5_btnDescargarImaxe);<br />
btnDescargarImaxe.setOnClickListener(new OnClickListener() {<br />
<br />
@Override<br />
public void onClick(View v) {<br />
// TODO Auto-generated method stub<br />
thread = new Thread(){<br />
<br />
@Override<br />
public void run(){<br />
descargarArquivo();<br />
}<br />
};<br />
thread.start();<br />
<br />
<br />
}<br />
});<br />
Button btnVisualizarImaxe=(Button) findViewById(R.id.ud5_btnVisualizarImaxe);<br />
btnVisualizarImaxe.setOnClickListener(new OnClickListener() {<br />
<br />
@Override<br />
public void onClick(View v) {<br />
// TODO Auto-generated method stub<br />
if ((thread!=null) && (!thread.isAlive())){<br />
ImageView imgviewImaxe = (ImageView)findViewById(R.id.ud5_imgvwImaxeDescargada);<br />
Bitmap bmpImaxe = BitmapFactory.decodeFile(rutaArquivo.getAbsolutePath());<br />
imgviewImaxe.setImageBitmap(bmpImaxe);<br />
}<br />
<br />
}<br />
});<br />
<br />
}<br />
<br />
@Override<br />
protected void onCreate(Bundle savedInstanceState) {<br />
super.onCreate(savedInstanceState);<br />
setContentView(R.layout.activity_ud5_01__comunicacion);<br />
<br />
conexion = comprobarRede();<br />
if (conexion==TIPOREDE.SENREDE){<br />
Toast.makeText(this, "NON SE PODE FACER ESTA PRACTICA SEN CONEXION A INTERNET", Toast.LENGTH_LONG).show();<br />
finish();<br />
}<br />
<br />
xestionarEventos();<br />
}<br />
}<br />
<br />
</syntaxhighlight> <br />
<br />
* Liñas 29,30,36-56: Como vimos na explicación inicial, comprobamos o tipo de conexión do noso dispositivo móbil. O método vai devolver un valor dun tipo ENUM (liña 29).<br />
* Liñas 59-108: Descargamos o arquivo como vimos na explicación inicial.<br />
* Liñas 122-124: Xestionamos o evento de Click sobre o botón 'Descargar Imaxe'. Creamos un fío novo de execución e chamamos ó método 'descargarArquivo'.<br />
* Liñas 137-141: Comprobamos que o fío rematou de executarse e nese caso cargamos a imaxe descargada.<br />
* Liñas 153-157: En caso de non estar conectado a ningunha rede finalizamos a activity.<br />
<br />
<br />
<br />
<br />
<br />
<br> -- [[Usuario:angelfg|Ángel D. Fernández González]] e [[Usuario:Carrion|Carlos Carrión Álvarez]] -- (2014).</div>Carrion