2022年

2022年发布的文章
  • GCC编译java

    java 语言是一门面向对象的编程语言,与 C++ 很相似,但是减少了很多相对来说比较难以理解的东西,例如操作符的重载、多继承、自动的强制类型转换等。java 语言中不使用指针,而是使用引用的方式,并且可以帮助我们自动管理内存,所以学习 java 语言会相对容易。
     
    java 语言和其他语言是有区别的,它对每个平台都有两种完全不同的目标代码格式,一种是 java 字节码(bytecode)格式的目标文件,可以在 java 的虚拟机中运行;另一种是可以将 java 字节码作为输入,生成本地的可执行的目标代码。
     
    我们今天需要了解的是 GCC 中的 java 编译器 gcj,为什么要发明出来这样一种编译器呢?可以总结出以下几点:

    1. 传统的 JVM(java 虚拟机)太慢了,因为它解释的是 class 文件中的 bytecode。
    2. 为了优化性能,引入JIT(Just-Time),JIT会分析代码,找出那些反复被调用到一定次数的方法和函数,然后直接把这个方法处理成汇编machine code,以后就可以直接运行机器码。
    3. 传统的 java 还有一个问题,文件部署很麻烦,需要有很多个 jar 文件,而不是一个可执行文件。并且 java 需要一个很大的运行环境,还有就是java 和 C/C++ 之间调用的时候很慢。

     
    与 java 程序相关的文件类型:

    • .a:库文件,包括静态链接的目标文件
    • .class:目标文件,包括可由 Java 虚拟机执行的字节码
    • .java:Java源文件
    • .o:二进制目标文件,格式和链接程序相符
    • .s:汇编语言源代码
    • .so:共享库,包含动态链接的目标文件

    1) 编译java程序

    编写一个简单的 Java 程序,代码展示如下:

    /*test.java*/
    public class Hello
    {
         public static void main(String arg[]){
         System.out.println(“hello world”);
         }
    }

    编译 java 程序使用如下命令:

    gcj –main=test -Wall test.java -o test

    使用“--main”选项是告诉程序必须使用 hello 类中的 main() 方法作为程序的起点。-o选项是为了指定生成文件的名字。如果不指定,编译器默认生成文件名为a.out。因为文件是二进制的可执行文件,所以文件的名称可以任意指定。

    java 语言允许所有的类都包含自己的主方法 main(),包含 main() 方法的 Java 类都可以运行。但在处理可执行程序时,必须明确指出独立的起点。

    2) 单一的源文件到类文件

    gcj 编译器可以编译 java 源文件生成 .class 文件,也就是类文件,我们都知道类文件可以在 java 虚拟机上运行。使用上面的 test.java 文件生成类文件test.class,编译命令如下:

    gcj -c -Wall test.java

    注意:编译时-o-c选项不要连用,这样可以保证输出的 .class 和输入的 .java 文件的文件名是相同的。写的类中必须包含public static void main() 方法,这样虚拟机可以使用下面的命令运行这个程序。

    在 java 虚拟机中运行程序使用命令如下:

    gij test

    在虚拟机中运行的时候要注意,文件不用添加 .class 后缀名,直接使用 gij 加上类文件的文件名就可以。

    3) 从源文件到二进制可执行文件的编译过程

    gcj 编译源文件的时候可以使用-c选项禁止链接操作,并且产生二进制目标文件,这样的二进制目标文件既可以链接生成可执行文件,又可以制作静态库文件。使用方式如下:

    gcj -c test.java

    上面的命令可以生成二进制目标文件 test.o,同时也可以使用-o选项指定生成的可执行文件的名字,使用如下:

    gcj -c test.java -o test

    4) java多文件编译

    将多个java源文件编译生成二进制可执行文件时,要分别编译每一个源文件,然后通过指出包含 main() 方法的源文件,最终将它们链接生成一个可执行文件。实例:

    /*Hello.java*/
    public class Hello{
         public static void main(String arg[]){
         SayHello hello = new SayHello();
         hello.add(“ni”);
         hello.add(“hao”);
         hello.add(“ma”);
         Say say = new Say();
         say.speak();
        }
    }
    /*SayHello.java*/
    public class SayHello{
         private String str = “”;
         public static void add(String newworld){
         if(str.length > 0)
                str+=” ”;
         str += newworld;
        }
    public String toString(){
         return (str);
        }
    }
    /*Say.java*/
    public class Say{
         private String string;
         Say(String str){
         string = str
        }
    public void speak(){
           System.out.println(string);
        }
    }

    我们通常使用的方法是将所有的源文件编译成目标文件,然后链接生成可执行文件,具体的命令使用如下:

    gcj -c Hello.java
    gcj -c SayHello.java
    gcj -c Say.java
    gcj –main=Hello Hello.o SayHello.o Say.o -o sayhello

    通过执行上述的命令可以得到最终目标文件 sayhello,当然这些命令也可以组合成一条命令来实现,使用如下:

    gcj –main=Hello Hello.java SayHello Say.java -o sayhello

    注意:无论使用这两种方法中的哪一种,都不要忘指定包含 main() 方法的程序的类名。

更多...

加载中...