Java匿名内部类访问全局变量和局部变量的注意事项

java 复制代码
import java.awt.BasicStroke;
import java.awt.Color;
import java.awt.FlowLayout;
import java.awt.Font;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.GridLayout;
import java.awt.Shape;
import java.awt.Stroke;
import java.awt.event.MouseEvent;
import java.awt.event.MouseMotionListener;
import java.awt.geom.Rectangle2D;

import javax.swing.*;
public class DemoAA extends JFrame{
	PanelAA panel;
	DemoAA(){
		super("练习监听");
		setBounds(600, 200, 600, 600);
		setDefaultCloseOperation(EXIT_ON_CLOSE);
		panel = new PanelAA();
		add(panel);
	}

	public static void main(String[] args) {
		// TODO Auto-generated method stub
		SwingUtilities.invokeLater(() ->{
			new DemoAA().setVisible(true);
		});
	}

	private class PanelAA extends JPanel{
		JLabel l1,l2;
		int x=0,y=0;
		PanelAA(){
			setLayout(new GridLayout(2, 1,0,10));
	        l1 = new JLabel("左上角文字", SwingConstants.LEFT); // 水平左对齐
	        l1.setVerticalAlignment(SwingConstants.TOP); // 垂直顶部对齐
			l1.setFont(new Font("黑体", Font.BOLD, 20));	
			add(l1);
			//鼠标监听
			addMouseMotionListener(new MouseMotionListener() {
				
				@Override
				public void mouseMoved(MouseEvent e) {
					// TODO Auto-generated method stub
					x =e.getX();
					y=e.getY();
					repaint();
				}
				
				@Override
				public void mouseDragged(MouseEvent e) {
					// TODO Auto-generated method stub
					
				}
			});
		}
		
		@Override
		protected void paintComponent(Graphics g) {
			// TODO Auto-generated method stub
			super.paintComponent(g);
			Graphics2D g2 = (Graphics2D)g;
			g2.setFont(new Font("黑体", Font.BOLD, 24));
			g2.setColor(Color.RED);
			g2.drawString("坐标 X:"+x+" Y:"+y, 10, 60);
		}

	}
}		

为什么匿名内部类能访问外部变量

背后的原理(简单版)

Java 为了让匿名内部类能访问外部变量,实际上在背后偷偷做了这些事:

  1. 自动传递 :Java 编译器会把外部类的 this 引用(指向 PanelAA 实例)传递给匿名内部类
  2. 隐形桥梁 :匿名内部类通过这个 this 引用,就能找到外部的 xy

如果是局部变量呢?

有趣的是,如果 xy 是方法内的局部变量(不是类的成员变量),规则就不同了:

java 复制代码
void someMethod() {
    int localX = 0; // 局部变量
    addMouseMotionListener(new MouseMotionListener() {
        public void mouseMoved(MouseEvent e) {
            localX = e.getX(); // 这会报错!不能修改局部变量
        }
    });
}

这时候 Java 会说:

🚨 "嘿!局部变量 localX 必须声明为 final 或等效于 final 才能被匿名内部类访问!"

为什么有这个区别?

  • 成员变量:属于对象,生命周期和对象一样长,安全可访问
  • 局部变量方法执行完就可能消失匿名内部类可能活得比它久,所以 Java 要限制
相关推荐
2601_950703941 分钟前
Java安全编程与静态分析实战
java
唐叔在学习2 分钟前
Python移动端应用消息提醒开发实践
开发语言·python
好家伙VCC2 分钟前
**发散创新:基于Python与OpenCV的视频流帧级分析实战**在当前人工智能与计算机视觉飞速发展的背景下
java·人工智能·python·计算机视觉
SimonKing3 分钟前
大V说’AI替代不了你’,但现实是——用AI的人正在替代你
java·后端·程序员
暴力求解3 分钟前
C++ ---string类(三)
开发语言·c++
Pocker_Spades_A9 分钟前
Python快速入门专业版(五十七)——POST请求与模拟登录:从表单分析到实战(以测试网站为例)
开发语言·python
一叶龙洲19 分钟前
Java中使用模板引擎(FreeMarker / Velocity) + Word XML导出复杂Word
xml·java·word
道清茗23 分钟前
【RH294知识点汇总】第 3 章 《 管理变量和事实 》1
开发语言·python
星空椰23 分钟前
JavaScript基础:运算符和流程控制
开发语言·javascript·ecmascript
Halo_tjn25 分钟前
Java 接口的定义重构学生管理系统
java·开发语言·算法