Unable to infer types in generic parameters of function constructor
php editor Youzi will answer a common question for everyone: the reason why the type cannot be inferred from the generic parameters of the constructor. In PHP, type inference for generic parameters is achieved through the default value of the parameter. However, in a constructor, since the parameters are passed in before the object is created, the type of the parameters cannot be inferred from the object's instance. This results in the inability to infer the type of the generic parameter in the constructor, and the type needs to be manually specified to solve this problem.
Question content
I have some generic code in go where I have a "master" type with a generic parameter and many "slave" types that should share the same generic parameters. The code looks similar to this:
type Doer[T any] interface { ModifyA(*A[T]) } type B[T any] struct { } func NewB[T any]() *B[T] { return new(B[T]) } func (b *B[T]) ModifyA(a *A[T]) { // Do a thing } type A[T any] struct{} func NewA[T any]() A[T] { return A[T]{} } func (a *A[T]) Run(doers ...Doer[T]) { for _, doer := range doers { doer.ModifyA(a) } } func main() { a := new(A[int]) a.Run(NewB()) // error here }
Basically, the user should define t
on a
and then t
on b
should be the same. This type of code works in other languages that support generics, but in go I get a cannot infer t
compile error at the commented line (see the go playground code here). It seems to me that the type parameter on a
is set to int
, so the type parameter on b
should also be set to int
. I could call newb[int]()
instead, but that seems too verbose to me. Why does this happen?
Solution
This is a variation on "Why can't the compiler infer the type parameters based on how the return type is used?" Answer: Because as of go 1.20, this is not how type inference works.
Type inferenceApplies to:
- Type parameter list
- Replacement map m initialized with known type parameters (if any)
- A (possibly empty) list of ordinary function parameters (only in the case of function calls)
If you check these rules one by one:
newb()
Is there a type parameter list? No. You didn't specify a type parameter when calling it.Are there other known type parameters that can be used to infer other type parameters? No, you didn't provide any type parameters at all. Note that this case applies to function calls where you provide a partial number of type parameters, for example:
func foo[T any, U *T]() {}
Copy after loginIn the above you can only provide
t
, for examplefloat64
, the compiler will construct the replacement map usingt -> float64
and then inferu -> *float64
-
Finally, is there a list of normal function parameters? no.
newb
is empty. - Proposal: Specification: A general approach to type inference, about changing the type inference algorithm to make it easier to implement future enhancements
- Inference based on assignment context, which depends on acceptance of the above proposal
That’s all. The compiler does not infer type parameters based on how the function's return type is used.
At the time of writing, relevant proposals under discussion are:
The above is the detailed content of Unable to infer types in generic parameters of function constructor. For more information, please follow other related articles on the PHP Chinese website!

Hot AI Tools

Undresser.AI Undress
AI-powered app for creating realistic nude photos

AI Clothes Remover
Online AI tool for removing clothes from photos.

Undress AI Tool
Undress images for free

Clothoff.io
AI clothes remover

AI Hentai Generator
Generate AI Hentai for free.

Hot Article

Hot Tools

Notepad++7.3.1
Easy-to-use and free code editor

SublimeText3 Chinese version
Chinese version, very easy to use

Zend Studio 13.0.1
Powerful PHP integrated development environment

Dreamweaver CS6
Visual web development tools

SublimeText3 Mac version
God-level code editing software (SublimeText3)

Hot Topics

In function inheritance, use "base class pointer" and "derived class pointer" to understand the inheritance mechanism: when the base class pointer points to the derived class object, upward transformation is performed and only the base class members are accessed. When a derived class pointer points to a base class object, a downward cast is performed (unsafe) and must be used with caution.

The shortcut keys for running Python code in Sublime Text are: Windows and Linux: Ctrl + BMac: Cmd + B Place the cursor in the code. Press the shortcut key. The code will be run using the system's default Python interpreter.

Notepad++ itself cannot run C language programs and requires an external compiler to compile and execute the code. In order to use an external compiler, you can follow the following steps to set it up: 1. Download and install the C language compiler; 2. Create a custom tool in Notepad++ and configure the compiler executable file path and parameters; 3. Create the C language program and save it with a .c file extension; 4. Select the C language program file and select a custom tool from the "Run" menu to compile; 5. View the compilation results and output a compilation error or success message. If the compilation is successful, an executable file will be generated.

The val keyword in Java is used to declare an immutable local variable, i.e. its value cannot be changed once assigned. Features are: Immutability: Once initialized, the val variable cannot be reassigned. Local scope: val variables are only visible within the block of code in which they are declared. Type inference: The Java compiler will infer the type of the val variable based on the assigned expression. Local variables only: val can only be used to declare local variables, not class fields or method parameters.

The "=" operator in the Java programming language is used to assign a value to a variable, storing the value on the right side of the expression in the variable on the left. Usage: variable = expression, where variable is the name of the variable that receives the assignment, and expression is the code segment that calculates or returns the value.

The const modifier indicates a constant and the value cannot be modified; the static modifier indicates the lifetime and scope of the variable. Data members modified by const cannot be modified after initialization. Variables modified by static are initialized when the program starts and destroyed when the program ends. They will exist even if there is no active object and can be accessed across functions. Local variables modified by const must be initialized when declared, while local variables modified by static can be initialized later. Const-modified class member variables must be initialized in the constructor or initialization list, and static-modified class member variables can be initialized outside the class.

To restrict type parameters in a Java generic method, use the syntax where Bound is the type or interface. As such, parameters only accept types that inherit from Bound or implement the Bound interface. For example, restrict T to a type that is comparable to itself.

The way the compiler differentiates between overloaded functions: by their signature, which is the type of each function parameter. Even if the function name and number of parameters are the same, the compiler can tell them apart as long as the parameter types are different.
