Neil's blog

Let's start from here


  • 首页

  • 归档

  • 标签

  • 关于

Bugsnag:监控应用程序错误,以改善客户体验和代码质量

发表于 2017-07-14 |

Bugsnag exception reporter for Android

“getstart”

Bugsnag的Android崩溃报告库会自动检测Android应用程序中的崩溃,收集诊断信息,并立即通知您的开发团队,帮助您尽快了解和解决问题。

主要特性

  • 自动报告未处理的异常和崩溃
  • 报告处理异常
  • 记录附加到崩溃报告的日志和添加对用户操作的洞察
  • 附上用户信息来确定多少人受到崩溃的影响

使用

  1. 创建Bugsnag账号
  2. 完成集成指南的说明来报告从你的应用程序中抛出的未处理的异常
  3. 使用 Bugsnag.notify 来报告已处理的异常
  4. 使用 configuration options 来自定义集成

支持

  • 阅读集成指南或者配置文件选项
  • 查询已开启或已关闭的类似问题
  • 报告错误或请求功能

Github : https://github.com/bugsnag/bugsnag-android#features

Bugsnag官网:https://app.bugsnag.com/accounts/sf-1/welcome

阿里百川Hotfix 1.4.0 Android接入

发表于 2017-07-12 |

一.Android Studio签名打包

自行百度。

二.基本配置依赖

maven仓库地址:

repositories {
   maven {
       url "http://repo.baichuan-android.taobao.com/content/groups/BaichuanRepositories"
   }
}

gradle坐标版本依赖:

dependencies {
    compile 'com.taobao.android:alisdk-hotfix:1.4.0'
}

三.配置权限

HotFix SDK使用到以下权限

<! -- 网络权限 -->
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<! -- 外部存储读权限 -->
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>

四.配置AndroidManifest文件

在AndroidManifest.xml中间的application节点下添加如下配置:

<meta-data
android:name="com.taobao.android.hotfix.APPSECRET"
android:value="your-app-secret" />
<meta-data
android:name="com.taobao.android.hotfix.RSASECRET"
android:value="your-rsa-secret" />

app-secret为阿里百川开发者申请百川平台HotFix服务申请得到的App Secret和RSA密钥

五.接入SDK

接入范例

initialize的调用应该尽可能的早. 强烈推荐在Application.onCreate()中进行SDk初始化以及查询服务器是否有可用补丁的操作.

HotFixManager.getInstance().setContext(this)
                .setAppVersion(appVersion)
                .setAppId(appId)
                .setAesKey(null)
                .setSupportHotpatch(true)
                .setEnableDebug(true)
                .setPatchLoadStatusStub(new PatchLoadStatusListener() {
                    @Override
                    public void onload(final int mode, final int code, final String info, final int handlePatchVersion) {
                        // 补丁加载回调通知
                        if (code == PatchStatusCode.CODE_SUCCESS_LOAD) {
                            // TODO: 10/24/16 表明补丁加载成功
                        } else if (code == PatchStatusCode.CODE_ERROR_NEEDRESTART) {
                            // TODO: 10/24/16 表明新补丁生效需要重启. 业务方可自行实现逻辑, 提示用户或者强制重启, 建议: 用户可以监听进入后台事件, 然后应用自杀
                        } else if (code == PatchStatusCode.CODE_ERROR_INNERENGINEFAIL) {
                            // 内部引擎加载异常, 推荐此时清空本地补丁, 但是不清空本地版本号, 防止失败补丁重复加载
                            //HotFixManager.getInstance().cleanPatches(false);
                        } else {
                            // TODO: 10/25/16 其它错误信息, 查看PatchStatusCode类说明
                        }
                    }
                }).initialize();

initialize()方法内部会强制调用queryNewHotPatch()方法, 所以此处不需要额外再调用queryNewHotPatch()方法

说明: initialize方法

该方法主要做些必要的初始化工作以及如果本地有补丁的话会加载补丁, 所以需要尽可能的早, 推荐在Application的onCreate方法中调用, 由于initialize方法参数越来越多变的原来越臃肿, 所以1.4.0版本修改了调用方式, initialize()方法调用之前你需要先调用如下几个方法, 方法调用说明如下:

setContext(this): Application上下文context 必选

setAppVersion(appVersion): 应用的版本号 必选

setAppId(appId): 百川上应用的唯一标识, 如何获取请查询获取SDK配置信息 必选

setAesKey(必须16位): 用户自定义aes秘钥, 此时平台无感知这个秘钥, 所以不用担心百川平台会利用你们的补丁做一些非法的事情. 这个参数值必须配合补丁工具的-y参数一起使用, 具体使用参见?Part2 生成patch补丁?的说明, 两者的值需要保持一致, 补丁才能正确被解密进而加载. 可选

setSupportHotpatch(true/false): 目前的版本热修复方案采用类似andfix本地hook方法方法所以热部署有一定的风险(方法正在被运行然后被patch了可能会导致native层的crash). 用户如果有实时生效的需求以及被patch的方法没有被高频调用那么这个参数可以设置为true. 第一个补丁将会即时生效 可选

setEnableDebug(true/false): 是否调试模式, 调试模式下会输出日志以及不进行补丁签名校验. 线下调试此参数可以设置为true, 查看日志过滤TAG:BCHotfix, 同时强制不对补丁进行签名校验, 所有就算补丁未签名或者签名失败也发现可以加载成功. 但是正式发布该参数必须设置为false, 需要对补丁签名校验, 否则就可能存在安全漏洞风险 可选

setPatchLoadStatusStub(new PatchLoadStatusListener()): 设置patch加载状态监听器, 该方法参数需要实现PatchLoadStatusListener接口, 接口说明见1.3.2.2说明 可选 
  • initialize(): sdk初始化方法 必选

  • PatchLoadStatusListener接口

    • 该接口需要自行实现并传入initialize方法中, 补丁加载状态会回调给该接口, 参数说明如下:

mode: 补丁模式, 0:正常请求模式 1:扫码模式 2:本地补丁模式
code: 补丁加载状态码, 详情查看PatchStatusCode类说明
info: 补丁加载详细说明, 详情查看PatchStatusCode类说明
handlePatchVersion: 当前处理的补丁版本号, 0:无 -1:本地补丁 其它:后台补丁

这里列举几个常见的code码说明, 详情查看SDK中PatchStatusCode类说明

  • code: 1 补丁加载成功

  • code: 6 服务端没有最新可用的补丁

  • code: 11 RSASECRET错误,官网中的密钥是否正确请检查
  • code: 12 当前应用已经存在一个旧补丁, 应用重启尝试加载新补丁
  • code: 13 补丁加载失败, 导致的原因很多种, 比如UnsatisfiedLinkError等异常, 此时应该严格检查logcat异常日志
  • code: 16 APPSECRET错误,官网中的密钥是否正确请检查
  • code: 18 一键清除补丁
  • code: 403 签名不匹配,可能是APPID APPSECRET填错,请检测

queryNewHotPatch方法

该方法主要用于查询服务器是否有新的可用补丁.

  • 首先initialize()方法内部会强制调用queryNewHotPatch()方法, 所以initialize()方法调用之后不需要再调用这个方法, 但是你可以在其它你需要的地方调用.

  • 同时SDK内部限制连续两次queryNewHotPatch()方法调用不能短于3s, 否则的话就会报code:19的错误码. 如果查询到可用的话, 首先下载补丁到本地, 然后应用原本没有补丁, 那么第一个补丁会立刻加载
    应用已经存在一个补丁, 首先会把之前的补丁文件删除, 然后不立刻加载, 而是等待下次应用重启再加载该补丁
    补丁在后台发布之后, 并不会主动下行推送到客户端, 需要手动调用queryNewHotPatch方法查询后台补丁是否可用.

  • 只会下载补丁版本号比当前应用存在的补丁版本号高的补丁, 比如当前应用已经下载了版本号为5的补丁, 那么只有后台发布的补丁版本号>5才会重新下载.

  • 同时1.4.0版本服务后台上线了“一键清除”补丁的功能, 所以如果后台点击了“一键清除”那么这个方法将会返回code:18的状态码. 此时本地补丁将会被强制清除, 同时不清除本地补丁版本号

cleanPatches(boolean force)方法

  • 参数force表示是否强制清空本地补丁版本号, 比如当前本地补丁版本号是10, 那么下次再次调用queryNewHotPatch方法时, 如果该参数为false: 不清除本地补丁版本号那么后台最新的补丁1就不会重新下载

  • 当然如果存在比10大的补丁版本仍然是可以下载下来的. 如果该参数为true: 清除本地补丁版本号, 本地补丁版本号将会被设置为0, 所以后台只要有任何发布的补丁都能够下载下来.

Set集合使用注意tips

发表于 2017-02-22 |

Set中的某些元素,当时使用边遍历,边删除的方法,报了以下异常:

ConcurrentModificationException

example:
    Set<CheckWork> set = this.getUserDao().getAll(qf).get(0).getActionCheckWorks();
    for(CheckWork checkWork : set){
        if(checkWork.getState()==1){
            set.remove(checkWork);
        }
    }

解决方案:

  1. 遍历删除List

    List<CheckWork> list = this.getUserDao().getAll();
    Iterator<CheckWork> chk_it = list.iterator();
    while(chk_it.hasNext()){
        CheckWork checkWork = chk_it.next();
        if(checkWork.getPlanState()==1){
            chk_it.remove();
        }
    }
    
  2. 遍历删除Set

    Set<CheckWork> set =  this.getUserDao().getAll().get(0).getActionCheckWorks();
            Iterator<CheckWork> it = set.iterator();
            while(it.hasNext()){
                CheckWork checkWork = it.next();
                if(checkWork.getState()==1){
                    it.remove();
                }
            }
    
  3. 遍历删除Set

    Set<CheckWork> set = this.getUserDao().getAll(qf).get(0).getActionCheckWorks();
    Set<CheckWork> delSet=new HashSet<>();
    for(CheckWork checkWork : set){
        if(checkWork.getState()==1){
            delSet.add(checkWork);
        }
    }
    set.removeAll(delSet);    
    

HTTP需要知道的知识点-补充

发表于 2016-11-19 |
  • 一、HTTP协议

    • 1.1协议的作用及版本

      • 1.1.1HTTP是Hyper Text Transfer Protocol(超文本–html传输协议)。
      • 1.1.2HTTP协议的作用用户描述客户端与服务器间的数据传递的。
      • 1.1.3:Http协议的版本:
        • 1.0:特点:每次请求服务器上的资源都要建立新的连接,响应完毕后都会关闭连接。是无状态的协议。
        • 1.1:特点:在一次TCP/IP连接的基础上可以发出多次请求和得到多次的响应。比1.0多了一些请求和响应头。
    • 1.2协议的组成

      • 1.2.1请求部分:

        GET /App1/1.html HTTP/1.1
        Accept: */*
        Accept-Language: zh-cn
        Accept-Encoding: gzip, deflate
        User-Agent: Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1; SV1; .NET CLR 2.0.50727; .NET CLR 3.0.4506.2152; .NET CLR 3.5.30729)
        Host: localhost:8080
        Connection: Keep-Alive
        
        username=admin&password=123
        

        请求部分由三部分组成的:
        请求行:位于第一行
        请求消息头:从第二行开始至第一个空行结束
        请求正文:从第一个空行之后的都是正文

      • 1.2.2响应部分:

        HTTP/1.1 200 OK
        Server: Apache-Coyote/1.1
        Accept-Ranges: bytes
        ETag: W/"65-1366335797484"
        Last-Modified: Fri, 19 Apr 2013 01:43:17 GMT
        Content-Type: text/html
        Content-Length: 65
        Date: Fri, 19 Apr 2013 02:06:23 GMT
        Warnning:w1
        Warnning: w2
        
        响应部分由三部分组成的:

        响应行:位于第一行

        响应消息头:从第二行开始至第一个空行结束

        响应正文:从第一个空行之后的都是正文

* 1.3协议详细讲解

    请求行:
    GET /App1/1.html HTTP/1.1
    GET:请求方式。默认的请求方式。其他常用的请求方式还有POST。
    GET的特点:默认的请求方式。

        /App1/c.html?username=sdsfds&password=234324
把表单的请求的数据放在了请求的URI的后面。?username=sdsfds&password=234324
这样不好:暴露数据;请求行长度有限。

    POST的特点(经常使用的):借助HTML中的form表单。<form action="c.html" method="post">
请求参数出现在正文部分。长度木有限制。相对安全。

二、HTTP/1.1:客户端使用的协议的版本

  • 响应行:HTTP/1.1 200 OK

    HTTP/1.1:服务器用的协议版本

  • 响应码:代表服务器处理的结果的一种表示

    200:正常
    302/307:重定向
    304:服务器的资源没有被修改
    404:请求的资源不存在
    500:服务器报错了
    
  • OK:响应码描述

  • 请求消息头:向服务器传递附加信息

    • Accept:通知服务器,浏览器可以接受的MIME类型。(文件系统中用文件扩展名区分数据的类型。网络上用MIME类型来区分数据类型。Tomcat\conf\web.mxl)
    • MIME类型名称:大类型/小类型
    • Accept-Charset:通知服务器,浏览器支持的字符集
    • Accept-Encoding:通知服务器,浏览器能够解码的数据压缩方式。比如:gzip
    • Accept-language:通知服务器,所希望的语言
    • Host:请求的主机和端口
    • Referer:是一个URL地址。取值是当前页面之前的那个页面地址的。防盗链
    • Content-Type:通知服务器,请求正文的MIME类型。
    • 取值:application/x-www-form-urlencoded默认值
      对应的是form表单的enctype属性
    • If-Modified-Since:通知服务器,缓存的文件的最后修改时间。
    • User-Agent:通知服务器,浏览器类型.
    • Content-Length:表示请求消息正文的长度
    • Connection:表示是否需要持久连接。如果服务器看到这里的值为“Keep -Alive”,或者看到请求使用的是HTTP 1.1(HTTP 1.1默认进行持久连接
    • Cookie:这是最重要的请求头信息之一(会话有关)
      响应消息头:
    • Location:通知客户端,指示新的资源的位置(结合302/307来用。请求重定向)
      Server:通知客户端,服务器的类型
    • Content-Encoding:通知客户端,响应正文的压缩编码方式。常用的是gzip。
    • Content-Length:通知客户端响应正文的数据大小
    • Content-Type:通知客户端响应正文的MIME类型
    • Refresh:让浏览器自动刷新。取值为整数(刷新的时间间隔,单位是秒)
    • Refresh:3
    • Refresh:3;URL=其他资源的URI
    • Content-Disposition:通知客户端,以下载的方式打开资源。
    • Content-Disposition:attachment;filename=1.jpg
    • Set-Cookie:SS=Q0=5Lb_nQ; path=/search服务器端发送的Cookie(会话有关)

    • Expires: -1 网页的有效时间。单位是毫秒

    • Cache-Control: no-cache (1.1)
    • Pragma: no-cache (1.0) 通知客户端不要缓存

关于HTTP需要理解的知识点

发表于 2016-11-13 |

关于HTTP需要知道的知识点

HTTP简介

HTTP协议是Hyper Text Transfer Protocol(超文本传输协议)的缩写,是用于从万维网(WWW:World Wide Web )服务器传输超文本到本地浏览器的传送协议。

HTTP是一个基于TCP/IP通信协议来传递数据(HTML 文件, 图片文件, 查询结果等)。

HTTP消息结构

客户端请求消息

客户端发送一个HTTP请求到服务器的请求消息包括以下格式:请求行(request line)、请求头部(header)、空行和请求数据四个部分组成,下图给出了请求报文的一般格式。

“请求报文”

服务器相应消息

HTTP响应也由四个部分组成,分别是:状态行、消息报头、空行和响应正文。

“服务器响应”

实例

使用GET来传递数据的实例:

客户端请求:

GET /hello.txt HTTP/1.1
User-Agent: curl/7.16.3 libcurl/7.16.3 OpenSSL/0.9.7l zlib/1.2.3
Host: www.example.com
Accept-Language: en, mi

服务端响应:

HTTP/1.1 200 OK
Date: Mon, 27 Jul 2009 12:28:53 GMT
Server: Apache
Last-Modified: Wed, 22 Jul 2009 19:15:56 GMT
ETag: "34aa387-d-1568eb00"
Accept-Ranges: bytes
Content-Length: 51
Vary: Accept-Encoding
Content-Type: text/plain

输出结果:

Hello World! My payload includes a trailing CRLF.

HTTP请求方法

  • HTTP1.0定义了三种请求方法: GET, POST 和 HEAD方法。

  • HTTP1.1新增了五种请求方法:OPTIONS, PUT, DELETE, TRACE 和 CONNECT 方法。

序号 方法 描述
1 GET 请求指定的页面信息,并返回实体主体。
2 HEAD 类似于get请求,只不过返回的响应中没有具体的内容,用于获取报头
3 POST 向指定资源提交数据进行处理请求(例如提交表单或者上传文件)。数据被包含在请求体中。POST请求可能会导致新的资源的建立和/或已有资源的修改。
4 PUT 从客户端向服务器传送的数据取代指定的文档的内容。
5 DELETE 请求服务器删除指定的页面。
6 CONNECT HTTP/1.1协议中预留给能够将连接改为管道方式的代理服务器。
7 OPTIONS 允许客户端查看服务器的性能。
8 TRACE 回显服务器收到的请求,主要用于测试或诊断。

HTTP响应头信息

HTTP请求头提供了关于请求,响应或者其他的发送实体的信息。

应答头 说明
Allow 服务器支持哪些请求方法(如GET、POST等)
Content-Encoding 文档的编码(Encode)方法。只有在解码之后才可以得到Content-Type头指定的内容类型。利用gzip压缩文档能够显著地减少HTML文档的下载时间。Java的GZIPOutputStream可以很方便地进行gzip压缩,但只有Unix上的Netscape和Windows上的IE 4、IE 5才支持它。因此,Servlet应该通过查看Accept-Encoding头(即request.getHeader(“Accept-Encoding”))检查浏览器是否支持gzip,为支持gzip的浏览器返回经gzip压缩的HTML页面,为其他浏览器返回普通页面。
Content-Length 表示内容长度。只有当浏览器使用持久HTTP连接时才需要这个数据。如果你想要利用持久连接的优势,可以把输出文档写入 ByteArrayOutputStream,完成后查看其大小,然后把该值放入Content-Length头,最后通过byteArrayStream.writeTo(response.getOutputStream()发送内容。
Content-Type 表示后面的文档属于什么MIME类型。Servlet默认为text/plain,但通常需要显式地指定为text/html。由于经常要设置Content-Type,因此HttpServletResponse提供了一个专用的方法setContentType。
Date 当前的GMT时间。你可以用setDateHeader来设置这个头以避免转换时间格式的麻烦。
Expires 应该在什么时候认为文档已经过期,从而不再缓存它
Last-Modified 文档的最后改动时间。客户可以通过If-Modified-Since请求头提供一个日期,该请求将被视为一个条件GET,只有改动时间迟于指定时间的文档才会返回,否则返回一个304(Not Modified)状态。Last-Modified也可用setDateHeader方法来设置。
Location 表示客户应当到哪里去提取文档。Location通常不是直接设置的,而是通过HttpServletResponse的sendRedirect方法,该方法同时设置状态代码为302。
Refresh 表示浏览器应该在多少时间之后刷新文档,以秒计。除了刷新当前文档之外,你还可以通过setHeader(“Refresh”, “5; URL=http://host/path")让浏览器读取指定的页面。注意这种功能通常是通过设置HTML页面HEAD区的<META HTTP-EQUIV=”Refresh” CONTENT=”5;URL=http://host/path">实现,这是因为,自动刷新或重定向对于那些不能使用CGI或Servlet的HTML编写者十分重要。但是,对于Servlet来说,直接设置Refresh头更加方便。注意Refresh的意义是"N秒之后刷新本页面或访问指定页面",而不是"每隔N秒刷新本页面或访问指定页面"。因此,连续刷新要求每次都发送一个Refresh头,而发送204状态代码则可以阻止浏览器继续刷新,不管是使用Refresh头还是<META HTTP-EQUIV=”Refresh” …>。注意Refresh头不属于HTTP 1.1正式规范的一部分,而是一个扩展,但Netscape和IE都支持它。
Server 服务器名字。Servlet一般不设置这个值,而是由Web服务器自己设置。
Set-Cookie 设置和页面关联的Cookie。Servlet不应使用response.setHeader(“Set-Cookie”, …),而是应使用HttpServletResponse提供的专用方法addCookie。参见下文有关Cookie设置的讨论。
WWW-Authenticate 客户应该在Authorization头中提供什么类型的授权信息?在包含401(Unauthorized)状态行的应答中这个头是必需的。例如,response.setHeader(“WWW-Authenticate”, “BASIC realm=\”executives\””)。注意Servlet一般不进行这方面的处理,而是让Web服务器的专门机制来控制受密码保护页面的访问(例如.htaccess)。

HTTP状态码

常见的HTTP状态码:

  • 200 - 请求成功
  • 301 - 资源(网页等)被永久转移到其它URL
  • 404 - 请求的资源(网页等)不存在
  • 500 - 内部服务器错误
1…11121314
Neil Liu

Neil Liu

优秀不够,你是否无可替代

68 日志
25 标签
GitHub
© 2019 Neil Liu
由 Hexo 强力驱动
主题 - NexT.Muse