Android【网络技术】

Android 网络技术

1. WebView的用法

在应用程序中加载和显示网页

xml文件,在布局中添加一个WebView

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <WebView
        android:id="@+id/web_view"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />
    
</LinearLayout>

MainActivity

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        WebView webView = (WebView)findViewById(R.id.web_view);
        webView.getSettings().setJavaScriptEnabled(true);
        webView.setWebViewClient(new WebViewClient());
        webView.loadUrl("http://www.baidu.com");
    }
}

1.首先用findViewById()方法获取到WebView实例

2. 调用WebView的getSettings()方法可以设置一些浏览器属性,这里只是调用了setJavaScriptEnabled()方法来让WebView支持JavaScript脚本

3. 调用setWebViewClient()方法,并传入一个WebViewClient实例。这段代码的作用是,当需要从一个网页跳到另外一个网页时,我们希望目标网页仍在当前WebView显示,而不是打开浏览器。

4. 接着调用loadUrl()方法,将网址传入,即可展示相应的网页内容

需要注意的是,要加入权限声明。

<uses-permission android:name="android.permission.INTERNET"/>

在这里插入图片描述

2. 使用http协议访问网络

Android 9 http及https的网络连接需要附加一些东西:参考How to allow all Network connection types HTTP and HTTPS in Android (9) Pie?

1. 使用HttpURLConnection

需要声明网络权限

1. 首先需要获取到HttpURLConnection()对象,一般只需要new一个URL对象,并传入目标网络地址,然后调用一下openConnection()方法即可

URL url = new URL("https://www.baidu.com");

HttpURLConnection connection = (HttpURLConnection)url.openConnection();

2. 得到了HttpURLConnection对象之后,我们可以设置一下请求方法,GET和POST。GET希望从服务器那里获取数据,而POST表示希望提交数据给服务器。

connection.setRequestMethod("GET");

3. 可以进行一些自由定制,连接超时,读取超时的毫秒数等等

connection.setConnectTimeout(8000);

connection.setReadTimeout(8000);

4. 获取服务器返回的输入流

InputStream in = connection.getInputStream();

5. 最后可以关闭HTTP连接

connection.disconnect();

示例

xml文件,布局中添加一个按钮

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">
    <Button
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:id="@+id/send_request"
        android:text="Send Request"/>
    <ScrollView
        android:layout_width="match_parent"
        android:layout_height="match_parent">
        <TextView
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:id="@+id/response_content"/>
    </ScrollView>
    
</LinearLayout>

ScrollView:由于手机屏幕的空间一般较小,有些时候过多的内容一屏是显示不下的,借助ScrollView可以以滚动的形式查看屏幕外的内容。

MainActivity

public class MainActivity extends AppCompatActivity implements  View.OnClickListener{
	TextView responseText;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Button sendRequest = (Button)findViewById(R.id.send_request);
        responseText = (TextView)findViewById(R.id.response_content);
        sendRequest.setOnClickListener(this);
    }

    @Override
    public void onClick(View view) {
        if(view.getId()==R.id.send_request){
            sendRequestWithHttpURLConnection();
        }
    }
    
    
    private void sendRequestWithHttpURLConnection(){
        //开启线程来发起网络请求
        new Thread(new Runnable(){
            public void run(){
                HttpURLConnection connection = null;
                BufferedReaader reader = null;
                try{
                    URL url = new URL("https://www.baidu.com");
                    connection = (HttpURLConnection)url.openConnection();
                    connection.setRequestMethod("GET");
                    connection.setConnectTimeout(8000);
                    connection.setReadTimeout(8000);
                    InputStream in = connection.getInputStream();
                    //下面对获取到的输入流进行读取
                    reader = new BufferedReader(new InputStreamReader(in));
                    StringBuilder response = new StringBuilder();
                    String line;
                    while((line = reader.readLine)!=null){
                        response.append(line);
                    }
                    showResponse(response.toString);
                }catch(Exception e){
                    e.printStackTrace();
                }
            }
        }).start;
    }
    
    private void showResponse(final String response){
        runOnUiThread(new Runnable() {
            @Override
            public void run() {
                responseText.setText(response);
            }
        });
    }


}

1. 在Send Request按钮的点击事件里调用了sendRequestWithHttpURLConnection()方法

2.在此方法中开启了一个子线程,调用HttpURLConnection()发出一条请求,目标地址是百度

3. 利用BufferedReader对服务器返回的流进行读取,将结果传入showResponse()方法中

4. 在showResponse()方法中调用了一个runOnUiThread()方法,因为Android不允许在子线程中进行UI操作,通过这个方法将线程切换到主线程

提交数据给服务器

只需要将HTTP请求的方法改成POST,并在获取输入流之前把要提交的数据写出即可。注意每条数据都要以键值对的形式存在,数据与数据之间用“&”符号隔开,比如我们要向服务器提交用户名和密码,如下:

connection.setRequestMethod("POST");
DataOutputStream out = new DataOutputStream(connection.getOutputStream);
out.writeBytes("username=admin&password=123456");

2. 使用OkHttp

在使用OkHttp之前需要在项目中添加依赖OkHttp库的依赖

implementation 'com.squareup.okhttp3:okhttp:4.1.0'

1. 首先需要创建一个OkHttpClient的实例

OkHttpClient client = new OkHttpClient();

2. 要想发起一条HTTP请求,需要创建一个Request对象

Request request = new Request.Builder().build();

3. 上述代码只是创建了一个空的Request对象,我们可以在build()之前通过连缀很多其他方法丰富这个Request对象

Request request = new Request.Builder()
			.url("https://www.baidu.com")
			.build;

4. 之后调用OkHttpClient的newCall()方法来创建一个Call对象,并调用它的execute()方法来发送请求并获取服务器返回的数据

Response response = client.newCall(request).execute();

其中response就是服务器返回的数据

5. 通过如下方法获得数据的具体内容

String responseData = response.body().string();

示例(xml文件不需要改动)

public class MainActivity extends AppCompatActivity implements  View.OnClickListener{
	TextView responseText;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Button sendRequest = (Button)findViewById(R.id.send_request);
        responseText = (TextView)findViewById(R.id.response_content);
        sendRequest.setOnClickListener(this);
    }

    @Override
    public void onClick(View view) {
        if(view.getId()==R.id.send_request){
            sendRequestWithOkHttp();
        }
    }

    private void sendRequestWithOkHttp() {
        new Thread(){
            public void run(){
                try {
                    OkHttpClient client = new OkHttpClient();
                    Request request = new Request.Builder()
                            .url("https://www.baidu.com")
                            .build();
                    Response response = client.newCall(request).execute();
                    String responseData = response.body().string();
                    showResponse(responseData);
                }catch(Exception e){
                    e.printStackTrace();
                }
            }
        }.start();
    }

	private void showResponse(final String response){
        runOnUiThread(new Runnable() {
            @Override
            public void run() {
                responseText.setText(response);
            }
        });
    }


}

提交数据给服务器

首先构建出一个RequestBody对象来存放带提交的数据

RequestBody requestBody= new FormBody.Builder()
		.add("username","admin")
		.add("password","123456")
		.build();

然后在Request.Builder()中调用一下post()方法,并将RequestBody对象传入

Request request = new Request.Builder()
		.url("https://www.baidu.com")
		.post(requestBody)
		.build();

3. 解析XML格式数据

在网络上传输数据时最常用的格式有两种:XML和JSON

我们搭建一个简单的Web服务器,在这个服务器上提供一段XML文本,然后我们在程序里去访问这个服务器,再对得到的XML文本进行解析。

1. 搭建简单的web服务器

Apache下载地址

进入网站,点击如图所示:

在这里插入图片描述

如下图所示,点击

在这里插入图片描述

选择下图框中任一版本下载

在这里插入图片描述

下载后是一个压缩包,将其解压

1. 配置

进入 Apache24/conf/httpd.conf

Define SRVROOT “D:\MyDownloads\httpd-2.4.39-lre-2.9.2-x64-vc14\Apache24”
ServerRoot “D:\MyDownloads\httpd-2.4.39-lre-2.9.2-x64-vc14\Apache24”

将双引号里面的内容改为当前Apache24文件夹的地址

Listen 80 ServerName

这两个后面都有端口号,开始设置为80,你也可以改为其他的,一般小于1024的端口号已经被占用了,这里我修改为18011

2 . 安装

以管理员身份打开控制台(右键点击开始),到bin目录下,输入 .\httpd -k install

安装成功后,运行ApacheMonitor.exe,如图

在这里插入图片描述
选择Apache2.4,点击start

在浏览器地址栏输入,localhost:端口号,若出现如下页面则表示启动成功

在这里插入图片描述

2. 开始

1. 进入htdocs文件夹,新建get_data.xml文件,编辑,并加入如下XML格式的内容

<apps>
	<app>
		<id>1</id>
		<name>Google Maps</name>
		<version>1.0</version>
	</app>
	<app>
		<id>2</id>
		<name>Chrome</name>
		<version>2.1</version>
	</app>
	<app>
		<id>3</id>
		<name>Google Play</name>
		<version>2.3</version>
	</app>
</apps>

Pull解析方式

public class MainActivity extends AppCompatActivity implements  View.OnClickListener{

    TextView responseText;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Button sendRequest = (Button)findViewById(R.id.send_request);
        responseText = (TextView)findViewById(R.id.response_content);
        sendRequest.setOnClickListener(this);
    }

    @Override
    public void onClick(View view) {
        if(view.getId()==R.id.send_request){
            sendRequestWithOkHttp();
        }
    }

    private void sendRequestWithOkHttp() {
        new Thread(){
            public void run(){
                try {
                    OkHttpClient client = new OkHttpClient();
                    Request request = new Request.Builder()
                            .url("http://10.0.2.2:18011/get_data.xml")
                            .build();
                    Response response = client.newCall(request).execute();
                    String responseData = response.body().string();
                    parseXMLWithPull(responseData);
                }catch(Exception e){
                    e.printStackTrace();
                }
            }
        }.start();
    }


    private void parseXMLWithPull(String xmlData){
        try {
            XmlPullParserFactory factory = XmlPullParserFactory.newInstance();
            XmlPullParser xmlPullParser = factory.newPullParser();
            xmlPullParser.setInput(new StringReader(xmlData));
            int eventType = xmlPullParser.getEventType();
            String id = "";
            String name = "";
            String version = "";
            while (eventType != XmlPullParser.END_DOCUMENT) {
                String nodeName = xmlPullParser.getName();
                switch (eventType) {
                    case XmlPullParser.START_TAG:
                        if ("id".equals(nodeName)) {
                            id = xmlPullParser.nextText();
                        } else if ("name".equals(nodeName)) {
                            name = xmlPullParser.nextText();
                        } else if ("version".equals(nodeName)) {
                            version = xmlPullParser.nextText();
                        }
                        break;
                    case XmlPullParser.END_TAG:
                        if ("app".equals(nodeName)) {
                            Log.d("MainActivity", "id is " + id);
                            Log.d("MainActivity", "name is " + name);
                            Log.d("MainActivity", "version is " + version);
                        }
                        break;
                    default:
                        break;
                }
                eventType = xmlPullParser.next();
            }

        }catch (Exception e){
            e.printStackTrace();
        }
    }
}

1 . 首先获取到一个XmlPullParserFactory的实例,并借助这个实例得到XmlPullParser对象

2 . 调用XmlPullParser的setInput()方法将服务器返回的XML数据设置进去

3. 通过getEventType()可以得到当前解析事件,然后在一个while循环中不断地解析

4. 如果当前的解析事件不等于XmlPullParser.END_DOCUMENT,说明解析没有完成,调用next()获取下一个解析事件

5 . 在while循环中,通过getName()得到当前节点的名字,如果发现节点名等于id,name或者version,就调用nextText()方法获取节点的具体内容

SAX解析

新建一个ContentHandler类继承DefaultHandler

public class ContentHandler extends DefaultHandler {
    private static final String TAG = "ContentHandler";
    private String nodeName;
    private StringBuilder id;
    private StringBuilder name;
    private StringBuilder version;
    @Override
    public void startDocument() throws SAXException {
        id = new StringBuilder();
        name = new StringBuilder();
        version = new StringBuilder();
    }

    @Override
    public void endDocument() throws SAXException {
        super.endDocument();
    }

    @Override
    public void startElement(String uri, String localName, String qName, Attributes attributes) throws SAXException {
        nodeName = localName;
    }

    @Override
    public void endElement(String uri, String localName, String qName) throws SAXException {
        if("app".equals(localName)){
            Log.d(TAG, "id is "+id.toString().trim());
            Log.d(TAG, "name is "+name.toString().trim());
            Log.d(TAG, "version is "+version.toString().trim());
            id.setLength(0);
            name.setLength(0);
            version.setLength(0);
        }
    }

    @Override
    public void characters(char[] ch, int start, int length) throws SAXException {
        if("id".equals(nodeName)){
            id.append(ch,start,length);
        }else if ("name".equals(nodeName)){
            name.append(ch,start,length);
        }else if ("version".equals(nodeName)){
            version.append(ch,start,length);
        }
    }
}



startDocument()方法会在开始XML解析的时候调用。

startElement()方法会在开始解析某个节点时调用

characters()方法会在获取节点中的内容时调用

endElement()方法会在完成解析某个节点时调用

endDocument()方法会在完成整个XML解析时调用

public class MainActivity extends AppCompatActivity implements  View.OnClickListener{

    TextView responseText;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Button sendRequest = (Button)findViewById(R.id.send_request);
        responseText = (TextView)findViewById(R.id.response_content);
        sendRequest.setOnClickListener(this);
    }

    @Override
    public void onClick(View view) {
        if(view.getId()==R.id.send_request){
            sendRequestWithOkHttp();
        }
    }

    private void sendRequestWithOkHttp() {
        new Thread(){
            public void run(){
                try {
                    OkHttpClient client = new OkHttpClient();
                    Request request = new Request.Builder()
                            .url("http://10.0.2.2:18011/get_data.xml")
                            .build();
                    Response response = client.newCall(request).execute();
                    String responseData = response.body().string();
                    parseXMLWithSAX(responseData);
                }catch(Exception e){
                    e.printStackTrace();
                }
            }
        }.start();
    }

    private void parseXMLWithSAX(String xmlData)  {
        try {
            SAXParserFactory factory = SAXParserFactory.newInstance();
            XMLReader xmlReader = factory.newSAXParser().getXMLReader();
            ContentHandler handler = new ContentHandler();
            xmlReader.setContentHandler(handler);
            xmlReader.parse(new InputSource(new StringReader(xmlData)));
        }catch (Exception e){
            e.printStackTrace();
        }
    }
}

parserXMLWithSAX方法先创建了一个SAXParserFactory对象,然后再获取XMLReader对象,接着将我们编写的ContentHandler的实例设置到XMLReader中,最后调用parse()方法

4. 解析JSON格式数据

1. 进入htdocs文件夹,新建get_data.json文件,编辑,并加入如下JSON格式的内容

[{"id":"5","version":"5.5","name":"clash of clans"},
{"id":"6","version":"7","name":"boom beach"},
{"id":"7","version":"3.5","name":"clash royale"}]
public class MainActivity extends AppCompatActivity implements  View.OnClickListener{

    TextView responseText;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Button sendRequest = (Button)findViewById(R.id.send_request);
        responseText = (TextView)findViewById(R.id.response_content);
        sendRequest.setOnClickListener(this);
    }

    @Override
    public void onClick(View view) {
        if(view.getId()==R.id.send_request){
            sendRequestWithOkHttp();
        }
    }

    private void sendRequestWithOkHttp() {
        new Thread(){
            public void run(){
                try {
                    OkHttpClient client = new OkHttpClient();
                    Request request = new Request.Builder()
                            .url("http://10.0.2.2:18011/get_data.json")
                            .build();
                    Response response = client.newCall(request).execute();
                    String responseData = response.body().string();
                    parseJSONWithJSONObject(responseData);
                }catch(Exception e){
                    e.printStackTrace();
                }
            }
        }.start();
    }

    private void parseJSONWithJSONObject(String jsonData){
        try {
            JSONArray jsonArray = new JSONArray(jsonData);
            for (int i = 0; i < jsonArray.length(); i++) {
                JSONObject jsonObject = jsonArray.getJSONObject(i);
                String id = jsonObject.getString("id");
                String name = jsonObject.getString("name");
                String version = jsonObject.getString("version");
                Log.d("MainActivity", "id is " + id);
                Log.d("MainActivity", "name is " + name);
                Log.d("MainActivity", "version is " + version);
            }
        }catch (Exception e){
            e.printStackTrace();
        }
    }

}

由于我们在服务器中定义的时一个JSON数组,因此这里首先将服务器返回的数据传入到了一个JSONArray对象

中,然后循环遍历这个JSONArray,从中取出的每一个元素都是JSONObject对象,每个对象中又会包含id,name,version这些数据

发布了31 篇原创文章 · 获赞 9 · 访问量 1611

猜你喜欢

转载自blog.csdn.net/qq_43621019/article/details/99613236