android入門學習-天氣預報app(一)

来源:https://www.cnblogs.com/lisd/archive/2018/07/13/9304794.html
-Advertisement-
Play Games

引言 學習《android第一行代碼》根據書本開發的天氣預報app,主要用於熟練操作android開發(android studio3.0平臺)。 今天主要分享一下從伺服器上獲取天氣信息,通過ListView綁定數據的操作(可以採用RecycleView),然後進行頁面點擊跳轉。 一、 伺服器返回數 ...


引言

學習《android第一行代碼》根據書本開發的天氣預報app,主要用於熟練操作android開發(android studio3.0平臺)。

今天主要分享一下從伺服器上獲取天氣信息,通過ListView綁定數據的操作(可以採用RecycleView),然後進行頁面點擊跳轉。

一、 伺服器返回數據預覽

通過本書作者提供的訪問地址,guolin.tech/api/china,直接打開可以訪問到全國各地的天氣信息,guolin.tech/api/china/16/116

(如點擊無法打開,請直接複製鏈接在瀏覽器中訪問)

[
{"id":1,"name":"北京"},{"id":2,"name":"上海"},{"id":3,"name":"天津"},{"id":4,"name":"重慶"},
{"id":5,"name":"香港"},{"id":6,"name":"澳門"},{"id":7,"name":"臺灣"},{"id":8,"name":"黑龍江"},
{"id":9,"name":"吉林"},{"id":10,"name":"遼寧"},{"id":11,"name":"內蒙古"},{"id":12,"name":"河北"},
{"id":13,"name":"河南"},{"id":14,"name":"山西"},{"id":15,"name":"山東"},{"id":16,"name":"江蘇"},
{"id":17,"name":"浙江"},{"id":18,"name":"福建"},{"id":19,"name":"江西"},{"id":20,"name":"安徽"},
{"id":21,"name":"湖北"},{"id":22,"name":"湖南"},{"id":23,"name":"廣東"},{"id":24,"name":"廣西"},
{"id":25,"name":"海南"},{"id":26,"name":"貴州"},{"id":27,"name":"雲南"},{"id":28,"name":"四川"},
{"id":29,"name":"西藏"},{"id":30,"name":"陝西"},{"id":31,"name":"寧夏"},{"id":32,"name":"甘肅"},
{"id":33,"name":"青海"},{"id":34,"name":"新疆"}
]

如上面看到的可以獲取到各省的信息,通過id可以進一步的獲取市縣的信息。

二、創建本地數據存儲

根據伺服器中獲取的數據屬性,創建Province、City、County三個數據表分別存儲省、市、縣的信息。

創建數據表採用Litepal,這裡需要導入外部庫,需要在build.gradle中添加

implementation 'org.litepal.android:core:1.4.1'
(註implemention為android3.0以上版本才有的,與compile並不完全一樣,這裡不多做解釋,2.0版本採用compile不影響
同時在載入外部庫時可能出現無法引用的問題,請清理項目後重新build)

 後面1.4.1是版本號,可以查詢官網獲取最新版本

下麵附上三個數據表類,由於是採用Litepal需要繼承DataSupport

 

 

  

public class Province extends DataSupport {

    private int id;
    private String provinceName;
    private int provinceCode;

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getProvinceName() {
        return provinceName;
    }

    public void setProvinceName(String provinceName) {
        this.provinceName = provinceName;
    }

    public int getProvinceCode() {
        return provinceCode;
    }

    public void setProvinceCode(int provinceCode) {
        this.provinceCode = provinceCode;
    }
}
Province省數據表
public class City extends DataSupport {
    private int id;
    private String cityName;
    private int cityCode;
    private int provinceId;

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getCityName() {
        return cityName;
    }

    public void setCityName(String cityName) {
        this.cityName = cityName;
    }

    public int getCityCode() {
        return cityCode;
    }

    public void setCityCode(int cityCode) {
        this.cityCode = cityCode;
    }

    public int getProvinceId() {
        return provinceId;
    }

    public void setProvinceId(int provinceId) {
        this.provinceId = provinceId;
    }
}
City市數據表
public class County extends DataSupport {
    private int id;
    private String countyName;
    private String weatherId;
    private int cityId;


    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getCountyName() {
        return countyName;
    }

    public void setCountyName(String countyName) {
        this.countyName = countyName;
    }

    public String getWeatherId() {
        return weatherId;
    }

    public void setWeatherId(String weatherId) {
        this.weatherId = weatherId;
    }

    public int getCityId() {
        return cityId;
    }

    public void setCityId(int cityId) {
        this.cityId = cityId;
    }
}
County縣數據表

這三個類用來生成本地資料庫

接下來新建一個litepal.xml配置文件,在main目錄下新建assets配置文件夾,添加litepal.xml

<litepal>
    <dbname value ="cool_weather"/>
    <version value = "1"/>
    <list>
        <mapping class="com.coolweather2.db.Province"/>
        <mapping class="com.coolweather2.db.County"/>
        <mapping class="com.coolweather2.db.City"/>
    </list>
</litepal>
litepal.xml

此文件通過Litepal創建了cool_weather資料庫同時添加了三個數據表

最後在androidmanifest.xml中添加LitepalApplication,如下

這樣Litepal可以全局調用Context,註意如果不加,預設啟動時會初始化Application類

三、編寫工具包類

3.1HttpUtil

public class HttpUtil {

    public static void sendOkHttpRequest(String address,okhttp3.Callback callback) {
        OkHttpClient client  = new OkHttpClient();
        //request屬於不能繼承的類初始化Builder靜態類調用url方法,最後調用Requset的build方法
        Request request = new Request.Builder().url(address).build();
        client.newCall(request).enqueue(callback);
    }
}
HttpUtil.class

主要用來想伺服器發送請求,創建request並設置好其地址,通過client發起請求並設置callback回調請求

3.2Utility

public class Utillty {
    /*解析和處理伺服器返回的省級數據*/
    public static boolean handleProvinceResponse(String response) {
        if(!TextUtils.isEmpty(response)) {
            try
            {
                JSONArray allProvinces = new JSONArray(response);
                for(int i = 0;i < allProvinces.length(); i++) {
                    JSONObject privinceObject = allProvinces.getJSONObject(i);
                    Province province = new Province();
                    province.setProvinceName(privinceObject.getString("name"));
                    province.setProvinceCode(privinceObject.getInt("id"));
                    province.save();
                }
                return true;
            }
            catch (JSONException ex) {
                ex.printStackTrace();
            }


        }
        return false;
    }


    /**
     * 解析和處理伺服器返回的市級數據
     * */
    public static boolean handleCityResponse(String response,int provinceId) {
        if(!TextUtils.isEmpty(response)) {
            try
            {
                JSONArray allCities = new JSONArray(response);
                for(int i = 0;i < allCities.length(); i++) {
                    JSONObject cityObject = allCities.getJSONObject(i);
                    City city = new City();
                    city.setCityName(cityObject.getString("name"));
                    city.setCityCode(cityObject.getInt("id"));
                    city.setProvinceId(provinceId);
                    city.save();
                }
                return true;
            }
            catch (JSONException ex) {
                ex.printStackTrace();
            }


        }
        return false;
    }

    /**
     * 解析和處理服器返回的縣級數據
     * */
    public static boolean handleCountyResponse(String response,int cityId) {
        if(!TextUtils.isEmpty(response)) {
            try
            {
                JSONArray allCountries = new JSONArray(response);
                for(int i = 0;i < allCountries.length(); i++) {
                    JSONObject countyObject = allCountries.getJSONObject(i);
                    County county = new County();
                    county.setCountyName(countyObject.getString("name"));
                    county.setWeatherId(countyObject.getString("weather_id"));
                    county.setCityId(cityId);
                    county.save();
                }
                return true;
            }
            catch (JSONException ex) {
                ex.printStackTrace();
            }


        }
        return false;
    }
Utility.class

主要處理從伺服器返回的數據,根據最開始的數據預覽可以看到,伺服器返回的數據是以Json格式輸出的

那麼這個工具類就是對json數據的分析處理,通過JSONArray數組將json數據保存在本地資料庫

有這裡的json數據比較簡單,直接進行解析,不採用GSON

四、創建碎片佈局

 碎片有利於代碼的復用,可以嘗試將自己的很多功能都通過碎片進行封裝

創建碎片會生成兩個文件,一個是碎片的佈局文件choose_area.xml,一個是ChooseAreaFragment.class

4.1佈局文件

<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    android:background="#fff">


    <RelativeLayout
        android:layout_width = "match_parent"
        android:layout_height = "?attr/actionBarSize"
        android:background="?attr/colorPrimary">
        <TextView
            android:id="@+id/title_text"
            android:layout_centerInParent="true"
            android:textColor="#fff"
            android:textSize="20sp"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content" />
        <Button
            android:id="@+id/back_button"
            android:layout_width="25dp"
            android:layout_marginLeft="10dp"
            android:layout_alignParentLeft="true"
            android:layout_centerVertical="true"
            android:layout_height="25dp"
            android:background="@drawable/ic_launcher_background"/>
    </RelativeLayout>


    <ListView
        android:id="@+id/list_view"
        android:layout_width="match_parent"
        android:layout_height="match_parent"></ListView>

</LinearLayout>
choose_area.xml

簡單的兩個佈局,整體用線性佈局垂直對齊方式,頭部標題欄採用RecycleLayout佈局,設置一個回調按鈕和一個標題文本框

底部一個listview用來顯示天氣列表

4.2ChooseAreaFragment.class

public class ChooseAreaFragment extends Fragment {

    public static final int LEVEL_PROVINCE = 0;
    public static final int LEVEL_CITY = 1;
    public static final int LEVEL_COUNTY = 2;

    private ProgressDialog progressDialog;
    private TextView titleText;
    private Button backButton;
    private ListView listView;
    private ArrayAdapter<String> adapter;
    private List<String> dataList = new ArrayList<>();

    private List<Province> provinceList;
    private List<City> cityList;
    private List<County> countyList;

    private Province selectedProvince;
    private City selectedCity;
    private int currentLevel;



    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.choose_area,container,false);
        titleText = (TextView)view.findViewById(R.id.title_text);
        backButton = (Button)view.findViewById(R.id.back_button);
        listView = (ListView)view.findViewById(R.id.list_view);

        adapter = new ArrayAdapter<String>(getContext(), R.layout.support_simple_spinner_dropdown_item,dataList);
        listView.setAdapter(adapter);
        return view;
    }

    @Override
    public void onActivityCreated(@Nullable Bundle savedInstanceState) {
        super.onActivityCreated(savedInstanceState);


        queryProvinces();

        listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
            @Override
            public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
                if(currentLevel == LEVEL_PROVINCE) {
                    selectedProvince = provinceList.get(position);
                    queryCities();
                }
                else if (currentLevel == LEVEL_CITY) {
                    selectedCity = cityList.get(position);
                    queryCountries();
                }
            }
        });

        backButton.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (currentLevel == LEVEL_COUNTY) {
                    queryCities();
                }
                else if (currentLevel == LEVEL_CITY) {
                    queryProvinces();
                }
            }
        });


    }


    /**
     * 查詢所有省,查資料庫,再查伺服器
     */
    private void queryProvinces() {
        titleText.setText("中國");
        backButton.setVisibility(View.GONE);
        provinceList = DataSupport.findAll(Province.class);
        if(provinceList.size()>0) {
            dataList.clear();
            for (Province province :
                    provinceList) {
                dataList.add(province.getProvinceName());
            }
            adapter.notifyDataSetChanged();
            listView.setSelection(0);
            currentLevel = LEVEL_PROVINCE;
        }
        else {
            //伺服器查詢
            String address = "http://guolin.tech/api/china";
            queryFromServer(address,"province");
        }
    }

    /**
     * 查詢市
     */

    private void queryCities() {
        titleText.setText(selectedProvince.getProvinceName());
        backButton.setVisibility(View.VISIBLE);
        cityList = DataSupport.where("provinceid = ?",String.valueOf(selectedProvince.getId()))
                .find(City.class);
        if (cityList.size()>0) {
            dataList.clear();
            for (City city :
                    cityList) {
                dataList.add(city.getCityName());
            }
            adapter.notifyDataSetChanged();
            listView.setSelection(0);
            currentLevel = LEVEL_CITY;
        }
        else {
            //查詢伺服器
            int provinceCode = selectedProvince.getProvinceCode();
            String address = "http://guolin.tech/api/china/" + provinceCode;
            queryFromServer(address,"city");
        }
    }

    /**
     * 查詢縣
     */

    private void queryCountries(){
        titleText.setText(selectedCity.getCityName());
        backButton.setVisibility(View.VISIBLE);
        countyList = DataSupport.where("cityid = ?",String.valueOf(selectedCity.getId()))
                .find(County.class);
        if (countyList.size()>0) {
            dataList.clear();
            for (County county :
                    countyList) {
                dataList.add(county.getCountyName());
            }
            adapter.notifyDataSetChanged();
            listView.setSelection(0);
            currentLevel = LEVEL_COUNTY;

        }
        else {
            //訪問伺服器
            int provinceCode = selectedProvince.getProvinceCode();
            int cityCode = selectedCity.getCityCode();
            String address = "http://guolin.tech/api/china/" + provinceCode + "/" + cityCode;
            Toast.makeText(getContext(),provinceCode +"/" +  cityCode,Toast.LENGTH_LONG).show();
            queryFromServer(address,"county ");
        }
    }


    /**
     * 從伺服器查詢
     */
    private void queryFromServer(String address,final String type) {
        showProgressDialog();
        Toast.makeText(getContext(),type,Toast.LENGTH_SHORT).show();
        HttpUtil.sendOkHttpRequest(address, new Callback() {
            @Override
            public void onFailure(Call call, IOException e) {
                getActivity().runOnUiThread(new Runnable() {
                    @Override
                    public void run() {
                        closeProgressDialog();
                        Toast.makeText(getContext(),"載入失敗!",Toast.LENGTH_SHORT).show();
                    }
                });
            }

            @Override
            public void onResponse(Call call, Response response) throws IOException {
                String responText = response.body().string();
                boolean result = false;
                if("province".equals(type)) {
                    result = Utillty.handleProvinceResponse(responText);
                }
                else if ("city".equals(type)) {
                    result = Utillty.handleCityResponse(responText,selectedProvince.getId());
                }
                else if ("county".equals(type)){
                    //Toast.makeText(getContext(),"aaaaa",Toast.LENGTH_LONG).show();
                    result = Utillty.handleCountyResponse(responText,selectedCity.getId());
                }
                //Toast.makeText(getContext(),"" + result,Toast.LENGTH_SHORT).show();
                if (result) {
                    getActivity().runOnUiThread(new Runnable() {
                        @Override
                        public void run() {
                            closeProgressDialog();
                            if("province".equals(type)) {
                                queryProvinces();
                            }
                            else if ("city".equals(type)) {
                                queryCities();
                            }
                            else if ("county".equals(type)){
                                queryCountries();
                            }
                        }
                    });
                }
            }
        });
    }

    /**
     * 顯示進度框
     */
    private void showProgressDialog() {
        if(progressDialog == null) {
            progressDialog = new ProgressDialog(getActivity());
            progressDialog.setMessage("正在載入");
            progressDialog.setCanceledOnTouchOutside(false);
        }
        progressDialog.show();
    }


    /**
     * 關閉進度對話框
     */
    private void closeProgressDialog() {
        if (progressDialog != null) {
            progressDialog.dismiss();
        }
    }
ChooseAreaFragment

這裡我把不需要用到的方法刪除,只留下onCreateView和onActivityCreated這兩個方法

一個表示創建佈局,主要用來載入佈局同時對部分數據進行初始化

onActivityCreated顯然是在佈局載入完之後才觸發的,主要的功能都在這裡面實現

主要實現的是listview的跳轉功能,判斷當前是那一級別的數據從而點擊時獲取下一級別的數據,

如當前點擊“江蘇”,首先返回按鈕顯示,標題改為“江蘇”,然後優先查詢資料庫信息,當沒有找到時再向伺服器發送請求

 五、主活動中載入碎片

修改main佈局中的文件

通過name在初始化時運行碎片

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".activity.MainActivity">

    <fragment
        android:id="@+id/choose_area_fragment"
        android:name="com.coolweather2.fragment.ChooseAreaFragment"

        android:layout_width="match_parent"
        android:layout_height="match_parent">

    </fragment>

</FrameLayout>
layout

註意由於碎片中設置過標題,需要在styles.xml中修改預設標題

 

最後需要設置網路訪問許可權(Androidmanifest.xml)

 

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 背景: 朋友的環境第二天突然訪問不了SQL Server,遠程SQL Server用戶無法登陸,但是本地SQL Server用戶登錄正常。 報錯: 用戶XX登錄失敗(MicroSoft SQL Server,錯誤18456) 排查: 對與無法連接伺服器的,一般的排查手段,也是最常用的手段。 1.因為 ...
  • 一、前言 背景:一些公立單位伺服器是不允許使用window7以上的系統的,而大部分都是window server系列的系統 首先大部分軟體安裝預設是在C盤,而一部分人安裝軟體是不喜歡預設安裝在C盤的,window7以上的系統安裝MySql都可以修改Data文件的存儲路徑。 但是我在window se ...
  • 目錄: (3)Android 官方網站 對 MediaPlayer的介紹 正文: Android 官方網站 對 MediaPlayer的介紹 MediaPlayer public class MediaPlayer extends Object implements VolumeAutomation ...
  • 啟動spark#啟動spark之前先要把hadoop節點啟動起來#cd /usr/hadoop/hadoop-3.1.0/#sbin/start-all.sh#jps //檢查啟動的線程是否已經把hadoop啟動起來了#cd /usr/spark/spark-2.3.1-bin-hadoop2.7# ...
  • 1. ImageView 繪製圖片 ...
  • 前提:已經通過USB設備線連接過電腦,併成功安裝驅動。 adb連接手機進行調試有兩種方式,一種是使用USB線,另一種是使用無線WiFi。 第一種 使用USB線連接 1. 在手機上啟用USB調試 2. 打開DOS命令視窗,輸入 cd D:\Android\sdk\platform-tools 回車 3 ...
  • Android的Service也運行在主線程,但是在服務裡面是沒法直接調用更改UI,如果需要服務傳遞消息給Activity,通過廣播是其中的一種方法: 一、在服務裡面發送廣播 通過intent傳送數據、通過setAction 設置Activity接收廣播時要過濾的動作名 二、在Activity中創建 ...
  • Mac 關於本機-磁碟管理,如果發現系統占用超過80g以上的小伙伴們可以做以下操作只需要以下4個步驟,輕鬆降到30g以內!!!!!!!(僅適用於安裝了Xcode的小伙伴) 打開Finder,command + shift +g (前往文件夾),刪除一下路徑的文件即可~ 1、 iOS DeviceSu ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...