美文网首页
2018-02-03 OKhttp设置UserAgent的那些事

2018-02-03 OKhttp设置UserAgent的那些事

作者: 心灵屋宿客 | 来源:发表于2018-02-03 15:52 被阅读0次

    首先科普一下,UserAgent中文名为用户代理,简称 UA,它是一个*特殊字符串头,使得服务器能够识别客户使用的操作系统及版本、CPU 类型、浏览器及版本、浏览器渲染引擎、浏览器语言、浏览器插件等。

    Okhttp走的并不是原生的http请求,因此它在header里面并没有真正的User-Agent,而是“okhttp/版本号”这样的字符串,因为后台需要统计信息,要求传入自定义的User-Agent。

    首先先看一下怎样正确获取User-Agent。

    0x00-正确获取User-Agent

    private static String getUserAgent() {
            String userAgent = "";
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) {
                try {
                    userAgent = WebSettings.getDefaultUserAgent(context);
                } catch (Exception e) {
                    userAgent = System.getProperty("http.agent");
                }
            } else {
                userAgent = System.getProperty("http.agent");
            }
            StringBuffer sb = new StringBuffer();
            for (int i = 0, length = userAgent.length(); i < length; i++) {
                char c = userAgent.charAt(i);
                if (c <= '\u001f' || c >= '\u007f') {
                    sb.append(String.format("\\u%04x", (int) c));
                } else {
                    sb.append(c);
                }
            }
            return sb.toString();
        }
    

    代码说明:
    1、在Api17之后可以通过WebSettings.getDefaultUserAgent(context)获取,但是经过测试极个别手机会出现找不到类的情况,因此try-catch一下,那么第二种方式是System.getProperty("http.agent"),这两种方式有什么不同呢?从结果上来看是第一种得到的信息更全一点;
    第一种:

    Mozilla/5.0 (Linux; Android 6.0.1; MI 4LTE Build/MMB29M; wv) AppleWebKit/537.36 (KHTML, like Gecko) 
    Version/4.0 Chrome/51.0.2704.81 Mobile Safari/537.36
    

    第二种:

    Dalvik/2.1.0 (Linux; U; Android 6.0.1; MI 4LTE MIUI/6.10.13)
    

    2、在一些国产手机上面这个User-Agent里面会包含中文,就会报错

    java.lang.IllegalArgumentException: Unexpected char 0x7231 at 33 in User-Agent value: Mozilla/5.0 (Linux; Android 5.1; 爱国者-X86 Build/KOT49H) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/30.0.0.0 Safari/537.36
    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2367)
    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2419)
    at android.app.ActivityThread.access$800(ActivityThread.java:151)
    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1342)
    at android.os.Handler.dispatchMessage(Handler.java:110)
    at android.os.Looper.loop(Looper.java:193)
    at android.app.ActivityThread.main(ActivityThread.java:5323)
    at java.lang.reflect.Method.invokeNative(Native Method)
    at java.lang.reflect.Method.invoke(Method.java:515)
    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:828)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:644)
    at dalvik.system.NativeStart.main(Native Method)
    Caused by: java.lang.IllegalArgumentException: Unexpected char 0x7231 at 33 in User-Agent value: Mozilla/5.0 
    (Linux; Android 5.1; 爱国者-X86 Build/KOT49H) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/30.0.0.0 Safari/537.36
    at okhttp3.Headers$Builder.checkNameAndValue(Headers.java:320)
    at okhttp3.Headers$Builder.add(Headers.java:270)
    at okhttp3.Request$Builder.addHeader(Request.java:175)
    

    什么原因引起的呢?okhttp3.Headers$Builder.checkNameAndValue进到这个方法里面可以看到okhttp对中文进行了过滤,如果不符合条件就抛出异常IllegalArgumentException

    private void checkNameAndValue(String name, String value) {
          if (name == null) throw new NullPointerException("name == null");
          if (name.isEmpty()) throw new IllegalArgumentException("name is empty");
          for (int i = 0, length = name.length(); i < length; i++) {
            char c = name.charAt(i);
            if (c <= '\u001f' || c >= '\u007f') {
              throw new IllegalArgumentException(Util.format(
                  "Unexpected char %#04x at %d in header name: %s", (int) c, i, name));
            }
          }
          if (value == null) throw new NullPointerException("value == null");
          for (int i = 0, length = value.length(); i < length; i++) {
            char c = value.charAt(i);
            if (c <= '\u001f' || c >= '\u007f') {
              throw new IllegalArgumentException(Util.format(
                  "Unexpected char %#04x at %d in %s value: %s", (int) c, i, name, value));
            }
          }
        }
    

    因此上段代码才会对返回结果进行过滤,如果不符合条件,会进行转码。

    0x01-给Okhttp设置User-Agent

    Request request = new Request.Builder().url(url).removeHeader("User-Agent").addHeader("User-Agent",
                    getUserAgent()).build();
    httpClient.newCall(request).enqueue(handler);
    

    然后看下需求,后台要求传入设备类型/APP版本号/设备型号/系统版本 这样的字段。那么就可以开始了。
    先分享一个工具类。

    /**
     * 系统工具类
     */
    public class SystemUtil {
    
        /**
         * 获取当前手机系统语言。
         *
         * @return 返回当前系统语言。例如:当前设置的是“中文-中国”,则返回“zh-CN”
         */
        public static String getSystemLanguage() {
            return Locale.getDefault().getLanguage();
        }
    
        /**
         * 获取当前系统上的语言列表(Locale列表)
         *
         * @return  语言列表
         */
        public static Locale[] getSystemLanguageList() {
            return Locale.getAvailableLocales();
        }
    
        /**
         * 获取当前手机系统版本号
         *
         * @return  系统版本号
         */
        public static String getSystemVersion() {
            return android.os.Build.VERSION.RELEASE;
        }
    
        /**
         * 获取手机型号
         *
         * @return  手机型号
         */
        public static String getSystemModel() {
            return android.os.Build.MODEL;
        }
    
        /**
         * 获取手机厂商
         *
         * @return  手机厂商
         */
        public static String getDeviceBrand() {
            return android.os.Build.BRAND;
        }
    
        /**
         * 获取APP版本
         * @param context
         * @return
         */
        public static String getVersionName(Context context) {
            PackageManager packageManager = context.getPackageManager();
            PackageInfo packageInfo;
            String versionName = "";
            try {
                packageInfo = packageManager.getPackageInfo(context.getPackageName(), 0);
                versionName = packageInfo.versionName;
            } catch (PackageManager.NameNotFoundException e) {
                e.printStackTrace();
            }
            return versionName;
        }
    
    }
    

    然后,有了工具类就可以设置UA了,

        private static String getUserAgent(Context context) {
            String userAgent = "";
    //        APP版本
            String versionName = SystemUtil.getVersionName(context);
    //        手机型号
            String systemModel = SystemUtil.getSystemModel();
    //        系统版本
            String systemVersion = SystemUtil.getSystemVersion();
            String deviceBrand = SystemUtil.getDeviceBrand();
            userAgent = "Android/" + versionName + "/" + deviceBrand + "" + systemModel + "/" + systemVersion;
            return userAgent;
        }
    

    然后给Okhttp设置给Okhttp设置User-Agent

    Request request = new Request.Builder().url(url).removeHeader("User-Agent").addHeader("User-Agent",
                    getUserAgent()).build();
    httpClient.newCall(request).enqueue(handler);
    

    至此,已经完成。

    部分内容来自简书。Okhttp设置User-Agent你可能没遇到的坑

    相关文章

      网友评论

          本文标题:2018-02-03 OKhttp设置UserAgent的那些事

          本文链接:https://www.haomeiwen.com/subject/bdiezxtx.html