基于OkHttp3封装网络请求框架

2023-05-16

 

  • 前言

 

网络请求可以说是开发一款移动APP最核心的基础功能了,通过实际工作中以及浏览了许多网络框架之后,本篇在这里分享慕课一位老师基于OkHttp封装的一个轻量的网络框架,至于为什么说它轻量,因为代码少啊!在这里会实现基本的get/post请求,并且支持https加密请求,通过封装方便使用的API,来简化我们的调用方式,对于文件的上传和下载在后续的功能中我会补上这一部分,因为时间关系,前期就没有把它做进去,因为是说的是封装,所以关于okhttp的基本使用这里就不做说明了,不了解的朋友们可以自行百度或者谷歌查找相关资料,这里甩出GitHub和官网的地址,大家可以简单的了解一下。

GitHub地址:https://github.com/square/okhttp

官网地址:http://square.github.io/okhttp/

一、为什么要封装?

首先来看不封装的情况下我们来完成一个最基本的网络请求该如何去做,来看代码:

 

//1、创建OkHttpClient对象
OkHttpClient mOkHttpClient = new OkHttpClient();
//2、创建一个Request
final Request request = new Request.Builder()
        .url("https://www.baidu.com/")
        .build();
//3、通过okhttpclient对象来构建Call对象
Call call = mOkHttpClient.newCall(request);
//4、请求加入调度
call.enqueue(new Callback() {
    
    //请求失败
    @Override
    public void onFailure(@NonNull Call call, @NonNull IOException e) {

    }
    
    //请求成功
    @Override
    public void onResponse(@NonNull Call call, @NonNull Response response) throws IOException {

    }
});

有人说了,这代码也不是很多啊,但是如果我们每个请求都这么写就显得很繁琐了,而且这还是最基本的get请求,所以实际工作中我们完全不可能这么使用,原因也说了:1、代码冗余;2:如果官方的api发生了更新,那么你每个网络请求的地方都要修改,这就有点得不偿失了;3:作为一个高逼格的程序猿,网络请求的逻辑全都暴露在了Activity里面,这肯定是不允许的,所以综合以上几点,我们肯定要对它做一个封装,接下来就来看看如何去封装。

 

二、封装思路说明

1、首先是封装我们的Request,创建一个类来接收请求参数和URL,然后返回给我们一个创建好的请求对象;2、然后是封装我们的okhttp核心部分,这部分首先要能完成请求的发送(主要是get和post请求,反正我在实际工作中是没有碰到过put和delete请求的),然后是配置我们的okhttp相关的参数,并且我们还添加了https请求的支持,这里是对所有https类型站点的信任;3、最后封装callback部分,我们为这部分封装了对成功和失败的回调处理以及相关的异常处理,还有就是okhttp在回调完成后此时还是处于子线程中的,这点是它框架的机制导致的,所以这时候我们还不能进行UI操作,我们需要把消息转发到主线程中去,最后为了方便应用层的处理,我们在框架层就提前将服务器返回的json数据解析成对应的实体对象,然后我们在应用层中就可以直接操作我们Bean类中的属性了,好了,思路就是这样,那说干就干,开始封装。

三、撸起袖子加油干——代码实战

这里我们需要创建一个框架Module,这个Module是一个Android Library,然后将封装的代码都放在这个Module中进行编写,然后让应用层的app Module依赖框架Module即可,这样不会将封装的具体实现逻辑暴露在应用层。

1、封装Request

封装我们的请求参数到一个HashMap中,这个类并不是我写的,是参考了之前使用的一个网络框架AsyncHttpClient中的代码,创建了两个线程安全的HashMap,将请求参数以键值对的形式放到我们的HashMap中,代码如下:

 

package com.archie.netlibrary.okhttp.request;

import java.io.FileNotFoundException;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

/**
 * 项目名:   NetTest2
 * 包名:     com.archie.netlibrary.okhttp.request
 * 文件名:   RequestParams
 * 创建者:   Jarchie
 * 创建时间: 17/12/13 上午10:28
 * 描述:     封装所有的请求参数到HashMap中
 * 此类是仿照AsyncHttpClient中的请求参数写的
 */

@SuppressWarnings({"unused", "RedundantIfStatement"})
public class RequestParams {
    //线程安全的HashMap
    public ConcurrentHashMap<String, String> urlParams = new ConcurrentHashMap<>();
    public ConcurrentHashMap<String, Object> fileParams = new ConcurrentHashMap<>();

    /**
     * Constructs a new empty {@code RequestParams} instance.
     */
    public RequestParams() {
        this((Map<String, String>) null);
    }

    /**
     * Constructs a new RequestParams instance containing the key/value string
     * params from the specified map.
     *
     * @param source the source key/value string map to add.
     */
    public RequestParams(Map<String, String> source) {
        if (source != null) {
            for (Map.Entry<String, String> entry : source.entrySet()) {
                put(entry.getKey(), entry.getValue());
            }
        }
    }

    /**
     * Constructs a new RequestParams instance and populate it with a single
     * initial key/value string param.
     *
     * @param key   the key name for the intial param.
     * @param value the value string for the initial param.
     */
    public RequestParams(final String key, final String value) {
        this(new HashMap<String, String>() {
            {
                put(key, value);
            }
        });
    }

    /**
     * Adds a key/value string pair to the request.
     *
     * @param key   the key name for the new param.
     * @param value the value string for the new param.
     */
    public void put(String key, String value) {
        if (key != null && value != null) {
            urlParams.put(key, value);
        }
    }

    public void put(String key, Object object) throws FileNotFoundException {
        if (key != null) {
            fileParams.put(key, object);
        }
    }

    public boolean hasParams() {
        if (urlParams.size() > 0 || fileParams.size() > 0) {
            return true;
        }
        return false;
    }

}

创建一个CommonRequest类,接收请求参数,然后生成Request对象,这里因为内容较少所有没有采用构建者的模式,二是采用公共静态的方法,在下一篇介绍对Retrofit的一个封装时我会采用构建者模式,因为它的封装过程比这个要稍微复杂一些,内容也更全面一些,敬请期待吧!这个类中我们创建了两个方法用于生成get和post类型的Request,都是通过循环遍历Map中的请求参数来完成的,不同的是,get请求的key-value是通过“&”拼接的,所以这里使用了StringBuilder来完成字符串的拼接,效率更高,post请求是通过okhttp中的FormBody对象的建造者来完成的,具体实现的代码如下:

 

package com.archie.netlibrary.okhttp.request;

import java.util.Map;

import okhttp3.FormBody;
import okhttp3.Request;

/**
 * 项目名:   NetTest2
 * 包名:     com.archie.netlibrary.okhttp.request
 * 文件名:   CommonRequest
 * 创建者:   Jarchie
 * 创建时间: 17/12/13 上午10:31
 * 描述:     接收请求参数,为我们生成Request对象
 */

public class CommonRequest {

    /**
     * 创建Get请求的Request
     *
     * @param url
     * @param params
     * @return 通过传入的参数返回一个Get类型的请求
     */
    public static Request getRequest(String url, RequestParams params) {
        StringBuilder urlBuilder = new StringBuilder(url).append("?");

        if (params != null) {
            for (Map.Entry<String, String> entry : params.urlParams.entrySet()) {
                urlBuilder
                        .append(entry.getKey())
                        .append("=")
                        .append(entry.getValue())
                        .append("&");
            }
        }

        return new Request.Builder().url(urlBuilder.substring(0, urlBuilder.length() - 1))
                .get().build();
    }

    /**
     * 创建Post请求的Request
     *
     * @param url
     * @param params
     * @return 返回一个创建好的Request对象
     */
    public static Request createPostRequest(String url, RequestParams params) {
        FormBody.Builder mFromBodyBuilder = new FormBody.Builder();

        if (params != null) {
            for (Map.Entry<String, String> entry : params.urlParams.entrySet()) {
                //将请求参数逐一遍历添加到我们的请求构建类中
                mFromBodyBuilder.add(entry.getKey(), entry.getValue());
            }
        }

        //通过请求构建类的build方法获取到真正的请求体对象
        FormBody mFormBody = mFromBodyBuilder.build();
        return new Request.Builder().url(url).post(mFormBody).build();
    }

}

2、封装OkHttp核心部分

 

这部分主要完成的功能是:1、请求的发送;2、请求参数的配置(这里配置了连接超时和读写超时时间,允许重定向);3、添加https的支持,在hostnameVerifier的回调中我们返回的是true,表明对所有类型的https证书的支持,无论是自己生成的还是购买的,对于sslSocketFactory的设置,我这里是参考网上的相关代码做了一个封装,https就是在TCP和HTTP协议之间加了一层SSL协议,是介于传输层和应用层之间的协议,需要注意的是加密算法的类型要与服务端的保持一致,一般为TSL/SSL,关于https我本人并不是太了解,讲的也不清楚,这里推荐给大家一篇鸿洋的文章,Android Https相关完全解析http://blog.csdn.net/lmj623565791/article/details/48129405。关于配置的一系列操作都是放在静态语句块中执行的,主要是通过OkHttpClient对象的Builder对象来设置的,最后我们创建了get()和post()两个方法,用于发送具体的请求,参数传入我们的Request和CallBack回调,具体代码如下:

 

package com.archie.netlibrary.okhttp;

import com.archie.netlibrary.okhttp.https.HttpsUtils;
import com.archie.netlibrary.okhttp.listener.DisposeDataHandle;
import com.archie.netlibrary.okhttp.response.CommonJsonCallback;

import java.util.concurrent.TimeUnit;

import javax.net.ssl.HostnameVerifier;
import javax.net.ssl.SSLSession;

import okhttp3.Call;
import okhttp3.OkHttpClient;
import okhttp3.Request;

/**
 * 项目名:   NetTest2
 * 包名:     com.archie.netlibrary.okhttp
 * 文件名:   CommonOkHttpClient
 * 创建者:   Jarchie
 * 创建时间: 17/12/13 上午10:25
 * 描述:     请求的发送,请求参数的配置,https支持
 */

public class CommonOkHttpClient {

    private static final int TIME_OUT = 30; //超时参数
    private static OkHttpClient mOkHttpClient;

    //为我们的Client配置参数,使用静态语句块来配置
    static {
        //创建我们Client对象的构建者
        OkHttpClient.Builder okHttpBuilder = new OkHttpClient.Builder();
        okHttpBuilder
                //为构建者填充超时时间
                .connectTimeout(TIME_OUT, TimeUnit.SECONDS)
                .readTimeout(TIME_OUT, TimeUnit.SECONDS)
                .writeTimeout(TIME_OUT, TimeUnit.SECONDS)
                //允许重定向
                .followRedirects(true)
                //添加https支持
                .hostnameVerifier(new HostnameVerifier() {
                    @Override
                    public boolean verify(String s, SSLSession sslSession) {
                        return true;
                    }
                })
                .sslSocketFactory(HttpsUtils.initSSLSocketFactory(), HttpsUtils.initTrustManager());
        mOkHttpClient = okHttpBuilder.build();
    }

    //发送具体的HTTP以及Https请求
    public static Call sendRequest(Request request, CommonJsonCallback commonCallback) {
        Call call = mOkHttpClient.newCall(request);
        call.enqueue(commonCallback);
        return call;
    }

    //GET请求
    public static Call get(Request request, DisposeDataHandle handle) {
        Call call = mOkHttpClient.newCall(request);
        call.enqueue(new CommonJsonCallback(handle));
        return call;
    }

    //POST请求
    public static Call post(Request request, DisposeDataHandle handle) {
        Call call = mOkHttpClient.newCall(request);
        call.enqueue(new CommonJsonCallback(handle));
        return call;
    }

}

 

关于HttpsUtils的代码,文末会给出项目源码,这里就不贴出了,不然代码太多了。

3、封装Callback

首先我们创建一个接口类,自定义事件监听的回调,用于处理成功和失败的请求,参数中使用的是Object类型,这样能够更灵活的处理数据:

 

public interface DisposeDataListener {

    //请求成功回调事件处理
    public void onSuccess(Object responseObj);

    //请求失败回调事件处理
    public void onFailure(Object responseObj);

}

接着新建一个类,将我们的事件回调和用于处理Json转换实体对象的字节码对象做一个封装,给出单参和双参的构造方法:

 

 

public class DisposeDataHandle {

    public DisposeDataListener mListener = null;
    public Class<?> mClass = null;

    public DisposeDataHandle(DisposeDataListener listener) {
        this.mListener = listener;
    }

    public DisposeDataHandle(DisposeDataListener listener, Class<?> clazz) {
        this.mListener = listener;
        this.mClass = clazz;
    }

}

接着我们再自定义一个异常类,返回错误码和错误信息到业务层:

 

public class OkHttpException extends Exception {
    private static final long serialVersionUID = 1L;

    private int ecode; //错误码
    private Object emsg; //错误消息

    public OkHttpException(int ecode, Object emsg) {
        this.ecode = ecode;
        this.emsg = emsg;
    }

    public int getEcode() {
        return ecode;
    }

    public Object getEmsg() {
        return emsg;
    }

}

最后我们创建一个类用于专门处理Json数据的响应,命名为CommonJsonCallback实现OKHTTP3中的Callback接口,重写onFailure和onResponse这两个失败和成功的回调函数,在这个类中我们定义了一些基本的常量,首先定义了与服务器字段的对应关系的常量,其次自定义了一些异常类型,这些字段需要和自己公司的后台开发人员进行商定,这里我只是简单的举个栗子,具体如下代码所示:

 

 

 

//与服务器的字段的一个对应关系
protected final String RESULT_CODE = "ecode"; //有返回则对于http请求来说是成功的,但还有可能是业务逻辑上的错误
protected final int RESULT_CODE_VALUE = 0;
protected final String ERROR_MSG = "emsg";
protected final String EMPTY_MSG = "";

//自定义异常类型
protected final int NETWORK_ERROR = -1; //the network relative error
protected final int JSON_ERROR = -2; //the JSON relative error
protected final int OTHER_ERROR = -3; //the unknow error

接着我们定义了一个Handler对象用于进行消息的转发,自定义回调监听用于对响应数据的回调处理,字节码文件用于对实体对象的转化,如下所示:

 

 

private Handler mDeliveryHandler; //进行消息的转发
private DisposeDataListener mListener;
private Class<?> mClass;

public CommonJsonCallback(DisposeDataHandle handle) {
    this.mListener = handle.mListener;
    this.mClass = handle.mClass;
    this.mDeliveryHandler = new Handler(Looper.getMainLooper());
}

接着是处理请求失败的回调,我们在这里用handler将异常信息通过自定义的回调监听发送到应用层中去处理,如下所示:

 

 

//请求失败的处理
@Override
public void onFailure(@NonNull Call call, @NonNull final IOException e) {
    mDeliveryHandler.post(new Runnable() {
        @Override
        public void run() {
            mListener.onFailure(new OkHttpException(NETWORK_ERROR, e));
        }
    });
}

最后来处理请求成功的回调,这也是最重要的一步,同样的我们首先是将响应数据发送到主线程中,然后判断字节码mClass对象是否为null,如果为空则表示我们不需要转换实体对象,直接将原始的Json数据回调到应用层,你可以直接操作json数据,如果mClass对象不为空,我这里直接通过谷歌的Gson将json数据解析成了对应的实体对象返回,这样我们在应用层可以直接操作实体对象,对于异常信息也是直接回调到应用层去处理的,具体代码如下:

 

 

//处理成功的响应
private void handleResponse(Object responseObj) {
    //为了保证代码的健壮性
    if (responseObj == null && responseObj.toString().trim().equals("")) {
        mListener.onFailure(new OkHttpException(NETWORK_ERROR, EMPTY_MSG));
        return;
    }
    try {
        JSONObject result = new JSONObject(responseObj.toString());
        if (result.has(RESULT_CODE)) {
            //从JSON对象中取出我们的响应码,如果为0,则是正确的响应
            if (result.getInt(RESULT_CODE) == RESULT_CODE_VALUE) {
                if (mClass == null) {
                    mListener.onSuccess(responseObj);
                } else { //需要转化为实体对象
                    Object obj = new Gson().fromJson((String) responseObj, mClass);
                    if (obj != null) { //表明正确的转为了实体对象
                        mListener.onSuccess(obj);
                    } else {
                        mListener.onFailure(new OkHttpException(JSON_ERROR, EMPTY_MSG));
                    }
                }
            } else { //将服务端返回的异常回调到应用层去处理
                mListener.onFailure(new OkHttpException(OTHER_ERROR, result.get(RESULT_CODE)));
            }
        }
    } catch (Exception e) {
        e.printStackTrace();
        mListener.onFailure(new OkHttpException(OTHER_ERROR, e.getMessage()));
    }
}

好了,这样我们就已经封装完成了基本的get/post请求,已经能够满足一般的开发要求了,关于框架层的封装就说这么多,接下来,我们去看看在应用层中该如何调用?

 

四、测试框架是否运行正常

1、准备服务器数据

我这里在本地环境下开了一个Tomcat,然后在它的webapps目录下新建了一个文件夹youdu,然后在里面新建了一个test.json的文件,在里面放入了一些简单的测试数据,并且我们在本地浏览器中访问这个地址,发现可以正常访问,数据如下:

2、根据对应的json数据,这里创建一个TestModel实体类,定义获取数据的各个字段,可以通过GsonFormat插件快速生成。

3、定义接口管理类,统一管理接口地址(实际项目中都要这么做):

 

public class HttpConstant {
    private static final String URL = "http://10.10.7.146:8080/";

    public static String HOME_RECOMMAND = URL + "youdu/test.json";
}

4、为了调用简单,我们在应用层中再做一层封装,定义一个RequestCenter类,在其中通过方法的重载,对外只暴露一个我们自定义的回调监听,其它几个参数都在这个类中传入即可,代码如下:

 

 

public class RequestCenter {

    //根据参数发送所有的get请求
    private static void getRequest(String url, RequestParams params,
                                   DisposeDataListener listener,
                                   Class<?> clazz){
        CommonOkHttpClient.get(CommonRequest.createGetRequest(url, params),
                new DisposeDataHandle(listener,clazz));
    }

    public static void requestRecommandData(DisposeDataListener listener){
        RequestCenter.getRequest(HttpConstant.HOME_RECOMMAND,null,listener, TestModel.class);
    }

}

5、这样我们在调用的时候就十分方便了,并且在回调方法中可以直接处理我们实体数据,具体的调用方法如下:

 

 

//测试网络框架
RequestCenter.requestRecommandData(new DisposeDataListener() {
    @Override
    public void onSuccess(Object responseObj) {
        TestModel model = (TestModel) responseObj;
        //设置名称字段显示到TextView上
        textView.setText(model.getData().getList().get(0).getUname());
    }

    @Override
    public void onFailure(Object responseObj) {
        Log.e("测试网络框架", "onFailure: " + responseObj.toString() );
    }
});

可以看到,我们的调用现在已经变的十分简单了,就这么几行代码就完成了数据请求解析以及展示的过程,来看一下程序运行的结果:

 

双手奉上项目源码地址:https://github.com/JArchie/NetTest2

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

基于OkHttp3封装网络请求框架 的相关文章

  • Supervisor的安装与使用

    简介 Supervisor 是用 Python 开发的一套通用的 进程管理程序 xff0c 能将一个普通的命令行进程变为后台daemon xff0c 并监控进程状态 xff0c 异常退出时能自动重启 它是通过fork exec的方式把这些被
  • devenv使用方法

    CD C CD C Program Files Microsoft Visual Studio NET 2003 Common7 IDE DEL D KTAPP KTUI1601 licx devenv build debug 34 D K
  • usb连接ubuntu,fdisk -l 查不出usb信息

    问题 xff1a 想要通过usb连接在虚拟机上的Ubuntu xff0c 但连接上后linux系统有图标但是却是灰色的 xff0c 在虚拟机 可移动设备 xff0c 那里的USB 断开与主机连接也是灰色 通过fdisk l 查不到usb信息
  • 树莓派——开启xrdp,换镜像,双击打开sh等等__一蓑烟雨任平生

    文章目录 前言一 树莓派是什么 xff1f 二 烧卡开机1 镜像用树莓派官网的镜像2 开机基本设置3 设置SSH和VNC4 查看本地IP5 设定固定IP 二 深度设置1 更换镜像源2 安装远程桌面3 修改sh脚本 Linux基本命令使用总结
  • 关于QImage无法设置透明色的问题

    明明在setPixelColor时 xff0c 设置的QColor为 xff08 254 254 254 0 xff09 xff0c 但是最后的结果却是白色 这是由于在设置QImage格式时 xff0c 该格式不支持透明色 xff0c 可以
  • 程序员必备的 40 个 VSCode 扩展

    在编写代码时 xff0c 一个富有成效的工作空间不仅仅是要找到合适的代码编辑器 开箱即用的VS Code是为开发人员制作的 根据2021年StackOverflow的调查 xff0c 71 06 的受访者将Visual Studio Cod
  • django debug toolbar 不显示

    python3 6 43 django2 0 xff0c 安装了django debug toolbar xff0c 根据官方文档配置之后却没看到调试栏 xff0c 增加了SHOW TOOLBAR CALLBACK字段后才显示 settin
  • 驳 GarbageMan 的《一个超复杂的简介递归》——对延迟计算的实验和思考

    这是一篇因骂战而起的博文 xff0c GarbageMan 在该文章回复中不仅对我进行了侮辱 xff0c 还涉及了我的母校 xff0c 特写此文用理性的分析和实验予以回击 在此也劝告 GarbageMan xff0c 没什么本事就别在那叫嚣
  • Debian_DNS服务搭建

    DNS是什么 xff1f DNS xff0c Domain Name System或者Domain Name Service xff08 域名系统或者余名服务 xff09 域名系统为Internet上的主机分配域名地址和IP地址 用户使用域
  • 解决MySQL无法插入中文数据问题(UTF-8编码)

    我花了好几个小时找过各种方法 xff0c 最终靠这个方法实现了中文插入 xff0c 我都快要喜极而泣了 xff0c 分享给大家 xff0c 真的很实用 一些关于查看和修改字符集的MySQL知识 xff1a 查看mysql的字符集 xff1a
  • PHP判断IP是否属于某个网段

    IP通过ip2long转换后可以进行比较 span class php span class hljs preprocessor lt php span span class hljs comment 判断IP是否在某个网络内 span c
  • phpstore 2016.1版激活方法

    激活请填写下面的地址吧 xff1a span class hljs label http span idea span class hljs preprocessor qinxi span 1992 span class hljs prep
  • Group by无法排序,但可以通过子查询实现

    lt pre name 61 34 code 34 class 61 34 sql 34 gt select from table where id in select max id from table where tid in 0 10
  • ext4文件系统恢复被删除的文件

    ext4magic工具 可以恢复出被rm f删除的文件 xff08 只要原始数据块未被新数据所覆盖 xff09 ext4magic device M d savedir 可参考 https sourceforge net projects
  • nginx反向代理经验整理

    location flag proxy pass http 127 0 0 1 19999 上面这段配置的反向代理实际访问路径是 usr www location flag proxy pass http 127 0 0 1 19999 上
  • Git一些使用注意事项

    Git创建远程空仓库时要注意加上 shared 61 group 即命令 xff1a 初始化远程空仓库 xff1a git init bare shared 61 group 给空仓库设置组共享 xff1a git config core
  • 使用WSL在Windows上安装Ubuntu

    1 清理环境 查看当前的wsl 状态 xff0c wsl list 可以列出当前系统中已安装的子系统 选择需要清理的系统 xff0c 然后用 wsl unregister lt DistributionName gt 即可完成卸载 将 ws
  • CentOS/Ubuntu/Debian常用版本更换国内源的方法

    Linux系统安装完后软件源一般都是国外服务器 xff0c 在国内特别慢 xff0c 这时候就需要更换国内的镜像源 如163 aliyun 还有各高校镜像源等 记住先备份原始源在更换源 xff0c 以防以后备用 一 centos 1 备份
  • C++ 的引用类型

    C 43 43 的引用类型 在翻旧文的时候 xff0c 发现这么一篇文章 xff1a 关于一道C 43 43 笔试题的纠结 xff0c 学计算机的伤不起啊 当时可能是觉得 Placement new 的语法1比较新鲜 xff0c 所以印象比
  • Debian 10(buster) 更换国内软件源

    今天安装了个debian10 xff0c 发现网上包括各大镜像网站提供的源地址都有点问题 xff0c 经测试 xff0c Debian 10 xff08 buster xff09 可用的国内软件源如下 xff08 阿里云源 xff09 xf

随机推荐

  • 华为交换机SNMP配置

    华为交换机SNMP配置 snmp服务配置 交换机内设置snmp一般只需要启动snmp服务和配置团体名称 xff0c 然后设置下版本就可以了 全局模式下 xff0c 配置命令 1 启动snmp服务 xff1a snmp agent 2 设置团
  • qt 菜单栏创建

    h文件内容 xff1a pragma once include lt QtWidgets QMainWindow gt include 34 ui QtWidgetsApplication2 h 34 include lt QLabel g
  • 立志学习编程的第一天 2021-05-06

    一直很遗憾大学没能报计算机专业 xff08 还一直觉得自己是被耽误的程序员来着 xff09 现在且用业余时间来试试 xff0c 且拭目以待吧 xff01 学习内容 xff1a Clojure for the Brave and True 工
  • 关于Clojure的Emacs配置 2021-05-07

    首先安装lein和Emacs xff1a https leiningen org http www gnu org software emacs 安装好Emacs之后 xff0c 找到 emacs d 对windows系统 xff0c 文件
  • Clojure基础语法学习笔记(一)

    首先推荐两个目前正在学的免费学习资源 xff1a Functional programming in Clojure Clojure for the Brave and True 都是英文的 xff0c 第一个是边学边练的形式 xff0c
  • matlab interp2函数详解

    切入正题 xff1a 下面是一个测试INTERP2 xff08 xff09 函数的MATLAB代码 function INTERP2 TEST I 61 2 3 6 8 3 5 1 7 4 2 9 6 6 8 1 3 X Y 61 mesh
  • word2016转mathtype

    用word2016自带公式编辑器给师兄敲了一堆公式之后 xff0c 发现毕业论文要求用mathtype 官网给的解决措施是 xff1a 1 http www mathtype cn wenti wordgongshi zhuanhuan h
  • 虚拟机中Linux扩容硬盘空间

    在初始安装CentOS时 xff0c 只给了硬盘空间30GB xff0c 现在因为需要 xff0c 所以需要扩容 1 关闭虚拟机中的系统 xff0c 打开虚拟机的设置 xff0c 修改磁盘空间到合适的大小 xff0c 再重启系统 2 打开终
  • Vue3 - setup语法糖

    与setup函数不同的是 xff0c 在script标签中添加setup 1 变量 方法不需要 return 出来 属性和方法也不用返回 xff0c 也不用写setup函数 xff0c 也不用写export default xff0c 甚至
  • Autoware 安装(源码)过程 与 踩坑记录(Ubuntu18.04)

    目录 autoware 源码安装 安装 ROS Melodic xff1a 设置软件源 设置密钥 xff1a 安装ROS xff1a rosdep xff1a 安装rosinstall 添加ROS环境变量 配置ROS环境变量 创建工作目录
  • 读论文:Feedback Network for Image Super-Resolution

    源码 xff1a https github com Paper99 SRFBN CVPR19 1 介绍 xff08 1 xff09 基于深度学习的方法的优势主要来自其两个关键因素 xff1a 深度和跳跃链接 第一 xff0c 保留更多的上下
  • Kolla-ansible部署OpenStack Train实践

    Kolla ansible部署OpenStack Train实践 前言系统环境设置安装pip和docker安装ansible安装kolla ansible配置文件修改执行部署 登录openstack写在最后部署过程中遇到的问题总结 前言 最
  • Windows 7 如何升级 PowerShell

    操作环境 xff1a Windows 7 旗舰版 Service Pack 1 x64 PowerShell 2 0 gt PowerShell 4 0 解决过程 xff1a 1 下载Windows6 1 KB2819745 x64 Mul
  • 最简单的算法:线性查找法

    目录 写在前面 一 什么是算法 二 线性查找法 2 1 实现线性查找法 2 2 思维拓展 使用泛型 2 3 自定义类测试泛型方法 2 4 循环不变量 三 复杂度分析 3 1 复杂度分析简介 3 2 常见的算法复杂度 四 算法性能测试 写在前
  • Android仿抖音主页效果实现

    目录 写在前面 一 准备工作 1 1 主页面布局 1 2 列表Item布局 1 3 列表Item适配器 二 自定义LayoutManager 三 实现播放 补充 xff1a 源码地址 xff1a https github com JArch
  • 数据结构基础之动态数组

    目录 前言 1 Java中的数组 2 实现动态数组 2 1 基本类结构设计 2 2 添加元素 2 3 查询 amp 修改元素 2 4 包含 amp 搜索 amp 删除 2 5 数组扩容 前言 今天我们来学习一下关于数据结构的一些基础知识 x
  • 数据结构基础之栈和队列

    目录 前言 1 栈 2 队列 2 1 实现队列 2 2 循环队列 前言 上一篇中我们介绍了数据结构基础中的 动态数组 xff0c 本篇我们继续来学习两种基本的数据结构 栈和队列 1 栈 特点 xff1a 栈也是一种线性结构 xff0c 相比
  • 数据结构基础之链表

    目录 前言 1 什么是链表 2 添加元素 3 虚拟头结点 4 查询 amp 修改元素 5 删除元素 附 xff1a 完整代码 前言 又到周末了 xff0c 修整了一天 xff0c 继续来写点东西吧 xff0c 今天 xff0c 我们来学习数
  • Android开发之局部广播的使用——LocalBroadcast

    一直以来都想着写一篇关于四大组件的文章 xff0c 可是一直懒一直都没去写 xff0c 今天终于抱起了电脑来敲一篇 这篇文章是关于安卓四大组件之一的广播的使用 xff0c 网上关于这方面的文章也是相当多 xff0c 我这里根据我一年多的工作
  • 基于OkHttp3封装网络请求框架

    前言 网络请求可以说是开发一款移动APP最核心的基础功能了 xff0c 通过实际工作中以及浏览了许多网络框架之后 xff0c 本篇在这里分享慕课一位老师基于OkHttp封装的一个轻量的网络框架 xff0c 至于为什么说它轻量 xff0c 因