Android学习系列笔记(五)

Stella981
• 阅读 873

##Android基础网络第二天

1 post方式提交数据乱码的解决

一般在公司开发客户端和服务端的编码要保持一致。
android端的默认编码是utf-8;

做url请求时需要对参数进行URLEncode编码.

URL url = new URL("http://192.168.13.83:8080/itheima74/servlet/LoginServlet?username="+URLEncoder.encode(username)+"&pwd="+URLEncoder.encode(password));


connection.setDoOutput(true);
connection.getOutputStream().write(parmes.getBytes());

2 get方式提交数据乱码解决

URLEncode

3 httpclient方式提交数据到服务器

 HttpClient:

get方式:
        //使用HttpClient请求服务器将用户密码发送服务器验证
            try{
            String path = "http://192.168.13.83:8080/itheima74/servlet/LoginServlet?username="+URLEncoder.encode(username,"utf-8")+"&pwd="+URLEncoder.encode(password,"utf-8");
            //1.创建一个httpClient对象
            HttpClient httpclient = new DefaultHttpClient();
            
            //2.设置请求的方式
            HttpGet httpget = new HttpGet(path);
            //3.执行一个http请求
            HttpResponse response = httpclient.execute(httpget);
            //4.获取请求的状态码,
            StatusLine statusLine = response.getStatusLine();
            int code = statusLine.getStatusCode();
            
            //5.判断状态码后获取内容
            if(code == 200){
                HttpEntity entity = response.getEntity();//获取实体内容,中封装的有http请求返回的流信息
                InputStream inputStream = entity.getContent();
                //将流信息转换成字符串
                String result = StreamUtils.streamToString(inputStream);
                
                Message msg = Message.obtain();
                msg.what = 1;
                msg.obj = result;
                handler.sendMessage(msg);
            }
            
            }catch (Exception e) {
                e.printStackTrace();
            }
post方式:


        //使用UrlConncetion请求服务器将用户密码发送服务器验证
            try{
                    String path = "http://192.168.13.83:8080/itheima74/servlet/LoginServlet";
                    //1.创建一个httpclient对象
                    HttpClient httpclient = new DefaultHttpClient();
                    //2.创建一个请求方式
                    HttpPost httppost = new HttpPost(path);
                    //创建集合封装数据
                    ArrayList<BasicNameValuePair> arrayList = new ArrayList<BasicNameValuePair>();
                    BasicNameValuePair nameValuePair = new BasicNameValuePair("username",username);
                    arrayList.add(nameValuePair);
                    BasicNameValuePair nameValuePair1 = new BasicNameValuePair("pwd",password);
                    arrayList.add(nameValuePair1);
                    
                    //创建一个Entity
                    UrlEncodedFormEntity entity = new UrlEncodedFormEntity(arrayList, "utf-8");
                    //设置请求时的内容
                    httppost.setEntity(entity);
                    
                    //3.执行一个请求,返回一个response对象
                    HttpResponse response = httpclient.execute(httppost);
                    //4.获取状态码
                    int code = response.getStatusLine().getStatusCode();
                    //5.判断并获取内容
                    if(code == 200){
                        HttpEntity entity1 = response.getEntity();//获取实体内容,中封装的有http请求返回的流信息
                        InputStream inputStream = entity1.getContent();
                        //将流信息转换成字符串
                        String result = StreamUtils.streamToString(inputStream);
                        Message msg = Message.obtain();
                        msg.what = 2;
                        msg.obj = result;
                        handler.sendMessage(msg);
                    }

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

4开源项目get post 方式提交 (asyncHttpClient)

    get方式:


                public static void requestNetForGetLogin(final Context context,final Handler handler ,final String username, final String password) {
            //使用HttpClient请求服务器将用户密码发送服务器验证
            try{
            String path = "http://192.168.13.83:8080/itheima74/servlet/LoginServlet?username="+URLEncoder.encode(username,"utf-8")+"&pwd="+URLEncoder.encode(password,"utf-8");
    
            //创建一个AsyncHttpClient对象
            AsyncHttpClient asyncHttpClient = new AsyncHttpClient();
            asyncHttpClient.get(path, new AsyncHttpResponseHandler() {
                
                [@Override](https://my.oschina.net/u/1162528)
                public void onSuccess(int statusCode, Header[] headers, byte[] responseBody) {
                    //statusCode:状态码    headers:头信息  responseBody:返回的内容,返回的实体
                    //判断状态码
                    if(statusCode == 200){
                        //获取结果
                        try {
                            String result = new String(responseBody,"utf-8");
                            Toast.makeText(context, result, 0).show();
                        } catch (UnsupportedEncodingException e) {
                            // TODO Auto-generated catch block
                            e.printStackTrace();
                        }
                    }
                }
                
                [@Override](https://my.oschina.net/u/1162528)
                public void onFailure(int statusCode, Header[] headers,
                        byte[] responseBody, Throwable error) {
                    
                    System.out.println("...............onFailure");
                    
                }
            });
            }catch (Exception e) {
                e.printStackTrace();
            }
}


    post方式:


                String path = "http://192.168.13.83:8080/itheima74/servlet/LoginServlet";

        AsyncHttpClient asyncHttpClient = new AsyncHttpClient();
        RequestParams params = new RequestParams();
        params.put("username", username);
        params.put("pwd", password);
        
        //url:   parmas:请求时携带的参数信息   responseHandler:是一个匿名内部类接受成功过失败
        asyncHttpClient.post(path, params, new AsyncHttpResponseHandler() {
            
            [@Override](https://my.oschina.net/u/1162528)
            public void onSuccess(int statusCode, Header[] headers, byte[] responseBody) {
                //statusCode:状态码    headers:头信息  responseBody:返回的内容,返回的实体
                
                //判断状态码
                if(statusCode == 200){
                    //获取结果
                    try {
                        String result = new String(responseBody,"utf-8");
                        Toast.makeText(context, result, 0).show();
                    } catch (UnsupportedEncodingException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }
                }
                
            }
            
            [@Override](https://my.oschina.net/u/1162528)
            public void onFailure(int statusCode, Header[] headers,
                    byte[] responseBody, Throwable error) {
                
            }
        });

5 文件上传的操作

    使用第三方utils做文件上传。
        
public void fileupload(View v){
    try{
    EditText et_filepath = (EditText) findViewById(R.id.et_filepath);
    //获取输入的文件地址
    String filepath = et_filepath.getText().toString().trim();
    
    //使用开源Utils做上传操作
    AsyncHttpClient asyncHttpClient = new AsyncHttpClient();
    
    RequestParams params = new RequestParams();
    params.put("filename", new File(filepath));
    //url : 请求服务器的url
    asyncHttpClient.post("http://192.168.13.83:8080/itheima74/servlet/UploaderServlet", params, new AsyncHttpResponseHandler() {
        
        [@Override](https://my.oschina.net/u/1162528)
        public void onSuccess(int statusCode, Header[] headers, byte[] responseBody) {
            if(statusCode == 200){
                Toast.makeText(MainActivity.this, "上传成功", 0).show();
            }
        }
        @Override
        public void onFailure(int statusCode, Header[] headers,
                byte[] responseBody, Throwable error) {
        }
    });
    
    }catch (Exception e) {
        e.printStackTrace();
    }
}

6 多线程加速下载的原理

    1.线程越多下载越快?? 不是。 与 本地网络带宽, 服务器资源的带宽 有关
    2.迅雷:3-5个。

    多线程下载的步骤:

    1.要知道服务端资源的大小。
    
        通过URLConnection请求服务器url获取。
        UrlConnection.getContentLength();//资源的大小
        
    2.在本地创建一个与服务端资源同样大小的一个文件(占位)
            //file :  文件; mode:文件的模式,rwd:直接写到底层设备,硬盘
            RandomAccessFile randomfile =new RandomAccessFile(File file,String mode)
                
            randomfile.setLength(long size);//创建一个文件和服务器资源一样大小
            
    3.要分配每个线程下载文件的开始位置和结束位置。
            
    4.开启线程去执行下载
            通过UrlConnection下载部分资源。
            注意:
             1.需要Range头,key:Range   value:bytes:0-499 
                    urlconnection.setRequestPropety("Range","bytes:0-499")
             2.需要设置每个线程在本地文件的保存的开始位置
                    RandomAccessFile randomfile =new RandomAccessFile(File file,String mode)
                    randomfile.seek(int startPostion);//本次线程下载保存的开始位置。
                    
    5.要知道每个线程下载完毕。

7 javase 多线程下载

8 多线程断点续传实现

9 Android版本多线程下载

        安智: sdcard没有判断。uc

10 开源项目实现多线程下载 (xutils)

    public void download(View v){
    EditText et_url = (EditText) findViewById(R.id.et_url);
    String url = et_url.getText().toString().trim();
    //1.创建httpUtils对象
    HttpUtils httpUtils = new HttpUtils();
    //2.调用download方法  url:下载的地址  target:下载的目录   callback:回调 
    httpUtils.download(url, "/sdcard/feiqiu/feiq.exe", new RequestCallBack<File>() {
        @Override
        public void onLoading(long total, long current, boolean isUploading) {
            System.out.println("total:"+total+";current:"+current);
            super.onLoading(total, current, isUploading);
        }
        @Override
        public void onSuccess(ResponseInfo<File> responseInfo) {
            System.out.println(responseInfo.result);
        }
        @Override
        public void onFailure(HttpException error, String msg) {
            // TODO Auto-generated method stub
            
        }
    });
}

2天网络内容大复习

总结:

第一天: android入门: 
        
        1.环境搭建 
                
            jdk+ eclipse+android sdk

        2,项目结构介绍
            熟悉各个目录的用途.
        3.五大布局

            LinearLayout + RelativeLayout
            控件: EditText TextVeiw Button ImageView ListView ScrollView CheckBox ProgressBar 
        4.adb的练习
            adb devices 
            adb install ; adb uninstall ; adb push ;adb pull;logcat ;adb shell
            adb kill-server ;adb start-server ;adb shell+input text  ;adb shell+monkey

        5.android系统架构
            4层;


        6. 电话拨号器

                Intent intent = new Intent();
                intent.setAction();
                intent.setData();
                startActivity();
        

第二天:数据的存储

        1.登录案例
            
            File file = context.getFilesDir();
            context.openFileOutput(String filename,String mode);//mode:Context.Mode_Private
            context.openFileInput(String filename);

        2.sdcard的使用
                1.权限
                2.硬性编码
                File file = Enviroment.getExternalStorageDirectory();//获取sdcard的目录文件
                3.判断sdcard状态
                    boolean state = Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)
                4.判断sdcard大小
                    File file = Enviroment.getExternalStorageDirectory();//获取sdcard的目录文件
                    long usablespace= file.getUsableSpace();
                    String usablespace_str= Formatter.foramatterFilesize(long size);

        4.SharedPreference的使用

                1.保存数据

                a. 获取一个sharedpreference对象
                        Sharedpreference sharedpreference = context.getShare..(String name ,String mode);
                b.得到一个Editor对象
                    Editor editor =     sharedpreference.edit();
                c.往Editor中添加数据
                        editor.putString(String key ,Stirng value);
                d.提交Editor
                    editor.commit();


                2.读取数据
                        
                a. 获取一个sharedpreference对象
                        Sharedpreference sharedpreference = context.getShare..(String name ,String mode);
                b.使用sharedpreference获取数据
                        sharedpreference.getString(String key ,String defValue);

        5.xml的生成与解析
            a.生成  XmlSerializer  模板代码
            b.解析 XmlPullParser 模板代码

第三天: 数据库 listview

    1.数据库的使用  对于一个数据库2个类, 1是帮助类 SqliteOpenHelper 2是Dao类封装增删改查

        1.创建数据库
            创建一个类集成SqliteOpenHelper 创建一个构造方法设置数据库的名称,版本,实现两个方法,oncteate:特别适合做表结构的初始化,数据库第一次创建的时候调用。
            onupgrade:特别适合做表结构的修改,数据库版本发生改变时调用
        2.增删改查
            a.创建一个帮助类的对象,并调用getReadableDatabase或getWriteableDatabase帮助获取一个SqliteDatabase数据库操作对象。

            b.使用SqliteDatabase数据库操作对象做增删改查
                2种方式
                1.调用execSql(String sql ,Object[] obj)可以做增删改,调用rawQuery(String sql ,Object[] obj)做查询,返回一个Cursor对象,遍历Cursor获取查询结果。
                2.调用insert delete update query做增删改查

                第一种方式适合做查询,第二种方式有了返回值,更适合做的增删改
                    
            
    2.listview的使用

                1.写布局
                2.找到listview,并设置条目的点击事件
                    listview.setOnItemClickListener(OnItemClickListener on);
                3.获取listview条目上要展示的数据到list集合中,传给adapter做控制显示
                4.写一个类集成BaseAdapter,实现四个方法,写一个构造方法接受list集合中的数据

                        getCount();返回listview要显示的条目数
                        getItem(int postion);返回listveiw上postion位置上的Bean对象
                        getId();返回postion
                        *****getView();告诉listview条目要显示的内容
                            1.模板代码优化listview
                            2.创建一个item布局转化成view对象作为getview的返回值
                            3.获取item布局中子控件
                            4.获取list集合中postion位置的Bean对象
                            5.将Bean中的数据设置给子控件做展示
                5.创建一个adapter,并设置给listview
                        listveiw.setAdapter(ListAdapter adapter)
                6.实现条目点击事件的操作

第四天:网络编程

    1.使用UrlConnection连接网络获取数据

            1.创建一个Url对象
            2.使用URL对象获取一个HttpUrlConnection
            3.设置connection对象的参数
            4.获取状态码
            5.获取流信息。


    2.主线程不能做耗时的操作(网络请求),只能在子线程中请求;子线程不能更新UI;解决办法:Handler

    3.使用handler
            1.主线程创建Handler,重写handlermessage方法
            2.子线程中创建Message对象,携带子线程获取的数据
            3.使用主线程的handlerf发送message到主线程
            4.主线程的handlerMessage方法接受数据,处理Ui

    4.流转图片

        Bitmap bitmap = BitmaptFatory.decodeStream(InputStream in);
    5.网络版新闻: 数据库  listview   URLConnection  Json解析  Handler        
            json解析:
                JsonObject  JsonArray
    6.get post请求网络
        post提交数据需要设置内容到写入流:
            connection.setDoOut(true);
            connection.getOutputStream().write(byte[]);
    7.解决乱码问题
            1.客户端和服务端编码保持一致
            2.提交的数据需要用URLEncode编码

第五天 网络请求2 HttpClient 文件上传 多线程下载

            1.HttpClent:
                1.创建一个DefaultHttpclient
                2.创建一个请求方式 HttpGet HttpPost
                    
                        Arraylist<BaicNameValueparire> list = new Arraylist<BaicNameValueparire>();
                        BaicNameValueparire valuse =    BaicNameValueparire()
                        valuse.put(String key ,String valuse);
                            list.add(valuse);
                        UrlEncodeFormEntity entity = new UrlEncodeFormEntity(List<NameValuepare>);
                        HttpPost.setEntity(entity);
                
                3.httpclient执行请求
                    httpeclient.execut(HttpGet httpget)
                4.获取响应码
                        httpclient.getStateLines().getstateCode();
                5.获取内容
                    InputStream in = httpclient.getEntry.getContent();

            2.文件上传 使用开源项目  AsycnHttpClient

            3.多线程断点续传下载        

                1.获取资源大小
                2.本地创建一个与服务端一致的文件            
                    RandmodAcecssFile
                3.根据线程数量计算每个线程下载的开始位置,结束位置

                4.开启线程进行真实的下载。    
                    header : Range  value :   bytes:0-499  
                    UrlConnection.setRequestProperty(String header , String value);
            
                5.将读取 的流分段写入文件
                        RandmodAcecssFile file = new RandmodAcecssFile(File file,String mode);
                        file.seek(int startIndex);//设置文件从哪里开始写入
                6.ProgressBar : 主要属性 : style   样式   max 最大进度   progress 当前进度
点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Wesley13 Wesley13
3年前
java将前端的json数组字符串转换为列表
记录下在前端通过ajax提交了一个json数组的字符串,在后端如何转换为列表。前端数据转化与请求varcontracts{id:'1',name:'yanggb合同1'},{id:'2',name:'yanggb合同2'},{id:'3',name:'yang
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
待兔 待兔
5个月前
手写Java HashMap源码
HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程22
Jacquelyn38 Jacquelyn38
3年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
Stella981 Stella981
3年前
Android So动态加载 优雅实现与原理分析
背景:漫品Android客户端集成适配转换功能(基于目标识别(So库35M)和人脸识别库(5M)),导致apk体积50M左右,为优化客户端体验,决定实现So文件动态加载.!(https://oscimg.oschina.net/oscnet/00d1ff90e4b34869664fef59e3ec3fdd20b.png)点击上方“蓝字”关注我
Wesley13 Wesley13
3年前
Unity横屏
Android下发现Unity里面的Player设置,并不能完全有效,比如打开了自动旋转,启动的时候还是会横屏,修改XML添加以下代码<applicationandroid:icon"@drawable/ic\_launcher"                    android:label"@string/app\_name"
Stella981 Stella981
3年前
Django中Admin中的一些参数配置
设置在列表中显示的字段,id为django模型默认的主键list_display('id','name','sex','profession','email','qq','phone','status','create_time')设置在列表可编辑字段list_editable
Wesley13 Wesley13
3年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_
Python进阶者 Python进阶者
11个月前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这