(一)操作方法和spring源码添加修改部分

事先说明:spring源码要下载好,会有修改spring的源码操作,本文和本作者所依赖的spring项目的版本是3.1.1,spring4及以上源码对应的类路径可能有所改变,需要自己找到要修改的类哦,类名应该是不会变的。望理解~~

操作步骤: ~具体说明看类文件注释~

1.运行ProduceKey.java的main方法 生成 key.key文件

2.ClassesEncryption类中,修改项目路径,修改key文件路径

3.执行ClassesEncryption的main方法加密文件

4.将OverrideClassLoader项目生成jar包,放在tomcat/lib下(自己网上看操作)

5.在tomcat/conf/context.xml 中Context 节点中添加
  <Loader loaderClass="com.loader.ReBuildClassLoader" delegate="false"></Loader>

6. 替换tomcat的文件,比如com/ 下的除了(com/webapp/entity)下所有文件

7.启动tomcat

加密修改spring 源码

1.org.springframework.context-3.1.1.RELEASE.jar
2.org.springframework.core-3.1.1.RELEASE.jar

其中org.springframework.context-3.1.1.RELEASE.jar 修改类

(1)ClassPathScanningCandidateComponentProvider.java

line:230 加入
packageSearchPath = packageSearchPath.replace(".class", ".zdywjmjw");
System.out.println("packageSearchPath"+packageSearchPath);
Resource[] resourcesZdywjmjw = this.resourcePatternResolver.getResources(packageSearchPath);
Resource[] resources = new Resource[resourcesClass.length+resourcesZdywjmjw.length];
System.arraycopy(resourcesClass, 0, resources, 0, resourcesClass.length); 
System.arraycopy(resourcesZdywjmjw, 0, resources, resourcesClass.length, resourcesZdywjmjw.length);

其中org.springframework.core-3.1.1.RELEASE.jar 修改类

(1)ClassPathResource.java

在类下面添加下列方法

private static boolean isWindowsOS() {
boolean isWindowsOS = false;
String osName = System.getProperty("os.name");
if (osName.toLowerCase().indexOf("windows") > -1) {
isWindowsOS = true;
}
return isWindowsOS;
}

// 把文件读入byte数组
private byte[] readFile(String filename) throws IOException {
File file = new File(filename);
long len = file.length();
byte data[] = new byte[(int) len];
FileInputStream fin = new FileInputStream(file);
int r = fin.read(data);
if (r != len)
throw new IOException("Only read " + r + " of " + len + " for " + file);
fin.close();
return data;
}

// 把byte数组写出到文件
private void writeFile(String filename, byte data[]) throws IOException {
FileOutputStream fout = new FileOutputStream(filename);
fout.write(data);
fout.close();
}

line 168行 修改
String basepath =this.getClass().getResource("/").getPath();
if(isWindowsOS()){
basepath = basepath.substring(1,basepath.length());//window下增加
}
String reloadedClass = basepath+this.path.replace(".class", ".zdywjmjw");
File file = new File(reloadedClass);
if (file.exists()) {
// 下面是定制部分
  try {
    String keyFilename = basepath+"key.key";
    // 读取密匙
    String algorithm = "DES";
    byte rawKey[] = readFile(keyFilename);
    DESKeySpec dks = new DESKeySpec(rawKey);
    SecretKeyFactory keyFactory = SecretKeyFactory.getInstance(algorithm);
    SecretKey key = keyFactory.generateSecret(dks);
    SecureRandom sr = new SecureRandom();
    Cipher cipher = Cipher.getInstance(algorithm);
    cipher.init(Cipher.DECRYPT_MODE, key, sr);
    // 读取经过加密的类文件
    byte classData[] = readFile(pclass);
    if (classData != null) {
      byte decryptedClassData[] = cipher.doFinal(classData); // 解密
      InputStream sbs = new ByteArrayInputStream(decryptedClassData); 
      return sbs;
    }
  } catch (Exception fnfe) {

  }
}

(2).SimpleMetadataReader.java

添加方法
private static boolean isWindowsOS() {
  boolean isWindowsOS = false;
  String osName = System.getProperty("os.name");
  if (osName.toLowerCase().indexOf("windows") > -1) {
    isWindowsOS = true;
  }
  return isWindowsOS;
}

public static byte[] readFile(String filename) throws IOException {
  File file = new File(filename);
  long len = file.length();
  byte data[] = new byte[(int) len];
  FileInputStream fin = new FileInputStream(file);
  int r = fin.read(data);
  if (r != len)
  throw new IOException("Only read " + r + " of " + len + " for " + file);
  fin.close();
  return data;
}

修改构造类

SimpleMetadataReader(Resource resource, ClassLoader classLoader) throws IOException {
  InputStream is = resource.getInputStream();
  ClassReader classReader = null;
  try {
    String filename = ""; 
    if (resource.getURI().toString().indexOf("jar:file") == -1 ) {
      String checkPath="";
      if(isWindowsOS()){
      checkPath = "classes\\com";
    }else{
      checkPath = "classes/com";
    }
    if(resource.getFile().getAbsolutePath().indexOf(checkPath)!=-1){
      filename = resource.getFile().getAbsolutePath().replace(".class", ".zdywjmjw"); 
      File file = new File(filename);
      if(file.exists()){
      //进行解密
      String basepath =this.getClass().getResource("/").getPath();
      if(isWindowsOS()){
        basepath = basepath.substring(1,basepath.length());
      }
      String keyFilename = basepath+"key.key";
      String algorithm = "DES";
      try{
        // 生成密匙
        SecureRandom sr = new SecureRandom();
        byte rawKey[] = readFile(keyFilename);
        DESKeySpec dks = new DESKeySpec(rawKey);
        SecretKeyFactory keyFactory = SecretKeyFactory.getInstance(algorithm);
        SecretKey key = keyFactory.generateSecret(dks);

        // 创建用于实际加密操作的Cipher对象
        Cipher cipher = Cipher.getInstance(algorithm);
        cipher.init(Cipher.DECRYPT_MODE, key, sr);
        byte classData[] = readFile(name);
        byte decryptedClassData[] = cipher.doFinal(classData);
        classReader = new ClassReader(decryptedClassData);
      } catch(NoSuchAlgorithmException e){
        e.printStackTrace();
      } catch (InvalidKeyException e) {
        e.printStackTrace();
      } catch (InvalidKeySpecException e) {
        e.printStackTrace();
      } catch (NoSuchPaddingException e) {
        e.printStackTrace();
      } catch (IllegalBlockSizeException e) {
        e.printStackTrace();
      } catch (BadPaddingException e) {
        e.printStackTrace();
      }
    }else{
      classReader = new ClassReader(is);
    }
   }else{
    classReader = new ClassReader(is);
   }
  }else{
    classReader = new ClassReader(is);
  }
  }finally {
    is.close();
  }

  AnnotationMetadataReadingVisitor visitor = new AnnotationMetadataReadingVisitor(classLoader);
  classReader.accept(visitor, true);

  this.annotationMetadata = visitor;
  // (since AnnotationMetadataReader extends ClassMetadataReadingVisitor)
  this.classMetadata = visitor;
  this.resource = resource;
}

(3)LocalVariableTableParameterNameDiscoverer.java

line 103 行修改
String cName = clazz.getResource("/").getPath()+clazz.getName().replace(".", "/")+".class";
is = clazz.getClassLoader().getResourceAsStream(cName);
if(is == null){
  // We couldn't load the class file, which is not fatal as it
  // simply means this method of discovering parameter names won't work.
  if (logger.isDebugEnabled()) {
    logger.debug("Cannot find '.class' file for class [" + clazz
    + "] - unable to determine constructors/methods parameter names");
  }
  return NO_DEBUG_INFO_MAP;
}

(二)项目相关

项目名称:OverrideClassLoader

项目打包方式:jar,是普通Java工程

项目结构:

com.key

com.loader

com.util

其中 key包下面文件有:ClassesEncryption.java、ProduceKey.java、GenerateEncryptedClassMain

loader包下面有文件:ReBuildClassLoader.java 

util包下面有文件:FileUtil.java

(1)ProduceKey.java

package com.key;

import java.security.SecureRandom;

import javax.crypto.KeyGenerator;
import javax.crypto.SecretKey;

import com.util.FileUtil;
/**
* 生成key文件类
* @author win7
*
*/
public class ProduceKey {

  public static void main(String args[]) throws Exception {

    String keyFilename = null;
    boolean windowsOS = isWindowsOS();//检查是否是window系统
    System.out.println(windowsOS);
    if (windowsOS) {
      keyFilename = "D://developer/apache-tomcat7/webapps/项目名称/WEB-INF/classes/key.key";//是本地开发的项目,tomcat的路径
    } else {
      keyFilename = "/usr/local/tomcat/webapps/项目名称/WEB-INF/classes/key.key";
    }

    String algorithm = "DES";

    // 生成密匙
    SecureRandom sr = new SecureRandom();
    KeyGenerator kg = KeyGenerator.getInstance(algorithm);
    kg.init(sr);
    SecretKey key = kg.generateKey();

    // 把密匙数据保存到文件,放入指定位置
    FileUtil.writeFile(keyFilename, key.getEncoded());
  }

  private static boolean isWindowsOS() {
    boolean isWindowsOS = false;
    String osName = System.getProperty("os.name");
    if (osName.toLowerCase().indexOf("windows") > -1) {
      isWindowsOS = true;
    }
    return isWindowsOS;
  }

}

(2)ClassesEncryption.java

package com.key;

import java.io.File;
import java.security.SecureRandom;
import java.util.ArrayList;
import java.util.List;

import javax.crypto.Cipher;
import javax.crypto.SecretKey;
import javax.crypto.SecretKeyFactory;
import javax.crypto.spec.DESKeySpec;

import com.util.FileUtil;
/**
* class文件加密类
* @author win7
*
*/
public class ClassesEncryption {
  public static List<File> filelist = new ArrayList<>();

  public static void main(String[] args) throws Exception {
  System.out.println("search start");

  // 得到要加密的项目下com目录下的class文件
  getFileList("D://developer/apache-tomcat7/webapps/项目名称/WEB-INF/classes/com");

  System.out.println("文件数量" + filelist.size());

  // 获取文件名秘钥
  String keyFilename = "D://developer/apache-tomcat7/webapps/项目名称/WEB-INF/classes/key.key";
  // 算法
  String algorithm = "DES";

  // -------下面开始生成密匙-------
  // 强加密随机数生成器
  SecureRandom sr = new SecureRandom();
  // 将加密后的key.key转成字节数组
  byte rawKey[] = FileUtil.readFile(keyFilename);
  // 创建一个 DESKeySpec 对象,使用 rawKey 中的前 8 个字节作为 DES 密钥的密钥内容。
  DESKeySpec dks = new DESKeySpec(rawKey);
  // 得到转换指定算法的秘密密钥的SecretKeyFactory对象。
  SecretKeyFactory keyFactory = SecretKeyFactory.getInstance(algorithm);
  // 根据提供的密钥规范(密钥材料)生成 SecretKey 对象
  SecretKey key = keyFactory.generateSecret(dks);

  // 创建用于实际加密操作的Cipher对象 , Cipher:此类为加密和解密提供密码功能
  Cipher ecipher = Cipher.getInstance(algorithm);
  // 参数:加密的常量,加密密钥, 算法参数
  ecipher.init(Cipher.ENCRYPT_MODE, key, sr);

    // 加密命令行中指定的每一个类
    for (int i = 0; i < filelist.size(); i++) {
      String filename = filelist.get(i).getAbsolutePath();
      byte classData[] = FileUtil.readFile(filename); // 读入类文件
      byte encryptedClassData[] = ecipher.doFinal(classData); // 加密
      String projectFile = filename.replace(".class", ".zdywjmjw");//zdyljzq:自定义文件名结尾
      FileUtil.writeFile(projectFile, encryptedClassData); // 保存加密后的内容
      System.out.println("Encrypted " + projectFile);

      File classFile = new File(filename);
      classFile.delete();
    }
  }

  public static void encrypt(String tomcat_home) throws Exception {
    System.out.println("search start");
    getFileList(tomcat_home + "webapps/项目名称/WEB-INF/classes/com");

    System.out.println("文件数量" + filelist.size());

    String keyFilename = tomcat_home + "webapps/项目名称/WEB-INF/classes/key.key";
    String algorithm = "DES";// 算法

    // -------下面开始生成密匙-------
    // 强加密随机数生成器
    SecureRandom sr = new SecureRandom();
    // 将加密后的key.key转成字节数组
    byte rawKey[] = FileUtil.readFile(keyFilename);
    // 创建一个 DESKeySpec 对象,使用 rawKey 中的前 8 个字节作为 DES 密钥的密钥内容。
    DESKeySpec dks = new DESKeySpec(rawKey);
    // 得到转换指定算法的秘密密钥的SecretKeyFactory对象。
    SecretKeyFactory keyFactory = SecretKeyFactory.getInstance(algorithm);
    // 根据提供的密钥规范(密钥材料)生成 SecretKey 对象
    SecretKey key = keyFactory.generateSecret(dks);

    // 创建用于实际加密操作的Cipher对象 , Cipher:此类为加密和解密提供密码功能
    Cipher ecipher = Cipher.getInstance(algorithm);
    // 参数:加密的常量,加密密钥, 算法参数
    ecipher.init(Cipher.ENCRYPT_MODE, key, sr);

    // 加密命令行中指定的每一个类
    for (int i = 0; i < filelist.size(); i++) {
      String filename = filelist.get(i).getAbsolutePath();
      byte classData[] = FileUtil.readFile(filename); // 读入类文件
      byte encryptedClassData[] = ecipher.doFinal(classData); // 加密
      String projectFile = filename.replace(".class", ".zdywjmjw");//zdyljzq:自定义文件名结尾
      FileUtil.writeFile(projectFile, encryptedClassData); // 保存加密后的内容
      System.out.println("Encrypted " + projectFile);

      File classFile = new File(filename);
      classFile.delete();

    }
  }

/**
* 注意:加密的时候不能加密实体,要不然项目就启动不成功或者异常
* @param strPath
* @return
*/
   public static List<File> getFileList(String strPath) {
      File dir = new File(strPath);
      File[] files = dir.listFiles(); // 该文件目录下文件全部放入数组
      if (files != null) {
        for (int i = 0; i < files.length; i++) {
          String fileName = files[i].getName();
          if (files[i].isDirectory()) { // 判断是文件还是文件夹
            getFileList(files[i].getAbsolutePath()); // 获取文件绝对路径
          } else if (fileName.endsWith(".class")) { // 判断文件名是否以.class结尾
            String strFileName = files[i].getAbsolutePath();
            // System.out.println("---" + strFileName);
            if (isWindowsOS()) {
              if (strFileName.indexOf("com\\项目名称\\webapp\\entity") == -1) {
                filelist.add(files[i]);
              }
            } else {
              if (strFileName.indexOf("com/项目名称/webapp/entity") == -1) {
              filelist.add(files[i]);
            }
          }

          } else {
          continue;
        }
      }

    }
    return filelist;
  }

  public static boolean isWindowsOS() {
    boolean isWindowsOS = false;
    String osName = System.getProperty("os.name");
    if (osName.toLowerCase().indexOf("windows") > -1) {
      isWindowsOS = true;
    }
    return isWindowsOS;
  }
}

(3)GenerateEncryptedClassMain.java

package com.key;

import java.util.Scanner;
/**
* main方法生成加密class文件
* @author win7
*
*/
public class GenerateEncryptedClassMain {
  public static void main(String[] args) {
    try {
      Scanner s = new Scanner(System.in);
      String str = null;
      System.out.println("请tomcat路径以'/'结尾:");//如: /usr/local/tomcat/
      str = s.next();
      if (str == null || str.trim().equals("")) {
        System.out.println("tomcat路径不能为空");
        return;
      }
      String tomcat_home = str;
      ClassesEncryption.encrypt(tomcat_home);
    } catch (Exception e) {
      e.printStackTrace();
    }
  }
}

(4)ReBuildClassLoader.java

package com.loader;

import java.io.ByteArrayInputStream;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.security.GeneralSecurityException;
import java.security.SecureRandom;

import javax.crypto.Cipher;
import javax.crypto.SecretKey;
import javax.crypto.SecretKeyFactory;
import javax.crypto.spec.DESKeySpec;

import org.apache.catalina.loader.WebappClassLoader;

import com.util.FileUtil;

/**
* 自定义类加载器
* WebappClassLoader:是tomcat的jar包提供,所以要添加 tomcat server Runtime

* @author win7
*
*/
public class ReBuildClassLoader extends WebappClassLoader {
  // 这些对象在构造函数中设置,以后loadClass()方法将利用它们解密类
  private SecretKey key;
  private Cipher cipher;

  public ReBuildClassLoader() {
    super();
  }

  public ReBuildClassLoader(ClassLoader parent) {
    super(parent);
  }

  @Override
  public Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {
    try {
      // 要创建的Class对象
      Class clasz = null;

      // 必需的步骤1:如果类已经在系统缓冲之中,不必再次装入它
      clasz = findLoadedClass(name);

      if (clasz != null)
        return clasz;

        // 下面是定制部分
      try {
        String basepath = "";
        if (isWindowsOS()) {
          basepath = "D://developer/apache-tomcat7/webapps/项目名称/WEB-INF/classes/";// 项目物理地址
        } else {
          basepath = "/usr/local/tomcat/webapps/项目名称/WEB-INF/classes/";
        }
        String cname = basepath + name.replace('.', '/') + ".zdywjmjw";
        File file = new File(cname);
        if (file.exists()) {
          String keyFilename = basepath + "key.key";
          // 读取密匙
          String algorithm = "DES";
          // System.err.println("[DecryptStart: reading key]");
          byte rawKey[] = FileUtil.readFile(keyFilename);
          DESKeySpec dks = new DESKeySpec(rawKey);
          SecretKeyFactory keyFactory = SecretKeyFactory.getInstance(algorithm);
          key = keyFactory.generateSecret(dks);
          SecureRandom sr = new SecureRandom();
          // System.err.println("[DecryptStart: creating cipher]");
          cipher = Cipher.getInstance(algorithm);
          cipher.init(Cipher.DECRYPT_MODE, key, sr);
          // 读取经过加密的类文件
          byte classData[] = FileUtil.readFile(cname);
          if (classData != null) {
            byte decryptedClassData[] = cipher.doFinal(classData); // 解密
            clasz = defineClass(name, decryptedClassData, 0, decryptedClassData.length); // 再把它转换成一个类
            // System.err.println("[DecryptStart: decrypting class + name + "]");
          }
        }
      } catch (FileNotFoundException fnfe) {

      }

      // 必需的步骤2:如果上面没有成功
      // 尝试用默认的ClassLoader装入它
      if (clasz == null)
        return super.loadClass(name, resolve);

        // 必需的步骤3:如有必要,则装入相关的类
        if (resolve && clasz != null)
          resolveClass(clasz);

          return clasz;// 把类返回给调用者

        } catch (IOException ie) {
          throw new ClassNotFoundException(ie.toString());
        } catch (GeneralSecurityException gse) {
          throw new ClassNotFoundException(gse.toString());
        }
    }

    @Override
    public Class<?> loadClass(String name) throws ClassNotFoundException {
      return super.loadClass(name);
    }

    @Override
    public InputStream getResourceAsStream(String arg0) {
      String pclass = arg0.replace(".class", ".zdywjmjw");
      File file = new File(pclass);
      if (file.exists()) {
      // 下面是定制部分
      try {
        String basepath = "";
        if (isWindowsOS()) {
          basepath = "D://developer/apache-tomcat7/webapps/项目名称/WEB-INF/classes/";// 项目物理地址
        } else {
          basepath = "/usr/local/tomcat/webapps/项目名称/WEB-INF/classes/";
        }
        String keyFilename = basepath + "key.key";
        // 读取密匙
        String algorithm = "DES";
        byte rawKey[] = FileUtil.readFile(keyFilename);
        DESKeySpec dks = new DESKeySpec(rawKey);
        SecretKeyFactory keyFactory = SecretKeyFactory.getInstance(algorithm);
        key = keyFactory.generateSecret(dks);
        SecureRandom sr = new SecureRandom();
        cipher = Cipher.getInstance(algorithm);
        cipher.init(Cipher.DECRYPT_MODE, key, sr);
        // 读取经过加密的类文件
        byte classData[] = FileUtil.readFile(pclass);
        if (classData != null) {
          byte decryptedClassData[] = cipher.doFinal(classData); // 解密
          InputStream sbs = new ByteArrayInputStream(decryptedClassData);
          return sbs;
          }
        } catch (Exception fnfe) {

        }
      }
      return super.getResourceAsStream(arg0);
    }

  private boolean isWindowsOS() {
    boolean isWindowsOS = false;
    String osName = System.getProperty("os.name");
    if (osName.toLowerCase().indexOf("windows") > -1) {
      isWindowsOS = true;
    }
      return isWindowsOS;
    }

  }

FileUtil.java

package com.util;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;

public class FileUtil {
  // 把文件读入byte数组
  static public byte[] readFile(String filename) throws IOException {
  File file = new File(filename);
  long len = file.length();
  byte data[] = new byte[(int) len];
  FileInputStream fin = new FileInputStream(file);
  int r = fin.read(data);
  if (r != len)
    throw new IOException("Only read " + r + " of " + len + " for " + file);
    fin.close();
    return data;
  }

  // 把byte数组写出到文件
  static public void writeFile(String filename, byte data[]) throws IOException {
    FileOutputStream fout = new FileOutputStream(filename);
    fout.write(data);
    fout.close();
  }
}

修改spring源码重写classloader实现项目加密的更多相关文章

  1. spring源码学习之:项目公共配置项解决方案

    一:项目中有一些key,value的简单配置 org.apache.commons.configuration.DatabaseConfiguration可以轻松解决 二:配置项目的xml中bean ...

  2. 修改和编译spring源码,构建jar(spring-context-4.0.2.RELEASE)

    上周在定位问题时,发现Spring容器实例化Bean的时候抛出异常,为了查看更详细的信息,决定修改spring-context-4.0.2.RELEASE.jar中的CommonAnnotationB ...

  3. Spring源码 02 项目搭建

    参考源 https://www.bilibili.com/video/BV1tR4y1F75R?spm_id_from=333.337.search-card.all.click https://ww ...

  4. 剑指Spring源码(二)

    这是春节后的第一篇博客,我在构思这篇博客的时候,一度想放弃,想想要不要换个东西写,因为毕竟个人水平有限,Spring源码实在博大精深,不是我这个菜的抠脚的菜鸡可以驾驭的,怕误人子弟,还有就是源码分析类 ...

  5. Spring源码分析 之浅谈设计模式

    一直想专门写个Spring源码的博客,工作了,可以全身性的投入到互联网行业中.虽然加班很严重,但是依然很开心.趁着凌晨有时间,总结总结. 首先spring,相信大家都很熟悉了. 1.轻量级  零配置, ...

  6. Spring源码深度解析之Spring MVC

    Spring源码深度解析之Spring MVC Spring框架提供了构建Web应用程序的全功能MVC模块.通过策略接口,Spring框架是高度可配置的,而且支持多种视图技术,例如JavaServer ...

  7. Spring源码阅读一

    引导: 众所周知,阅读spring源码最开始的就是去了解spring bean的生命周期:bean的生命周期是怎么样的呢,见图知意: 大致流程: 首先后通过BeanDefinitionReader读取 ...

  8. spring源码分析(一)IoC、DI

    创建日期:2016.08.06 修改日期:2016.08.07 - 2016.08.12 交流QQ:992591601 参考书籍:<spring源码深度解析>.<spring技术内幕 ...

  9. 【Spring源码分析】Bean加载流程概览

    代码入口 之前写文章都会啰啰嗦嗦一大堆再开始,进入[Spring源码分析]这个板块就直接切入正题了. 很多朋友可能想看Spring源码,但是不知道应当如何入手去看,这个可以理解:Java开发者通常从事 ...

随机推荐

  1. Python基于Python实现批量上传文件或目录到不同的Linux服务器

    基于Python实现批量上传文件或目录到不同的Linux服务器   by:授客 QQ:1033553122 实现功能 1 测试环境 1 使用方法 1 1. 编辑配置文件conf/rootpath_fo ...

  2. Android为TV端助力 最简单的自定义圆点view

    首先创建一个选择器,用来判断圆点状态,可以根本自己的需求改 <selector xmlns:android="http://schemas.android.com/apk/res/an ...

  3. listview reclyerview上下拉刷新

    x写控件挺麻烦的,因为有很多细节要处理好,列表控件使用太频繁了,网上也各种自定义的方法,一般的listview自定义肯定会联想到加个头部,然后监听事件加动画,其实方式很多种,今天记录的方式是另外一种方 ...

  4. OpenVDB for Mitsuba

    https://github.com/zhoub/mitsuba-vdb

  5. Android包管理机制(二)PackageInstaller安装APK

    前言 在本系列上一篇文章Android包管理机制(一)PackageInstaller的初始化中我们学习了PackageInstaller是如何初始化的,这一篇文章我们接着学习PackageInsta ...

  6. Nginx 负载均衡4种模式

    转载请标明出处:http://blog.csdn.net/zhaoyanjun6/article/details/80541464 本文出自[赵彦军的博客] 4 种负载均衡算法 upstream 支持 ...

  7. Git忽略规则及.gitignore规则不生效的解决办法(转)

    在git中如果想忽略掉某个文件,不让这个文件提交到版本库中,可以使用修改根目录中 .gitignore 文件的方法(如无,则需自己手工建立此文件).这个文件每一行保存了一个匹配的规则例如: 1 2 3 ...

  8. cmd输出控制台传递的参数

    public class Test2{ public static void main(String[] args){ System.out.println(args[0]); System.out. ...

  9. ASP.NET -- WebForm -- 缓存Cache的使用

    ASP.NET -- WebForm --  缓存Cache的使用 把数据从数据库或文件中读取出来,放在内存中,后面的用户直接从内存中取数据,速度快.适用于经常被查询.但不经常变动的数据. 1. Te ...

  10. 【2018.05.09 Python学习及实践】个人项目中使用的Python库备忘-持续更新

    科研中无论是使用C/C++.Python.Matlab,如果能找到合适的库可谓是事半功倍: 有时候忙活半天才发现本身就有成熟的库可用,自己实现的在功能.性能.安全性上都远远不及,虽然锻炼了能力,但存在 ...