社区首页 >问答首页 >用开关语句可以无限循环吗?

用开关语句可以无限循环吗?
EN

Stack Overflow用户
提问于 2015-10-23 04:53:06
回答 4查看 437关注 0票数 4

我正在编写一个代码质量工具。我正在扫描源代码和编译类,寻找潜在的无限循环。

我无法想象有一种方式的源代码切换语句可以无限期地循环。我说错了吗?

转换语句编译为lookupswitchtableswitch操作码。出于安全考虑,我需要检查编译类,并且在质量控制程序处理编译类之前允许字节码修改。说过,是否有可能只通过修改类或用汇编程序生成类来无限循环?

我已经处理了所有其他分支说明和陈述。

你的帮助会很感激的。

编辑:结论:

正如我所怀疑的,并根据这里提供的答案,源代码中的switch语句只能向前分支,但是字节码中的任何分支指令都有可能向后跳(假设字节码修改)。

EN

回答 4

Stack Overflow用户

回答已采纳

发布于 2015-10-23 05:13:09

话虽如此,通过修改类或用汇编程序生成类,是否有可能通过使用这些操作码来无限循环呢?

要有一个无限的循环,你必须回到某个地方。如果您修改了字节代码,那么在任何时候添加或更改跳转返回的地方都会发生这种情况。如果不是,它不能是一个循环,无限或其他。

票数 1
EN

Stack Overflow用户

发布于 2015-10-23 23:31:00

有趣的是,您可以使用字节码版本1.6 (50)来完成这一任务,但如果验证失败,则不能使用字节码版本1.7 (51)。此代码(需要ASM5)工作正常,具有无限循环:

代码语言:javascript
代码运行次数:0
复制
import org.objectweb.asm.ClassWriter;
import org.objectweb.asm.Label;
import org.objectweb.asm.MethodVisitor;
import static org.objectweb.asm.Opcodes.*;

public class LookupTest {
    public static void main(String[] args) throws InstantiationException,
            IllegalAccessException, ClassNotFoundException {
        new ClassLoader() {
            @Override
            protected Class<?> findClass(String name)
                    throws ClassNotFoundException {
                ClassWriter cw = new ClassWriter(ClassWriter.COMPUTE_MAXS
                        | ClassWriter.COMPUTE_FRAMES);
                // Create public class extending java.lang.Object
                cw.visit(V1_6, ACC_PUBLIC | ACC_SUPER, name, null,
                        "java/lang/Object", null);
                // Create default constructor
                MethodVisitor mv = cw.visitMethod(ACC_PUBLIC, "<init>", "()V",
                        null, null);
                mv.visitCode();
                // Call superclass constructor (this is required)
                mv.visitVarInsn(ALOAD, 0);
                mv.visitMethodInsn(INVOKESPECIAL, "java/lang/Object", "<init>",
                        "()V", false);
                // Create branch target
                Label target = new Label();
                mv.visitLabel(target);
                // System.out.println("Hello");
                mv.visitFieldInsn(GETSTATIC, "java/lang/System", "out",
                        "Ljava/io/PrintStream;");
                mv.visitLdcInsn("Hello");
                mv.visitMethodInsn(INVOKEVIRTUAL, "java/io/PrintStream",
                        "println", "(Ljava/lang/String;)V", false);
                // switch(0) {
                mv.visitInsn(ICONST_0);
                // default: goto target;
                // }
                mv.visitLookupSwitchInsn(target, new int[0], new Label[0]);
                mv.visitMaxs(-1, -1);
                mv.visitEnd();
                cw.visitEnd();
                byte[] bytes = cw.toByteArray();
                return defineClass(name, bytes, 0, bytes.length);
            }
        }.loadClass("LookupGotoTest").newInstance();
    }
}

但是,如果将V1_6替换为V1_7,则会出现以下错误:

代码语言:javascript
代码运行次数:0
复制
Exception in thread "main" java.lang.VerifyError: Bad instruction
Exception Details:
  Location:
    LookupGotoTest.<init>()V @13: lookupswitch
  Reason:
    Error exists in the bytecode
  Bytecode:
    0x0000000: 2ab7 0008 b200 0e12 10b6 0016 03ab 0000
    0x0000010: ffff fff7 0000 0000                    
  Stackmap Table:
    full_frame(@4,{Object[#2]},{})

    at java.lang.Class.getDeclaredConstructors0(Native Method)
    at java.lang.Class.privateGetDeclaredConstructors(Class.java:2658)
    at java.lang.Class.getConstructor0(Class.java:3062)
    at java.lang.Class.newInstance(Class.java:403)
    at LookupTest.main(LookupTest.java:46)

但是,如果我选择前进跳转并添加goto指令,那么即使使用1.7字节码,它也能正常工作:

代码语言:javascript
代码运行次数:0
复制
Label target2 = new Label();
// switch(0) {
mv.visitInsn(ICONST_0);
// default: goto target2;
// }
mv.visitLookupSwitchInsn(target2, new int[0], new Label[0]);
mv.visitLabel(target2);
// goto target
mv.visitJumpInsn(GOTO, target);

这种差异是由于不同的验证过程造成的:Java1.6之前的Java类没有StackMapTable,由类型推理验证,而版本1.7或更高的类由类型检查验证,类型检查对包括查找开关在内的各个指令都有单独的严格规则。

目前,我还不清楚在1.7+中是否真的不允许这样的指令,或者ASM只是生成了不正确的StackMapTable。

正如@Holger和@apangin所指出的,这可能是一个ASM缺陷,可以通过mv.visitLookupSwitchInsn(target, new int[]{1}, new Label[]{target});添加至少一个案例分支。因此,最后:是的,您可以使用任何字节码版本在交换机中生成反向分支。

票数 2
EN

Stack Overflow用户

发布于 2015-10-23 07:35:15

在字节码级别,基本上一切都是gotos。表转换或查找开关指令只是要跳转到的偏移量列表。如果你想的话你可以让它向后跳。您不能让它直接跳转到自己,但这只是因为它每次从堆栈中弹出一个int。如果以int push作为前缀,则可以使用2指令循环。

票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/33302816

复制
相关文章
[接口测试 - 基础篇] 05 好讨厌的xml解析
概述 什么是XML? XML 指可扩展标记语言(eXtensible Markup Language)。 XML 被设计用来传输和存储数据。 XML是一套定义语义标记的规则,这些标记将文档分成许多部件并对这些部件加以标识。 它也是元标记语言,即定义了用于定义其他与特定领域有关的、语义的、结构化的标记语言的句法语言。 xml构成 XML由3个部分构成,它们分别是: 文档类型定义(Document Type Definition,DTD),即XML的布局语言 可扩展的样式语言(Extensible Style
苦叶子
2018/04/09
9720
libpcap试玩
libpcap驱动了tcpdump,和wireshark这类抓包工具.提供了高度灵活的包过滤语言. 据wikipedia,高性能的包过滤最早是在bsd上作为一个问题被解决,被称为bpf,在内核实现了一个解释器,进行包匹配,用户态提供一个字符设备, linux作为后来者,支持与bsd基本相同的packet filter,称为lpf,不同的是,linux是通过在一个raw socket来支持包过滤的,通过setsockopt来SO_ATTACH_FILTER,挂载过滤器. strace 可知,libpcap实际上进行了如下syscall:
byronhe
2021/06/25
7160
Winshark:一款用于控制ETW的Wireshark插件
Winshark Winshark是一款用于控制ETW的Wireshark插件,ETW(Event Tracing for Windows)提供了一种对用户层应用程序和内核层驱动创建的事件对象的跟踪记录机制。为开发者提供了一套快速、可靠、通用的一系列事件跟踪特性。Microsoft Message Analyzer早就已经过时了,而且它的下载包早在2019年11月25日也被微软从其官网上移除了。Wireshark建立了一个庞大的网络协议剖析器工具库,为了帮助广大研究人员更好地收集和分析各种类型的网络日志,W
FB客服
2023/04/26
1K0
Winshark:一款用于控制ETW的Wireshark插件
18C 也不能避免 SQL 解析的 Bug
在 Oracle 12.2 版本和新发布的18.0版本中存在一个 SQL 解析的 bug,导致了数据库后台报 ora-07445 或者 ora-00600 错误。报 ora-07445 时,可导致数据库断开当前会话连接,无法进行 SQL 操作,当报 ora-00600 时,会话没有断开,但无法完成解析返回结果。
数据和云01
2019/05/26
8030
电能表国标DLT698协议解析
响应:85 01 01 40 01 02 00 01 09 06 12 34 56 78 90 12 00 00
科控物联
2023/09/01
2.4K0
电能表国标DLT698协议解析
18C 也不能避免 SQL 解析的 Bug
作者简介 苏星开 云和恩墨南区交付技术顾问,曾服务过通信、能源生产、金融等行业客户,擅长 SQL 审核和优化,DataGuard 容灾等。 1 概述 在 Oracle 12.2 版本和新发布的18.0
数据和云
2018/03/07
1.1K0
18C 也不能避免 SQL 解析的 Bug
讨厌的ALG
这几天测试FreeSWITCH的Bypass Media功能,FreeSWITCH在公网上,客户端在私网,发现SDP数据被篡改,影响通话。
杜金房
2020/12/21
1.1K0
javascript的一些bug建议收藏
JavaScript是如今最受欢迎的编程语言之一,但受欢迎同时就是该语言自身的各种特性带来的副作用,无论该语言多美妙,每天还是有成千上万的程序员弄出一堆bug。先不要嘲笑别人,或许你也是其中之一。
全栈程序员站长
2022/07/15
2440
尽量避免bug的一些手法
最近参与了几个需求开发,BUG很少,有些需求没BUG,有些才一个BUG,搞的测试人员还发牢骚说:
Java团长
2019/01/23
8100
DLT645-2007
对于电能来说,DI0是结算日的信息,现在的就是写0,上一结算日的就写 01,上12结算日就写 0C
科控物联
2022/06/13
1.1K0
DLT645-2007
DLT645调试
DLT645-2007 端子2和10接单相电;通讯是24和25就可以测试了。 各种软件测试: 直接串口调试软件测试 厂家古老的测试软件 各个品牌网关测试 都只能一个变量一个变量的读???
科控物联
2022/06/13
5190
DLT645调试
ajaxFileUpload.js 的一些Bug
这里以前提到过 http://blog.csdn.net/qq_30930805/article/details/62427726
試毅-思伟
2018/09/06
6930
LAMP Linux路由和Libpcap配置
1、Ubuntu安装:sda会覆盖MBR,选择这个,sda1会导致找不到系统 2、配置路由 在/etc/rc.local 开机时候自动加载 sudo iptables -F  //清除所有规则 sudo iptables -P INPUT ACCEPT sudo iptables -P FORWARD ACCEPT sudo iptables -t nat POSTROUTING -o eth1 -j MASQUERADE 具体内容忘记了 eth1 为 wan口网卡
星哥玩云
2022/06/30
4.2K0
Libpcap PACKET_MMAP内存分配
libpcap为了提高效率,调用setsockopt(handle->fd, SOL_PACKET, PACKET_RX_RING,(void *) &req, sizeof(req))时采用kmalloc分配内存。 可以参考: https://www.kernel.org/doc/Documentation/networking/packet_mmap.txt kmalloc底层依赖linux的slab内存分配机制,在2.6.22内核之后,slub取代slab成为默认的内存分配器。空间和时间上都有所提升。
happy123.me
2018/06/04
1.4K0
JSON金额解析BUG的解决过程
这是在我们开发的一个支付系统中暴露的一个BUG,问题本身比较简单,有意思的是解决问题的过程。将过程分享出来,希望能够对大家有所帮助。
程序猿讲故事
2019/09/27
1.1K0
JSON金额解析BUG的解决过程
Shell之讨厌的正则
思想(KISS)相当重要。KISS(keep it simple stupid)。 这是其实不难,只要按照一下思路进行就ok。
后场技术
2020/09/03
4750
Shell之讨厌的正则
BibTeX条目类型
使用 BibTeX 时,各大参考文献检索网站经常会给出不同的 BibTeX 条目类型,本文便就 BibTeX 的各种条目类型及其说明进行摘录。
hotarugali
2022/11/23
4330
深度学习跟踪DLT (deep learning tracker)
粒子滤波是对预测粒子进行评价,添加不同的权重,越接近于真实状态的粒子,其权重越大;否则,就加的权重小一些。步骤:
代码的路
2022/08/23
7500
解决ueditor上传视频、音频的一些bug
使用ueditor上传视频和音频功能之前一直没有用,也没有去搞因为用不太着。现在想上传个视频就搞了一下都是在网上找的方法一步一步的试,花了几个小时终于好了。记录下怎么改的。
sunonzj
2022/06/21
1.9K0
知名系统中的一些有趣bug
产品的绝大部分bug,会在测试阶段被消灭,但仍然有不少的bug,脱离测试工程师的魔掌,展现在了用户面前。有些bug十分影响用户体验,不过有些bug,反而会娱乐大众,让人笑翻了天。
良月柒
2019/03/20
5970
知名系统中的一些有趣bug

相似问题

防止用户切换TabItem +讨厌的bug?

20

React讨厌的bug,我搞不懂

222

KD的构建程序中的讨厌的bug

16

grails ajax功能--一个“讨厌的bug”

10

代码中的一些错误- libpcap

33
添加站长 进交流群

领取专属 10元无门槛券

AI混元助手 在线答疑

扫码加入开发者社群
关注 腾讯云开发者公众号

洞察 腾讯核心技术

剖析业界实践案例

扫码关注腾讯云开发者公众号
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文