Введение
Всем привет. Сегодня изучал поисковик с таким запросом «как работать с MySQL в ADT Bundle» и, если честно сказать, совершенно ничего не нашел. Решил написать об этом здесь.
Этот урок может помочь любому, от начинающего разработчика под Android, до профи, который начал работать с БД в своем приложении.
Создание проекта в Eclipse (далее ADT)
Итак, поехали. Заходим, создаем новое приложение в нашем ADT под названием AndroMySQL, активити используем: mainActivite.
Создание backend
Нам нужно написать серверную часть. В своем уроке я буду использовать для работы серверной части PHP, MySQL, PHPMyAdmin, Linux Debian.
Шаг 1
Ставим сервер, я использовал хостера
Шаг 2
Теперь, когда наш сайт (далее site.ru) доступен, мы можем приступить к делу и взяться за написание серверной части нашего проекта.
Заходим в корень site.ru (через FTP/SFTP или проводник) и создаем там файл andro.php. В итоге созданный файл должен быть виден по такой ссылке: www.site.ru/andro.php. Далее создаем файл add_prod.php (аналогично первому файлу add_prod.php должен быть доступен по ссылке www.site.ru/add_prod.php).
Шаг 3
Заходим в наш PHPMyAdmin (чаще всего это site.ru/phpmyadmin) и создаем новую базу под название andro. В этой базе нам нужно создать таблицу под названием products, которая будет состоять из сущностей: id_product, title, short, price.
id_product — номер продукта (значение типа int, длина 128, AUTOINCREMENT);
title — название продукта (значение типа VarChar, длина 512);
short — описание продукта (значение типа VarChar, длина 2048);
price — цена продукта (значение типа int, длина 128).
Шаг 4
Открываем файл andro.php для редактирования (для уменьшения вероятности появления ошибки сделайте кодировку файла ANSI). В этом файле нам нужно будет прописать функционал получения данных из нашей БД и вывод их в формате JSON (если вам неизвестно — эта аббревиатура, то для начала прочитайте о нем). Итак, файл должен быть такого содержания:
//Для начала подключаемся к серверу MySQL
$connect = mysql_connect("server_ip","user","password"); // Подключаемся к БД, данные для подключения введите свои
//Выбираем базу данных в которой будем работать
$msDB = mysql_select_db("andro"); // Мы выбрали бд andro, так как мы её используем в нашем уроке
// Строим SQL запрос для БД
$query = "SELECT * FROM products WHERE 1"; //Оброщаемся к таблице products и получаем все сущности таблицы, WHERE 1 обозначает, что выводить нужно все, можно любую там цифру ввести, можно вообще пустым оставить
//Выполняем наш SQL
$result = mysql_query($query); // Переменная result получает ресурс (заметьте ресурс, не массив) с данными вернувшимися от бд
if(mysql_num_rows($query) >= 1) { // Проверяем есть ли хоть одна запись в бд по данному запросу
$iter = 0; // Создаем переменную для дальнейшего создания массива
$arr = Array(); //Создаем пустой массив в который будем дополнять наши данные о продуктах
//Функция mysql_fetch_assoc строит ассоциативный массив из полученных данных, но построение и присвоение может осуществляться только по одной записи. Именно по этой причине нам придется считывать по одиночно записи и добавлять их в наш массив.
while($data = mysql_fetch_assoc($query)) { //Достаем строку и присваиваем темповой переменной data
$arr['products'][$iter] = $data; //Дополняем наш массив arr данными из бд
++$iter; //увеличиваем счетчик
//Это не единственный вариант создания данного массива, можно также использовать функцию php - array_push, но мы пойдем по такому пути
} //Заканчиваем цикл
//Функция json_encode(array mixed_value[,parameters]) эта функция преобразует массив данных в JSON строку
print json_encode($arr); //Преобразовываем наш массив в JSON
} // Закрываем IF
mysql_close($connect); // Закрываем соединение с базой
Пока не заходим на andro.php через сайт, так как значений у нас в БД еще нет, поэтому отредактируем ранее созданный файл add_prod.php.
Шаг 5
Открываем файл add_prod.php для редактирования (также про кодировку не забываем). Содержимое файла должно быть таким:
<form action="" method="get">
Название: <input type="text" name="title" /><br>
Описание: <input type="text" name="short" /><br>
Цена: <input type="text" name="price" /><br>
<input type="hidden" name="add" value="yes" />
<input type="submit" value="Добавить запись" />
</form>
<?
$connect = mysql_connect("server_ip_mysql","user","password"); // Подключаемся к бд
$MsSDB = mysql_select_db("andro"); // Выбираем бд
if($_GET["add"] == "yes") {
foreach($_GET as $key => $value) { // Присваиваем две временные переменные для обработки пришедших данных
$_GET[$key] = addslashes($value); // Обрабатываем все данные
}
$query = mysql_query("INSERT INTO products (title,short,price) VALUES ('$_GET[title]','$_GET[short]','$_GET[price]')"); //Выполняем запрос
}
?>
Шаг 6
Теперь нужно перейти на www.site.ru/add_prod.php и добавить записи (количество на ваше усмотрение).
Шаг 7
После добавления зайдите на www.site.ru/andro.php и в результате на странице должно получиться что-то вроде этого.
Итак, мы закончили с серверной частью, теперь перейдем к так называемому frontend'у или к части приложения Android.
Создание приложения Android
Итак, в начале урока мы уже создали форму для дальнейшей разработки, давайте перейдем в ADT.
Шаг 1
Перейдем в activity_main.xml и добавим туда это:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
tools:context=".MainActivity" >
<ListView
android:id="@+id/listView1"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_alignParentLeft="true"
android:layout_alignParentTop="true"
android:layout_marginLeft="19dp"
android:layout_marginTop="15dp" >
</ListView>
</RelativeLayout>
Далее создадим еще один layout под название list_item.xml (для создания нового layout перейдите в File->New->Android XML File. Resource Type: Layot, Project: AndroMySQL, File: list_item, Root Element: LinearLayout) и добавим в него это:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:orientation="vertical"
android:padding="10dp"
android:paddingLeft="10dp"
android:paddingRight="10dp" >
<!-- Поле названия-->
<TextView
android:id="@+id/name"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:paddingBottom="2dip"
android:paddingTop="6dip"
android:textColor="#43bd00"
android:textSize="16sp"
android:textStyle="bold" />
<!-- Поле описание -->
<TextView
android:id="@+id/short"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:paddingBottom="2dip"
android:textColor="#acacac" />
<!-- Поле цены -->
<TextView
android:id="@+id/price"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:paddingBottom="0dip"
android:textColor="#acacac" />
</LinearLayout>
Шаг 2
Для работы нам нужен новый класс (class) для получения JSON сформированной строки из нашего andro.php. Переходим в File->New->Class, там вводим:
Source folder: AndroMySQL/src,
Package: com.example.andromysql (или ваш класс),
Name: ServiceHandler,
Super Class: java.lang.Object
В ServiceHandler.java добавляем текст:
package com.example.andromysql; //подключаем наш проект
//Подключаем библиотеки нужные для работы нашего нового класса
import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.util.List;
import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse;
import org.apache.http.NameValuePair;
import org.apache.http.client.ClientProtocolException;
import org.apache.http.client.entity.UrlEncodedFormEntity;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.client.utils.URLEncodedUtils;
import org.apache.http.impl.client.DefaultHttpClient;
import org.apache.http.util.EntityUtils;
public class ServiceHandler {
static String response = null;
public final static int GET = 1;
public final static int POST = 2;
public ServiceHandler() {
}
/**
* Making service call
* @url - ссылка на запрос
* @method - http метод для получения значений
* */
public String makeServiceCall(String url, int method) {
return this.makeServiceCall(url, method, null);
}
/**
* Making service call
* @url - url для запроса
* @method - http метод для получения значений
* @params - http параметры
* */
public String makeServiceCall(String url, int method,
List<NameValuePair> params) {
try {
//Клиент HTTP
DefaultHttpClient httpClient = new DefaultHttpClient();
HttpEntity httpEntity = null;
HttpResponse httpResponse = null;
// Проверяем HTTP на метод POST
if (method == POST) {
HttpPost httpPost = new HttpPost(url);
// добавляем параметры
if (params != null) {
httpPost.setEntity(new UrlEncodedFormEntity(params));
}
httpResponse = httpClient.execute(httpPost);
} else if (method == GET) {
// конкатерируем параметры к URL
if (params != null) {
String paramString = URLEncodedUtils
.format(params, "utf-8");
url += "?" + paramString;
}
HttpGet httpGet = new HttpGet(url);
httpResponse = httpClient.execute(httpGet);
}
httpEntity = httpResponse.getEntity();
response = EntityUtils.toString(httpEntity);
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
} catch (ClientProtocolException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return response;
}
}
Сохраняем наш класс и переходим в MainActivity.java
Шаг 3
В MainActivity.java нам нужно добавить такое содержимое:
package com.example.andromysql;
//Подключаем нужные библиотеки
import java.util.ArrayList;
import java.util.HashMap;
import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;
import android.app.ListActivity;
import android.app.ProgressDialog;
import android.os.AsyncTask;
import android.os.Bundle;
import android.util.Log;
import android.widget.ListAdapter;
import android.widget.ListView;
import android.widget.SimpleAdapter;
public class MainActivity extends ListActivity {
private ProgressDialog pDialog;
// URL где хранится наш отформатированный JSON
private static String url = "http://www.develsi.com/view.php"; // поменяйте значение на свое
// JSON Теги, чтобы в будущем оброщаться в методах
private static final String TAG_PRODUCTS = "products";
private static final String TAG_ID = "id_product";
private static final String TAG_NAME = "title";
private static final String TAG_SHORT = "short";
private static final String TAG_PRICE = "price";
// JSONArray - тип данных : массив из JSON строки
JSONArray products = null;
// Hashmap для ListView
ArrayList<HashMap<String, String>> productList;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
contactList = new ArrayList<HashMap<String, String>>();
ListView lv = getListView();
// Вызываем асинхронное получение JSON
new GetContacts().execute();
}
/**
* Асинхронный класс для получения JSON строки с HTTP страницы
* */
private class GetContacts extends AsyncTask<Void, Void, Void> {
@Override
protected void onPreExecute() {
super.onPreExecute();
// Показываем диалог загрузки на время получения данных
pDialog = new ProgressDialog(MainActivity.this);
pDialog.setMessage("Загрузка...");
pDialog.setCancelable(false);
pDialog.show();
}
@Override
protected Void doInBackground(Void... arg0) {
// Создаем экземпляр класса ServiceHandler
ServiceHandler sh = new ServiceHandler();
// Делаем запрос на наш url и получаем ответ в переменную типа String sh
String jsonStr = sh.makeServiceCall(url, ServiceHandler.GET);
Log.d("Response: ", "> " + jsonStr);// Выводим в консоль результат, то что пришло с сервера
//Лучше читать, что именно пришло, так как бывают ошибки из-за лишних символов, пробел и перенос строки - это тоже символ.
if (jsonStr != null) { //Проверяем не пустой ли нам пришел ответ
try {
JSONObject jsonObj = new JSONObject(jsonStr); // Создаем переменную типа JSONObject
// Получаем массив продуктов из нашего объекта
products = jsonObj.getJSONArray(TAG_PRODUCTS);
// Проходим по всем продуктом циклом, для создания ассоциативного массива
for (int i = 0; i < products.length(); i++) {
JSONObject p = products.getJSONObject(i);
//Создаем некоторые переменные для дальнейшей работы
String id = p.getString(TAG_ID); // достаем номер продукта
String name = p.getString(TAG_NAME); //достаем название продукта
String email = p.getString(TAG_SHORT); //достаем описание продукта
String address = p.getString(TAG_PRICE); //достаем цену продукта
// Временная переменная hashmap для одиночной записи продукта
HashMap<String, String> product = new HashMap<String, String>();
// Добовляем значения в наш HashMap в виде key => value
product.put(TAG_ID, id);
product.put(TAG_NAME, name);
product.put(TAG_EMAIL, email);
// Добовляем продукты в наш лист продуктов
productList.add(product);
}
} catch (JSONException e) {
e.printStackTrace();
}
} else {
Log.e("ServiceHandler", "Не удалось получить никаких данных из адреса");
}
return null;
}
@Override
protected void onPostExecute(Void result) {
super.onPostExecute(result);
// Закрываем наш диалог загрузки
if (pDialog.isShowing())
pDialog.dismiss();
/**
* Обновляем запарсенные данные JSON в наш ListView
* */
ListAdapter adapter = new SimpleAdapter(
MainActivity.this, contactList,
R.layout.list_item, new String[] { TAG_NAME, TAG_SHORT }, new int[] { R.id.name,
R.id.short});
// Я не стал использовать в примере еще 2 переменные так как мне это не нужно на данный момент, но если вы желаете включить их в показ, то просто добавьте строку TAG_PRICE в new String[] {...} и R.id.price в new int[] {...}
setListAdapter(adapter); // Теперь через адаптер добовляем наш получившийся массив в ListView
}
}
}
Итог
После всех этих трудоемких шагов вы можете запустить ваше приложение на тестовом девайсе, предварительно включив сервер и проверив, выдает ли страница andro.php JSON код. Если вы все выполнили правильно, в итоге у вас должно получится так:
1. Запускаете приложение.
2. Открывается диалог загрузки.
3. Диалог загрузки пропадает и появляется список названий и описаний продуктов.
4. При нажатии кнопки назад происходит выход из приложения.
Спасибо всем за внимание.
Автор: ctaciv
Здравствуйте! Можно ли обноыить данную статью, а то:
import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse;
import org.apache.http.NameValuePair;
import org.apache.http.client.ClientProtocolException;
import org.apache.http.client.entity.UrlEncodedFormEntity;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.client.utils.URLEncodedUtils;
import org.apache.http.impl.client.DefaultHttpClient;
import org.apache.http.util.EntityUtils;
Эти библиотеки устарели и не работают. Спасибо!
Федор,
https://www.android-examples.com/the-import-org-apache-http-httpresponce-cannot-resolved-error-solution/
Исправьте ошибку,
в строках
if(mysql_num_rows($query) >= 1) {
while($data = mysql_fetch_assoc($query)) {
вместо переменной $query нужна $result результат запроса к бд
Проблема!!! ниже представленная часть кода отображается в браузере
$value) {
$_GET[$key] = addslashes($value);
}
$query = mysql_query(“INSERT INTO products (title,short,price) VALUES (‘$_GET[title]’,’$_GET[short]’,’$_GET[price]’)”);
}
?>
в чем может быть проблема?
Название:
Описание:
Цена:
$value) { // Присваиваем две временные переменные для $
$_GET[$key] = addslashes($value); // Обрабатываем все данные
$query = mysql_query(“INSERT INTO products (title,short,price) VALUES ($
}
}