首先要明确一点

static修饰方法、成员变量是属于类自己的,并不属于特定的某个对象。所以我们可以像这样使用static修饰的方法、变量:ClassName.methodName、ClassName.variable,而不用先new一个Class出来。通过new出来的对象就是特定的某一个对象。所以还有这么一说,不能在静态方法内使用非静态的成员变量,因为这个成员变量不是类自己的,而是某一个对象的,你不能确定在使用这个成员变量的时候这个对象是否存在。

解释为什么非静态内部类里面不能定义静态方法、静态成员变量静态初始化块

通过查看jvm类加载的过程不难理解这个问题

加载——>连接——>类初始化——>类实例化——>使用——>卸载
1.加载
将.class文件载入到方法区。这样常量和类相关信息还有方法已经在方法区
2.连接
(1)验证
验证合法性,保证能让JVM正常执行
(2)准备
为静态变量在方法区开辟内存,并初始化。基本类型设为0,非基本类型设为NULL,常量设为程序中的
赋值(只有常量设为程序中值,非常量的静态变量都是赋JVM默认值,而非程序设定的值)
(3)解析
把常量池中的常量从符号引用变成直接引用。
3.类初始化
执行static的赋值语句。包括static变量的赋值,以及static代码块。自上而下执行。
4.实例化
执行成员变量的赋值语句和构造函数。包括成员变量的赋值,以及{}代码块。自上而下执行。构造函数最后执行 
5.使用
每个线程一个栈,调用方法,就往里面压一个Stack Frame。方法执行结束后则弹出。实例化对象后,对象
是在堆中,栈中有个引用指向堆中的对象,堆中的对象有个地址指向方法区中的类的信息
6.卸载

在类加载的时候,静态属性和代码块会先加载,那么按照这个逻辑,非静态内部类里面的静态属性也要优先于这个内部类加载,但这个时候这个内部类都还没有初始化,这就出现矛盾了。

总结

其实归根结底,还是类与对象的区别,静态属性不依赖于对象,所以访问修改的时候不需要依赖当前有没有存活的对象。
内部类其实也可以认为是外部类的一个成员变量,想要访问内部类,必须先实例化外部类,然后通过外部类才能访问内部类。
只要是成员变量,就需要依赖具体的对象,这个时候如果在非静态内部类里面声明静态属性就会破坏了这一逻辑,所以java语言在语义层面不允许我们那么做,这其实不是技术问题,是一个语言的逻辑和语义问题。