fffffffffffff

本文详细介绍Ant自动化构建工具的配置及使用方法,包括环境搭建、构建文件编写等,并通过实例演示如何利用Ant完成编译、打包及部署等工作。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

构建ant环境
要使用ant首先要构建一个ant环境,步骤很简单:
1),安装jdk,设置JAVA_HOME ,PATH ,CLASS_PATH(这些应该是看这篇文章的人应该知道的)
2),下载ant 地址http://www.apache.org/找一个你喜欢的版本,或者干脆最新的版本
3),解压ant 你得到的是一个压缩包,解压缩它,并把它放在一个尽量简单的目录,例如D:\ant-1.6
4),设置ANT_HOME PATH中添加ANT_HOME目录下的bin目录
5),测试一下你的设置,开始-->运行-->cmd进入命令行-->键入 ant 回车,如果看到
Buildfile: build.xml does not exist!
Build failed
那么恭喜你你已经完成ant的设置


Ant的构建文件
当开始一个新的项目时,首先应该编写Ant构建文件。构建文件定义了构建过程,并被团队开发

中每个人使用。Ant构建文件默认命名为build.xml,也可以取其他的名字。只不过在运行的时候

把这个命名当作参数传给Ant。构建文件可以放在任何的位置。一般做法是放在项目顶层目录中

,这样可以保持项目的简洁和清晰。下面是一个典型的项目层次结构。
(1) src存放文件。
(2) class存放编译后的文件。
(3) lib存放第三方JAR包。
(4) dist存放打包,发布以后的代码。
Ant构建文件是XML文件。每个构建文件定义一个唯一的项目(Project元素)。每个项目下可以定

义很多目标(target元素),这些目标之间可以有依赖关系。当执行这类目标时,需要执行他们所

依赖的目标。
每个目标中可以定义多个任务,目标中还定义了所要执行的任务序列。Ant在构建目标时必须调

用所定义的任务。任务定义了Ant实际执行的命令。Ant中的任务可以为3类。
(1) 核心任务。核心任务是Ant自带的任务。
(2) 可选任务。可选任务实来自第三方的任务,因此需要一个附加的JAR文件。
(3) 用户自定义的任务。用户自定义的任务实用户自己开发的任务。
1.<project>标签
每个构建文件对应一个项目。<project>标签时构建文件的根标签。它可以有多个内在属性,

就如代码中所示,其各个属性的含义分别如下。
(1) default表示默认的运行目标,这个属性是必须的。
(2) basedir表示项目的基准目录。
(3) name表示项目名。
(4) description表示项目的描述。
每个构建文件都对应于一个项目,但是大型项目经常包含大量的子项目,每一个子项目都可以有

自己的构建文件。

2.<target>标签
一个项目标签下可以有一个或多个target标签。一个target标签可以依赖其他的target标签。例

如,有一个target用于编译程序,另一个target用于声称可执行文件。在生成可执行文件之前必

须先编译该文件,因策可执行文件的target依赖于编译程序的target。Target的所有属性如下。
(1).name表示标明,这个属性是必须的。
(2).depends表示依赖的目标。
(3)if表示仅当属性设置时才执行。
(4)unless表示当属性没有设置时才执行。
(5)description表示项目的描述。
Ant的depends属性指定了target的执行顺序。Ant会依照depends属性中target出现顺序依次执行

每个target。在执行之前,首先需要执行它所依赖的target。程序中的名为run的target的

depends属性compile,而名为compile的target的depends属性是prepare,所以这几个target执

行的顺序是prepare->compile->run。
一个target只能被执行一次,即使有多个target依赖于它。如果没有if或unless属性,target总

会被执行。

3.<mkdir>标签
该标签用于创建一个目录,它有一个属性dir用来指定所创建的目录名,其代码如下:
<mkdir dir=”${class.root}”/>
通过以上代码就创建了一个目录,这个目录已经被前面的property标签所指定。

4<jar>标签
该标签用来生成一个JAR文件,其属性如下。
(1) destfile表示JAR文件名。
(2) basedir表示被归档的文件名。
(3) includes表示别归档的文件模式。
(4) exchudes表示被排除的文件模式。

5.<javac标签>
该标签用于编译一个或一组java文件,其属性如下。
(1).srcdir表示源程序的目录。
(2).destdir表示class文件的输出目录。
(3).include表示被编译的文件的模式。
(4).excludes表示被排除的文件的模式。
(5).classpath表示所使用的类路径。
(6).debug表示包含的调试信息。
(7).optimize表示是否使用优化。
(8).verbose 表示提供详细的输出信息。
(9).fileonerror表示当碰到错误就自动停止。

6.<java>标签
该标签用来执行编译生成的.class文件,其属性如下。
(1).classname 表示将执行的类名。
(2).jar表示包含该类的JAR文件名。
(3).classpath所表示用到的类路径。
(4).fork表示在一个新的虚拟机中运行该类。
(5).failonerror表示当出现错误时自动停止。
(6).output 表示输出文件。
(7).append表示追加或者覆盖默认文件。

7.<delete>标签
该标签用于删除一个文件或一组文件,去属性如下。
(1)/file表示要删除的文件。
(2).dir表示要删除的目录。
(3).includeEmptyDirs 表示指定是否要删除空目录,默认值是删除。
(4).failonerror 表示指定当碰到错误是否停止,默认值是自动停止。
(5).verbose表示指定是否列出所删除的文件,默认值为不列出。

8.<copy>标签
该标签用于文件或文件集的拷贝,其属性如下。
(1).file 表示源文件。
(2).tofile 表示目标文件。
(3).todir 表示目标目录。
(4).overwrite 表示指定是否覆盖目标文件,默认值是不覆盖。
(5).includeEmptyDirs 表示制定是否拷贝空目录,默认值为拷贝。
(6).failonerror 表示指定如目标没有发现是否自动停止,默认值是停止。
(7).verbose 表示制定是否显示详细信息,默认值不显示。

Ant的数据类型
在构建文件中为了标识文件或文件组,经常需要使用数据类型。数据类型包含在

org.apache.tool.ant.types包中。下面镜简单介绍构建文件中一些常用的数据类型。

1. argument 类型
由Ant构建文件调用的程序,可以通过<arg>元素向其传递命令行参数,如apply,exec和java任

务均可接受嵌套<arg>元素,可以为各自的过程调用指定参数。以下是<arg>的所有属性。
(1).values 是一个命令参数。如果参数种有空格,但又想将它作为单独一个值,则使用此属性


(2).file表示一个参数的文件名。在构建文件中,此文件名相对于当前的工作目录。
(3).line表示用空格分隔的多个参数列表。
(4).path表示路径。

2.ervironment 类型
由Ant构建文件调用的外部命令或程序,<env>元素制定了哪些环境变量要传递给正在执行的系

统命令,<env>元素可以接受以下属性。
(1).file表示环境变量值得文件名。此文件名要被转换位一个绝对路径。
(2).path表示环境变量的路径。Ant会将它转换为一个本地约定。
(3).value 表示环境变量的一个直接变量。
(4).key 表示环境变量名。
注意 file path 或 value只能取一个。

3.filelist类型
Filelist 是一个支持命名的文件列表的数据类型,包含在一个filelist类型中的文件不一定是

存在的文件。以下是其所有的属性。
(1).dir是用于计算绝对文件名的目录。
(2).files 是用逗号分隔的文件名列表。
(3).refid 是对某处定义的一个<filelist>的引用。
注意 dir 和 files 都是必要的,除非指定了refid(这种情况下,dir和files都不允许使用)。

4.fileset类型
Fileset 数据类型定义了一组文件,并通常表示为<fileset>元素。不过,许多ant任务构建成了

隐式的fileset,这说明他们支持所有的fileset属性和嵌套元素。以下为fileset 的属性列表。
(1).dir表示fileset 的基目录。
(2).casesensitive的值如果为false,那么匹配文件名时,fileset不是区分大小写的,其默认

值为true.
(3).defaultexcludes 用来确定是否使用默认的排除模式,默认为true。
(4).excludes 是用逗号分隔的需要派出的文件模式列表。
(5).excludesfile 表示每行包含一个排除模式的文件的文件名。
(6).includes 是用逗号分隔的,需要包含的文件模式列表。
(7).includesfile 表示每行包括一个包含模式的文件名。

5.patternset 类型
Fileset 是对文件的分组,而patternset是对模式的分组,他们是紧密相关的概念。

<patternset>支持4个属性:includes excludex includexfile 和 excludesfile,与fileset相

同。Patternset 还允许以下嵌套元素:include,exclude,includefile 和 excludesfile.

6.filterset 类型
Filterset定义了一组过滤器,这些过滤器将在文件移动或复制时完成文件的文本替换。
主要属性如下:
(1).begintoken 表示嵌套过滤器所搜索的记号,这是标识其开始的字符串。
(2).endtoken表示嵌套过滤器所搜索的记号这是标识其结束的字符串。
(3).id是过滤器的唯一标志符。
(4).refid是对构建文件中某处定义一个过滤器的引用。

7.Path类型
Path元素用来表示一个类路径,不过它还可以用于表示其他的路径。在用作揖个属性时,路经中

的各项用分号或冒号隔开。在构建的时候,此分隔符将代替当前平台中所有的路径分隔符,其拥

有的属性如下。
(1).location 表示一个文件或目录。Ant在内部将此扩展为一个绝对路径。
(2).refid 是对当前构建文件中某处定义的一个path的引用。
(3).path表示一个文件或路径名列表。

8.mapper类型
Mapper类型定义了一组输入文件和一组输出文件间的关系,其属性如下。
(1).classname 表示实现mapper类的类名。当内置mapper不满足要求时,用于创建定制mapper.
(2).classpath表示查找一个定制mapper时所用的类型路径。
(3).classpathref是对某处定义的一个类路径的引用。
(4).from属性的含义取决于所用的mapper.
(5).to属性的含义取决于所用的mapper.
(6).type属性的取值为identity,flatten glob merge regexp 其中之一,它定义了要是用的

内置mapper的类型。

就像每个语言都有HelloWorld一样,一个最简单的应用能让人感受一下Ant
1,首先你要知道你要干什么,我现在想做的事情是:
编写一些程序
编译它们
把它打包成jar包
把他们放在应该放置的地方
运行它们
这里为了简单起见只写一个程序,就是HelloWorld.java程序代码如下:
package test.ant;
public class HelloWorld{
public static void main(String[] args){
System.out.println("Hello world1");
}
};

2,为了达到上边的目的,你可以手动的用javac 、copy 、jar、java来完成,但是考虑一下如果你有成百上千个类,在多次调试,部署的时候,一次次的javac 、copy、jar、java那将是一份辛苦的工作。现在看看ant怎么优雅的完成它们。

要运行ant需要有一个build.xml虽然不一定要叫这个名字,但是建议你这么做
下边就是一个完整的build.xml,然后我们来详细的解释每一句
<?xml version="1.0" encoding="UTF-8" ?>
<project name="HelloWorld" default="run" basedir=".">
<property name="src" value="src"/>
<property name="dest" value="classes"/>
<property name="hello_jar" value="hello1.jar"/>
<target name="init">
<mkdir dir="${dest}"/>
</target>
<target name="compile" depends="init">
<javac srcdir="${src}" destdir="${dest}"/>
</target>
<target name="build" depends="compile">
<jar jarfile="${hello_jar}" basedir="${dest}"/>
</target>
<target name="run" depends="build">
<java classname="test.ant.HelloWorld" classpath="${hello_jar}"/>
</target>
<target name="clean">
<delete dir="${dest}" />
<delete file="${hello_jar}" />
</target>
<target name="rerun" depends="clean,run">
<ant target="clean" />
<ant target="run" />
</target>
</project>

解释:
<?xml version="1.0" encoding="UTF-8" ?>
build.xml中的第一句话,没有实际的意义

<project name="HelloWorld" default="run" basedir=".">
</project>
ant的所有内容必须包含在这个里边,name是你给它取的名字,basedir故名思意就是工作的根目录 .代表当前目录。default代表默认要做的事情。

<property name="src" value="src"/>
类似程序中的变量,为什么这么做想一下变量的作用

<target name="compile" depends="init">
<javac srcdir="${src}" destdir="${dest}"/>
</target>
把你想做的每一件事情写成一个target ,它有一个名字,depends是它所依赖的target,在执行这个target 例如这里的compile之前ant会先检查init是否曾经被执行过,如果执行过则直接直接执行compile,如果没有则会先执行它依赖的 target例如这里的init,然后在执行这个target

如我们的计划
编译:
<target name="compile" depends="init">
<javac srcdir="${src}" destdir="${dest}"/>
</target>

做jar包:
<target name="build" depends="compile">
<jar jarfile="${hello_jar}" basedir="${dest}"/>
</target>
运行:
<target name="run" depends="build">
<java classname="test.ant.HelloWorld" classpath="${hello_jar}"/>
</target>
为了不用拷贝,我们可以在最开始定义好目标文件夹,这样ant直接把结果就放在目标文件夹中了
新建文件夹:
<target name="init">
<mkdir dir="${dest}"/>
</target>
为了更多一点的功能体现,又加入了两个target
删除生成的文件
<target name="clean">
<delete dir="${dest}" />
<delete file="${hello_jar}" />
</target>
再次运行,这里显示了如何在一个target里边调用其他的target
<target name="rerun" depends="clean,run">
<ant target="clean" />
<ant target="run" />
</target>

Build文件
<?xml version='1.0' encoding='gb2312'?> <!--申明版本信息-->
<project name="TryAnt" default="deploy">

<!--properties-->
<property name="src.dir" value="src" />
<property name="class.dir" value="bin" />
<property name="lib.dir" value="lib" />
<property name="dist.dir" value="dist" />
<property name="doc.dir" value="doc" />
<property name="tomcat.dir" value="F:\apache-tomcat-6.0.18" />
<property name="deploy.dir" value="${tomcat.dir}/webapps/myweb" />
<property name="deploylib.dir" value="${deploy.dir}/web-inf/lib" />

<!--定义classPath-->
<path id="master-classpath">
<fileset file="${lib.dir}/*.jar" />
<pathelement path="${class.dir}" />
</path>

<!--初始化文件夹init--><!--如果不存在就自动会创建否则不创建-->
<target name="init">
<mkdir dir="${class.dir}" />
<mkdir dir="${lib.dir}" />
<mkdir dir="${dist.dir}" />
<mkdir dir="${doc.dir}" />
</target>

<!--编译-->
<target name="compile" depends="init" description="注释">
<javac srcdir="${src.dir}" destdir="${class.dir}" target="1.5">
<classpath refid="master-classpath" />
</javac>
</target>
<!--打包jar-->
<target name="pack" depends="compile" description="this is make jar">
<jar destfile="${dist.dir}/study.jar" basedir="${class.dir}" />
</target>

<!-- 发布 从dist.dir中拷贝jar包到指定的deploylib.dir的路径下-->
<target name="deploy" depends="pack">
<copy todir="${deploylib.dir}">
<fileset dir="${dist.dir}" />
</copy>
</target>

<!--输出文档-->
<target name="doc" depends="pack">
<javadoc destdir="${doc.dir}" author="true" version="true" use="true" windowtitle="TryAntAPI">
<packageset dir="${src.dir}" />
</javadoc>
</target>
</project>
def send_udp_broadcast(self, source_ip): """以指定的源 IP 和随机源端口发送 UDP 广播""" data_to_send = bytes.fromhex('fffffffffffff8e43bdc0032000158494e4a455853500000000000000000') # UDP负载 source_port = random.randint(1024, 65535) # 随机选择源端口 sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) sock.setsockopt(socket.SOL_SOCKET, socket.SO_BROADCAST, 1) # 允许广播 sock.bind((source_ip, 0)) # 绑定到源 IP,端口设为 0 让系统选择 sock.settimeout(2) # 设置超时为 2 秒,以给设备更多时间回复 try: print(f"Sending broadcast packet from {source_ip}:{source_port} to 255.255.255.255:{self.target_port}") sock.sendto(data_to_send, ('255.255.255.255', self.target_port)) # 发送广播数据包 receiver = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) receiver.setsockopt(socket.SOL_SOCKET, socket.SO_BROADCAST, 1) # 允许广播 receiver.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) # 允许端口复用 receiver.bind(('0.0.0.0', 541)) # 绑定所有网络接口 receiver.settimeout(2) # 接收响应 while True: try: response, addr = receiver.recvfrom(1024) response_str = response.decode(errors='ignore') # 尝试解码响应 print(f"Received response from {addr[0]}:{addr[1]} - it appears to be open!") print(f"Response data: {response_str}") # 打印响应内容 if addr[0] not in self.open_hosts: self.open_hosts.append(addr[0]) # 保存响应的 IP 地址 except receiver.timeout: print("No more responses received, exiting the receive loop.") # 打印超时消息 break # 超时,退出循环 except Exception as e: print(f"Error during broadcast: {e}") finally: sock.close() receiver.close()执行这一段代码会报一个异常错误catching classes that do not inherit from BaseException is not allowedError during broadcast
03-21
分析这段log,给出解决方案 06-03 23:35:42.727 12302 12302 F DEBUG : pid: 5435, tid: 9809, name: applyRouting >>> com.android.nfc <<< 06-03 23:35:42.727 12302 12302 F DEBUG : uid: 1027 06-03 23:35:42.727 12302 12302 F DEBUG : tagged_addr_ctrl: 0000000000000001 (PR_TAGGED_ADDR_ENABLE) 06-03 23:35:42.727 12302 12302 F DEBUG : signal 6 (SIGABRT), code -1 (SI_QUEUE), fault addr -------- 06-03 23:35:42.727 12302 12302 F DEBUG : Abort message: 'JNI FatalError called: applyRouting' 06-03 23:35:42.727 12302 12302 F DEBUG : x0 0000000000000000 x1 0000000000002651 x2 0000000000000006 x3 000000727ee32210 06-03 23:35:42.727 12302 12302 F DEBUG : x4 71446b607360451f x5 71446b607360451f x6 71446b607360451f x7 7f7f7f7f7f7f7f7f 06-03 23:35:42.727 12302 12302 F DEBUG : x8 00000000000000f0 x9 000000752b052418 x10 0000000000000001 x11 000000752b0a4070 06-03 23:35:42.727 12302 12302 F DEBUG : x12 000000727ee30c28 x13 0000000000000039 x14 fffffffffffff06d x15 000000727ee30cf0 06-03 23:35:42.727 12302 12302 F DEBUG : x16 000000752b10dfe8 x17 000000752b0f7ac0 x18 000000727dd9c008 x19 000000000000153b 06-03 23:35:42.727 12302 12302 F DEBUG : x20 0000000000002651 x21 00000000ffffffff x22 b4000072c33700d0 x23 000000727ee32a80 06-03 23:35:42.727 12302 12302 F DEBUG : x24 000000000000002f x25 000000005c041798 x26 b4000073c332a870 x27 b400007353344490 06-03 23:35:42.727 12302 12302 F DEBUG : x28 0000000000110000 x29 000000727ee32290 06-03 23:35:42.727 12302 12302 F DEBUG : lr 000000752b08c998 sp 000000727ee321f0 pc 000000752b08c9c4 pst 0000000000001000 06-03 23:35:42.727 12302 12302 F DEBUG : 15 total frames 06-03 23:35:42.727 12302 12302 F DEBUG : backtrace: 06-03 23:35:42.727 12302 12302 F DEBUG : #00 pc 000000000005d9c4 /apex/com.android.runtime/lib64/bionic/libc.so (abort+164) (BuildId: 41b260160dbd4aae2b04f90d937b7e4f) 06-03 23:35:42.727 12302 12302 F DEBUG : #01 pc 0000000000014abc /system/lib64/libbase.so (android::base::DefaultAborter(char const*)+12) (BuildId: 60e5261000aaaa4206a16ca816e8d913) 06-03 23:35:42.727 12302 12302 F DEBUG : #02 pc 00000000000a53ec /system/lib64/libbinder.so (android::base::SetAborter(std::__1::function<void (char const*)>&&)::$_0::__invoke(char const*) (.__uniq.304115623023563947864004479300660821567)+60) (BuildId: 22440d9a55c014f3ddc28a80f336d7c7) 06-03 23:35:42.727 12302 12302 F DEBUG : #03 pc 00000000000156d0 /apex/com.android.art/lib64/libbase.so (android::base::LogMessage::~LogMessage()+516) (BuildId: 1470f61c05962eb04fafe76bd58bf664) 06-03 23:35:42.727 12302 12302 F DEBUG : #04 pc 0000000000890c08 /apex/com.android.art/lib64/libart.so (art::JNI<false>::FatalError(_JNIEnv*, char const*)+196) (BuildId: c35c9ebf7bb06435e4b31977d87bd5d5) 06-03 23:35:42.727 12302 12302 F DEBUG : #05 pc 000000000013c03c /system_ext/lib64/libstnfc_nci_jni.so (android::stNfcManager_doAbort(_JNIEnv*, _jobject*, _jstring*)+60) (BuildId: bc256ab9c81d664a26e98b7c906517da) 06-03 23:35:42.727 12302 12302 F DEBUG : #06 pc 00000000000990bc /system/framework/arm64/boot.oat (art_jni_trampoline+124) (BuildId: 99295d3fa7cfe561936e19bb043a43bf66b8d323) 06-03 23:35:42.727 12302 12302 F DEBUG : #07 pc 0000000000044460 /data/dalvik-cache/arm64/system_ext@priv-app@Nfc_st@Nfc_st.apk@classes.dex (com.android.nfc.NfcService$WatchDogThread.run+544) 06-03 23:35:42.727 12302 12302 F DEBUG : #08 pc 000000000036d574 /apex/com.android.art/lib64/libart.so (art_quick_invoke_stub+612) (BuildId: c35c9ebf7bb06435e4b31977d87bd5d5) 06-03 23:35:42.727 12302 12302 F DEBUG : #09 pc 0000000000358bc0 /apex/com.android.art/lib64/libart.so (art::ArtMethod::Invoke(art::Thread*, unsigned int*, unsigned int, art::JValue*, char const*)+132) (BuildId: c35c9ebf7bb06435e4b31977d87bd5d5) 06-03 23:35:42.727 12302 12302 F DEBUG : #10 pc 0000000000944608 /apex/com.android.art/lib64/libart.so (art::detail::ShortyTraits<(char)86>::Type art::ArtMethod::InvokeInstance<(char)86>(art::Thread*, art::ObjPtr<art::mirror::Object>, art::detail::ShortyTraits<>::Type...)+60) (BuildId: c35c9ebf7bb06435e4b31977d87bd5d5) 06-03 23:35:42.727 12302 12302 F DEBUG : #11 pc 0000000000625d24 /apex/com.android.art/lib64/libart.so (art::Thread::CreateCallback(void*)+1344) (BuildId: c35c9ebf7bb06435e4b31977d87bd5d5) 06-03 23:35:42.727 12302 12302 F DEBUG : #12 pc 00000000006257d4 /apex/com.android.art/lib64/libart.so (art::Thread::CreateCallbackWithUffdGc(void*)+8) (BuildId: c35c9ebf7bb06435e4b31977d87bd5d5) 06-03 23:35:42.727 12302 12302 F DEBUG : #13 pc 0000000000070098 /apex/com.android.runtime/lib64/bionic/libc.so (__pthread_start(void*)+200) (BuildId: 41b260160dbd4aae2b04f90d937b7e4f) 06-03 23:35:42.727 12302 12302 F DEBUG : #14 pc 0000000000061410 /apex/com.android.runtime/lib64/bionic/libc.so (__start_thread+64) (BuildId: 41b260160dbd4aae2b04f90d937b7e4f)
06-10
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值