「java匿名委托」匿名函数 java

博主:adminadmin 2022-12-02 13:21:08 74

本篇文章给大家谈谈java匿名委托,以及匿名函数 java对应的知识点,希望对各位有所帮助,不要忘了收藏本站喔。

本文目录一览:

java 匿名类

abstract class Animal {

public abstract void cry();

}

class TestCry {

public void testCry(Animal animal) {

animal.cry();

}

}

public class Example{

public static void main(String[] args) {

TestCry cry = new TestCry () ;

cry.testCry(new Animal (){

public void cry() {

System.out.println("Cat is crying!");

}

});

cry.testCry(new Animal (){

public void cry() {

System.out.println("Dog is crying!");

}

});

// ... 省略其他

}

}

先看段伪代码

abstract class Father(){

....

}

public class Test{

Father f1 = new Father(){ .... } //这里就是有个匿名内部类

}

一般来说,new 一个对象时小括号后应该是分号,也就是new出对象该语句就结束了。

但是出现匿名内部类就不一样,小括号后跟的是大括号,大括号中是该new 出对象的具体的实现方法。

因为我们知道,一个抽象类是不能直接new 的,必须先有实现类了我们才能new出它的实现类。

上面的伪代码就是表示new 的是Father的实现类,这个实现类是个匿名内部类。

其实拆分上面的匿名内部类可为

class SonOne extends Father{

... //这里的代码和上面匿名内部类,大括号中的代码是一样的

}

public class Test{

Father f1 = new SonOne() ;

}

C#和JAVA中Ticks的不同含义

.NET(C#) Java

基本类型 基本类型

C#中有无符号数,Java没有。

C#中有值类型,且可自己定义值类型的结构体(struct)。

Java中的基本类型(或叫基元类型)即为值类型,但Java没有结构体,所以不能自定义值类型。

C#中的值类型(包括所有基本类型)间接继承自Object,有自己的方法可以调用;Java中的值类型(即基本类型)不继承自Object,只是简单的数据,没有方法可以调用。

C#中int等同于System.Int32,是值类型;bool等同于System.Boolean;等。

Java中int是基本类型,是值类型,而Integer是引用类型,Integer是int的包装器,int自身没有方法,Integer有一些方法;int与Integer之间可隐式转换(导致装箱和拆箱),但当Integer值为null的时候会在运行时抛出异常。boolean等类似。

Java中的int与Integer的对应在C#中类似int和Nullableint的对应,它们的后者都是前者的包装,且后者可以等于null。但Nullableint实际上仍然是值类型的(所以仍然很轻量级),所以从内存上讲C#中int和Object的对应更接近Java的对应一些。C#中Nullableint到int的转换必须显式进行,因为Nullableint中的值为null时会引发运行时异常。

其他基本类型与之类似。

委托,事件

[无]

C#中的委托可以认为是方法的类型化,于是可以将方法放在变量里传递。事件是对委托做了一层包装。

Java通过接口来实现C#中委托和事件的功能,可通过匿名类来达到C#中匿名委托的作用(同样也能实现闭包的功能)。

另,C#中也有匿名类,但C#和Java的匿名类刚好各做各的事情:Java中的匿名类只有方法没有数据;C#中的匿名类只有数据没有方法。

非托管

[无]

C#可以有非托管代码,可以有指针等。Java没有。

索引器

[无]

C#有索引器,可方便容器类实现类似数组的效果。Java没有,Java的容器基本上用put,get,set等方法达到同样效果。

属性

[无]

C#的属性通过在内部定义get/set方法,使外部使用时像是在使用变量字段,但其实是在调用get/set方法,以达到透明的封装数据的目的。

Java没有属性的概念。Java通过约定为字段XX添加getXX,setXX方法达到同样的目的。

预编译指令

[无]

C#有预编译指令可方便调试,且有ConditionalAttribute来描述方法。Java没有。

操作符重载

[无]

C#可重载操作符。Java没有。

Java自己重载了String的+和+=,但没有重载==,这是我这段时间犯的最多的错误。C#中String的==是比较值相等,Java中==是Object的默认行为:比较引用相等,要比较值相等得用equals方法。(这么多年编程以来,我似乎从来没有遇到过要比较两个字符串变量的引用相等。对于比较值相等来讲,==符号比equals方法调用看上去优雅得多,况且方法调用还得注意空指针的情况)

内部类 内部类

Java的内部类可以直接访问外部类的实例成员。

C#的不行。C#的内部类等同于Java的静态内部类。

goto、switch [goto]、switch

C#允许用goto。Java的goto是保留关键字,不能使用。但Java允许有标签,在有嵌套循环时可以在continue、break后面跟标签名。

C#的switch可以使用long、String;Java不可以。

Java的switch中的case子句在后面没有跟break的情况下直接跳到下一个case子句;

C#中只有在前一个case没有任何代码的情况下才允许不写break直接跳到下一个case,C#中可以通过goto跳转到另一case。

enum enum

C#中的枚举是值类型,且其基于数值类型(默认基于int),可设置枚举项对应的数字,不能在其中添加方法等任何其他成员。

Java中的枚举是引用类型(Java除了基本类型外,任何类型都是引用类型),不是基于数值类型。除了不能继承外,它跟普通类差别不大,可以添加成员方法和成员变量等(当然也就可以重写toString方法)。

C#和Java的枚举都可以用于switch。

可以将C#的枚举作为数值看待而直接进行位运算,因此可以在一个变量中存储多个位标记。

Java的枚举跟数值没有直接关系,因此不能直接这么用。Java用EnumSet来存储枚举标志,不需要直接使用位运算,更远离底层。

override @Override

C#能被重写的方法必须添加virtual关键字声明为虚方法,派生类重写子类方法时添加override关键字。

Java默认方法都可被重写,派生类和子类方法签名一样时被认为是重写。要声明不能被重写的方法需在方法前加final关键字。重写时可以在方法前添加标注(即C#中的定制特性)@Override,这样一旦此方法找不到被重写的方法时编译器会报错,以防止拼写错误。

定制特性 标注

C#用中括号[]将定制特性括起来。Java用@打头,后面跟定制特性的名字。

泛型 泛型

Java中泛型实现使用的擦除机制,为类型参数传入类型并不导致新类型出现,即传入了类型参数后在运行时仍然完全不知道类型参数的具体类型,它的目的是为了兼容非泛型(所以可以在泛型和非泛型之间隐式转换,会有编译警告但不会有编译错误,这当然其实并不安全);这同时衍生了一系列问题:不能定义泛型类型参数的数组如T[],不能通过new T()的方式实例化泛型,等。

Java的泛型不支持值类型(使用的话会被自动包装成引用类型)。

C#的泛型在类型参数传入类型后会产生一个新类型(虽然CLR的优化机制会使引用类型共享同样的代码),可以在运行时得到类型参数的类型信息。可以定义泛型数组,可以添加约束使其可以new。C#的泛型可以使用值类型(不会被装箱)。

对于Java的泛型,简单的讲,它的好处只在编译时,运行时没有任何泛型的意义。当你在使用已有的泛型类时,这通常能满足要求;但如果你要自己定义泛型类,那你得知道它有多少你觉得它应该可以但事实上不可以的事情。

参数引用传递 [无]

C#允许使用关键字out,ref显式指定参数传递方式为引用传递。

Java只有值传递。

@字符串 [无]

C#在写字符串时可以在引号前加个@符号来取消/的转义作用。

Java没有。

?? [无]

C#的??二元操作符当前面的表达式不为null时返回前面表达式的值,前面表达式为null时返回后面表达式的值。

Java没有。

using import

C#可以用using为命名空间或类指定别名。(using还有Dispose的使用方式,与命名空间无关)

Java的import可以引入类或包(即C#的命名空间),static import可以引入类的成员。

初始化 初始化

C#调用基类构造函数的语法为:

SubClass() : base() { }

Java调用基类构造函数的语法为:

SubClass(){

super();

}

C#和Java都可以用类似的语法调用同一个类的其他构造函数。(分别将base和super换成this)

Java有代码块概念,会在构造函数之前执行(基类的构造函数之后)。

在成员变量声明时赋值,Java允许其赋值表达式中引用前面声明的另一个变量,如:

private int x = 1;

private int y = x + 10;

这里变量y的赋值语句有变量x。

C#不允许这样做。

interface interface

Java的接口内允许有内部类、静态字段等。

C#不允许。

readonly,const final

C#的const是绝对的常量,必须在声明语句中同时赋值,只有数值、枚举和String可以声明为const。const的值会内联到各个使用的地方。

C#的readonly表示变量在构造函数执行完之后是不能再变化的。它只约束变量本身,而无法约束变量引用(如果它是引用类型或者有成员是引用类型)的对象。

Java中的final(在约束变量的时候)看上去更像readonly。

但C#的readonly和const有个区别,readonly的int是不能作为switch的case语句的,const的可以。

而Java的final则是:有时候可以有时候不可以----编译时可以得到明确值的可以,反之不可以。如:

final int x = 1; // 这个可以

final int y = new Random().nextInt(); // 这个不可以

那么可以理解为:编译时能得到明确值的时候,final等同于C#的const(不清楚Java在这个情况下是否会内联,估计不会);编译时无法得到明确值的时候,final等同于C#的readonly。

[无] throws

Java在可能抛出异常时,除了RuntimeException(包括派生类),都要么捕获,要么在方法声明中用throws关键字声明出来表示继续抛出。

C#没有采用这种强制处理机制。

功能相同但语法有差异的

namespace == package (Java的package对文件结构也有要求;C#没有)

internal == [默认] (Java中不写访问修饰符即表示访问权限是package;C#默认是private。C#的internal protected在Java中没有。)

lock == synchronized (Java中synchronized可以修饰方法,C#可以用定制特性[MethodImplAttribute(MethodImplOptions.Synchronized)]达到同样效果)

: == extends,implements

base == super

is == instanceof (C#有as,Java没有)

typeof == .class

[SerializableAttribute]定制特性 == Serializable接口

[NonSerializedAttribute]定制特性 == transient

params == ... (可变数目参数)

java语言中委托的实现方法

委托是C、OC和C#中的才有,在java中是叫接口(interface )。

实现接口可以使用关键字implements,假设有接口Animal,那么实现接口代码示范如下:

class cat implements Animal

{

   public void shout ()

   {

        int(“喵喵”);

    }

}

java中的委托

委托模式是软件设计模式中的一项基本技巧。在委托模式中,有两个对象参与处理同一个请求,接受请求的对象将请求委托给另一个对象来处理。委托模式是一项基本技巧,许多其他的模式,如状态模式、策略模式、访问者模式本质上是在更特殊的场合采用了委托模式。委托模式使得我们可以用聚合来替代继承,它还使我们可以模拟mixin。

“委托”在C#中是一个语言级特性,而在Java语言中没有直接的对应,但是我们可以通过动态代理来实现委托!代码如下:

import java.lang.reflect.InvocationHandler;

import java.lang.reflect.Method;

import java.lang.reflect.Proxy;

/*

* @author Liusheng

* 实现“委托”模式,用户需要实现InvocationHandler接口;

* 参考:

*/

public abstract class Delegator implements InvocationHandler {

//--------------------------------------------

protected Object obj_orgin = null; //原始对象

protected Object obj_proxy = null; //代理对象

//--------------------------------------------

public Delegator() {

}

public Delegator(Object orgin) {

this.createProxy(orgin);

}

/*

* 完成原始对象和委托对象的实例化

* @param orgin 原始对象实例

*/

protected Object createProxy(Object orgin) {

obj_orgin = orgin;

//下面语句中orgin.getClass().getClassLoader()为加载器,orgin.getClass().getInterfaces()为接口集

obj_proxy = Proxy.newProxyInstance(orgin.getClass().getClassLoader(), orgin.getClass().getInterfaces(), this); //委托

return obj_proxy;

}

/*

* 对带有指定参数的指定对象调用由此 Method 对象表示的底层方法,具体请参见Java API

* @param args 参数

* @param method 方法类实例

*/

protected Object invokeSuper(Method method, Object[] args) throws Throwable {

return method.invoke(obj_orgin, args);

}

//--------------实现InvocationHandler接口,要求覆盖------------

//下面实现的方法是当委托的类调用toString()方法时,操作其他方法而不是该类默认的toString(),这个类的其他方法则不会。

public Object invoke(Object obj, Method method, Object[] args) throws Throwable {

// 缺省实现:委托给obj_orgin完成对应的操作

if (method.getName().equals("toString")) { //对其做额外处理

return this.invokeSuper(method, args) + "$Proxy";

} else { //注意,调用原始对象的方法,而不是代理的(obj==obj_proxy)

return this.invokeSuper(method, args);

}

}

}

下面的代码,则是作为一个委托的例子,实现Map的功能。

import java.io.IOException;

import java.lang.reflect.Method;

import java.util.Hashtable;

import java.util.Map;

import org.apache.commons.logging.Log;

import org.apache.commons.logging.LogFactory;

import com.bs2.core.UtilLog;

/**

* @author Liusheng

* 本代码主要用于演示RelegateTo的使用方法

*/

public class Delegator4Map extends Delegator {

private static Log _log = LogFactory.getLog(Delegator4Map.class);

private Map orginClass = null; //原始对象

private Map proxyClass = null; //代理对象

public Map getOrgin() {

return orginClass;

}

public Map getProxy() {

return proxyClass;

}

public Delegator4Map(Map orgin) {

super(orgin);

orginClass = orgin;

proxyClass = (Map) super.obj_proxy;

}

public Object invoke(Object obj, Method method, Object[] args) throws Throwable {

if (method.getName().equals("size")) { //修改size处理逻辑

Object res2 = new Integer(-1);

System.out.println("调用委托的方法");

return res2;

} else {

System.out.println("调用原始的方法");

return super.invoke(obj, method, args);

}

}

public static void main(String[] args) throws IOException {

Delegator4Map rtm = new Delegator4Map(new Hashtable());

Map m = rtm.getProxy();

m.size();

}

}

java的匿名函数怎么用

Java 应用程序按值传递参数(引用类型或基本类型),其实都是传递他们的一份拷贝.而不是数据本身.(不是像 C++ 中那样对原始值进行操作。)

例1:

Java代码

//在函数中传递基本数据类型,

public class Test {

public static void change(int i, int j) {

int temp = i;

i = j;

j = temp;

}

public static void main(String[] args) {

int a = 3;

int b = 4;

change(a, b);

System.out.println("a=" + a);

System.out.println("b=" + b);

}

}

结果为:

a=3

b=4

原因就是 参数中传递的是 基本类型 a 和 b 的拷贝,在函数中交换的也是那份拷贝的值 而不是数据本身;

例2:

Java代码

//传的是引用数据类型

public class Test {

public static void change(int[] counts) {

counts[0] = 6;

System.out.println(counts[0]);

}

public static void main(String[] args) {

int[] count = { 1, 2, 3, 4, 5 };

change(count);

}

}

在方法中 传递引用数据类型int数组,实际上传递的是其引用count的拷贝,他们都指向数组对象,在方法中可以改变数组对象的内容。即:对复制的引用所调用的方法更改的是同一个对象。

JAVA中内部匿名类以及接口的使用和作用?

Java中内部匿名类的使用

Java中内部匿名类用的最多的地方也许就是在Frame中加入Listner了吧。

如下:

import java.awt.*;

import java.awt.event.*;

public class QFrame extends Frame {

public QFrame() {

this.setTitle("my application");

addWindowListener(new WindowAdapter() {

public void windowClosing(WindowEvent e) {

dispose();

System.exit(0);

}

});

this.setBounds(10,10,200,200);

}

}

内部匿名类,就是建立一个内部的类,但没有给你命名,也就是没有引用实例的变量。

new WindowAdapter() {

public void windowClosing(WindowEvent e) {

dispose();

System.exit(0);

}

}

new 是建立一个 WindowAdapter对象 ,后面一个 {} 表示这个括号中的操作作用于这个默认的对名象,

而上面的Java程序中后面是一个函数体。这种用法是 Delphi 和 VB 所没有的。

这个用法的作用是:创建一个对象的实例,并且 override 它的一个函数。

打开 WindowAdapter 的代码可以发现。它是一个抽象类。它是对 WindowListener 接口的一个实现。

Frame.addWindowListner(); 的参数是一个 WindowListner ,而实现上是传一个从WindowAdapter 派生出的一个匿名类。

---------------------------------------------------------------

以上观点,纯属个人看法。

如有不同意见,留下名来。

多谢。

----------------------------------------------------------------

补记, java 中的内部匿名类与 Delphi 中的 with 是不是一样的。with 只是指定一个默认的操作对象,是从编辑上来说,减少输入录(执行上也许也会快点)。而 java 中的内部匿名类,着重在于派生一个新类,但不给这个新类指定一个类名。

代码例子如下:

01

02

03 class AClass {

04 String name = new String("");

05

06 AClass() {

07 }

08

09 AClass(String name) {

10 this();

11 this.name = name;

12 }

13

14 public void print() {

15 System.out.println("The name = " + name);

16 }

17 }

18

19 public class QTest {

20 public static void main(String[] args) {

21 (new AClass("RedSoft"){

22 public void print() {

23 System.out.println("the anonymose class print");

24 super.print();

25 }

26 }).print();

27 }

28 }

运行结果:

the anonymose class print

The name = RedSoft

接口因为java不像c++那样能够多继承 所以有了接口,原理和c++里的多继承是一样的,还有就是规范设计

关于java匿名委托和匿名函数 java的介绍到此就结束了,不知道你从中找到你需要的信息了吗 ?如果你还想了解更多这方面的信息,记得收藏关注本站。

The End

发布于:2022-12-02,除非注明,否则均为首码项目网原创文章,转载请注明出处。