00-JAVA基础-动态编译

动态编译

JAVA 6 引入了动态编译机制。Java 动态编译是指在运行时将Java源代码编译成可执行的字节码。这通常使用Java的内置编译器API javax.tools.JavaCompiler 来实现。

动态编译的应用场景

  • 可以做一个浏览器编写java代码,上传服务器编译和运行的在线测评系统
  • 服务器动态加载某些类文件进行编译

动态编译的两种实现方式

  • JAVA6之前,可以通过Runtime调用javac,启动新的进程去操作

    Runtime run = Runtime.getRuntime();
    Process process = run.exec("javac D:/myjava/demo/HelloWorld.java");
    
  • JAVA6之后,可以使用JavaCompiler实现

      public static in compileFile(String sourceFile){
          JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
          int result = compiler.run(null,null,null,sourceFile);
          System.out.println(result == 0 ? "编译通过" : "编译失败")
          return result;
      }
    

compiler.run(null,null,null,sourceFile)参数说明:

第一个参数:为java编译器提供参数
第二个参数:得到Java编译器的输出信息
第三个参数:接受编译器的错误信息
第四个参数:可变参数(是一个String[]数组)能传入一个或多个Java源文件
返回值:0 表示编译成 ,非0 表示编译失败

使用runtime.exec方法编译

package demo2;

import java.io.*;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLClassLoader;

/**
 * 使用Runtime编译,通过反射调用main方法
 *
 * @author Anna.
 * @date 2024/4/4 11:13
 */
public class DyanmaicDemo {

  public static void main(String[] args) throws Exception {
    String path = DyanmaicDemo.class.getResource("").getPath();
    String javaName = "Hello";

    // 创建文件
    File tempFile = createFile(path, javaName);

    // 使用Runtime 编译
    compilation(tempFile);

    // 反射执行文件
    extracted(path, javaName);
  }

  /**
   * 创建文件
   *
   * @param path
   * @param javaName
   * @return java.io.File
   * @author Anna.
   * @date 2024/4/4 13:05
   */
  private static File createFile(String path, String javaName) throws IOException {
    // 创建编译内容
    StringBuffer sb = new StringBuffer();
    sb.append("public class ").append(javaName).append("{")
            .append("public static void main(String[] args){")
            .append("System.out.println(\"hello world !!!\");")
            .append("}")
            .append("}");

    // 写入临时文件
    File tempFile = new File(path + javaName + ".java");
    FileWriter fileWriter = new FileWriter(tempFile);
    fileWriter.write(sb.toString());
    fileWriter.close();
    return tempFile;
  }

  /**
   * 编译
   *
   * @param tempFile
   * @return void
   * @author Anna.
   * @date 2024/4/4 13:05
   */
  private static void compilation(File tempFile) throws IOException, InterruptedException {
    String replace = tempFile.getAbsolutePath().replace(tempFile.getName(), "");
    replace = replace.substring(0, replace.length() - 1);

    String str = "javac " + tempFile.getAbsolutePath();
    Runtime runtime = Runtime.getRuntime();

    Process process = runtime.exec(str);

    // 读取命令的标准输出
    BufferedReader reader = new BufferedReader(new InputStreamReader(process.getInputStream()));
    String line;
    while ((line = reader.readLine()) != null) {
      System.out.println(line);
    }

    // 读取命令的错误输出
    BufferedReader errorReader = new BufferedReader(new InputStreamReader(process.getErrorStream()));
    while ((line = errorReader.readLine()) != null) {
      System.err.println(line);
    }

    // 等待进程结束并获取退出值
    int exitValue = process.waitFor();
    if (exitValue == 0) {
      System.out.println("Compilation is successful");
    } else {
      System.out.println("Compilation Failed");
    }
  }

  /**
   * 反射执行main方法
   *
   * @param path
   * @param javaName
   * @return void
   * @author Anna.
   * @date 2024/4/4 13:05
   */
  private static void extracted(String path, String javaName) throws MalformedURLException, ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException {
    URLClassLoader urlClassLoader = new URLClassLoader(new URL[]{new URL("file:/" + path)});
    Class<?> clazz = urlClassLoader.loadClass(javaName);
    Method mainMethod = clazz.getMethod("main", String[].class);
    // 注意:由于可变参数是JDK5之后才有的,下面代码如果new String[]{"1","2"}强制在转换才Object,则会被编译成 mainMethod.invoke(null,"1","2"),从而导致找不到方法。
    // 因此,如果传参则徐亚加上(Object),避免这个问题
    mainMethod.invoke(null, (Object) new String[]{"1", "2"});
  }

}

执行结果

在这里插入图片描述

使用JavaCompiler实现

package demo1;

import javax.tools.*;
import java.io.File;
import java.io.IOException;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.net.MalformedURLException;
import java.net.URI;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.Arrays;
import java.util.Collections;

/**
 * 使用JavaCompiler编译,通过反射调用main方法
 *
 * @author Anna.
 * @date 2024/4/4 10:55
 */
public class DyanmicDemo1 {
  public static void main(String[] args) throws Exception {
    // 创建编译内容
    String javaName = "Hello";

    // 获取JavaFileObject
    JavaFileObject source = getJavaFileObject(javaName);

    // 编译
    compiler(source);

    // 执行
    extracted(javaName);
  }

  /**
   * 获取JavaFileObject
   *
   * @param javaName
   * @return javax.tools.JavaFileObject
   * @author Anna.
   * @date 2024/4/4 13:56
   */
  private static JavaFileObject getJavaFileObject(String javaName) {
    StringBuffer sb = new StringBuffer();
    sb.append("public class ").append(javaName).append("{")
            .append("public static void main(String[] args){")
            .append("System.out.println(\"hello world !!!\");")
            .append("}")
            .append("}");

    // 创建一个Java源代码文件 string:///是一个特殊的URI协议,用于表示源代码内容直接来自一个字符串,而不是来自文件系统中的一个文件。 好处是,你可以完全在内存中处理源代码,无需涉及文件系统的I/O操作。
    JavaFileObject source = new SimpleJavaFileObject(URI.create("string:///" + javaName + ".java"), JavaFileObject.Kind.SOURCE) {
      @Override
      public CharSequence getCharContent(boolean ignoreEncodingErrors) {
        return sb.toString();
      }
    };
    return source;
  }

  /**
   * 编译
   *
   * @param source
   * @return void
   * @author Anna.
   * @date 2024/4/4 13:49
   */
  private static void compiler(JavaFileObject source) {
    // 获取系统Java编译器
    JavaCompiler systemJavaCompiler = ToolProvider.getSystemJavaCompiler();
    // 创建诊断收集器
    DiagnosticCollector<JavaFileObject> diagnostics = new DiagnosticCollector<>();

    // 创建编译任务
    StandardJavaFileManager fileManager = systemJavaCompiler.getStandardFileManager(diagnostics, null, null);

    // 设置输出目录
    Iterable<? extends File> locations = fileManager.getLocation(StandardLocation.CLASS_OUTPUT);
    try {
      fileManager.setLocation(StandardLocation.CLASS_OUTPUT, Collections.singleton(new File(DyanmicDemo1.class.getClassLoader().getResource("").getPath())));
    } catch (IOException e) {
      e.printStackTrace();
      return;
    }

    Iterable<? extends JavaFileObject> compilationUnits = Arrays.asList(source);
    JavaCompiler.CompilationTask task = systemJavaCompiler.getTask(null, fileManager, diagnostics, null, null, compilationUnits);

    // 执行编译任务
    boolean success = task.call();
    // 关闭文件管理器
    try {
      fileManager.close();
    } catch (IOException e) {
      e.printStackTrace();
    }

    // 处理编译诊断信息
    for (Diagnostic<?> diagnostic : diagnostics.getDiagnostics()) {
      System.out.println(diagnostic);
      System.out.println(diagnostic.getKind() + ": " + diagnostic.getMessage(null));
    }

    System.out.println(success ? "编译通过" : "编译失败");
  }

  /**
   * 反射执行
   *
   * @param javaName
   * @return void
   * @author Anna.
   * @date 2024/4/4 13:50
   */
  private static void extracted(String javaName) throws MalformedURLException, ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException {
    URLClassLoader urlClassLoader = new URLClassLoader(new URL[]{new URL("file:/" + DyanmicDemo1.class.getClassLoader().getResource("").getPath())});
    Class<?> clazz = urlClassLoader.loadClass(javaName);
    Method mainMethod = clazz.getMethod("main", String[].class);
    // 注意:由于可变参数是JDK5之后才有的,下面代码如果new String[]{"1","2"}强制在转换才Object,则会被编译成 mainMethod.invoke(null,"1","2"),从而导致找不到方法。
    // 因此,如果传参则徐亚加上(Object),避免这个问题
    mainMethod.invoke(null, (Object) new String[]{"1", "2"});
  }

}

执行结果

在这里插入图片描述

注意:

Runtime不仅仅是可以用于执行javac,当然可以用来执行其他命令,这里就不进一步说明了

CommandUtil调用系统命令工具类

import org.apache.commons.exec.*;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.OutputStream;

/**
 * 调用系统命令工具类
 *
 * @author Anna.
 * @date 2021/9/8 16:05
 */
public class CommandUtil {
    private static Logger logger = LoggerFactory.getLogger(CommandUtil.class);
    private static final String DEFAULT_CHARSET = "UTF-8";
    private static final Long TIMEOUT = 10000L;

    /**
     * 执行指定命令
     *
     * @param command 命令
     * @return 命令执行完成返回结果
     * @throws RuntimeException 失败时抛出异常,由调用者捕获处理
     */
    public synchronized static String exeCommand(String command) throws RuntimeException {
        try (ByteArrayOutputStream out = new ByteArrayOutputStream()) {
            int exitCode = exeCommand(command, out);
            if (exitCode == 0) {
                logger.info("命令运行成功:" + System.currentTimeMillis());
            } else {
                logger.info("命令运行失败:" + System.currentTimeMillis());
            }
            return out.toString(DEFAULT_CHARSET);
        } catch (Exception e) {
            logger.info(e.getMessage());
            throw new RuntimeException(e.getMessage());
        }
    }

    /**
     * 执行指定命令,输出结果到指定输出流中
     *
     * @param command 命令
     * @param out     执行结果输出流
     * @return 执行结果状态码:执行成功返回0
     * @throws ExecuteException 失败时抛出异常,由调用者捕获处理
     * @throws IOException      失败时抛出异常,由调用者捕获处理
     */
    public synchronized static int exeCommand(String command, OutputStream out) throws ExecuteException, IOException {
        CommandLine commandLine = CommandLine.parse(command);
        PumpStreamHandler pumpStreamHandler = null;
        if (null == out) {
            pumpStreamHandler = new PumpStreamHandler();
        } else {
            pumpStreamHandler = new PumpStreamHandler(out);
        }
        // 设置超时时间为10秒
        ExecuteWatchdog watchdog = new ExecuteWatchdog(TIMEOUT);
        DefaultExecutor executor = new DefaultExecutor();
        executor.setStreamHandler(pumpStreamHandler);
        executor.setWatchdog(watchdog);
        return executor.execute(commandLine);
    }

    public static void main(String[] args) {
        String out = null;
        try {
//            out = CommandUtil.exeCommand("ipconfig");
            out = CommandUtil.exeCommand("D:\\SoftWare\\ffmpeg\\bin\\ffmpeg.exe -y  -i D:/test/1631090617181_191979483992900.mp3  -acodec pcm_s16le -f s16le -ac 1 -ar 16000 D:/test/PCM1631090617181_191979483992900.mp3.pcm");
        } catch (Exception e) {
            e.printStackTrace();
        }
        System.out.println(out);
        //CommandUtil.executeCommand("kill -9 3104");
    }


}

gitee源码

git clone https://gitee.com/dchh/JavaStudyWorkSpaces.git

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/518920.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

我为什么会选择Vim来开发Go项目及Vim IDE安装配置和操作

你好&#xff0c;我是孔令飞&#xff0c;字节跳动云原生资深研发、前腾讯云原生技术专家。《企业级 Go 项目开发实战》、《从零开发企业级 Go 应用》作者&#xff0c;欢迎加入 孔令飞的云原生实战营&#xff0c;助你进阶 Go 云原生高级开发工程师。 作为一名 Golang 开发&…

我的需求分析方法论

或网上看了无数博客文章、技术视频&#xff0c;或购买金装版本技术书籍&#xff0c;看过无数原理原则、各种各样经典方法论&#xff0c;真正在实际开发工作中&#xff0c;本能去遵守和执行的又留下多少呢。 启动一个新系统时&#xff0c;我们可能还会去花些时间遵循这些原理原则…

前端学习之DOM编程-docmument对象、操作DOM对像内容、操作DOM对象属性方式、操作DOM对象的样式

docmument对象 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>document对象</title> </head> <body><div id"container" nameparent><ul name"parent&qu…

k8s CNI Calico 网络模式总结

目录 calico架构图 IPIP模式下的架构图 calico 核心组件 Overlay 网络模式&#xff1a; Pod IP对外暴露 不对外暴露&#xff1a; 实现对外暴露的方法&#xff1a; overlay模式下的网络MTU Iptables & ipvs overlay的主要缺点&#xff1a; Full-mesh Unoverla…

DXP学习003-PCB编辑器的环境参数及电路板参数相关设置

目录 一&#xff0c;dxp的pcb编辑器环境 1&#xff0c;创建新的PCB设计文档 2&#xff0c;PCB编辑器界面 1&#xff09;布线工具栏 2&#xff09;公用工具栏 3&#xff09;层标签栏 ​☀ 3&#xff0c;PCB设计面板 1&#xff09;打开pcb设计面板 4&#xff0c;PCB观察…

重温OKHTTP源码

本文基于OkHttp4.12.0源码分析 官方地址 概括 本篇主要是对okhttp开源库的一个详细解析&#xff0c;包含详细的请求流程分析、各大拦截器的解读等。 使用方法 同步请求&#xff1a;创建一个OKHttpClient对象&#xff0c;一个Request对象&#xff0c;然后利用它们创建一个Ca…

免费微信小程序源码分享~搭起来改一下就可以【创业】

【前言】现在很多人都想做微信小程序创业搞钱&#xff0c;但是苦于不会开发或过高的开发成本只能放弃&#xff0c;下面我收集了几套微信小程序的源码供各位有梦想的同学免费使用~ 这些小程序代码都包含了客户端和管理端&#xff0c;你搭建起来就可以开始创业搞钱了~ 下载链接&a…

PostgreSQL 文章下架 与 热更新和填充可以提升数据库性能

开头还是介绍一下群&#xff0c;如果感兴趣PolarDB ,MongoDB ,MySQL ,PostgreSQL ,Redis, Oceanbase, Sql Server等有问题&#xff0c;有需求都可以加群群内有各大数据库行业大咖&#xff0c;CTO&#xff0c;可以解决你的问题。加群请联系 liuaustin3 &#xff0c;&#xff08;…

4月4日今日预告:printf+scanf+分支循环,if语句,else悬空问题,加油,干干干这篇文章三个小时半了,从愚人节被告知今天就有课程-今日4/3,

今天中午知道要爆肝的C语言的&#xff0c;今天本来作业好多的&#xff1b; 干了&#xff0c;家人们 做一些补充&#xff1a; 一&#xff1a;printf() 参数与占位符对应关系 printf() 参数与占位符是⼀⼀对应关系&#xff0c;如果有 n 个占位符&#xff0c; printf() 的参数…

使用docker-tc对host容器进行限流

docker-tc是一个github开源项目&#xff0c;项目地址是https://github.com/lukaszlach/docker-tc。 运行docker-tc docker run -d \ --name docker-tc \ --network host \ --cap-add NET_ADMIN \ --restart always \ -v /var/run/docker.sock:/var/run/docker.sock \ -v /var…

通过vite创建项目

一、VUE3官网 Vue.js - 渐进式 JavaScript 框架 | Vue.js (vuejs.org) 二、通过Vite创建项目 1、在cmd窗口下&#xff0c;全局安装vite //使用国内镜像源 npm config set registryhttps://registry.npmmirror.com//安装最新版vite npm install -g vitelatest Vite | 下一代…

阿里云、腾讯云、华为云优惠券领取攻略

随着云计算技术的日益成熟和普及&#xff0c;越来越多的企业和个人开始选择使用云服务商来满足自己的数据存储、计算和处理需求。阿里云、腾讯云、华为云作为国内领先的云服务商&#xff0c;提供了丰富多样的云产品和服务。而为了吸引更多用户&#xff0c;它们也时常会推出各种…

4.4学习总结

一.线段树概念 一.定义: 线段树是一种二叉搜索树&#xff0c;而二叉搜索树&#xff0c;首先满足二叉树&#xff0c;即每个结点最多有两颗子树&#xff0c;并且是一颗搜索树&#xff0c;我们要知道&#xff0c;线段树的每个结点都存储了一个区间&#xff0c;也可以理解成一个线…

文件系统监视库(watchdog)

Python Watchdog库是一个用于监视文件系统变化的Python第三方库。以下是关于Watchdog库的详细介绍&#xff1a; 功能&#xff1a;Watchdog库能够监控文件和目录的创建、修改、删除和移动等操作。它通过使用底层原生API&#xff08;如Windows的ReadDirectoryChangesW、Linux 2.6…

Golang学习笔记

Golang学习笔记 安装Golang 来源&#xff1a;linux 安装 golang - 知乎 (zhihu.com) 由于我用的是linux系统&#xff0c;所以本文采用linux的安装方式介绍&#xff0c;如果你使用的是Windows/Mac 也可以看下该文章&#xff0c;或者自己去下列地址进行操作。 Download and in…

react中配置webpack:使用@代表src目录

在vue的项目中可以使用表示src目录&#xff0c;使用该符号表示绝对路径&#xff0c;那么在react中想要使用怎么办呢&#xff1f; 在react中使用表示src目录是需要在webpack中配置的&#xff0c;在核心模块node_modules-》react-scripts-》config-》webpack.config.js中搜索找到…

基于SSM的品牌银饰售卖平台(有报告)。Javaee项目。ssm项目。

演示视频&#xff1a; 基于SSM的品牌银饰售卖平台&#xff08;有报告&#xff09;。Javaee项目。ssm项目。 项目介绍&#xff1a; 采用M&#xff08;model&#xff09;V&#xff08;view&#xff09;C&#xff08;controller&#xff09;三层体系结构&#xff0c;通过Spring …

Windows11下Docker使用记录(一)

Docker使用记录&#xff08;一&#xff09; 简单介绍Docker安装Docker 常用命令Docker 可视化Docker 使用GPU可视化rviz、gazebo 在进行ROS项目开发时&#xff0c;如果只有一台Windows电脑&#xff0c;我们可以考虑使用WSL或Docker来搭建ROS环境。在尝试了两种方式后&#xff0…

代码随想录第31天 | 455.分发饼干 、376. 摆动序列、53. 最大子序和

一、前言 参考文献&#xff1a;代码随想录 今天的内容是贪心算法&#xff0c;这个算法分为两个极端&#xff0c;一个极端是很简单&#xff0c;靠常识就可以解出来&#xff0c;另外一个是&#xff0c;你怎么想也想不出来&#xff0c;只能看题解的那种。 and 对第一天和第二天…

Spring源码解析上

spring源码解析 整体架构 defaultListableBeanFactory xmlBeanDefinitionReader 创建XmlBeanFactory 对资源文件进行加载–Resource 利用LoadBeandefinitions(resource)方法加载配置中的bean loadBeandefinitions加载步骤 doLoadBeanDefinition xml配置模式 validationMode 获…