Java8 introduces Lambda expressions, which allow developers to pass functions as arguments to a method or treat the code itself as data. Lambda expressions are used to make applications concise and compact. Many languages (Groovy, Scala, etc.) have supported Lambda expressions since their design. But Java uses anonymous inner classes instead. Finally, with the help of the powerful community, we found a compromise Lambda implementation scheme, which can achieve a concise and compact language structure.
The evolution of anonymous inner classes to Lambda
An anonymous inner class is a class without a name that exists inside a class or method. When we need to use a class only once to create and use a combination, we can choose the anonymous inner class, saving us the steps of defining the class.
An anonymous inner class inherits a class or implements an interface, or an anonymous object that subclasses the class or implements the interface. Let’s look at an example of an anonymous inner class
package com.java8;
/* Define and use anonymous inner classes */
public class NoNameClass {
public static void main(String[] args) {
Model m = new Model(){
@Override
public void func(a) {
System.out.println("Method implementation"); }}; m.func(); }}// The interface to be implemented
interface Model{
void func(a);
}
Copy the code
Equivalent Lambda code
package com.java8;
/* Define and use Lambda simplifications */
public class NoNameClass {
public static void main(String[] args) {
Model m = new Model(){()->{
System.out.println("Method implementation"); }}; m.func(); }}Copy the code
As you can see, Lambda expressions are used instead of anonymous inner class code, making the code simpler and more compact.
Second, the grammar
(parameters) -> expression 或 (parameters) ->{ statements; }
-
Optional type declaration
There is no need to declare parameter types, and the compiler can uniformly identify parameter values.
-
Optional parameter parentheses
Parentheses are not required for one parameter, but parentheses are required for multiple parameters.
-
Optional braces
If the body contains a statement, you do not need braces.
-
Optional return keyword
If the body returns only one expression value, the compiler automatically returns the value. The braces need to indicate that the expression returns a numeric value.
Examples of Lambda expressions:
3. The form of Lambda
With Lambda, implementation methods can have arguments or return values, and if no argument type is specified, the compiler can infer that.
1. No parameter return value
Generates any integer between [1,10]
interface Model2{
int func(a);
}
Model2 md2 = () -> {return (int)(Math.random()*10+1)};
Copy the code
Rewriting Lambda requires corresponding abstract methods, using () placeholders when there are no arguments, and omitting return and {} when the expression is a single line of code.
Lambda is equivalent to:
Model2 md2 = () -> (int)(Math.random()*10+1);
Copy the code
2. Parameter with return value
Returns a string that describes a number.
interface Model3{
String func(int a);
}
Model3 md3 = (int a) -> {
return "This is a number " + a;
};
Copy the code
The type of the parameter can be omitted, which is inferred by the compiler, and () can also be omitted.
Lambda is equivalent to:
md3 = a -> "This is a number " + a;
Copy the code
Omit the argument type, the parenthesis, and the implementation body’s parenthesis and return.
3. Take multiple parameters
Computes the operation of two numbers based on the input operator and returns the result
interface Model4{
String func(int a, int b, String oper);
}
Model4 md4 = (a, b, s) -> {
String res = "";
if("+".equals(s)){
res = ( a+b ) + "";
}else if("-".equals(s)){
res = ( a-b ) + "";
}else if("*".equals(s)){
res = ( a*b ) + "";
}else if("/".equals(s)){
res = ( a/b ) + ""; // We can divide by 0
}else{
res = "Error in operation.";
}
return res;
};
System.out.println(md4.func(1.1."+"));
Copy the code
The above example is a Lambda expression for multiple arguments, with the type of each argument omitted and the compiler automatically inferences. The {} of the implementation body cannot be omitted in multiple statements.
Lambda as a parameter
Prior to Java8, an interface could be passed in as a method parameter and must be executed with an instance of the interface implementation class. Starting with java8, lambdas can be implemented as interface methods and passed in as arguments, eliminating object creation in both form and reality. Make the code more compact, simple and efficient.
Defines the interface
In the interface, there must be one and only one abstract method to determine the Lambda template
// A method with no parameters and no return value
interface LambdaInterface1{
void printString(a);
}
// A method with arguments that return no value
interface LambdaInterface2{
void printString(String str);
}
Copy the code
Define methods to receive parameters
You need an interface as an argument in a method
/ / no arguments
public static void testLambda(LambdaInterface1 lam1){
lam1.printString();
}
/ / take ginseng
public static void testLambda2(String s,LambdaInterface2 lam2){
lam2.printString(s);
}
Copy the code
Lambda expressions are passed in as arguments
// Lambda takes no arguments
testLambda(()->{
System.out.println("It can be simple, it can be complicated.");
});
// Lambda as an argument
testLambdaParam("hello",(a)->{
System.out.println(a);
});
Copy the code
5. Use variables in Lambda
You can define your own local variables in a Lambda, you can use local variables of an outer method, and you can use attributes. This is not hard to understand, since it is a method implementation, only write a block of code, so it is not excessive to use the local variables of the method itself and the class attributes.
public static void main(String[] args) {
List<String> strs = new ArrayList<String>(){
{
add("aaa");
add("bbb");
add("ccc"); }};int j = 1;
strs.forEach((str)->{
int i = 0;
System.out.println(str + "" + i + "" + j);
});
}
Copy the code
Lambda type inference
Type checking
The type of Lambda is inferred from the context in which Lambda is used. The arguments to Lambda expressions correspond to the arguments and return value types of methods in functional interfaces. The type required by Lambda expressions, or the functional interface that Lambda implements, is called the target type.
Type inference
Use the target type to check whether a Lambda can be used in a particular context, and infer the type of Lambda parameters.
Lambda expression practice
1, hot commodity sequencing
Sorting for the development of long for you may not be strange, if the original you have done e-commerce projects, I believe that the e-commerce scene under the commodity record sorting operation is very emotional, we use Lambda to see the operation of hot commodity sorting.
Test data The following uses mobile phone test data as an example
/** * The actual development data is usually fetched from the database * test data is used here */
Goods g01=new Goods(1."Millet 9".1789.200, BigDecimal.valueOf(2500));
Goods g02=new Goods(2."Huawei Mate20".5000.3000, BigDecimal.valueOf(7000));
Goods g03=new Goods(3."OPPO R17".2000.2827, BigDecimal.valueOf(1500));
Goods g04=new Goods(4."The meizu Note9." ".2000.1600, BigDecimal.valueOf(1600));
Goods g05=new Goods(5."One plus 6 t".8000.5000, BigDecimal.valueOf(3500));
List<Goods> goods= Arrays.asList(g01,g02,g03,g04,g05);
Copy the code
The collections.sort static method implements sorting
Collections.sort(goods,(g1,g2)->g1.getSale()-g2.getSale());
Copy the code
The list.sort default method implements collection sorting
// Use Lambda to sort the item records by volume
goods.sort((g1,g2)->g1.getSale()-g2.getSale());
Copy the code
The stream. sorted method implements element sorting
// Multi-condition sort Lambda configuration Stream sales + price sort if sales are equal, sort by price
goods =goods.stream().sorted((g1,g2)->g1.getSale()-g2.getSale())
.sorted((g1,g2)->g1.getPrice().compareTo(g2.getPrice()))
.collect(Collectors.toList());
Copy the code
2. Log output optimization
For project development log printing is an unattainable module, whether in the development stage or after the project deployment online, the output of log information is an important reference index for developers and operation and maintenance personnel.
Log output scenario The user module UserService is used as an example. The log output code before optimization is as follows:
public String login(String userName, String userPwd) {
logger.info("UserService receives parameter -->" + userName + "," + userPwd);
/** * Logins omit */
return "login";
}
Copy the code
Set the log level to DEBUG to view parameters received by the backend during development. When the log level is set to INFO, the debug log should not be output. In addition, when the debug method is called, the string parameters passed in need to be concatenated. The situation can get worse when accessing the mall project activity: all debug messages are printed out with a lot of string concatenation, which affects the performance of the entire application.
Log output Scenario This section uses the user module UserService as an example. The log output code is optimized
- Determine the log output level before output logs
- Delay log content output with Lambda
/** * Add info method * Determine the log printing level * Output log information when the condition is true *@param logger
* @param message
*/
public void info(Log logger, Supplier<String> message){
if(logger.isInfoEnabled()){ logger.info(message.get()); }}public String login(String userName, String userPwd) {
//logger.info(" + userName + "," + userPwd);
// Delay Lambda expression execution only with certainty
info(logger,()->"UserService receives parameter -->" + userName + "," + userPwd);
return "login";
}
Copy the code
Advantages and usage scenarios of Lambda
The introduction of Lambda expressions replaces anonymous inner classes, making code concise and compact, while the inertness of Lambda improves application performance at development time.
For Lambda application scenarios, the code structure is usually combined with functional interfaces to make the development of functional oriented programming, which is also a new idea introduced in Java8 – functional programming (described later). At the same time, it will be combined with the interface default method mentioned earlier to withdraw into the application development.