关于 WebView 缓存的重写

本贴最后更新于 2853 天前,其中的信息可能已经时异事殊

在现在的 app 中,越来越多的公司开始使用 webview 来进行一些活动页面的展示,甚至一些公司开始使用 webview 做为主要显示组件,把所有的内容都使用 H5 来呈现,这样一来,就对 WebView 的加载速度开始有越来越高的要求,我们要讨论的是 webview 的原本缓存机制所存在的弊端和如何复写 WebView 的缓存机制。

首先说一下 webview 的自带缓存机制的弊端:
webview 的自带缓存机制是无差别缓存,也就是说,不管是页面,样式还是图片,都会缓存到本地,刷新 webview 的缓存一般分为以下几种:

LOAD_CACHE_ONLY: 不使用网络,只读取本地缓存数据
LOAD_DEFAULT: 根据 cache-control 决定是否从网络上取数据。
LOAD_CACHE_NORMAL: API level 17 中已经废弃, 从 API level 11 开始作用同 LOAD_DEFAULT 模式
LOAD_NO_CACHE: 不使用缓存,只从网络获取数据.
LOAD_CACHE_ELSE_NETWORK,只要本地有,无论是否过期,或者 no-cache,都使用缓存中的数据。

反正不管以上几种具体情况如何,都肯定不是我们想要的。

我们想要的机制是:
1.缓存自己想要缓存的内容。
2.指定一个缓存策略,在需要的时候重新去服务器获取最新数据

于是我想到了以下方法
重写

Java 代码

  1. WebViewClient.shouldInterceptRequest(WebView view,WebResourceRequest request)

方法
首先需要新建一个类,继承 WebViewClient:

Java 代码

  1. public class DVDWebViewClient extends WebViewClient

然后实现

Java 代码

  1. public WebResourceResponse shouldInterceptRequest(WebView view,WebResourceRequest request)

方法
shouldInterceptRequest 方法会将所有页面的资源 URL 都一一列举出来,这样一来就好办了,我们似乎只需要缓存自己想要缓存的 url 就可以了。
然后事实是不是这样的呢?

Java 代码

  1. @Override
  2.  public WebResourceResponse shouldInterceptRequest(WebView view,  
    
  3.                                                    WebResourceRequest request) {  
    
  4.      //获取本地的URL主域名  
    
  5.      String curDomain = request.getUrl().getHost();  
    
  6.      //这行LOG可以不看  
    
  7.      if (BuildConfig.DEBUG) {  
    
  8.          Log.d(LOG_TAG, "curDomain " + curDomain + " request headers " + request.getRequestHeaders());  
    
  9.          for (Map.Entry entry : request.getRequestHeaders().entrySet()) {  
    
  10.              Log.d(LOG_TAG, "key=" + entry.getKey() + " #####value=" + entry.getValue() + "\n");  
    
  11.          }  
    
  12.      }  
    
  13.      //取不到domain就直接返回,把接下俩的动作交给webview自己处理  
    
  14.      if (curDomain == null || !isPicUrl(curDomain)) {  
    
  15.          return null;  
    
  16.      }  
    
  17.      if (BuildConfig.DEBUG) {  
    
  18.          Log.d(LOG_TAG, "shouldInterceptRequest url " + request.getUrl().toString());  
    
  19.      }  
    
  20.      //读取当前webview正准备加载URL资源  
    
  21.      String url = request.getUrl().toString();  
    
  22.      try {  
    
  23.         //根据资源url获取一个你要缓存到本地的文件名,一般是URL的MD5  
    
  24.          String resFileName = getResourcesFileName(url);  
    
  25.          if (resFileName == null || "".equals(resFileName)) {  
    
  26.              return null;  
    
  27.          }  
    
  28.         //这里是处理本地缓存的URL,缓存到本地,或者已经缓存过的就直接返回而不去网络进行加载  
    
  29.          this.dvdUrlCache.register(url, getResourcesFileName(url),  
    
  30.                  request.getRequestHeaders().get("Accept"), "UTF-8", DVDUrlCache.ONE_MONTH);  
    
  31.          return this.dvdUrlCache.load(url);  
    
  32.      } catch (Exception e) {  
    
  33.          Log.e(LOG_TAG, "", e);  
    
  34.      }  
    
  35.      return null;  
    
  36.  }  
    

接下来我们看下 DVDUrlCache 的实现:
DVDUrlCache 主要做了这么几件事:
1.封装一个内部类 CacheEntry,做一些基本信息存储

Java 代码

  1. private static class CacheEntry {
  2.     //用作存储的URL  
    
  3.      public String url;  
    
  4.      //本地保存的文件名称  
    
  5.      public String fileName;  
    
  6.      //标记资源的头部,通过request参数取回  
    
  7.      String mimeType;  
    
  8.      //需要缓存的资源文件的编码  
    
  9.      public String encoding;  
    
  10.      //缓存最大有效时间  
    
  11.      long maxAgeMillis;  
    
  12.      private CacheEntry(String url, String fileName,  
    
  13.                         String mimeType, String encoding, long maxAgeMillis) {  
    
  14.          this.url = url;  
    
  15.          this.fileName = fileName;  
    
  16.          this.mimeType = mimeType;  
    
  17.          this.encoding = encoding;  
    
  18.          this.maxAgeMillis = maxAgeMillis;  
    
  19.      }  
    
  20.  }  
    

接下来是类的构造放方法以及需要映射的 map

Java 代码

  1. //Key 为 URL
  2. private Map cacheEntries = new HashMap<>();
  3. //缓存路径的根目录
  4.  private File rootDir = null;  
    
  5.  DVDUrlCache() {  
    
  6. //获取本地缓存路径,这个请在调试中自行修改
  7.      this.rootDir = DiskUtil.getDiskCacheDir(DVDApplicationContext.getInstance().getApplicationContext());  
    
  8.  }  
    
  9. //资源注册,参考 DVDWebViewClient 的调用
  10. public void register(String url, String cacheFileName,
  11.                       String mimeType, String encoding,  
    
  12.                       long maxAgeMillis) {  
    
  13.      CacheEntry entry = new CacheEntry(url, cacheFileName, mimeType, encoding, maxAgeMillis);  
    
  14.      this.cacheEntries.put(url, entry);  
    
  15.  }  
    

然后是核心内容

Java 代码

  1. public WebResourceResponse load(final String url) {
  2.      try {  
    
  3.          final CacheEntry cacheEntry = this.cacheEntries.get(url);  
    
  4.          if (cacheEntry == null) {  
    
  5.              return null;  
    
  6.          }  
    
  7.          final File cachedFile = new File(this.rootDir.getPath() + File.separator + cacheEntry.fileName);  
    
  8.          if (BuildConfig.DEBUG) {  
    
  9.              Log.d(LOG_TAG, "cachedFile is " + cachedFile);  
    
  10.          }  
    
  11.          if (cachedFile.exists() && isReadFromCache(url)) {  
    
  12.              //还没有下载完,在快速切换URL的时候,可能会有很多task并没有及时完成,所以这里需要一个map用于存储正在下载的URL,下载完成后需要移除相应的task  
    
  13.              if (queueMap.containsKey(url)) {                 
    
  14.                  return null;  
    
  15.              }  
    
  16. //过期后直接删除本地缓存内容
  17.              long cacheEntryAge = System.currentTimeMillis() - cachedFile.lastModified();  
    
  18.              if (cacheEntryAge > cacheEntry.maxAgeMillis) {  
    
  19.                  cachedFile.delete();  
    
  20.                  if (BuildConfig.DEBUG) {  
    
  21.                      Log.d(LOG_TAG, "Deleting from cache: " + url);  
    
  22.                  }  
    
  23.                  return null;  
    
  24.              }  
    
  25.              //cached file exists and is not too old. Return file.  
    
  26.              if (BuildConfig.DEBUG) {  
    
  27.                  Log.d(LOG_TAG, url + " ### cache file : " + cachedFile.getAbsolutePath());  
    
  28.              }  
    
  29.              return new WebResourceResponse(  
    
  30.                      cacheEntry.mimeType, cacheEntry.encoding, new FileInputStream(cachedFile));  
    
  31.          } else {  
    
  32.              if (!queueMap.containsKey(url)) {  
    
  33.                  queueMap.put(url, new Callable() {  
    
  34.                      @Override  
    
  35.                      public Boolean call() throws Exception {  
    
  36.                          return downloadAndStore(url, cacheEntry);  
    
  37.                      }  
    
  38.                  });  
    
  39.                  final FutureTask futureTask = ThreadPoolManager.getInstance().addTaskCallback(queueMap.get(url));  
    
  40.                  ThreadPoolManager.getInstance().addTask(new Runnable() {  
    
  41.                      @Override  
    
  42.                      public void run() {  
    
  43.                          try {  
    
  44.                              if (futureTask.get()) {  
    
  45.                                  if (BuildConfig.DEBUG) {  
    
  46.                                      Log.d(LOG_TAG, "remove " + url);  
    
  47.                                  }  
    
  48.                                  queueMap.remove(url);  
    
  49.                              }  
    
  50.                          } catch (InterruptedException | ExecutionException e) {  
    
  51.                              Log.d(LOG_TAG, "", e);  
    
  52.                          }  
    
  53.                      }  
    
  54.                  });  
    
  55.              }  
    
  56.          }  
    
  57.      } catch (Exception e) {  
    
  58.          Log.d(LOG_TAG, "Error reading file over network: ", e);  
    
  59.      }  
    
  60.      return null;  
    
  61.  }  
    
  62. //这个方法是资源下载
  63.  private boolean downloadAndStore(final String url, final CacheEntry cacheEntry)  
    
  64.          throws IOException {  
    
  65.      FileOutputStream fileOutputStream = null;  
    
  66.      InputStream urlInput = null;  
    
  67.      try {  
    
  68.          URL urlObj = new URL(url);  
    
  69.          URLConnection urlConnection = urlObj.openConnection();  
    
  70.          urlInput = urlConnection.getInputStream();  
    
  71.          String tempFilePath = DVDUrlCache.this.rootDir.getPath() + File.separator + cacheEntry.fileName + ".temp";  
    
  72.          File tempFile = new File(tempFilePath);  
    
  73.          fileOutputStream = new FileOutputStream(tempFile);  
    
  74.          byte[] buffer = new byte[1024];  
    
  75.          int length;  
    
  76.          while ((length = urlInput.read(buffer)) > 0) {  
    
  77.              fileOutputStream.write(buffer, 0, length);  
    
  78.          }  
    
  79.          fileOutputStream.flush();  
    
  80.          File lastFile = new File(tempFilePath.replace(".temp", ""));  
    
  81.          boolean renameResult = tempFile.renameTo(lastFile);  
    
  82.          if (!renameResult) {  
    
  83.              Log.w(LOG_TAG, "rename file failed, " + tempFilePath);  
    
  84.          }  
    
  85. // Log.d(LOG_TAG, "Cache file: " + cacheEntry.fileName + " stored. ");
  86.          return true;  
    
  87.      } catch (Exception e) {  
    
  88.          Log.e(LOG_TAG, "", e);  
    
  89.      } finally {  
    
  90.          if (urlInput != null) {  
    
  91.              urlInput.close();  
    
  92.          }  
    
  93.          if (fileOutputStream != null) {  
    
  94.              fileOutputStream.close();  
    
  95.          }  
    
  96.      }  
    
  97.      return false;  
    
  98.  }  
    
  99.  private boolean isReadFromCache(String url) {  
    
  100.      return true;  
    
  101.  }  
    

完整的 DVDURLCache 代码,方便大家直接 copy

Java 代码

  1. package com.davdian.seller.util.WebUtil;
  2. import android.util.Log;
  3. import android.webkit.WebResourceResponse;
  4. import com.davdian.seller.BuildConfig;
  5. import com.davdian.seller.global.DVDApplicationContext;
  6. import com.davdian.seller.util.DiskUtil;
  7. import com.davdian.seller.util.ThreadPoolManager;
  8. import java.io.*;
  9. import java.net.URL;
  10. import java.net.URLConnection;
  11. import java.util.HashMap;
  12. import java.util.LinkedHashMap;
  13. import java.util.Map;
  14. import java.util.concurrent.Callable;
  15. import java.util.concurrent.ExecutionException;
  16. import java.util.concurrent.FutureTask;
  17. /**
    • 只缓存图片的自定义接口
    • Created by hongminghuangfu on 16/9/3.
  18. */
  19. public class DVDUrlCache {
  20.  private static final String LOG_TAG = "DVDUrlCache";  
    
  21.  private static final long ONE_SECOND = 1000L;  
    
  22.  private static final long ONE_MINUTE = 60L * ONE_SECOND;  
    
  23.  static final long ONE_HOUR = 60 * ONE_MINUTE;  
    
  24.  static final long ONE_DAY = 24 * ONE_HOUR;  
    
  25.  static final long ONE_MONTH = 30 * ONE_DAY;  
    
  26.  private static final LinkedHashMap> queueMap = new LinkedHashMap<>();  
    
  27.  private static class CacheEntry {  
    
  28.      public String url;  
    
  29.      public String fileName;  
    
  30.      String mimeType;  
    
  31.      public String encoding;  
    
  32.      long maxAgeMillis;  
    
  33.      private CacheEntry(String url, String fileName,  
    
  34.                         String mimeType, String encoding, long maxAgeMillis) {  
    
  35.          this.url = url;  
    
  36.          this.fileName = fileName;  
    
  37.          this.mimeType = mimeType;  
    
  38.          this.encoding = encoding;  
    
  39.          this.maxAgeMillis = maxAgeMillis;  
    
  40.      }  
    
  41.  }  
    
  42.  private Map cacheEntries = new HashMap<>();  
    
  43.  private File rootDir = null;  
    
  44.  DVDUrlCache() {  
    
  45. //本地缓存路径,请在调试中自行修改
  46.      this.rootDir = DiskUtil.getDiskCacheDir(DVDApplicationContext.getInstance().getApplicationContext());  
    
  47.  }  
    
  48.  public void register(String url, String cacheFileName,  
    
  49.                       String mimeType, String encoding,  
    
  50.                       long maxAgeMillis) {  
    
  51.      CacheEntry entry = new CacheEntry(url, cacheFileName, mimeType, encoding, maxAgeMillis);  
    
  52.      this.cacheEntries.put(url, entry);  
    
  53.  }  
    
  54.  public WebResourceResponse load(final String url) {  
    
  55.      try {  
    
  56.          final CacheEntry cacheEntry = this.cacheEntries.get(url);  
    
  57.          if (cacheEntry == null) {  
    
  58.              return null;  
    
  59.          }  
    
  60.          final File cachedFile = new File(this.rootDir.getPath() + File.separator + cacheEntry.fileName);  
    
  61.          if (BuildConfig.DEBUG) {  
    
  62.              Log.d(LOG_TAG, "cachedFile is " + cachedFile);  
    
  63.          }  
    
  64.          if (cachedFile.exists() && isReadFromCache(url)) {  
    
  65.              //还没有下载完  
    
  66.              if (queueMap.containsKey(url)) {                  
    
  67.                  return null;  
    
  68.              }  
    
  69.              long cacheEntryAge = System.currentTimeMillis() - cachedFile.lastModified();  
    
  70.              if (cacheEntryAge > cacheEntry.maxAgeMillis) {  
    
  71.                  cachedFile.delete();  
    
  72.                  if (BuildConfig.DEBUG) {  
    
  73.                      Log.d(LOG_TAG, "Deleting from cache: " + url);  
    
  74.                  }  
    
  75.                  return null;  
    
  76.              }  
    
  77.              //cached file exists and is not too old. Return file.  
    
  78.              if (BuildConfig.DEBUG) {  
    
  79.                  Log.d(LOG_TAG, url + " ### cache file : " + cachedFile.getAbsolutePath());  
    
  80.              }  
    
  81.              return new WebResourceResponse(  
    
  82.                      cacheEntry.mimeType, cacheEntry.encoding, new FileInputStream(cachedFile));  
    
  83.          } else {  
    
  84.              if (!queueMap.containsKey(url)) {  
    
  85.                  queueMap.put(url, new Callable() {  
    
  86.                      @Override  
    
  87.                      public Boolean call() throws Exception {  
    
  88.                          return downloadAndStore(url, cacheEntry);  
    
  89.                      }  
    
  90.                  });  
    
  91.                  final FutureTask futureTask = ThreadPoolManager.getInstance().addTaskCallback(queueMap.get(url));  
    
  92.                  ThreadPoolManager.getInstance().addTask(new Runnable() {  
    
  93.                      @Override  
    
  94.                      public void run() {  
    
  95.                          try {  
    
  96.                              if (futureTask.get()) {  
    
  97.                                  if (BuildConfig.DEBUG) {  
    
  98.                                      Log.d(LOG_TAG, "remove " + url);  
    
  99.                                  }  
    
  100.                                  queueMap.remove(url);  
    
  101.                              }  
    
  102.                          } catch (InterruptedException | ExecutionException e) {  
    
  103.                              Log.d(LOG_TAG, "", e);  
    
  104.                          }  
    
  105.                      }  
    
  106.                  });  
    
  107.              }  
    
  108.          }  
    
  109.      } catch (Exception e) {  
    
  110.          Log.d(LOG_TAG, "Error reading file over network: ", e);  
    
  111.      }  
    
  112.      return null;  
    
  113.  }  
    
  114.  private boolean downloadAndStore(final String url, final CacheEntry cacheEntry)  
    
  115.          throws IOException {  
    
  116.      FileOutputStream fileOutputStream = null;  
    
  117.      InputStream urlInput = null;  
    
  118.      try {  
    
  119.          URL urlObj = new URL(url);  
    
  120.          URLConnection urlConnection = urlObj.openConnection();  
    
  121.          urlInput = urlConnection.getInputStream();  
    
  122.          String tempFilePath = DVDUrlCache.this.rootDir.getPath() + File.separator + cacheEntry.fileName + ".temp";  
    
  123.          File tempFile = new File(tempFilePath);  
    
  124.          fileOutputStream = new FileOutputStream(tempFile);  
    
  125.          byte[] buffer = new byte[1024];  
    
  126.          int length;  
    
  127.          while ((length = urlInput.read(buffer)) > 0) {  
    
  128.              fileOutputStream.write(buffer, 0, length);  
    
  129.          }  
    
  130.          fileOutputStream.flush();  
    
  131.          File lastFile = new File(tempFilePath.replace(".temp", ""));  
    
  132.          boolean renameResult = tempFile.renameTo(lastFile);  
    
  133.          if (!renameResult) {  
    
  134.              Log.w(LOG_TAG, "rename file failed, " + tempFilePath);  
    
  135.          }  
    
  136. // Log.d(LOG_TAG, "Cache file: " + cacheEntry.fileName + " stored. ");
  137.          return true;  
    
  138.      } catch (Exception e) {  
    
  139.          Log.e(LOG_TAG, "", e);  
    
  140.      } finally {  
    
  141.          if (urlInput != null) {  
    
  142.              urlInput.close();  
    
  143.          }  
    
  144.          if (fileOutputStream != null) {  
    
  145.              fileOutputStream.close();  
    
  146.          }  
    
  147.      }  
    
  148.      return false;  
    
  149.  }  
    
  150.  private boolean isReadFromCache(String url) {  
    
  151.      return true;  
    
  152.  }  
    
  153. }

ThreadPoolManager 很简单:

Java 代码

  1. package com.davdian.seller.util;
  2. import android.util.Log;
  3. import java.util.concurrent.Callable;
  4. import java.util.concurrent.ExecutorService;
  5. import java.util.concurrent.Executors;
  6. import java.util.concurrent.FutureTask;
  7. /**
    • 线程池
    • Created by hongminghuangfu on 16/9/9.
  8. */
  9. public class ThreadPoolManager {
  10.  private static final String LOG_TAG = "ThreadPoolManager";  
    
  11.  private static final ThreadPoolManager instance = new ThreadPoolManager();  
    
  12.  private ExecutorService threadPool = Executors.newFixedThreadPool(100);  
    
  13.  public static ThreadPoolManager getInstance() {  
    
  14.      return instance;  
    
  15.  }  
    
  16.  /** 
    
  17.   * @param runnable 不返回执行结果的异步任务 
    
  18.   */  
    
  19.  public void addTask(Runnable runnable) {  
    
  20.      try {  
    
  21.          if (runnable != null) {  
    
  22.              threadPool.execute(runnable);  
    
  23.          }  
    
  24.      } catch (Exception e) {  
    
  25.          Log.e(LOG_TAG, "", e);  
    
  26.      }  
    
  27.  }  
    
  28.  /** 
    
  29.   * @param callback 异步任务 
    
  30.   * @return 你可以获取相应的执行结果 
    
  31.   */  
    
  32.  public FutureTask addTaskCallback(Callable callback) {  
    
  33.      if (callback == null) {  
    
  34.          return null;  
    
  35.      } else {  
    
  36.          FutureTask futureTask = new FutureTask<>(callback);  
    
  37.          threadPool.submit(futureTask);  
    
  38.          return futureTask;  
    
  39.      }  
    
  40.  }  
    
  41. // 这是一个 demo,如果你看不懂,可以打开跑一下
  42. // public static void main(String args[]) {
  43. // FutureTask ft = ThreadPoolManager.getInstance().addTaskCallback(new Callable() {
  44. // @Override
  45. // public Object call() throws Exception {
  46. // int sum = 0;
  47. // for (int i = 0; i < 1000; i++) {
  48. // sum++;
  49. // }
  50. // return sum;
  51. // }
  52. // });
  53. // try {
  54. // System.out.println("执行结果是:" + ft.get());
  55. // } catch (InterruptedException | ExecutionException e) {
  56. // e.printStackTrace();
  57. // }
  58. //
  59. // }
  60. }
  • Android

    Android 是一种以 Linux 为基础的开放源码操作系统,主要使用于便携设备。2005 年由 Google 收购注资,并拉拢多家制造商组成开放手机联盟开发改良,逐渐扩展到到平板电脑及其他领域上。

    334 引用 • 323 回帖
  • Java

    Java 是一种可以撰写跨平台应用软件的面向对象的程序设计语言,是由 Sun Microsystems 公司于 1995 年 5 月推出的。Java 技术具有卓越的通用性、高效性、平台移植性和安全性。

    3190 引用 • 8214 回帖

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...
  • 被你的图片骗进来了。为嘛要用图片打那么多的星呢? emoji 多好,还好看。 ⭐️ 🌟

推荐标签 标签

  • TextBundle

    TextBundle 文件格式旨在应用程序之间交换 Markdown 或 Fountain 之类的纯文本文件时,提供更无缝的用户体验。

    1 引用 • 2 回帖 • 55 关注
  • Vue.js

    Vue.js(读音 /vju ː/,类似于 view)是一个构建数据驱动的 Web 界面库。Vue.js 的目标是通过尽可能简单的 API 实现响应的数据绑定和组合的视图组件。

    265 引用 • 666 回帖
  • CAP

    CAP 指的是在一个分布式系统中, Consistency(一致性)、 Availability(可用性)、Partition tolerance(分区容错性),三者不可兼得。

    11 引用 • 5 回帖 • 615 关注
  • Dubbo

    Dubbo 是一个分布式服务框架,致力于提供高性能和透明化的 RPC 远程服务调用方案,是 [阿里巴巴] SOA 服务化治理方案的核心框架,每天为 2,000+ 个服务提供 3,000,000,000+ 次访问量支持,并被广泛应用于阿里巴巴集团的各成员站点。

    60 引用 • 82 回帖 • 605 关注
  • Flutter

    Flutter 是谷歌的移动 UI 框架,可以快速在 iOS 和 Android 上构建高质量的原生用户界面。 Flutter 可以与现有的代码一起工作,它正在被越来越多的开发者和组织使用,并且 Flutter 是完全免费、开源的。

    39 引用 • 92 回帖 • 2 关注
  • GraphQL

    GraphQL 是一个用于 API 的查询语言,是一个使用基于类型系统来执行查询的服务端运行时(类型系统由你的数据定义)。GraphQL 并没有和任何特定数据库或者存储引擎绑定,而是依靠你现有的代码和数据支撑。

    4 引用 • 3 回帖 • 11 关注
  • webpack

    webpack 是一个用于前端开发的模块加载器和打包工具,它能把各种资源,例如 JS、CSS(less/sass)、图片等都作为模块来使用和处理。

    41 引用 • 130 回帖 • 255 关注
  • PWA

    PWA(Progressive Web App)是 Google 在 2015 年提出、2016 年 6 月开始推广的项目。它结合了一系列现代 Web 技术,在网页应用中实现和原生应用相近的用户体验。

    14 引用 • 69 回帖 • 160 关注
  • Sandbox

    如果帖子标签含有 Sandbox ,则该帖子会被视为“测试帖”,主要用于测试社区功能,排查 bug 等,该标签下内容不定期进行清理。

    412 引用 • 1246 回帖 • 589 关注
  • InfluxDB

    InfluxDB 是一个开源的没有外部依赖的时间序列数据库。适用于记录度量,事件及实时分析。

    2 引用 • 78 关注
  • 博客

    记录并分享人生的经历。

    273 引用 • 2388 回帖
  • MySQL

    MySQL 是一个关系型数据库管理系统,由瑞典 MySQL AB 公司开发,目前属于 Oracle 公司。MySQL 是最流行的关系型数据库管理系统之一。

    692 引用 • 535 回帖
  • 30Seconds

    📙 前端知识精选集,包含 HTML、CSS、JavaScript、React、Node、安全等方面,每天仅需 30 秒。

    • 精选常见面试题,帮助您准备下一次面试
    • 精选常见交互,帮助您拥有简洁酷炫的站点
    • 精选有用的 React 片段,帮助你获取最佳实践
    • 精选常见代码集,帮助您提高打码效率
    • 整理前端界的最新资讯,邀您一同探索新世界
    488 引用 • 384 回帖 • 3 关注
  • 区块链

    区块链是分布式数据存储、点对点传输、共识机制、加密算法等计算机技术的新型应用模式。所谓共识机制是区块链系统中实现不同节点之间建立信任、获取权益的数学算法 。

    91 引用 • 751 回帖
  • 心情

    心是产生任何想法的源泉,心本体会陷入到对自己本体不能理解的状态中,因为心能产生任何想法,不能分出对错,不能分出自己。

    59 引用 • 369 回帖 • 2 关注
  • GitBook

    GitBook 使您的团队可以轻松编写和维护高质量的文档。 分享知识,提高团队的工作效率,让用户满意。

    3 引用 • 8 回帖
  • 京东

    京东是中国最大的自营式电商企业,2015 年第一季度在中国自营式 B2C 电商市场的占有率为 56.3%。2014 年 5 月,京东在美国纳斯达克证券交易所正式挂牌上市(股票代码:JD),是中国第一个成功赴美上市的大型综合型电商平台,与腾讯、百度等中国互联网巨头共同跻身全球前十大互联网公司排行榜。

    14 引用 • 102 回帖 • 350 关注
  • 前端

    前端技术一般分为前端设计和前端开发,前端设计可以理解为网站的视觉设计,前端开发则是网站的前台代码实现,包括 HTML、CSS 以及 JavaScript 等。

    247 引用 • 1348 回帖
  • BND

    BND(Baidu Netdisk Downloader)是一款图形界面的百度网盘不限速下载器,支持 Windows、Linux 和 Mac,详细介绍请看这里

    107 引用 • 1281 回帖 • 37 关注
  • 音乐

    你听到信仰的声音了么?

    61 引用 • 511 回帖 • 2 关注
  • Swift

    Swift 是苹果于 2014 年 WWDC(苹果开发者大会)发布的开发语言,可与 Objective-C 共同运行于 Mac OS 和 iOS 平台,用于搭建基于苹果平台的应用程序。

    36 引用 • 37 回帖 • 534 关注
  • 安全

    安全永远都不是一个小问题。

    201 引用 • 816 回帖
  • golang

    Go 语言是 Google 推出的一种全新的编程语言,可以在不损失应用程序性能的情况下降低代码的复杂性。谷歌首席软件工程师罗布派克(Rob Pike)说:我们之所以开发 Go,是因为过去 10 多年间软件开发的难度令人沮丧。Go 是谷歌 2009 发布的第二款编程语言。

    497 引用 • 1388 回帖 • 280 关注
  • BAE

    百度应用引擎(Baidu App Engine)提供了 PHP、Java、Python 的执行环境,以及云存储、消息服务、云数据库等全面的云服务。它可以让开发者实现自动地部署和管理应用,并且提供动态扩容和负载均衡的运行环境,让开发者不用考虑高成本的运维工作,只需专注于业务逻辑,大大降低了开发者学习和迁移的成本。

    19 引用 • 75 回帖 • 647 关注
  • JVM

    JVM(Java Virtual Machine)Java 虚拟机是一个微型操作系统,有自己的硬件构架体系,还有相应的指令系统。能够识别 Java 独特的 .class 文件(字节码),能够将这些文件中的信息读取出来,使得 Java 程序只需要生成 Java 虚拟机上的字节码后就能在不同操作系统平台上进行运行。

    180 引用 • 120 回帖 • 1 关注
  • WebComponents

    Web Components 是 W3C 定义的标准,它给了前端开发者扩展浏览器标签的能力,可以方便地定制可复用组件,更好的进行模块化开发,解放了前端开发者的生产力。

    1 引用 • 3 关注
  • danl
    150 关注