Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
lwm1986
roslyn
提交
e616ed7f
R
roslyn
项目概览
lwm1986
/
roslyn
与 Fork 源项目一致
从无法访问的项目Fork
通知
1
Star
0
Fork
0
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
0
列表
看板
标记
里程碑
合并请求
0
DevOps
流水线
流水线任务
计划
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
R
roslyn
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
0
Issue
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
DevOps
DevOps
流水线
流水线任务
计划
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
流水线任务
提交
Issue看板
体验新版 GitCode,发现更多精彩内容 >>
未验证
提交
e616ed7f
编写于
11月 17, 2017
作者:
S
Sam Harwell
提交者:
GitHub
11月 17, 2017
浏览文件
操作
浏览文件
下载
差异文件
Merge pull request #20240 from alrz/smart-as-p2
Look for type check in local declaration and return statements
上级
4fd056df
e19060e3
变更
3
隐藏空白更改
内联
并排
Showing
3 changed file
with
63 addition
and
45 deletion
+63
-45
src/EditorFeatures/CSharpTest/UsePatternMatching/CSharpAsAndNullCheckTests_FixAllTests.cs
...ePatternMatching/CSharpAsAndNullCheckTests_FixAllTests.cs
+10
-9
src/Features/CSharp/Portable/UsePatternMatching/CSharpAsAndNullCheckCodeFixProvider.cs
...UsePatternMatching/CSharpAsAndNullCheckCodeFixProvider.cs
+9
-5
src/Features/CSharp/Portable/UsePatternMatching/CSharpAsAndNullCheckDiagnosticAnalyzer.cs
...PatternMatching/CSharpAsAndNullCheckDiagnosticAnalyzer.cs
+44
-31
未找到文件。
src/EditorFeatures/CSharpTest/UsePatternMatching/CSharpAsAndNullCheckTests_FixAllTests.cs
浏览文件 @
e616ed7f
...
...
@@ -14,10 +14,9 @@ public async Task FixAllInDocument1()
await
TestInRegularAndScriptAsync
(
@"class C
{
void
M()
int
M()
{
string a;
int[] b;
{|FixAllInDocument:var|} x = o as string;
if (x != null)
{
...
...
@@ -32,14 +31,16 @@ void M()
{
}
if ((b = o as int[]) != null)
{
}
var c = o as string;
var d = c != null ? 1 : 0;
var e = o as string;
return e != null ? 1 : 0;
}
}"
,
@"class C
{
void
M()
int
M()
{
if (o is string x)
{
...
...
@@ -53,9 +54,9 @@ void M()
{
}
if (o is int[] b)
{
}
var d = o is string c ? 1 : 0;
return o is string e ? 1 : 0;
}
}"
);
}
...
...
src/Features/CSharp/Portable/UsePatternMatching/CSharpAsAndNullCheckCodeFixProvider.cs
浏览文件 @
e616ed7f
...
...
@@ -53,8 +53,12 @@ private static ExpressionSyntax GetCondition(SyntaxNode node)
return
((
WhileStatementSyntax
)
node
).
Condition
;
case
SyntaxKind
.
IfStatement
:
return
((
IfStatementSyntax
)
node
).
Condition
;
case
SyntaxKind
.
ReturnStatement
:
return
((
ReturnStatementSyntax
)
node
).
Expression
;
case
SyntaxKind
.
LocalDeclarationStatement
:
return
((
LocalDeclarationStatementSyntax
)
node
).
Declaration
.
Variables
[
0
].
Initializer
.
Value
;
default
:
throw
ExceptionUtilities
.
Un
reachable
;
throw
ExceptionUtilities
.
Un
expectedValue
(
node
.
Kind
())
;
}
}
...
...
@@ -64,12 +68,12 @@ private static ExpressionSyntax GetCondition(SyntaxNode node)
CancellationToken
cancellationToken
)
{
var
localDeclarationLocation
=
diagnostic
.
AdditionalLocations
[
0
];
var
ifOrWhile
StatementLocation
=
diagnostic
.
AdditionalLocations
[
1
];
var
target
StatementLocation
=
diagnostic
.
AdditionalLocations
[
1
];
var
conditionLocation
=
diagnostic
.
AdditionalLocations
[
2
];
var
asExpressionLocation
=
diagnostic
.
AdditionalLocations
[
3
];
var
localDeclaration
=
(
LocalDeclarationStatementSyntax
)
localDeclarationLocation
.
FindNode
(
cancellationToken
);
var
ifOrWhileStatement
=
(
StatementSyntax
)
ifOrWhile
StatementLocation
.
FindNode
(
cancellationToken
);
var
targetStatement
=
(
StatementSyntax
)
target
StatementLocation
.
FindNode
(
cancellationToken
);
var
conditionPart
=
(
BinaryExpressionSyntax
)
conditionLocation
.
FindNode
(
cancellationToken
);
var
asExpression
=
(
BinaryExpressionSyntax
)
asExpressionLocation
.
FindNode
(
cancellationToken
);
...
...
@@ -79,7 +83,7 @@ private static ExpressionSyntax GetCondition(SyntaxNode node)
SyntaxFactory
.
SingleVariableDesignation
(
localDeclaration
.
Declaration
.
Variables
[
0
].
Identifier
.
WithoutTrivia
())));
var
currentCondition
=
GetCondition
(
ifOrWhile
Statement
);
var
currentCondition
=
GetCondition
(
target
Statement
);
var
updatedCondition
=
currentCondition
.
ReplaceNode
(
conditionPart
,
updatedConditionPart
);
var
block
=
(
BlockSyntax
)
localDeclaration
.
Parent
;
...
...
@@ -93,7 +97,7 @@ private static ExpressionSyntax GetCondition(SyntaxNode node)
(
s
,
g
)
=>
s
.
WithPrependedNonIndentationTriviaFrom
(
localDeclaration
));
editor
.
RemoveNode
(
localDeclaration
,
SyntaxRemoveOptions
.
KeepUnbalancedDirectives
);
editor
.
ReplaceNode
(
ifOrWhile
Statement
,
(
currentStatement
,
g
)
=>
editor
.
ReplaceNode
(
target
Statement
,
(
currentStatement
,
g
)
=>
{
var
updatedStatement
=
currentStatement
.
ReplaceNode
(
GetCondition
(
currentStatement
),
updatedCondition
);
return
updatedStatement
.
WithAdditionalAnnotations
(
Formatter
.
Annotation
);
...
...
src/Features/CSharp/Portable/UsePatternMatching/CSharpAsAndNullCheckDiagnosticAnalyzer.cs
浏览文件 @
e616ed7f
...
...
@@ -39,7 +39,10 @@ public CSharpAsAndNullCheckDiagnosticAnalyzer()
protected
override
void
InitializeWorker
(
AnalysisContext
context
)
=>
context
.
RegisterSyntaxNodeAction
(
SyntaxNodeAction
,
SyntaxKind
.
IfStatement
,
SyntaxKind
.
WhileStatement
);
SyntaxKind
.
IfStatement
,
SyntaxKind
.
WhileStatement
,
SyntaxKind
.
ReturnStatement
,
SyntaxKind
.
LocalDeclarationStatement
);
private
void
SyntaxNodeAction
(
SyntaxNodeAnalysisContext
syntaxContext
)
{
...
...
@@ -68,34 +71,33 @@ private void SyntaxNodeAction(SyntaxNodeAnalysisContext syntaxContext)
return
;
}
var
ifOrWhile
Statement
=
(
StatementSyntax
)
node
;
var
leftmostCondition
=
GetLeftmostCondition
(
ifOrWhile
Statement
);
if
(!
leftmostCondition
.
IsKind
(
SyntaxKind
.
NotEqualsExpression
,
out
BinaryExpressionSyntax
comparison
))
var
target
Statement
=
(
StatementSyntax
)
node
;
var
leftmostCondition
=
GetLeftmostCondition
(
target
Statement
);
if
(!
leftmostCondition
.
IsKind
(
SyntaxKind
.
NotEqualsExpression
,
out
BinaryExpressionSyntax
notEquals
))
{
return
;
}
var
operand
=
GetNullCheckOperand
(
comparison
.
Left
,
comparison
.
Right
)?.
WalkDownParentheses
();
var
operand
=
GetNullCheckOperand
(
notEquals
.
Left
,
notEquals
.
Right
)?.
WalkDownParentheses
();
if
(
operand
==
null
)
{
return
;
}
// if/while has to be in a block so we can at least look for a preceding local variable declaration.
if
(!
ifOrWhile
Statement
.
Parent
.
IsKind
(
SyntaxKind
.
Block
,
out
BlockSyntax
parentBlock
))
if
(!
target
Statement
.
Parent
.
IsKind
(
SyntaxKind
.
Block
,
out
BlockSyntax
parentBlock
))
{
return
;
}
var
blockStatements
=
parentBlock
.
Statements
;
if
(!
TryGetTypeCheckParts
(
operand
,
ifOrWhileStatement
,
blockStatements
,
if
(!
TryGetTypeCheckParts
(
operand
,
targetStatement
,
parentBlock
,
out
var
declarator
,
out
var
asExpression
))
{
return
;
}
var
semanticModel
=
syntaxContext
.
SemanticModel
;
if
(
semanticModel
.
GetSymbolInfo
(
comparison
).
GetAnySymbol
().
IsUserDefinedOperator
())
if
(
semanticModel
.
GetSymbolInfo
(
notEquals
).
GetAnySymbol
().
IsUserDefinedOperator
())
{
return
;
}
...
...
@@ -139,7 +141,7 @@ private void SyntaxNodeAction(SyntaxNodeAnalysisContext syntaxContext)
// in the Else branch of the IfStatement, or after the IfStatement. Make sure
// that doesn't cause definite assignment issues.
if
(
IsAccessedBeforeAssignment
(
semanticModel
,
localSymbol
,
declarationStatement
,
ifOrWhileStatement
,
blockStatements
,
cancellationToken
))
declarationStatement
,
targetStatement
,
parentBlock
,
cancellationToken
))
{
return
;
}
...
...
@@ -147,7 +149,7 @@ private void SyntaxNodeAction(SyntaxNodeAnalysisContext syntaxContext)
// Looks good!
var
additionalLocations
=
ImmutableArray
.
Create
(
declarationStatement
.
GetLocation
(),
ifOrWhile
Statement
.
GetLocation
(),
target
Statement
.
GetLocation
(),
leftmostCondition
.
GetLocation
(),
asExpression
.
GetLocation
());
...
...
@@ -162,22 +164,23 @@ private void SyntaxNodeAction(SyntaxNodeAnalysisContext syntaxContext)
SemanticModel
semanticModel
,
ISymbol
localVariable
,
StatementSyntax
declarationStatement
,
StatementSyntax
usage
Statement
,
SyntaxList
<
StatementSyntax
>
blockStatements
,
StatementSyntax
target
Statement
,
BlockSyntax
parentBlock
,
CancellationToken
cancellationToken
)
{
var
isAssigned
=
false
;
var
isAccessedBeforeAssignment
=
false
;
var
usageIndex
=
blockStatements
.
IndexOf
(
usageStatement
);
var
declarationIndex
=
blockStatements
.
IndexOf
(
declarationStatement
);
var
statements
=
parentBlock
.
Statements
;
var
targetIndex
=
statements
.
IndexOf
(
targetStatement
);
var
declarationIndex
=
statements
.
IndexOf
(
declarationStatement
);
// Since we're going to remove this declaration-statement,
// we need to first ensure that it's not used up to the target statement.
for
(
var
index
=
declarationIndex
+
1
;
index
<
usage
Index
;
index
++)
for
(
var
index
=
declarationIndex
+
1
;
index
<
target
Index
;
index
++)
{
CheckDefiniteAssignment
(
semanticModel
,
localVariable
,
blockS
tatements
[
index
],
semanticModel
,
localVariable
,
s
tatements
[
index
],
out
isAssigned
,
out
isAccessedBeforeAssignment
,
cancellationToken
);
...
...
@@ -189,7 +192,7 @@ private void SyntaxNodeAction(SyntaxNodeAnalysisContext syntaxContext)
// In case of an if-statement, we need to check if the variable
// is being accessed before assignment in the else clause.
if
(
usageStatement
is
IfStatementSyntax
ifStatement
)
if
(
targetStatement
.
IsKind
(
SyntaxKind
.
IfStatement
,
out
IfStatementSyntax
ifStatement
)
)
{
CheckDefiniteAssignment
(
semanticModel
,
localVariable
,
ifStatement
.
Else
,
...
...
@@ -208,10 +211,10 @@ private void SyntaxNodeAction(SyntaxNodeAnalysisContext syntaxContext)
}
// Make sure that no access is made to the variable before assignment in the subsequent statements
for
(
int
index
=
usageIndex
+
1
,
n
=
blockS
tatements
.
Count
;
index
<
n
;
index
++)
for
(
int
index
=
targetIndex
+
1
,
n
=
s
tatements
.
Count
;
index
<
n
;
index
++)
{
CheckDefiniteAssignment
(
semanticModel
,
localVariable
,
blockS
tatements
[
index
],
semanticModel
,
localVariable
,
s
tatements
[
index
],
out
isAssigned
,
out
isAccessedBeforeAssignment
,
cancellationToken
);
...
...
@@ -224,7 +227,7 @@ private void SyntaxNodeAction(SyntaxNodeAnalysisContext syntaxContext)
{
// The scope of pattern variables in a while-statement does not leak out to
// the enclosing block so we bail also if there is any assignments afterwards.
return
usage
Statement
.
Kind
()
==
SyntaxKind
.
WhileStatement
;
return
target
Statement
.
Kind
()
==
SyntaxKind
.
WhileStatement
;
}
}
...
...
@@ -261,8 +264,8 @@ private void SyntaxNodeAction(SyntaxNodeAnalysisContext syntaxContext)
private
static
bool
TryGetTypeCheckParts
(
SyntaxNode
operand
,
StatementSyntax
usage
Statement
,
SyntaxList
<
StatementSyntax
>
blockStatements
,
StatementSyntax
target
Statement
,
BlockSyntax
parentBlock
,
out
SyntaxNode
variableDeclarator
,
out
SyntaxNode
asExpression
)
{
...
...
@@ -279,12 +282,12 @@ private void SyntaxNodeAction(SyntaxNodeAnalysisContext syntaxContext)
//
// That's because in this case, unlike the original code, we're type-checking in every iteration
// so we do not replace simple null check with the "is" operator if it's in a while loop
case
SyntaxKind
.
IdentifierName
when
usage
Statement
.
Kind
()
!=
SyntaxKind
.
WhileStatement
:
case
SyntaxKind
.
IdentifierName
when
target
Statement
.
Kind
()
!=
SyntaxKind
.
WhileStatement
:
{
// var x = e as T;
// if (x != null) F(x);
var
identifier
=
(
IdentifierNameSyntax
)
operand
;
var
declarator
=
TryFindVariableDeclarator
(
identifier
,
usageStatement
,
blockStatements
);
var
declarator
=
TryFindVariableDeclarator
(
identifier
,
targetStatement
,
parentBlock
);
var
initializerValue
=
declarator
?.
Initializer
?.
Value
;
if
(!
initializerValue
.
IsKind
(
SyntaxKind
.
AsExpression
))
{
...
...
@@ -308,7 +311,7 @@ private void SyntaxNodeAction(SyntaxNodeAnalysisContext syntaxContext)
}
var
identifier
=
(
IdentifierNameSyntax
)
assignment
.
Left
;
var
declarator
=
TryFindVariableDeclarator
(
identifier
,
usageStatement
,
blockStatements
);
var
declarator
=
TryFindVariableDeclarator
(
identifier
,
targetStatement
,
parentBlock
);
if
(
declarator
==
null
)
{
break
;
...
...
@@ -326,12 +329,13 @@ private void SyntaxNodeAction(SyntaxNodeAnalysisContext syntaxContext)
}
private
static
VariableDeclaratorSyntax
TryFindVariableDeclarator
(
IdentifierNameSyntax
identifier
,
StatementSyntax
usageStatement
,
SyntaxList
<
StatementSyntax
>
blockStatements
)
IdentifierNameSyntax
identifier
,
StatementSyntax
targetStatement
,
BlockSyntax
parentBlock
)
{
var
usageIndex
=
blockStatements
.
IndexOf
(
usageStatement
);
for
(
var
index
=
usageIndex
-
1
;
index
>=
0
;
index
--)
var
statement
=
parentBlock
.
Statements
;
var
targetIndex
=
statement
.
IndexOf
(
targetStatement
);
for
(
var
index
=
targetIndex
-
1
;
index
>=
0
;
index
--)
{
if
(!
blockStatements
[
index
].
IsKind
(
SyntaxKind
.
LocalDeclarationStatement
,
if
(!
statement
[
index
].
IsKind
(
SyntaxKind
.
LocalDeclarationStatement
,
out
LocalDeclarationStatementSyntax
declarationStatement
))
{
continue
;
...
...
@@ -369,7 +373,7 @@ private static SyntaxNode GetLeftmostCondition(SyntaxNode node)
{
while
(
true
)
{
switch
(
node
.
Kind
())
switch
(
node
?
.
Kind
())
{
case
SyntaxKind
.
WhileStatement
:
node
=
((
WhileStatementSyntax
)
node
).
Condition
;
...
...
@@ -377,6 +381,15 @@ private static SyntaxNode GetLeftmostCondition(SyntaxNode node)
case
SyntaxKind
.
IfStatement
:
node
=
((
IfStatementSyntax
)
node
).
Condition
;
continue
;
case
SyntaxKind
.
ReturnStatement
:
node
=
((
ReturnStatementSyntax
)
node
).
Expression
;
continue
;
case
SyntaxKind
.
LocalDeclarationStatement
:
var
declarators
=
((
LocalDeclarationStatementSyntax
)
node
).
Declaration
.
Variables
;
// We require this to be the only declarator in the declaration statement
// to simplify definitive assignment check and the code fix for now
node
=
declarators
.
Count
==
1
?
declarators
[
0
].
Initializer
?.
Value
:
null
;
continue
;
case
SyntaxKind
.
ParenthesizedExpression
:
node
=
((
ParenthesizedExpressionSyntax
)
node
).
Expression
;
continue
;
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录